Skip to content
| Marketplace
Sign in
Visual Studio Code>Visualization>FieldglassNew to Visual Studio Code? Get it now.
Fieldglass

Fieldglass

Preview

Fieldglass

|
21 installs
| (0) | Free
View and render GRIB1, GRIB2, and NetCDF weather data files in VS Code
Installation
Launch VS Code Quick Open (Ctrl+P), paste the following command, and press enter.
Copied to clipboard
More Info

Fieldglass

CI Coverage License: MIT OR Apache-2.0

A Visual Studio Code extension for viewing meteorological data files (GRIB1, GRIB2, and NetCDF) right in the editor. It runs on a native Rust module and needs no extra dependencies to install. Fieldglass bundles a prebuilt binary for each supported platform and selects the right one automatically.

Fieldglass viewer rendering a multi-message GRIB1 file

Latest release

Status

This is an early public release (beta). For GRIB1, you can read every message's metadata and render its grid as a 2-D color image, either in the grid's own coordinates or reprojected onto a map. Five projection targets are available (source, equirectangular, Web Mercator, orthographic, and polar stereographic), and you can overlay coastlines and a latitude/longitude grid. GRIB2 reads metadata for every message and renders values for simple-packed messages. NetCDF reads the full structure of classic files (CDF-1/2/5): dimensions, variables, and global attributes. Still to come: value decode for more GRIB2 packings and for NetCDF, deep NetCDF-4 / HDF5 parsing, and metadata editing. The feature matrix below shows exactly what works today.

Feature matrix

Feature GRIB1 GRIB2 NetCDF
Format detection from magic bytes ✅ ✅ ✅
File-extension association (.grb / .grib* / .nc*) ✅ ✅ ✅
Open via Reopen Editor With… for unrecognized files ✅ ✅ ✅
Indicator / header section parsing ✅ ✅ ✅ classic / 🚧 NetCDF-4
Per-message metadata (parameter, level, time, forecast period) ✅ ✅ (§0–§4) ✅ classic (dims / vars / attrs)
Grid description (lat/lon, Gaussian, polar stereo, Lambert) ✅ 🚧 lat/lon (3.0), rotated lat/lon (3.1), Mercator (3.10), polar stereo (3.20), Lambert (3.30), Gaussian (3.40), space view (3.90) ❌ Not yet
WMO ON388 lookups (parameter, centre, level type) ✅ 🚧 Tables 0.0 / 1.2 / 1.3 / 1.4 / 3.1 / 3.2 / 4.1 / 4.2 / 4.3 / 4.4 / 4.5 / 4.6 / 4.10 + centres (subset) n/a
Tabular metadata viewer ✅ ✅ (§0–§4) ✅ classic / 🚧 NetCDF-4
Binary data section decoding (Rust API) ✅ 🚧 simple packing (5.0); 5.2 / 5.3 / 5.4 / 5.40 / 5.41 / 5.42 follow-up ❌ Not yet
Metadata editing ❌ Not yet ❌ Not yet ❌ Not yet
2-D grid rendering with colormap ✅ 🚧 simple-packed messages only ❌ Not yet
Render-panel projection picker (5 targets) ✅ ✅ (simple-packed messages) n/a
Coastline + lat/lon grid overlay ✅ ✅ (simple-packed messages) n/a
Resampling picker (nearest / bilinear) ✅ ✅ n/a

Format-agnostic features:

  • Hex and ASCII fallback view for files whose contents are not a recognized format. ✅
  • Files without a recognized extension can still be opened through Reopen Editor With… → Fieldglass Viewer.

GRIB1 packing modes

A GRIB1 file's BDS (Binary Data Section) flag bits select one of several packing schemes. Decoding (and therefore 2-D rendering) covers only some of them today; metadata and format detection do not depend on the packing mode and work on every file. For an unsupported variant, the decoder returns an error naming the eccodes-style packingType, so you can match it against eccodes' definitions.

