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, ¶m)?;
35
36 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 let alias = rust_grib_decoder::param_alias::ParamAlias::for_param(¶m);
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 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 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 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}