feat(license): registration + 1-year licenses + tier scaffolding
A complete offline licensing layer (no internet at any step): Core - src/license/ — schema (License, Tier, FeatureFlag), HMAC crypto, JSON storage, LicenseManager singleton with activate/renew/ deactivate/issue_trial. Tier-scaffolded so future SKUs can carve per-tool feature sets without consumer-code edits. - scripts/generate_license.py — creator-only key generator. Mints a DTLIC1: blob the buyer pastes into the activation page. GUI - New activation form component (src/gui/components/activation.py). - hide_streamlit_chrome() now inline-renders the activation form when no valid license is present (every page short-circuits to the form until activated). - Sidebar shows tier + days remaining; renewal warning under 30 days. - New pages/_Activate.py for revisiting the form after activation. CLI - src/license_cli.py — activate / renew / status / trial / deactivate commands. Exempt from the guard. - src/cli_license_guard.py — drop-in guard call added to every tool CLI's main(). Lets --help through; respects DATATOOLS_DEV_MODE. i18n - New activation.* and license.* keys in en.json + es.json (page title, form labels, status badges, renewal warnings, error messages). Pack parity test stays green. Test infrastructure - tests/conftest.py autouse fixture sets DATATOOLS_DEV_MODE=1 so the existing 1916 tests continue to pass. - isolated_license_path / activated_license_manager / unactivated_license_manager fixtures for tests that want to drive the real check. Tests (+79) - tests/test_license.py (40): schema, crypto roundtrip, blob encode/decode, tier→feature mapping, activation flow, name/email mismatch rejection, tamper detection, expiration, renewal, dev-mode bypass. - tests/test_license_cli.py (26): every license_cli command + subprocess tests confirming every tool CLI refuses to run without a license, --help always works, DEV_MODE bypasses. - tests/gui/test_activation.py (13): gate blocks without license, passes with trial, activation form submission unlocks the gate, sidebar status, renewal warning, i18n. Total: 1916 → 1995 tests. All pass under the strict warning filter. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
34
src/gui/pages/_Activate.py
Normal file
34
src/gui/pages/_Activate.py
Normal file
@@ -0,0 +1,34 @@
|
||||
"""Activate — license registration + renewal.
|
||||
|
||||
Lives in the sidebar nav under an underscore-prefixed filename so
|
||||
Streamlit sorts it above the numbered tool pages. The chrome's
|
||||
license gate also injects the activation form inline on any other
|
||||
page when no valid license is present; this page exists so a user
|
||||
can revisit the form without hitting an expiration first (e.g., to
|
||||
review renewal status or deactivate the device).
|
||||
"""
|
||||
|
||||
from __future__ import annotations
|
||||
|
||||
import sys
|
||||
from pathlib import Path
|
||||
|
||||
import streamlit as st
|
||||
|
||||
_project_root = Path(__file__).resolve().parent.parent.parent.parent
|
||||
if str(_project_root) not in sys.path:
|
||||
sys.path.insert(0, str(_project_root))
|
||||
|
||||
from src.gui.components import hide_streamlit_chrome, render_activation_form
|
||||
from src.i18n import t
|
||||
|
||||
st.set_page_config(
|
||||
page_title=t("activation.page_title"),
|
||||
page_icon="🔑",
|
||||
layout="wide",
|
||||
)
|
||||
|
||||
# ``gate_license=False`` keeps the chrome from re-rendering the
|
||||
# activation form on top of the form we're about to render below.
|
||||
hide_streamlit_chrome(gate_license=False)
|
||||
render_activation_form(key_prefix="page")
|
||||
Reference in New Issue
Block a user