BDS packing eccodes packingType Decode Notes
Simple grid-point packing grid_simple ✅ The bulk of CMC, NCEP non-operational data, and pygrib sample sets.
Constant field (bits_per_value = 0) grid_simple ✅ Special case of simple packing.
Second-order, no SPD grid_second_order_no_SPD ✅ Decodes via the shared general-extended path (order 0). Cross-validated against eccodes 2.34 with a hand-built oracle fixture (eccodes won't encode this order but decodes it).
Second-order, SPD-1 grid_second_order_SPD1 ✅ Decodes via the shared general-extended path (order 1). Cross-validated against eccodes 2.34 with a hand-built oracle fixture.
Second-order, SPD-2 (ECMWF default) grid_second_order ✅ Most common in ECMWF MARS-derived files. Cross-validated against eccodes 2.34.
Second-order, SPD-3 grid_second_order_SPD3 ✅ Cross-validated against eccodes 2.34 (re-encoded from the SPD-2 fixture), with and without boustrophedonic row ordering.
Second-order, row-by-row grid_second_order_row_by_row ✅ Classic WMO layout: one group per row, per-row widths, no SPD. Cross-validated against eccodes 2.34 with a hand-built oracle fixture.
Second-order, constant width grid_second_order_constant_width ✅ Classic WMO layout: explicit secondary bitmap + single shared width. Cross-validated against eccodes 2.34 with a hand-built oracle fixture.
Second-order, general (legacy) grid_second_order_general_grib1 ✅ Classic WMO layout: secondary-bitmap-delimited variable-length groups + per-group widths. Cross-validated against eccodes 2.34 with a hand-built oracle fixture.
IEEE 754 raw floats grid_ieee ✅ Values stored verbatim as big-endian floats; 32-bit (precision = 1) and 64-bit (precision = 2). Cross-validated against eccodes 2.34. 128-bit (precision = 3) is unsupported; eccodes returns NOT_IMPLEMENTED for it too.
Matrix-of-values (scalar form) grid_simple_matrix ✅ matrixOfValues = 0: a simple-packed body behind the matrix sub-header (what eccodes emits for packingType=grid_simple_matrix). Cross-validated against eccodes 2.34.
Matrix-of-values (true matrix) grid_simple_matrix ✅ matrixOfValues = 1: an NR×NC matrix at every grid point, via secondary bitmaps. Decoded through Grib1Reader::decode_matrix_message (not a single 2-D field, so it bypasses the scalar path). eccodes 2.34 can neither encode nor decode this variant, so it's validated against the eccodes definition/accessor source + a hand-computed fixture rather than a grib_get_data oracle.
Spherical-harmonic coefficients spectral_simple / spectral_complex ❌ Planned IFS analyses; needs an inverse Legendre transform.
JPEG 2000 / PNG grid_jpeg / grid_png ❌ n/a Not defined for GRIB1 edition 1 (eccodes' packingType concept lists them only to avoid set-failures); no encoder and no obtainable fixture. Common in GRIB2, tracked there.

Known limitations

This is a beta. Things to be aware of:

  • No metadata editing in the viewer. The Rust API has byte-level patching for the forecast period (P1) and the webview retains the full undo/redo wiring, but the editable affordance is hidden in beta until general PDS-field editing lands. For now Fieldglass is a read-only viewer.
  • The globe-style projections use fixed presets. The projection picker offers five targets: source (the grid in its native shape), equirectangular, Web Mercator, orthographic, and polar stereographic, plus an optional coastline and lat/lon grid overlay. GRIB1 lat/lon, Gaussian, Lambert, and polar stereographic grids, and GRIB2 polar stereographic (§3.20) grids, all reproject into the flat (lat/lon-box) targets. The two globe-style targets (orthographic and polar stereographic) are framed by a fixed center or hemisphere preset rather than free-form numbers; arbitrary projection centers are a follow-up.
  • GRIB2: full §0–§7 parsing, simple-packed value decode. .grb2 / .grib2 files enumerate messages and show edition, discipline, total length, originating centre, reference time, production status, data type, parameter name, level, forecast time, and grid geometry (templates 3.0 / 3.1 / 3.10 / 3.20 / 3.30 / 3.40 / 3.90: regular lat/lon, rotated lat/lon, Mercator, polar stereographic, Lambert Conformal, regular and reduced Gaussian, and space-view perspective). Value decoding works for simple packing (DRS template 5.0). Complex packing (5.2 / 5.3), IEEE (5.4), JPEG 2000 (5.40), PNG (5.41), and CCSDS (5.42) are tracked under separate issues; those messages still parse to the section level, but decode_message_values returns an UnsupportedSection error naming the template.
  • NetCDF-4 / HDF5: header probe only. Classic NetCDF (CDF-1 / CDF-2 / CDF-5) parses fully and renders dimensions, global attributes, and variables. NetCDF-4 / HDF5 files are validated and report the superblock version, but deep traversal (groups, datasets, attributes through the HDF5 object header tree) is a follow-up.
  • GRIB1 GDS coverage: Lat/Lon, Gaussian, Polar Stereographic, and Lambert Conformal grids are parsed. Reduced grids, rotated/oblique projections, and predefined grids (grid_number != 255) are not yet supported and will render as unsupported.
  • Parameter table coverage: WMO ON388 Table 2 (versions 1–3) only. ECMWF local tables (versions 128+) and other centre-specific extensions resolve as Unknown.
  • Large files: the extension reads the whole file into memory via vscode.workspace.fs.readFile to keep remote/virtual workspaces working. Multi-GB GRIB archives are not the target use case yet.

Installation

Pre-built binaries for all supported platforms are bundled inside a single .vsix package. The extension selects the correct binary at runtime based on the host platform and architecture.

Supported platforms:

  • Linux x64 (glibc), Linux arm64 (glibc)
  • macOS x64, macOS arm64
  • Windows x64, Windows arm64

macOS

  1. Download the latest fieldglass-x.y.z.vsix from the releases page.
  2. Open VS Code, run "Extensions: Install from VSIX..." from the command palette, and select the downloaded file. Alternatively, from a terminal:
    code --install-extension fieldglass-x.y.z.vsix
    
  3. Reload the VS Code window.

Linux

  1. Download the latest fieldglass-x.y.z.vsix from the releases page.
  2. Open VS Code, run "Extensions: Install from VSIX..." from the command palette, and select the downloaded file. Alternatively, from a terminal:
    code --install-extension fieldglass-x.y.z.vsix
    
  3. Reload the VS Code window.

Windows

  1. Download the latest fieldglass-x.y.z.vsix from the releases page.
  2. Open VS Code, run "Extensions: Install from VSIX..." from the command palette, and select the downloaded file. Alternatively, from PowerShell or Command Prompt:
    code --install-extension fieldglass-x.y.z.vsix
    
  3. Reload the VS Code window.

Usage

Open any file with a supported extension. VS Code will use Fieldglass as the default editor and render a metadata table for each message in the file. To open an unrecognized file, right-click the file in the Explorer and choose "Open With...", then select "Fieldglass Viewer".

Development

Prerequisites

  • Rust (stable toolchain, edition 2024 — Rust 1.85 or newer)
  • Node.js 22 or newer
  • Python 3.10 or newer (only for the dev tooling — pre-commit, semgrep)
  • Visual Studio Code 1.85 or newer

Repository layout

Path Purpose
crates/fieldglass-core Format-agnostic traits and shared metadata types.
crates/fieldglass-grib1 GRIB1 parser, organized by section (is.rs, pds.rs, gds.rs, bds.rs) and WMO table lookups (tables.rs).
crates/fieldglass-grib2 GRIB2 reader — full §0–§7 parsing, message enumeration, and value decoding for simple packing (DRS template 5.0). Grid templates 3.0 / 3.1 / 3.10 / 3.20 / 3.30 / 3.40 / 3.90; product templates 4.0 / 4.8 / 4.11. Complex / JPEG 2000 / PNG / CCSDS packings are tracked under separate issues.
crates/fieldglass-netcdf NetCDF reader — full classic (CDF-1/2/5) header parser; HDF5 / NetCDF-4 superblock probe with deep traversal as a follow-up.
crates/fieldglass-napi Node.js bindings exposed via napi-rs. The only crate that knows about Node.
extension/ TypeScript VS Code extension. Registers a custom read-only editor and renders a webview.

Initial setup

git clone git@github.com:D0ubleD0uble/fieldglass.git
cd fieldglass
pipx install pre-commit              # or: pip install --user pre-commit
npm install                          # installs @napi-rs/cli and activates git hooks

The root npm install installs @napi-rs/cli (used to build the native module) and runs an npm prepare step that activates the repository's git hooks (see Pre-commit hooks below). If pre-commit isn't on PATH the prepare step prints a hint and continues; you can install it later and re-run npm install.

Building the native module

The compiled binary must be present in extension/bin/ for the extension to load. From the repository root:

cd crates/fieldglass-napi
npx napi build --platform --release --output-dir ../../extension/bin

This produces a file such as extension/bin/fieldglass.linux-x64-gnu.node along with extension/bin/index.d.ts. Repeat after changing any Rust code.

Building the extension

cd extension
npm install
npm run compile

For continuous compilation during development, run npm run watch instead.

Running the extension

Open the repository in VS Code and press F5. An Extension Development Host window will launch with Fieldglass loaded. Open any supported file in that window to test changes.

Tests

Run the full Rust test suite:

cargo test

Run tests for a single crate or a single test by name substring:

cargo test -p fieldglass-grib1
cargo test -p fieldglass-grib1 parse_pds

Extension integration tests

The TypeScript side has a small @vscode/test-electron suite that boots a real VS Code extension host and exercises the editor + render-panel paths end-to-end. The render-panel tests in particular round-trip a real gridReady payload through webview.postMessage so the wire format between the extension host and the webview is exercised under the same serializer as production. Coverage spans all three primary file formats (GRIB1 / GRIB2 render, NetCDF metadata) plus a regression test for the Buffer-vs-Uint8Array wire-format constraint.

cd extension
npm test

Tests require a display; CI runs them under xvfb-run -a npm test.

eccodes reference snapshots

fieldglass-grib2 cross-checks every bundled GRIB2 fixture against the eccodes reference implementation via JSON snapshots. The snapshots live next to each fixture as tests/fixtures/{name}.grib2.eccodes.ref.json and are checked into git, so the test (crates/fieldglass-grib2/tests/eccodes_reference.rs) has zero runtime dependencies — it just compares our parser output to the stored JSON, one curated WMO key at a time.

You only need eccodes when regenerating the snapshots, typically after adding a fixture or upgrading the reference. Install it via your package manager (Debian/Ubuntu: apt install libeccodes-tools; macOS: brew install eccodes), then run:

python3 tools/regenerate-eccodes-snapshots.py

The script invokes grib_dump -j per fixture, filters down to the curated subset of keys in CURATED_KEYS, and rewrites each snapshot. The diff is human-reviewable in PRs. To grow coverage to a new WMO key, add it to both CURATED_KEYS (in the script) and the dispatch match in tests/eccodes_reference.rs::assert_message_matches, then re-run the script.

Linting

cargo clippy --all-targets --workspace -- -D warnings
cargo fmt --all -- --check

The fieldglass-napi crate also enables #![deny(clippy::all)], so warnings there are hard errors regardless.

Pre-commit hooks

The repository uses the pre-commit framework. Its config is at .pre-commit-config.yaml. The framework auto-fetches and isolates the pinned versions of shellcheck, actionlint, gitleaks, and semgrep, so you only need pre-commit itself plus the Rust + Node toolchains.

One-time setup (per clone):

pipx install pre-commit                # preferred — works everywhere
# or, on systems without pipx:
pip install --user pre-commit          # may need --break-system-packages on PEP 668 distros
# or, fully isolated venv:
python3 -m venv ~/.venvs/fieldglass && ~/.venvs/fieldglass/bin/pip install pre-commit

cd /path/to/fieldglass
npm install                            # auto-runs `pre-commit install --hook-type pre-commit --hook-type pre-push`

Optional but recommended (the hooks gracefully report-and-fail if missing):

cargo install --locked cargo-deny      # advisory / license / source policy

What runs:

Stage Hook What it does
pre-commit cargo fmt --check, cargo clippy -- -D warnings, tsc --noEmit Fast lints — usually under 3 s on incremental builds.
pre-commit check-yaml, check-json, check-toml, end-of-file-fixer, trailing-whitespace, check-merge-conflict, check-added-large-files File-hygiene polish.
pre-commit shellcheck, actionlint, gitleaks Lint shell scripts, GitHub Actions YAML, and scan staged diff for secrets.
pre-push cargo test --workspace, cargo deny check, npm audit --omit=dev, semgrep scan Slower correctness + security checks.

Bypass with git commit --no-verify / git push --no-verify when you really must — CI (below) runs the same checks at full strength regardless.

Continuous integration

The pre-commit job in .github/workflows/ci.yml installs the same toolchain (Rust, Node, Python, cargo-deny) and runs pre-commit run --all-files for both the commit and push stages — so local hooks and CI run exactly the same checks, no drift. A second job builds the native module and compiles the extension as a smoke test. Tagged versions matching v* trigger a release build that compiles the native module for all six supported targets, packages the .vsix, and publishes it to GitHub Releases.

Security and static analysis

Three additional Tier-1 scanners run on every push and PR (results land in the repo's Security tab):

  • Semgrep SAST in .github/workflows/semgrep.yml — pattern-based security rules across Rust + TypeScript (p/default, p/security-audit, p/owasp-top-ten, p/rust, p/typescript, p/secrets). The same scan also runs locally on git push.
  • CodeQL semantic analysis in .github/workflows/codeql.yml — JavaScript/TypeScript and Rust, with the security-extended query suite.
  • Dependabot in .github/dependabot.yml — weekly version + security update PRs across cargo, npm (extension + root), and github-actions.

The Semgrep and CodeQL workflows are gated on github.event.repository.visibility == 'public' because GitHub Code Scanning's SARIF upload requires either a public repo or GitHub Advanced Security. They sit dormant on private repos and self-activate the moment the repo is flipped public. Dependabot runs regardless of visibility.

Adding a new GRIB metadata field

The same flow applies to either edition — only the source section module differs.

  1. Parse the field in the relevant section module:
    • GRIB1 → crates/fieldglass-grib1/src/{is,pds,gds,bds}.rs
    • GRIB2 → crates/fieldglass-grib2/src/{is,ids,lus,gds,pds,drs,bms,ds}.rs
  2. Expose the value on the section struct.
  3. Populate it on MessageMeta in crates/fieldglass-napi/src/lib.rs (both open_grib1 and open_grib2 produce the same MessageMeta shape, so a new field usually lands in both paths).
  4. Add the corresponding camelCase field to the MessageMeta interface in extension/src/provider.ts.
  5. Render it in the webview table in the same file.

WMO ON388 / FM 92 lookup tables live in crates/fieldglass-grib1/src/tables.rs (GRIB1) and crates/fieldglass-grib2/src/tables.rs (GRIB2) — extend the tables there rather than hardcoding strings at the napi or TypeScript layer. The napi-rs bindings automatically convert snake_case Rust field names to camelCase TypeScript fields.

License

Fieldglass is dual-licensed under either of:

  • MIT License (LICENSE-MIT or https://opensource.org/licenses/MIT)
  • Apache License, Version 2.0 (LICENSE-APACHE or https://www.apache.org/licenses/LICENSE-2.0)

at your option.

Unless you explicitly state otherwise, any contribution intentionally submitted for inclusion in Fieldglass by you, as defined in the Apache-2.0 license, shall be dual-licensed as above, without any additional terms or conditions.

  • Contact us
  • Jobs
  • Privacy
  • Manage cookies
  • Terms of use
  • Trademarks
© 2026 Microsoft