extract_param/
extract_param.rs

1use std::env;
2use std::error::Error;
3use std::path::Path;
4
5use rust_grib_decoder::{
6    find_candidates_for_param, read_first_grib2_latlon_grid, read_grib2_section5_template42_params_by_message,
7    decode_template42_try_message_payload, summarize_file,
8};
9
10#[cfg(feature = "grib-support")]
11use grib::{from_reader, Grib2SubmessageDecoder};
12
13fn main() -> Result<(), Box<dyn Error>> {
14    let args: Vec<_> = env::args_os().skip(1).collect();
15    if args.len() < 4 {
16        eprintln!("usage: extract_param <file.grib2> <param> <lon> <lat>");
17        eprintln!("example: extract_param data.grib2 msl 113.363 22.962");
18        return Ok(());
19    }
20
21    let file_s = args[0].to_string_lossy().to_string();
22    let path = Path::new(&file_s);
23    let param = args[1].to_string_lossy().to_string();
24    let lon: f64 = args[2].to_string_lossy().parse()?;
25    let lat: f64 = args[3].to_string_lossy().parse()?;
26
27    println!("Summarizing file: {}", path.display());
28    let fs = summarize_file(path)?;
29    for m in fs.messages.iter().take(10) {
30        println!("msg {}: key='{}' repr={:?} payload_len={} decode_ok={}", m.message_index, m.key, m.repr_template, m.payload_len, m.decode_ok);
31    }
32
33    println!("\nFinding candidates for param '{}':", param);
34    let candidates = find_candidates_for_param(path, &param)?;
35
36    // Read grid early so fallback paths can use it
37    let grid = read_first_grib2_latlon_grid(path)?;
38
39    if candidates.is_empty() {
40        println!("No candidates found for '{}'. Trying filename prefix fallback...", param);
41        // Try filename prefix fallback using the ParamAlias file_prefixes
42        let alias = rust_grib_decoder::param_alias::ParamAlias::for_param(&param);
43        if let Some(fname) = path.file_name().and_then(|s| s.to_str()) {
44            for prefix in alias.file_prefixes.iter() {
45                if fname.starts_with(prefix) {
46                    println!("Filename '{}' starts with prefix '{}', attempting message 0 decode", fname, prefix);
47                    // Try to decode message 0 using template42 Section5 params if present
48                    let params_vec = read_grib2_section5_template42_params_by_message(path)?;
49                    if let Some(Some(params)) = params_vec.get(0) {
50                        if let Ok(values) = decode_template42_try_message_payload(path, 0, grid.ni, grid.nj, params) {
51                            let val = interp_value(&values, grid.ni, grid.nj, grid.lon1_deg, grid.lat1_deg, grid.di_deg, grid.dj_deg, lon, lat);
52                            println!("Decoded {} samples from message 0 — interpolated value at {}x{} = {}", values.len(), lon, lat, val);
53                            return Ok(());
54                        } else {
55                            eprintln!("template42 rust-aec decode failed for message 0");
56                        }
57                    }
58                    #[cfg(feature = "grib-support")]
59                    {
60                        use grib::{from_reader, Grib2SubmessageDecoder};
61                        use std::fs::File;
62                        use std::io::BufReader;
63                        let f = File::open(path)?;
64                        let r = BufReader::new(f);
65                        let grib2 = from_reader(r)?;
66                        for ((midx, _sidx), sub) in grib2.iter() {
67                            if midx == 0 {
68                                let (w, h) = sub.grid_shape()?;
69                                let decoder = Grib2SubmessageDecoder::from(sub)?;
70                                match decoder.dispatch() {
71                                    Ok(decoded) => {
72                                        let vals: Vec<f32> = decoded.collect();
73                                        let val = interp_value(&vals, w, h, grid.lon1_deg, grid.lat1_deg, grid.di_deg, grid.dj_deg, lon, lat);
74                                        println!("grib crate decoded {} samples (msg 0) — interpolated value = {}", vals.len(), val);
75                                        return Ok(());
76                                    }
77                                    Err(e) => eprintln!("grib crate decode error (msg 0): {}", e),
78                                }
79                            }
80                        }
81                    }
82                }
83            }
84        }
85
86        println!("No decode succeeded using filename prefix fallback.");
87        return Ok(());
88    }
89
90    for (idx, reason, key, ft) in candidates.iter().take(3) {
91        println!("candidate: msg={} reason={:?} key='{}' ft={:?}", idx, reason, key, ft);
92    }
93
94    // Choose the first candidate and attempt to decode it
95    let (msg_idx, _, key, _ft) = &candidates[0];
96    println!("\nAttempting to decode candidate msg {} ('{}')", msg_idx, key);
97
98    let grid = read_first_grib2_latlon_grid(path)?;
99    println!("Grid: {}x{} lon1={} lat1={} di={} dj={}", grid.ni, grid.nj, grid.lon1_deg, grid.lat1_deg, grid.di_deg, grid.dj_deg);
100
101    // Try template-42 rust-aec decode using Section5 params if available
102    let params_vec = read_grib2_section5_template42_params_by_message(path)?;
103    if let Some(Some(params)) = params_vec.get(*msg_idx) {
104        println!("Found template42 params for msg {} — trying rust-aec decode", msg_idx);
105        match decode_template42_try_message_payload(path, *msg_idx, grid.ni, grid.nj, params) {
106            Ok(values) => {
107                let val = interp_value(&values, grid.ni, grid.nj, grid.lon1_deg, grid.lat1_deg, grid.di_deg, grid.dj_deg, lon, lat);
108                println!("Decoded {} samples — interpolated value at {}x{} = {}", values.len(), lon, lat, val);
109                return Ok(());
110            }
111            Err(e) => eprintln!("rust-aec decode failed: {}", e),
112        }
113    }
114
115    #[cfg(feature = "grib-support")]
116    {
117        println!("Attempting to decode using `grib` crate for msg {}", msg_idx);
118        use std::fs::File;
119        use std::io::BufReader;
120        let f = File::open(path)?;
121        let r = BufReader::new(f);
122        let grib2 = from_reader(r)?;
123        for ((midx, _sidx), sub) in grib2.iter() {
124            if midx == *msg_idx {
125                let (w, h) = sub.grid_shape()?;
126                let decoder = Grib2SubmessageDecoder::from(sub)?;
127                match decoder.dispatch() {
128                    Ok(decoded) => {
129                        let vals: Vec<f32> = decoded.collect();
130                        let val = interp_value(&vals, w, h, grid.lon1_deg, grid.lat1_deg, grid.di_deg, grid.dj_deg, lon, lat);
131                        println!("grib crate decoded {} samples — interpolated value = {}", vals.len(), val);
132                        return Ok(());
133                    }
134                    Err(e) => eprintln!("grib crate decode error: {}", e),
135                }
136            }
137        }
138    }
139
140    println!("Could not decode the selected message (no template42 params and grib-support not enabled or decoding failed).");
141    Ok(())
142}
143
144fn interp_value(values: &[f32], width: usize, height: usize, lon1: f64, lat1: f64, di: f64, dj: f64, lon: f64, lat: f64) -> f64 {
145    let mut dlon = lon - lon1;
146    while dlon < 0.0 { dlon += 360.0; }
147    while dlon >= 360.0 { dlon -= 360.0; }
148    let i_f = dlon / di;
149    let j_f = (lat1 - lat) / dj;
150    if !i_f.is_finite() || !j_f.is_finite() { return f64::NAN; }
151    let ni = width as isize; let nj = height as isize;
152
153    let i0 = i_f.floor() as isize; let j0 = j_f.floor() as isize; let i1 = i0 + 1; let j1 = j0 + 1;
154    let dx = i_f - (i0 as f64); let dy = j_f - (j0 as f64);
155    let i0w = (((i0 % ni) + ni) % ni) as usize; let i1w = (((i1 % ni) + ni) % ni) as usize;
156    let j0c = j0.clamp(0, nj - 1) as usize; let j1c = j1.clamp(0, nj - 1) as usize;
157    let idx00 = j0c.saturating_mul(width).saturating_add(i0w);
158    let idx10 = j0c.saturating_mul(width).saturating_add(i1w);
159    let idx01 = j1c.saturating_mul(width).saturating_add(i0w);
160    let idx11 = j1c.saturating_mul(width).saturating_add(i1w);
161    let s00 = values.get(idx00).copied().unwrap_or(f32::NAN);
162    let s10 = values.get(idx10).copied().unwrap_or(f32::NAN);
163    let s01 = values.get(idx01).copied().unwrap_or(f32::NAN);
164    let s11 = values.get(idx11).copied().unwrap_or(f32::NAN);
165    if !s00.is_finite() || !s10.is_finite() || !s01.is_finite() || !s11.is_finite() { return f64::NAN; }
166    let v0 = s00 * (1.0 - dx as f32) + s10 * (dx as f32);
167    let v1 = s01 * (1.0 - dx as f32) + s11 * (dx as f32);
168    let v = v0 * (1.0 - dy as f32) + v1 * (dy as f32);
169    v as f64
170}