Skip to main content

Power system State Estimation: pure-Python WLS/IPM on numpy/scipy, input via a versioned file contract (npz).

Project description

gridstate

Power-system State Estimation in pure Python.

gridstate reconstructs the most likely operating state of an electrical network from SCADA telemetry and a topology model: voltage magnitudes and angles at every bus, plus the derived quantities — branch power flows, currents and nodal injections.

The estimator is implemented on top of numpy and scipy only — there are no proprietary or vendor dependencies. Input and output cross a single, explicit, versioned data contract (a .npz file), so the package is fully self-contained and decoupled from any particular network-model format.

 measurements (value + σ)        ┌──────────────────────┐        V, δ  (state)
 topology, impedances, taps  ──► │  gridstate.run_se    │ ──►   P, Q, I (branches)
 (SEInput, .npz contract)        │  WLS / IPM estimator │        injections, residuals
                                 └──────────────────────┘        (SEOutput contract)

Status

Pre-alpha. The core pipeline runs end to end:

  • p.u. conversion of the working model (gridstate.units);
  • bus-admittance assembly Ybus / Yf / Yt (gridstate.ybus);
  • measurement vector z, weights R, measurement index (gridstate.z_vector);
  • WLS Gauss–Newton solve — measurement function h(E), Jacobian H = ∂h/∂E, step ΔE = (HᵀR⁻¹H)⁻¹HᵀR⁻¹r (gridstate.algebra, gridstate.algorithms.wls), with SHGM-IRLS robust re-weighting;
  • IPM interior-point estimator as an alternative solver;
  • result write-back (V, δ, branch P/Q, currents, injections) into the output contract;
  • validation: χ² test (gridstate.validation.chi2_test), bad-data removal by normalized residuals / rn_max (gridstate.validation.bad_data), observability analysis (gridstate.validation.observability).

Tests: 445 passing on Python 3.10–3.12, dependency-free (numpy/scipy only).

Installation

python -m venv .venv
source .venv/bin/activate
pip install -e ".[dev]"

Runtime dependencies are just numpy>=1.24 and scipy>=1.10. Python 3.10+.

Quick start

The public boundary is a .npz data-contract file. Load it, run the estimator, read the results:

from gridstate import load_se_input_npz, run_se

# 1. Load a state-estimation input contract (working model + precomputed plans).
se_input = load_se_input_npz("path/to/case.npz")

# 2. Run the estimator (WLS by default; production pipeline defaults applied).
out = run_se(se_input)

print(out.success, out.iterations, out.objective_value)

# 3. Read per-object results from the output contract.
bus = out.node(node_id=1001)        # {'voltage_magnitude': ..., 'voltage_angle': ..., ...}
flow = out.branch(branch_id=42)     # {'power_from_p': ..., 'current_from': ..., ...}

run_se returns an SEOutput: structured arrays nodes / branches / measurements (id + result columns) plus convergence scalars (success, iterations, objective_value, algorithm) and the state vectors v_pu / delta_rad.

To estimate from a model you already hold in memory, wrap it with load_se_input(model) (skips the format-dependent input stages) or call the lower-level estimate(model, algorithm="wls", ...) directly.

Data contract

Input and output are governed by an explicit, versioned schema (gridstate.contract):

  • SE_INPUT / SE_OUTPUT — table/column declarations with roles (INPUT / WORKING / OUTPUT) for nodes, branches, measurements, generators, and the raw side tables.
  • CONTRACT_VERSION follows SemVer; is_data_compatible(...) gates whether a given .npz can be consumed.
  • load_se_input_npz / save_se_input round-trip the contract to disk.
  • validate_input(...) checks a model against the contract and fails early on missing required columns or an incompatible version.

This contract is what lets gridstate stand alone: the estimator never depends on a concrete external network-model class — only on the data it declares it needs.

Validation

from gridstate import chi2_analysis, remove_bad_data, analyze_observability

report = analyze_observability(se_input.model)
assert report.is_observable, report.diagnostics

chi2 = chi2_analysis(se_input.model, chi2_prob_false=0.05)
if chi2.bad_data_present:
    cleaned = remove_bad_data(se_input.model, rn_max_threshold=3.0)
    print("removed measurements:", cleaned.removed_meas_ids)

Public API

Exported from the top-level package:

Symbol Purpose
run_se, run_pipeline run the estimation pipeline (contract / direct)
estimate low-level single-solve entry point
load_se_input_npz, load_se_input, save_se_input data-contract I/O
SEInput, SEOutput, SEResult input / output / detailed result objects
PipelineConfig, pipeline_manifest pipeline configuration & step manifest
chi2_analysis, remove_bad_data, analyze_observability validation

Development

make install-dev    # editable install + dev/test extras + pre-commit hooks
make test           # pytest tests/
make lint           # ruff check + ruff format --check
make type-check     # mypy gridstate
make check          # format + lint + type-check + test

Continuous integration (lint, the 3.10–3.12 test matrix, and a build) runs on GitHub Actions; see .github/workflows/.

License

gridstate is released under the MIT License — see LICENSE.

The estimation core (WLS/IPM linear algebra, χ² test and bad-data detection) is adapted from pandapower, which is distributed under the BSD 3-Clause License. The full third-party notice and the list of adapted files are reproduced in the "Third-Party Software Notices" section of LICENSE.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

gridstate-1.0.0.tar.gz (345.1 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

gridstate-1.0.0-py3-none-any.whl (227.6 kB view details)

Uploaded Python 3

File details

Details for the file gridstate-1.0.0.tar.gz.

File metadata

  • Download URL: gridstate-1.0.0.tar.gz
  • Upload date:
  • Size: 345.1 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for gridstate-1.0.0.tar.gz
Algorithm Hash digest
SHA256 4892d1be4361fa09b074dfef0fee7514e2c91152b6337f1b845407e49ebe2f75
MD5 6336625221ceb49e451beb7aa9e3340a
BLAKE2b-256 22efcd1b293f889cae7f69dd92414125dda32cde6003a0616399991b536f62fd

See more details on using hashes here.

Provenance

The following attestation bundles were made for gridstate-1.0.0.tar.gz:

Publisher: release.yml on Genajoin/gridstate

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file gridstate-1.0.0-py3-none-any.whl.

File metadata

  • Download URL: gridstate-1.0.0-py3-none-any.whl
  • Upload date:
  • Size: 227.6 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for gridstate-1.0.0-py3-none-any.whl
Algorithm Hash digest
SHA256 4da9fb735e0ab05581c3bef9925834c6e87d299099dbe10830322db5e93cdda9
MD5 b45cd40ad428f4e95d3a351847d54629
BLAKE2b-256 250be47815d33c67de664f2a9d86ff08be263267e90e1aec6c6e67574914598d

See more details on using hashes here.

Provenance

The following attestation bundles were made for gridstate-1.0.0-py3-none-any.whl:

Publisher: release.yml on Genajoin/gridstate

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page