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-0.1.0.tar.gz (338.0 kB view details)

Uploaded Source

Built Distribution

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

gridstate-0.1.0-py3-none-any.whl (224.5 kB view details)

Uploaded Python 3

File details

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

File metadata

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

File hashes

Hashes for gridstate-0.1.0.tar.gz
Algorithm Hash digest
SHA256 5d6c94382c897a5248f3d0afac6b60a6ee2c66c99e85a39f58ce56b3c7896b41
MD5 e2732321d2a2c93ef0b76fed14f34a3f
BLAKE2b-256 e0635be43640b9d9c07fdf2e430f6b2cf06f1ff7569ea06863d6a6068227bd1a

See more details on using hashes here.

Provenance

The following attestation bundles were made for gridstate-0.1.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-0.1.0-py3-none-any.whl.

File metadata

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

File hashes

Hashes for gridstate-0.1.0-py3-none-any.whl
Algorithm Hash digest
SHA256 ec9e4285ce038f05f184610d39024694a4977269c63de529a716201164a5fa97
MD5 5e15efa1b46440b1c02e581421f8d812
BLAKE2b-256 f24f7303b97beff92258e38b402c639eec1a60a8725a72de4f3f0af95bae70e9

See more details on using hashes here.

Provenance

The following attestation bundles were made for gridstate-0.1.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