Skip to content

Latest commit

 

History

History
264 lines (177 loc) · 16.4 KB

File metadata and controls

264 lines (177 loc) · 16.4 KB

Changelog

All notable changes to this project will be documented in this file.

The format is based on Keep a Changelog, and this project adheres to Semantic Versioning.

Convention: Each release section uses emoji headers (## ✨ Added, ## 🔄 Changed, etc.) so that .github/workflows/release.yml can extract them directly into GitHub Releases without reformatting. The [Unreleased] section uses plain headers for drafting convenience; rename the headers to emoji form when cutting a release.

[Unreleased]

Added

  • TBD

Changed

  • TBD

Fixed

  • TBD

[0.3.0] - 2026-04-15

📦 CertMonitor v0.3.0 – Zero-Dependency Milestone & Chain Validator

Release Date: April 15, 2026 Repository: bradh11/certmonitor


🚀 Overview

CertMonitor v0.3.0 is a zero-dependency milestone. The Rust extension's entire X.509 / DER parser is now written in-house against the Rust standard library — no third-party parsing crates in the runtime dependency tree. The Rust dependency count drops from 48 crates to 20, with every remaining crate being pyo3 or a build-time helper.

The parser is annotated #![forbid(unsafe_code)] at the crate root, returns Result on every code path (no panics on malformed input), and has been fuzz-tested against 1.7 billion adversarial byte sequences with zero crashes.

This release also ships the chain validator for structural inspection of TLS certificate chains, and fixes two latent bugs in the public key info output.


✨ Added

  • chain validator (#14): structural validation of the full TLS certificate chain. Flags missing intermediates, out-of-order chains, expired members, weak signature algorithms (SHA-1, MD5), non-CA intermediates, and unexpected self-signed leaves. Registered but disabled by default — opt in via enabled_validators=["chain"] or ENABLED_VALIDATORS=...,chain. Chain retrieval requires Python 3.10+; returns a clear error on older interpreters.
  • certinfo.analyze_chain (Rust): parses a full DER chain in a single PyO3 call and returns per-cert details plus subject/issuer and SKI/AKI linkage.
  • SSLHandler.fetch_raw_cert now additionally returns chain_der and chain_error, populated via SSLSocket.get_verified_chain() on Python 3.13+ and the stable _sslobj.get_unverified_chain() fallback on 3.10–3.12.
  • In-tree DER / X.509 parser (#22): a strict-DER, no-unsafe, panic-free parser under rust_certinfo/src/{der,x509}/. The der/ layer (TLV reader, OID decoder, time parser, string decoders) is reusable for future ASN.1-based capabilities. The x509/ layer (Certificate, Name, SPKI, AlgorithmIdentifier, Extensions) composes those primitives into RFC 5280 structures.
  • Fuzz harness (#25): make fuzz (60-second smoke run) and make fuzz-long (1-hour soak) Makefile targets. Pre-release soak run results: 1.7 billion iterations, 310 code-coverage points, 503 libfuzzer features explored, zero crashes. Requires nightly Rust + cargo-fuzz. Manual pre-release gate, not CI.
  • 130-cert real-world corpus (tests/test_certinfo_corpus.py): snapshot tests run every public certinfo entry point against captured certs from 101 production hosts on every CI run.
  • 56 in-module Rust unit tests covering DER primitives, OID round-trips, time parsing, Name/RDN walking, SPKI dispatch, and extension parsing.
  • python Cargo feature on the certinfo crate (default on). Disabling it drops the PyO3 layer entirely and builds only the pure-Rust parser core — used by the fuzz crate.
  • scripts/bench_chain.py: opt-in benchmark with a microbench of analyze_chain (~400 µs/call) and a 101-host concurrent pipeline test.

🔄 Changed

  • Zero non-pyo3 Rust dependencies. x509-parser and base64 are gone. The Rust dependency tree shrinks from 48 crates to 20 — every remaining crate is pyo3 or a build-time helper. cargo audit surface drops accordingly.
  • Cargo.toml crate-type is now ["cdylib", "rlib"]. The cdylib is the same Python wheel target; rlib lets the fuzz crate link the parser as a normal Rust library. No published-wheel surface change.
  • certinfo::Certificate::from_der and certinfo::ParseError are now pub at the crate root for use by the fuzz crate and future in-tree Rust consumers. The PyO3 boundary and Python-facing API are unchanged.

🛠️ Fixed

  • EC curve field now correctly contains the curve OID. Previous builds emitted the algorithm OID 1.2.840.10045.2.1 (id-ecPublicKey) in the field literally named curve. The new parser extracts the curve OID from algorithm.parameters: 1.2.840.10045.3.1.7 for P-256, 1.3.132.0.34 for P-384, 1.3.132.0.35 for P-521. Visible behavior change for any caller reading public_key_info["curve"].
  • RSA modulus bit length is no longer over-counted by 8 bits. Previous builds reported RSA-2048 / 3072 / 4096 keys as 2056 / 3080 / 4104 due to including the DER-mandated leading-zero sign byte. The new parser reports the canonical 2048 / 3072 / 4096. Visible in public_key_info["size"].

📚 Documentation

  • New per-validator docs page: docs/validators/chain.md.
  • README: new "Why Trust CertMonitor" section with fuzz results, zero-dep guarantee, forbid(unsafe_code), and coverage numbers.
  • Comprehensive documentation at certmonitor.readthedocs.io.

🐍 Python Compatibility

Tested with Python 3.8 through 3.13 with 99% code coverage across all supported versions. The chain validator requires Python 3.10+ for chain retrieval; all other features work on 3.8+.


📝 License

This project is licensed under the MIT License. See the LICENSE file for details.

Full Changelog: https://github.com/bradh11/certmonitor/compare/v0.2.0...v0.3.0

[0.2.0] - 2026-04-13

📦 CertMonitor v0.2.0 – Dynamic Validator Args & sensitive_date validator

Release Date: April 13, 2026 Repository: bradh11/certmonitor


🚀 Overview

CertMonitor v0.2.0 overhauls how validators receive arguments. New validators can now declare their user arguments directly on the validate() method signature — the dispatcher discovers them automatically and no core changes are needed. As part of the same effort, the sensitive_date validator — which has been sitting on develop since #15 back in June 2025 — finally makes it into a release, and gets ergonomic input forms, a structured match field, and structured error handling along the way.

This is a minor version bump to reflect the scale of the changes, not because of any hard break in the public API. Existing callers using validator_args={"subject_alt_names": [...]} still work with a DeprecationWarning, and no validator output shape has changed for users.


✨ Added

  • Dynamic validator argument dispatch (#18): validators declare their user-configurable arguments directly on the validate() method signature, and CertMonitor.validate(validator_args=...) discovers them automatically. New validators get argument passing for free — zero core changes needed.
  • CertMonitor.describe_validators(): new introspection helper that returns every registered validator's name, docstring, and argument schema (name, annotation, default). Useful for building CLI --help pages, config validators, or dashboards.
  • sensitive_date validator finally ships: flags certificates that expire on weekends, leap days, or user-specified dates (e.g. Black Friday, Cyber Monday, go-live dates).
  • sensitive_date input ergonomics: the dates argument accepts SensitiveDate named tuples, plain date / datetime values, ISO 8601 strings ("2025-12-25"), or (name, date) tuples — all mixable in a single call. No need to import SensitiveDate from a deeply nested module path just to pass a list of blackout dates.
  • sensitive_date_matches structured field: matching sensitive dates are surfaced as a machine-readable list of {"name", "date"} entries in addition to the existing human-readable warnings strings.
  • Weekend / leap-day warning strings: when the sensitive_date validator flags a weekend or leap-day expiry, a human-readable warning line is now emitted alongside the existing boolean fields, so log output is self-explanatory when is_valid is false.
  • Shared parse_not_after helper (certmonitor/validators/_utils.py): centralizes the notAfter format string shared by expiration and sensitive_date.

🔄 Changed

  • Validator author contract: user arguments on validate() must be keyword-only, type-annotated, and have a default value. Enforcement runs in BaseCertValidator / BaseCipherValidator __init_subclass__ at import time, so a malformed validator raises TypeError the moment its module is imported. No user-facing impact — every built-in validator conforms, and the dispatcher continues to accept the pre-0.2.0 validator_args call style via a deprecation shim.
  • subject_alt_names and sensitive_date signatures migrated to keyword-only user arguments (alternate_names=..., dates=...). Existing users of monitor.validate(validator_args={...}) are unaffected; callers invoking the validator classes directly with positional arguments need the keyword form.
  • validator_args canonical form is now a nested dict: validator_args={"subject_alt_names": {"alternate_names": [...]}}. The pre-0.2.0 bare-list form still works and is transparently rewritten by the dispatcher — with a DeprecationWarning — so no user code needs to change immediately.
  • sensitive_date error handling: malformed dates input (wrong type, invalid ISO string, bad tuple shape) now returns a structured error dict instead of raising TypeError, matching the rest of the validator suite.
  • expiration validator: now uses the shared parse_not_after helper; behavior unchanged.
  • mkdocs.yml: added the previously-missing SensitiveDate nav entry so the validator's auto-generated reference page is reachable.
  • docs/usage/validator_args.md: rewritten to document the canonical nested-dict form, describe_validators(), the bare-list deprecation, and a worked sensitive_date example showing all four input forms.
  • Rust toolchain floor moved to rustc >= 1.88.0 (transitively via the time 0.3.47 security bump, see below). Affects contributors and source builds only — published wheels are unaffected.

⚠️ Deprecated

  • Bare-list shorthand for single-argument validators (validator_args={"subject_alt_names": [...]}) still works but now emits a DeprecationWarning. Migrate to the canonical nested-dict form. Scheduled for removal in a future release.

🔒 Security

  • RUSTSEC-2026-0009: bumped the time crate from 0.3.41 to 0.3.47 (transitively via x509-parser) to address the denial-of-service-via-stack-exhaustion advisory.

🛠️ Fixed

  • subject_alt_names core dispatch: the hardcoded if validator.name == "subject_alt_names" special case in core.validate() is gone — replaced with the generic argument-resolution helper used by every validator.
  • CHANGELOG.md: backfilled the missing [0.1.4] section from the published release notes so the historical record is complete.

📚 Documentation

Comprehensive documentation is available at certmonitor.readthedocs.io.


🐍 Python Compatibility

Tested with Python 3.8 and above with 98%+ code coverage across all supported versions.


📝 License

This project is licensed under the MIT License. See the LICENSE file for details.

Full Changelog: https://github.com/bradh11/certmonitor/compare/v0.1.4...v0.2.0

[0.1.4] - 2025-06-02

Focus: test coverage and CI optimization (#16).

Added

  • Achieved 99% test coverage (up from 95%) with comprehensive edge case testing.
  • Instance convenience methods for improved developer experience:
    • monitor.get_enabled_validators() — get validators enabled for this specific instance.
    • monitor.list_validators() — get all available validators.
  • Enhanced test suite with 323 tests covering all edge cases.

Changed

  • Streamlined security scanning: removed heavy semgrep dependency, kept focused bandit scanning.
  • Improved test descriptions — removed line-number references for maintainable, functionality-focused tests.
  • Enhanced validator configuration: proper distinction between empty lists vs config defaults.

Fixed

  • Default validator behavior: enabled_validators=[] now properly means "no validators", vs None meaning "use defaults" (see #16).
  • Configuration environment handling: proper string parsing for the ENABLED_VALIDATORS environment variable.
  • Test coverage gaps — targeted testing for previously uncovered edge cases: SSL handler retry exception scenarios, certificate parsing fallback mechanisms, handler-None conditions in raw data operations, and public key parsing error paths.

[0.1.3] - 2025-05-25

Added

  • Comprehensive GitHub workflows and templates with develop/main branch strategy
  • Enhanced type hints throughout entire codebase (zero mypy errors)
  • Modularized test suite for better maintainability
  • ReadTheDocs integration with proper configuration
  • Consolidated CI/CD pipeline with conditional job execution
  • Unified Makefile commands for Python + Rust development workflow
    • make format - Format both Python and Rust code
    • make format-check - Check formatting for both languages
    • make lint - Lint both Python and Rust code
    • make security - Run security vulnerability check (cargo audit)
    • Individual language commands: python-format, python-lint, rust-format, rust-lint
    • Enhanced make test with 9-step CI-equivalent comprehensive testing including security checks
    • Improved make help with clear categorized command documentation

Changed

  • Improved code organization and structure
  • Enhanced documentation and contributing guidelines
  • Updated all workflows to use develop/main branch strategy
  • Removed redundant CI configurations (quality.yml, security.yml, rust.yml, docs.yml)
  • Streamlined dependency management (removed Dependabot for stdlib-only project)
  • Enhanced local development experience with unified format/lint commands
  • Makefile now provides comprehensive Python + Rust development workflow

Fixed

  • All mypy type errors across 20 source files using proper type annotations and runtime checking
  • Import sorting and code quality issues
  • SSL handler connection logic and check_connection() functionality
  • Python 3.8 compatibility issues (datetime.UTC → datetime.timezone.utc)
  • CI workflow syntax errors and redundant documentation building
  • Fixed incorrect reporting on root certificate validation
  • Security vulnerability RUSTSEC-2025-0020 by upgrading PyO3 from 0.20.0 to 0.24.1
  • Rust code compatibility with PyO3 0.24.x API changes (updated module binding syntax)

[0.1.2] - 2025-05-11

Added

  • ABI3 support for Python wheels for better compatibility

Changed

  • Improved wheel building process for cross-version compatibility

[0.1.1] - 2025-05-11 [DEPRECATED]

Note: This release has been deprecated. Please use v0.1.2 or later.

Added

  • Python 3.13 support in testing matrix
  • Trusted publisher configuration for PyPI releases

Changed

  • Enhanced CI/CD pipeline with proper release permissions
  • Updated package structure (moved Rust library outside of Python package)

Fixed

  • CI workflow issues and version bumping process

[0.1.0] - 2025-05-11 [DEPRECATED]

Note: This release has been deprecated due to Rust build failures. Please use v0.1.2 or later.

Added

  • Initial release of certmonitor
  • Certificate validation and monitoring capabilities
  • SSL/TLS certificate analysis
  • SSH certificate support
  • Hybrid Python-Rust implementation for performance
  • Virtual environment setup in CI publish job

Known Issues

  • Rust build failures preventing proper installation