plugin-scanner 2.0.175__tar.gz → 2.0.177__tar.gz
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/PKG-INFO +1 -1
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/approval-center-primitives.tsx +169 -34
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/apps/app-detail-workspace.tsx +13 -13
- plugin_scanner-2.0.177/dashboard/src/fleet-workspace.tsx +210 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/home-dashboard.tsx +18 -15
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/receipts-workspace.tsx +58 -69
- plugin_scanner-2.0.177/dashboard/src/settings-workspace.tsx +705 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/pyproject.toml +1 -1
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/pyproject.toml.bak +1 -1
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/daemon/static/assets/guard-dashboard.js +422 -678
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/daemon/static/assets/index.css +198 -142
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/version.py +1 -1
- plugin_scanner-2.0.175/dashboard/src/fleet-workspace.tsx +0 -205
- plugin_scanner-2.0.175/dashboard/src/settings-workspace.tsx +0 -891
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/.clusterfuzzlite/Dockerfile +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/.clusterfuzzlite/build.sh +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/.clusterfuzzlite/project.yaml +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/.clusterfuzzlite/requirements-atheris.txt +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/.dockerignore +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/.github/CODEOWNERS +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/.github/ISSUE_TEMPLATE/bug-report.yml +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/.github/ISSUE_TEMPLATE/config.yml +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/.github/ISSUE_TEMPLATE/feature-request.yml +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/.github/dependabot.yml +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/.github/workflows/ci.yml +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/.github/workflows/codeql.yml +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/.github/workflows/dependabot-uv-lock.yml +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/.github/workflows/fuzz.yml +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/.github/workflows/harness-smoke.yml +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/.github/workflows/publish.yml +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/.github/workflows/scorecard.yml +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/.gitignore +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/.pre-commit-hooks.yaml +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/CONTRIBUTING.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/Dockerfile +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/LICENSE +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/README.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/SECURITY.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/index.html +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/package.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/pnpm-lock.yaml +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/public/apple-touch-icon.png +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/public/brand/Logo_Icon_Dark.png +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/public/brand/Logo_Whole.png +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/public/favicon-16x16.png +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/public/favicon-32x32.png +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/public/favicon.ico +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/app.tsx +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/approval-center-layout.test.ts +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/approval-center-layout.tsx +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/approval-center-mobile.test.ts +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/approval-center-review-cards.tsx +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/approval-center-utils.ts +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/data-flow-evidence-card.tsx +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/error-boundary.tsx +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/guard-api.test.ts +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/guard-api.ts +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/guard-demo.ts +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/guard-types.ts +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/help-modal.tsx +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/home-dashboard.test.ts +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/main.tsx +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/queue-chip-filter.tsx +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/queue-state.test.ts +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/queue-state.ts +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/receipts-workspace.test.ts +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/review-workspace.tsx +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/risk-signal-cards.test.ts +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/risk-signal-cards.tsx +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/runtime-overview.test.ts +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/runtime-overview.tsx +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/scanner-evidence-badge.tsx +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/settings-workspace.test.ts +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/styles.css +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/use-focus-trap.ts +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/vite-env.d.ts +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/watched-app-card.tsx +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/tsconfig.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/vite.config.ts +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/docker-requirements.txt +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/docs/guard/approval-audit.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/docs/guard/architecture.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/docs/guard/get-started.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/docs/guard/harness-support.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/docs/guard/local-vs-cloud.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/docs/guard/release-checklist.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/docs/guard/release-notes.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/docs/guard/smoke-tests.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/docs/guard/testing-matrix.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/docs/trust/mcp-trust-draft.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/docs/trust/plugin-trust-draft.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/docs/trust/skill-trust-local.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/fuzzers/manifest_fuzzer.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/requirements.txt +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/schemas/plugin-quality.v1.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/schemas/scan-result.v1.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/schemas/verify-result.v1.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/__init__.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/action_runner.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/argparse_utils.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/checks/__init__.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/checks/best_practices.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/checks/claude.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/checks/code_quality.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/checks/ecosystem_common.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/checks/gemini.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/checks/manifest.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/checks/manifest_support.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/checks/marketplace.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/checks/mcp_security.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/checks/opencode.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/checks/operational_security.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/checks/security.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/checks/skill_security.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/cli.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/cli_ui.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/config.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/ecosystems/__init__.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/ecosystems/base.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/ecosystems/claude.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/ecosystems/codex.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/ecosystems/detect.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/ecosystems/gemini.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/ecosystems/opencode.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/ecosystems/registry.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/ecosystems/types.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/github_reporting.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/__init__.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/access_graph_events.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/adapters/__init__.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/adapters/antigravity.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/adapters/base.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/adapters/claude_code.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/adapters/cloud_identity.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/adapters/codex.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/adapters/contracts.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/adapters/copilot.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/adapters/cursor.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/adapters/gemini.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/adapters/hermes.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/adapters/mcp_servers.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/adapters/openclaw.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/adapters/openclaw_config.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/adapters/openclaw_support.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/adapters/opencode.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/adapters/opencode_artifacts.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/advisory_model.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/approvals.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/bridge/__init__.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/capabilities.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/cli/__init__.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/cli/approval_commands.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/cli/bootstrap.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/cli/commands.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/cli/connect_flow.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/cli/install_commands.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/cli/product.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/cli/prompt.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/cli/render.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/cli/update_commands.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/codex_config.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/config.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/consumer/__init__.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/consumer/service.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/daemon/__init__.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/daemon/client.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/daemon/manager.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/daemon/server.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/daemon/static/apple-touch-icon.png +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/daemon/static/brand/Logo_Icon_Dark.png +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/daemon/static/brand/Logo_Whole.png +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/daemon/static/favicon-16x16.png +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/daemon/static/favicon-32x32.png +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/daemon/static/favicon.ico +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/daemon/static/index.html +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/edge_events.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/incident.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/insights.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/inventory_cisco.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/inventory_contract.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/launcher.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/mcp_tool_calls.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/models.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/policy/__init__.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/policy/engine.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/protect.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/proxy/__init__.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/proxy/remote.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/proxy/runtime_mcp.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/proxy/stdio.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/receipts/__init__.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/receipts/manager.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/redaction.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/risk.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/runtime/__init__.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/runtime/action_identity.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/runtime/actions.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/runtime/advisory_escalation.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/runtime/advisory_matchers.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/runtime/cisco_evidence.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/runtime/cisco_preflight.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/runtime/composition_rules.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/runtime/data_flow.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/runtime/data_flow_rules.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/runtime/data_flow_variables.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/runtime/decisions.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/runtime/detectors.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/runtime/false_positive_rules.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/runtime/mcp_protection.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/runtime/persistence_rules.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/runtime/prompt_injection.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/runtime/runner.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/runtime/safe_decode.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/runtime/sandbox.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/runtime/scanner_cache.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/runtime/secret_file_requests.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/runtime/secret_sensitivity.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/runtime/secret_sources.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/runtime/shell_commands.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/runtime/signals.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/runtime/skill_protection.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/runtime/supply_chain.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/runtime/surface_server.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/runtime/temp_files.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/runtime/threat_intel.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/schemas/__init__.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/schemas/consumer_mode.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/schemas/guard_event_v1.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/schemas/surface_server.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/shims.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/store.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/store_approvals.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/store_connect.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/store_evidence.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/store_threat_intel.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/guard/types.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/integrations/__init__.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/integrations/cisco_mcp_scanner.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/integrations/cisco_skill_scanner.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/lint_fixes.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/marketplace_support.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/models.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/path_support.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/policy.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/quality_artifact.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/repo_detect.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/reporting.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/rules/__init__.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/rules/registry.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/rules/specs.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/scanner.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/submission.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/suppressions.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/trust_domain_scoring.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/trust_helpers.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/trust_mcp_scoring.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/trust_models.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/trust_plugin_scoring.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/trust_scoring.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/trust_skill_scoring.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/trust_specs.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/src/codex_plugin_scanner/verification.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/__init__.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/conftest.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/__init__.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/bad-plugin/.codex-plugin/plugin.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/bad-plugin/.mcp.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/bad-plugin/secrets.js +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/claude-plugin-good/.claude-plugin/plugin.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/claude-plugin-good/LICENSE +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/claude-plugin-good/README.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/claude-plugin-good/SECURITY.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/claude-plugin-good/hooks/hooks.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/claude-plugin-good/skills/example/SKILL.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/code-quality-bad/evil.js +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/code-quality-bad/inject.js +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/gemini-extension-good/GEMINI.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/gemini-extension-good/LICENSE +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/gemini-extension-good/README.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/gemini-extension-good/SECURITY.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/gemini-extension-good/commands/hello.toml +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/gemini-extension-good/gemini-extension.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/good-plugin/.codex-plugin/plugin.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/good-plugin/.codexignore +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/good-plugin/LICENSE +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/good-plugin/README.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/good-plugin/SECURITY.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/good-plugin/assets/icon.svg +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/good-plugin/assets/logo.svg +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/good-plugin/assets/screenshot.svg +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/good-plugin/skills/example/SKILL.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/guard-codex-malicious-mcp/.codex/config.toml +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/guard-red-team/README.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/guard-red-team/benign-docs-fake-token.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/guard-red-team/benign-health-endpoint.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/guard-red-team/benign-nvmrc-fake-creds.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/guard-red-team/benign-source-search.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/guard-red-team/canary-exfil-encoded.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/guard-red-team/canary-exfil.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/guard-red-team/expected-decisions.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/guard-red-team/malicious-dockerfile.txt +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/guard-red-team/malicious-encoded-shell-exfil.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/guard-red-team/malicious-github-action.yml +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/guard-red-team/malicious-mcp-delete.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/guard-red-team/malicious-mcp-secret-read.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/guard-red-team/malicious-mcp-skill-exfil.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/guard-red-team/malicious-npm-postinstall.js +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/guard-red-team/malicious-prompt-env-read.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/guard-red-team/malicious-prompt-guard-bypass.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/guard-red-team/malicious-prompt-npmrc-read.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/guard-red-team/malicious-python-setup.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/guard-red-team/smoke-evidence-template.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/hermes-plugin-evil/config.yaml +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/hermes-plugin-evil/mcp_servers.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/hermes-plugin-evil/skills/security/malicious/SKILL.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/hermes-plugin-evil/skills/stealth/sneaky/SKILL.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/hermes-plugin-evil/skills/stealth/sneaky/references/api-setup.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/hermes-plugin-evil/skills/stealth/sneaky/scripts/deploy.sh +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/hermes-plugin-evil/skills/utils/benign/SKILL.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/malformed-json/.codex-plugin/plugin.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/malicious-skill-plugin/.codex-plugin/plugin.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/malicious-skill-plugin/.codexignore +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/malicious-skill-plugin/LICENSE +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/malicious-skill-plugin/README.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/malicious-skill-plugin/SECURITY.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/malicious-skill-plugin/skills/leaky-skill/SKILL.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/mcp-canary-server.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/minimal-plugin/.codex-plugin/plugin.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/missing-fields/.codex-plugin/plugin.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/mit-license/LICENSE +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/multi-ecosystem-repo/codex-plugin/.codex-plugin/plugin.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/multi-ecosystem-repo/codex-plugin/LICENSE +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/multi-ecosystem-repo/codex-plugin/README.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/multi-ecosystem-repo/codex-plugin/SECURITY.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/multi-ecosystem-repo/gemini-ext/README.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/multi-ecosystem-repo/gemini-ext/gemini-extension.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/multi-plugin-repo/.agents/plugins/marketplace.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/multi-plugin-repo/plugins/alpha-plugin/.codex-plugin/plugin.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/multi-plugin-repo/plugins/alpha-plugin/.codexignore +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/multi-plugin-repo/plugins/alpha-plugin/LICENSE +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/multi-plugin-repo/plugins/alpha-plugin/README.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/multi-plugin-repo/plugins/alpha-plugin/SECURITY.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/multi-plugin-repo/plugins/alpha-plugin/skills/example/SKILL.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/multi-plugin-repo/plugins/beta-plugin/.codex-plugin/plugin.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/multi-plugin-repo/plugins/beta-plugin/skills/example/SKILL.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/no-version/.codex-plugin/plugin.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/opencode-good/.opencode/commands/hello.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/opencode-good/.opencode/plugins/example.ts +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/opencode-good/LICENSE +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/opencode-good/README.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/opencode-good/SECURITY.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/opencode-good/opencode.jsonc +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/skills-missing-dir/.codex-plugin/plugin.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/skills-no-frontmatter/.codex-plugin/plugin.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/skills-no-frontmatter/skills/bad-skill/SKILL.md +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/supply-chain/benign-npm-package.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/supply-chain/benign-pnpm-package.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/supply-chain/benign-pyproject.toml +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/supply-chain/malicious-Dockerfile +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/supply-chain/malicious-action.yml +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/supply-chain/malicious-npm-package.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/supply-chain/malicious-setup.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/with-marketplace/.codex-plugin/plugin.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/with-marketplace/marketplace-broken.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/fixtures/with-marketplace/marketplace.json +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test-trust-scoring.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test-trust-specs.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_action_runner.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_best_practices.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_cisco_install_surfaces.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_cli.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_code_quality.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_config.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_coverage_remaining.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_ecosystems.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_edge_cases.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_final_coverage.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_access_graph.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_action_identity.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_advisory_escalation.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_approval_continuity.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_approval_copy_commands.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_approval_store_dedup.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_approval_store_scale.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_approvals.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_bootstrap.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_bypass_detector.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_canary_fixtures.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_capabilities.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_cisco_evidence.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_cisco_runtime_cli.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_claude_adapter.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_cli.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_cloud_local_sync.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_codex_e2e.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_codex_install.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_codex_proxy.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_config_paths.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_connect_flow.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_consumer_mode.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_copilot_adapter.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_copilot_proxy.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_daemon_cli.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_daemon_manager.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_daemon_perf.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_daemon_registry.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_daemon_repair_perf.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_daemon_wake.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_data_flow.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_decision_propagation.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_detector_fp.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_event_schema_v1.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_events.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_evidence_store.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_harness_contracts.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_harness_setup.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_harness_smoke.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_insights.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_inventory_cisco.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_inventory_contract.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_launch_env.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_mcp_detectors.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_mcp_protection.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_opencode_proxy.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_policy_dedup.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_product_flow.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_prompt_injection.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_protect.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_queue_api_contract.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_queue_contract.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_red_team.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_red_team_e2e.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_redaction.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_render.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_resolution_copy.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_risk.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_runtime.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_runtime_action_harnesses.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_runtime_actions.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_runtime_decisions.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_runtime_detectors.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_runtime_signals.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_safe_decode.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_sandbox.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_settings_presets.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_skill_protection.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_store_migrations.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_supply_chain.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_surface_server.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_threat_intel.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_verdicts.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_guard_web_recovery.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_hermes_adapter.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_integration.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_lint_fixes.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_live_cisco_smoke.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_manifest.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_marketplace.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_mcp_security.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_openclaw_adapter.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_operational_security.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_policy.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_quality_artifact.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_rule_registry.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_scanner.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_schema_contracts.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_security.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_security_ops.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_skill_security.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_submission.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_trust_scoring.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_trust_specs.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_verification.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/tests/test_versioning.py +0 -0
- {plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/uv.lock +0 -0
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
Metadata-Version: 2.4
|
|
2
2
|
Name: plugin-scanner
|
|
3
|
-
Version: 2.0.
|
|
3
|
+
Version: 2.0.177
|
|
4
4
|
Summary: Lint, verify, and gate plugin ecosystems for maintainers, CI, and publish workflows.
|
|
5
5
|
Project-URL: Homepage, https://github.com/hashgraph-online/ai-plugin-scanner
|
|
6
6
|
Project-URL: Repository, https://github.com/hashgraph-online/ai-plugin-scanner
|
{plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/approval-center-primitives.tsx
RENAMED
|
@@ -16,6 +16,8 @@ import {
|
|
|
16
16
|
HiMiniCheckCircle,
|
|
17
17
|
HiMiniArrowRight,
|
|
18
18
|
HiMiniExclamationTriangle,
|
|
19
|
+
HiMiniChevronDown,
|
|
20
|
+
HiMiniChevronUp,
|
|
19
21
|
} from "react-icons/hi2";
|
|
20
22
|
|
|
21
23
|
import { guardAwareHref } from "./guard-api";
|
|
@@ -237,7 +239,7 @@ export function Surface(props: {
|
|
|
237
239
|
const toneClass = surfaceToneClass(props.tone);
|
|
238
240
|
return (
|
|
239
241
|
<section
|
|
240
|
-
className={`guard-surface-in rounded-
|
|
242
|
+
className={`guard-surface-in rounded-xl border p-4 sm:p-5 ${toneClass}${props.className ? ` ${props.className}` : ""}`}
|
|
241
243
|
>
|
|
242
244
|
{props.children}
|
|
243
245
|
</section>
|
|
@@ -296,8 +298,8 @@ export function EmptyState(props: {
|
|
|
296
298
|
}) {
|
|
297
299
|
const isTeach = props.tone === "teach";
|
|
298
300
|
return (
|
|
299
|
-
<div className={`
|
|
300
|
-
<div className={`mb-5 flex h-14 w-14 items-center justify-center rounded-full ${isTeach ? "bg-brand-blue/10" : "bg-
|
|
301
|
+
<div className={`flex flex-col items-center justify-center py-12 text-center sm:py-16 ${isTeach ? "rounded-2xl border border-brand-blue/10 bg-gradient-to-br from-white to-brand-blue/[0.02] px-6" : "px-6"}`}>
|
|
302
|
+
<div className={`mb-5 flex h-14 w-14 items-center justify-center rounded-full ${isTeach ? "bg-brand-blue/10" : "bg-slate-100"}`}>
|
|
301
303
|
<HiMiniInformationCircle className={`h-7 w-7 ${isTeach ? "text-brand-blue" : "text-slate-400"}`} aria-hidden="true" />
|
|
302
304
|
</div>
|
|
303
305
|
<h3 className="text-lg font-semibold tracking-tight text-brand-dark">{props.title}</h3>
|
|
@@ -472,12 +474,12 @@ function SidebarAction(props: { href: string; children: ReactNode; icon: ReactNo
|
|
|
472
474
|
}
|
|
473
475
|
|
|
474
476
|
function surfaceToneClass(tone: "default" | "accent" | "success" | "warning" | "danger" | "attention" | undefined): string {
|
|
475
|
-
if (tone === "accent") return "border-brand-blue/
|
|
476
|
-
if (tone === "success") return "border-brand-green/
|
|
477
|
-
if (tone === "warning") return "border-brand-blue/
|
|
478
|
-
if (tone === "danger") return "border-brand-purple/
|
|
479
|
-
if (tone === "attention") return "border-brand-attention/
|
|
480
|
-
return "border-
|
|
477
|
+
if (tone === "accent") return "border-brand-blue/15 bg-gradient-to-b from-white to-blue-50/30";
|
|
478
|
+
if (tone === "success") return "border-brand-green/15 bg-brand-green-bg/20";
|
|
479
|
+
if (tone === "warning") return "border-brand-blue/20 bg-brand-blue/[0.03]";
|
|
480
|
+
if (tone === "danger") return "border-brand-purple/20 bg-brand-purple/[0.03]";
|
|
481
|
+
if (tone === "attention") return "border-brand-attention/15 bg-brand-attention-bg/60";
|
|
482
|
+
return "border-slate-100 bg-white/60";
|
|
481
483
|
}
|
|
482
484
|
|
|
483
485
|
function badgeToneClass(tone: "default" | "success" | "warning" | "info" | "destructive" | "attention" | undefined): string {
|
|
@@ -665,11 +667,10 @@ export function GuardHero(props: {
|
|
|
665
667
|
|
|
666
668
|
return (
|
|
667
669
|
<section
|
|
668
|
-
className={`guard-surface-in relative overflow-hidden rounded-
|
|
670
|
+
className={`guard-surface-in relative overflow-hidden rounded-2xl border border-brand-blue/10 ${bgClass} p-5 sm:p-6 lg:p-7`}
|
|
669
671
|
role="region"
|
|
670
672
|
aria-label="Protection status"
|
|
671
673
|
>
|
|
672
|
-
<div className="pointer-events-none absolute right-10 top-8 h-24 w-24 rounded-full bg-brand-blue/20 blur-3xl" />
|
|
673
674
|
<div className="relative space-y-4">
|
|
674
675
|
<div className="flex flex-wrap items-center gap-2">
|
|
675
676
|
<SectionLabel>Protection status</SectionLabel>
|
|
@@ -696,25 +697,19 @@ export function GuardHero(props: {
|
|
|
696
697
|
}
|
|
697
698
|
|
|
698
699
|
export function ProofStrip(props: {
|
|
699
|
-
items: Array<{ label: string; value: string | number; tone?: "blue" | "green" | "purple" | "slate"; icon?: ReactNode;
|
|
700
|
+
items: Array<{ label: string; value: string | number; tone?: "blue" | "green" | "purple" | "slate"; icon?: ReactNode; hint?: string }>;
|
|
700
701
|
}) {
|
|
701
702
|
return (
|
|
702
|
-
<div className="
|
|
703
|
-
{props.items.map((item) =>
|
|
704
|
-
|
|
705
|
-
|
|
706
|
-
|
|
707
|
-
|
|
708
|
-
|
|
709
|
-
<div className="flex items-center gap-2">
|
|
710
|
-
<p className="font-mono text-[10px] font-semibold uppercase tracking-[0.18em] text-muted-foreground">
|
|
711
|
-
{item.label}
|
|
712
|
-
</p>
|
|
713
|
-
{item.icon}
|
|
703
|
+
<div className="grid gap-x-8 gap-y-4 sm:grid-cols-2 lg:grid-cols-4">
|
|
704
|
+
{props.items.map((item) => {
|
|
705
|
+
const toneColor = item.tone === "blue" ? "text-brand-blue" : item.tone === "green" ? "text-emerald-600" : item.tone === "purple" ? "text-brand-purple" : "text-brand-dark";
|
|
706
|
+
return (
|
|
707
|
+
<div key={item.label} className="flex flex-col" title={item.hint}>
|
|
708
|
+
<p className="text-[11px] font-semibold uppercase tracking-wider text-slate-400">{item.label}</p>
|
|
709
|
+
<p className={`text-2xl font-semibold tracking-tight ${toneColor}`}>{item.value}</p>
|
|
714
710
|
</div>
|
|
715
|
-
|
|
716
|
-
|
|
717
|
-
))}
|
|
711
|
+
);
|
|
712
|
+
})}
|
|
718
713
|
</div>
|
|
719
714
|
);
|
|
720
715
|
}
|
|
@@ -727,22 +722,162 @@ export function NextActionCard(props: {
|
|
|
727
722
|
}) {
|
|
728
723
|
const borderClass =
|
|
729
724
|
props.tone === "green"
|
|
730
|
-
? "border-brand-green/
|
|
725
|
+
? "border-brand-green/20"
|
|
731
726
|
: props.tone === "purple"
|
|
732
|
-
? "border-brand-purple/
|
|
733
|
-
: "border-brand-blue/
|
|
727
|
+
? "border-brand-purple/20"
|
|
728
|
+
: "border-brand-blue/20";
|
|
734
729
|
const bgClass =
|
|
735
730
|
props.tone === "green"
|
|
736
|
-
? "bg-brand-green-bg/
|
|
731
|
+
? "bg-brand-green-bg/20"
|
|
737
732
|
: props.tone === "purple"
|
|
738
|
-
? "bg-brand-purple/[0.
|
|
739
|
-
: "bg-brand-blue/[0.
|
|
733
|
+
? "bg-brand-purple/[0.03]"
|
|
734
|
+
: "bg-brand-blue/[0.03]";
|
|
740
735
|
|
|
741
736
|
return (
|
|
742
|
-
<div className={`
|
|
737
|
+
<div className={`rounded-xl border ${borderClass} ${bgClass} p-4 sm:p-5`}>
|
|
743
738
|
<SectionLabel>{props.title}</SectionLabel>
|
|
744
739
|
<p className="mt-2 text-sm leading-relaxed text-brand-dark/70">{props.body}</p>
|
|
745
740
|
<div className="mt-4">{props.cta}</div>
|
|
746
741
|
</div>
|
|
747
742
|
);
|
|
748
743
|
}
|
|
744
|
+
|
|
745
|
+
export function SegmentedControl<T extends string>(props: {
|
|
746
|
+
options: Array<{ value: T; label: string }>;
|
|
747
|
+
value: T;
|
|
748
|
+
onChange: (value: T) => void;
|
|
749
|
+
}) {
|
|
750
|
+
return (
|
|
751
|
+
<div className="inline-flex rounded-lg border border-slate-200 bg-slate-50 p-0.5">
|
|
752
|
+
{props.options.map((opt) => (
|
|
753
|
+
<button
|
|
754
|
+
key={opt.value}
|
|
755
|
+
type="button"
|
|
756
|
+
onClick={() => props.onChange(opt.value)}
|
|
757
|
+
className={`rounded-md px-2.5 py-1 text-xs font-medium transition-colors ${
|
|
758
|
+
props.value === opt.value
|
|
759
|
+
? "bg-white text-brand-dark shadow-sm"
|
|
760
|
+
: "text-slate-500 hover:text-brand-dark"
|
|
761
|
+
}`}
|
|
762
|
+
>
|
|
763
|
+
{opt.label}
|
|
764
|
+
</button>
|
|
765
|
+
))}
|
|
766
|
+
</div>
|
|
767
|
+
);
|
|
768
|
+
}
|
|
769
|
+
|
|
770
|
+
export function ListRow(props: {
|
|
771
|
+
children: ReactNode;
|
|
772
|
+
accent?: "green" | "attention" | "blue" | "none";
|
|
773
|
+
onClick?: () => void;
|
|
774
|
+
href?: string;
|
|
775
|
+
}) {
|
|
776
|
+
const accentClass =
|
|
777
|
+
props.accent === "green"
|
|
778
|
+
? "border-l-2 border-emerald-500 pl-3"
|
|
779
|
+
: props.accent === "attention"
|
|
780
|
+
? "border-l-2 border-brand-attention pl-3"
|
|
781
|
+
: props.accent === "blue"
|
|
782
|
+
? "border-l-2 border-brand-blue pl-3"
|
|
783
|
+
: "pl-3.5";
|
|
784
|
+
const clickable = props.onClick !== undefined || props.href !== undefined;
|
|
785
|
+
const content = (
|
|
786
|
+
<div
|
|
787
|
+
className={`flex items-center gap-3 border-b border-slate-100 py-2.5 transition-colors hover:bg-slate-50/60 ${accentClass} ${
|
|
788
|
+
clickable ? "cursor-pointer" : ""
|
|
789
|
+
}`}
|
|
790
|
+
onClick={props.onClick}
|
|
791
|
+
role={clickable ? "button" : undefined}
|
|
792
|
+
tabIndex={clickable ? 0 : undefined}
|
|
793
|
+
onKeyDown={
|
|
794
|
+
clickable
|
|
795
|
+
? (e) => {
|
|
796
|
+
if (e.key === "Enter" || e.key === " ") {
|
|
797
|
+
e.preventDefault();
|
|
798
|
+
props.onClick?.();
|
|
799
|
+
}
|
|
800
|
+
}
|
|
801
|
+
: undefined
|
|
802
|
+
}
|
|
803
|
+
>
|
|
804
|
+
{props.children}
|
|
805
|
+
</div>
|
|
806
|
+
);
|
|
807
|
+
if (props.href) {
|
|
808
|
+
return (
|
|
809
|
+
<a href={guardAwareHref(props.href)} className="block no-underline">
|
|
810
|
+
{content}
|
|
811
|
+
</a>
|
|
812
|
+
);
|
|
813
|
+
}
|
|
814
|
+
return content;
|
|
815
|
+
}
|
|
816
|
+
|
|
817
|
+
export function TabBar<T extends string>(props: {
|
|
818
|
+
tabs: Array<{ value: T; label: string }>;
|
|
819
|
+
active: T;
|
|
820
|
+
onChange: (value: T) => void;
|
|
821
|
+
}) {
|
|
822
|
+
return (
|
|
823
|
+
<div className="flex gap-1 rounded-lg border border-slate-200 bg-slate-50 p-0.5">
|
|
824
|
+
{props.tabs.map((tab) => (
|
|
825
|
+
<button
|
|
826
|
+
key={tab.value}
|
|
827
|
+
type="button"
|
|
828
|
+
onClick={() => props.onChange(tab.value)}
|
|
829
|
+
className={`rounded-md px-3 py-1.5 text-sm font-medium transition-colors ${
|
|
830
|
+
props.active === tab.value
|
|
831
|
+
? "bg-white text-brand-dark shadow-sm"
|
|
832
|
+
: "text-slate-500 hover:text-brand-dark"
|
|
833
|
+
}`}
|
|
834
|
+
>
|
|
835
|
+
{tab.label}
|
|
836
|
+
</button>
|
|
837
|
+
))}
|
|
838
|
+
</div>
|
|
839
|
+
);
|
|
840
|
+
}
|
|
841
|
+
|
|
842
|
+
export function AccordionSection(props: {
|
|
843
|
+
title: string;
|
|
844
|
+
subtitle?: string;
|
|
845
|
+
expanded: boolean;
|
|
846
|
+
onToggle: () => void;
|
|
847
|
+
children: ReactNode;
|
|
848
|
+
}) {
|
|
849
|
+
return (
|
|
850
|
+
<div className="rounded-xl border border-slate-100 overflow-hidden">
|
|
851
|
+
<button
|
|
852
|
+
onClick={props.onToggle}
|
|
853
|
+
className="flex w-full items-center justify-between gap-3 px-4 py-3 text-left transition-colors hover:bg-slate-50/60"
|
|
854
|
+
aria-expanded={props.expanded}
|
|
855
|
+
>
|
|
856
|
+
<div>
|
|
857
|
+
<p className="text-sm font-semibold text-brand-dark">{props.title}</p>
|
|
858
|
+
{props.subtitle ? <p className="text-xs text-slate-400">{props.subtitle}</p> : null}
|
|
859
|
+
</div>
|
|
860
|
+
{props.expanded ? (
|
|
861
|
+
<HiMiniChevronUp className="h-4 w-4 text-slate-400" aria-hidden="true" />
|
|
862
|
+
) : (
|
|
863
|
+
<HiMiniChevronDown className="h-4 w-4 text-slate-400" aria-hidden="true" />
|
|
864
|
+
)}
|
|
865
|
+
</button>
|
|
866
|
+
{props.expanded && (
|
|
867
|
+
<div className="border-t border-slate-100 px-4 py-4">
|
|
868
|
+
{props.children}
|
|
869
|
+
</div>
|
|
870
|
+
)}
|
|
871
|
+
</div>
|
|
872
|
+
);
|
|
873
|
+
}
|
|
874
|
+
|
|
875
|
+
export function StickyActionBar(props: {
|
|
876
|
+
children: ReactNode;
|
|
877
|
+
}) {
|
|
878
|
+
return (
|
|
879
|
+
<div className="sticky bottom-4 z-30 rounded-xl border border-slate-200 bg-white/95 p-4 shadow-lg backdrop-blur">
|
|
880
|
+
{props.children}
|
|
881
|
+
</div>
|
|
882
|
+
);
|
|
883
|
+
}
|
{plugin_scanner-2.0.175 → plugin_scanner-2.0.177}/dashboard/src/apps/app-detail-workspace.tsx
RENAMED
|
@@ -377,7 +377,7 @@ function AppOverviewTab(props: {
|
|
|
377
377
|
return (
|
|
378
378
|
<div className="grid gap-6 lg:grid-cols-[minmax(0,1.3fr)_minmax(0,0.9fr)]">
|
|
379
379
|
<section className="space-y-6">
|
|
380
|
-
<div className="rounded-
|
|
380
|
+
<div className="rounded-xl border border-slate-100 p-4 sm:p-5">
|
|
381
381
|
<div className="flex items-center justify-between gap-3">
|
|
382
382
|
<div>
|
|
383
383
|
<SectionLabel>Status</SectionLabel>
|
|
@@ -428,7 +428,7 @@ function AppOverviewTab(props: {
|
|
|
428
428
|
</div>
|
|
429
429
|
|
|
430
430
|
{props.pendingItems.length > 0 && (
|
|
431
|
-
<div className="rounded-
|
|
431
|
+
<div className="rounded-xl border border-brand-blue/10 bg-brand-blue/[0.03] p-4 sm:p-5">
|
|
432
432
|
<SectionLabel>Pending review</SectionLabel>
|
|
433
433
|
<p className="mt-2 text-sm text-muted-foreground">
|
|
434
434
|
These actions from {harnessDisplayName(props.harness)} need your decision.
|
|
@@ -458,7 +458,7 @@ function AppOverviewTab(props: {
|
|
|
458
458
|
|
|
459
459
|
<section className="space-y-6">
|
|
460
460
|
{props.harnessReceipts.length > 0 && (
|
|
461
|
-
<div className="rounded-
|
|
461
|
+
<div className="rounded-xl border border-slate-100 p-4 sm:p-5">
|
|
462
462
|
<SectionLabel>Recent events</SectionLabel>
|
|
463
463
|
<p className="mt-2 text-sm text-muted-foreground">
|
|
464
464
|
What Guard decided recently.
|
|
@@ -490,7 +490,7 @@ function AppOverviewTab(props: {
|
|
|
490
490
|
)}
|
|
491
491
|
|
|
492
492
|
{props.harnessInventory.length > 0 && (
|
|
493
|
-
<div className="rounded-
|
|
493
|
+
<div className="rounded-xl border border-slate-100 p-4 sm:p-5">
|
|
494
494
|
<SectionLabel>Discovered items</SectionLabel>
|
|
495
495
|
<p className="mt-2 text-sm text-muted-foreground">
|
|
496
496
|
Tools and plugins Guard found in this app.
|
|
@@ -612,7 +612,7 @@ function AppActivityTab(props: {
|
|
|
612
612
|
return (
|
|
613
613
|
<div className="space-y-6">
|
|
614
614
|
{props.queueError && (
|
|
615
|
-
<div className="guard-fade-in rounded-
|
|
615
|
+
<div className="guard-fade-in rounded-xl border border-brand-attention/10 bg-brand-attention/[0.03] p-4 sm:p-5">
|
|
616
616
|
<div className="flex items-start gap-3">
|
|
617
617
|
<HiMiniExclamationTriangle className="mt-0.5 h-5 w-5 shrink-0 text-brand-attention" aria-hidden="true" />
|
|
618
618
|
<div className="flex-1">
|
|
@@ -628,7 +628,7 @@ function AppActivityTab(props: {
|
|
|
628
628
|
</div>
|
|
629
629
|
</div>
|
|
630
630
|
)}
|
|
631
|
-
<div className="rounded-
|
|
631
|
+
<div className="rounded-xl border border-slate-100 p-4 sm:p-5">
|
|
632
632
|
<div className="flex flex-wrap items-center gap-2">
|
|
633
633
|
{(
|
|
634
634
|
[
|
|
@@ -747,7 +747,7 @@ function AppActivityTab(props: {
|
|
|
747
747
|
function ReceiptGroup({ title, items, selectedIds, onToggle }: { title: string; items: GuardReceipt[]; selectedIds: Set<string>; onToggle: (id: string) => void }) {
|
|
748
748
|
if (items.length === 0) return null;
|
|
749
749
|
return (
|
|
750
|
-
<div className="rounded-
|
|
750
|
+
<div className="rounded-xl border border-slate-100 p-4 sm:p-5">
|
|
751
751
|
<div className="flex items-center justify-between">
|
|
752
752
|
<SectionLabel>{title}</SectionLabel>
|
|
753
753
|
<span className="text-xs text-muted-foreground">{items.length} events</span>
|
|
@@ -866,7 +866,7 @@ function AppSettingsTab(props: {
|
|
|
866
866
|
<div className="grid gap-6 lg:grid-cols-[minmax(0,1fr)_minmax(0,0.8fr)]">
|
|
867
867
|
<div className="space-y-6">
|
|
868
868
|
{props.policyError && (
|
|
869
|
-
<div className="guard-fade-in rounded-
|
|
869
|
+
<div className="guard-fade-in rounded-xl border border-brand-attention/10 bg-brand-attention/[0.03] p-4 sm:p-5">
|
|
870
870
|
<div className="flex items-start gap-3">
|
|
871
871
|
<HiMiniExclamationTriangle className="mt-0.5 h-5 w-5 shrink-0 text-brand-attention" aria-hidden="true" />
|
|
872
872
|
<div className="flex-1">
|
|
@@ -882,7 +882,7 @@ function AppSettingsTab(props: {
|
|
|
882
882
|
</div>
|
|
883
883
|
</div>
|
|
884
884
|
)}
|
|
885
|
-
<div className="rounded-
|
|
885
|
+
<div className="rounded-xl border border-slate-100 p-4 sm:p-5">
|
|
886
886
|
<div className="flex items-center justify-between gap-3">
|
|
887
887
|
<SectionLabel>Remembered decisions</SectionLabel>
|
|
888
888
|
{props.harnessPolicies.length > 0 && props.onClearAppPolicies && (
|
|
@@ -937,7 +937,7 @@ function AppSettingsTab(props: {
|
|
|
937
937
|
|
|
938
938
|
{/* Clear confirmation */}
|
|
939
939
|
{showClearConfirm && (
|
|
940
|
-
<div ref={confirmRef} className="guard-fade-in rounded-
|
|
940
|
+
<div ref={confirmRef} className="guard-fade-in rounded-xl border border-brand-attention/10 bg-brand-attention/[0.03] p-4 sm:p-5">
|
|
941
941
|
<div className="flex items-start gap-3">
|
|
942
942
|
<HiMiniExclamationTriangle className="mt-0.5 h-5 w-5 shrink-0 text-brand-attention" aria-hidden="true" />
|
|
943
943
|
<div>
|
|
@@ -979,7 +979,7 @@ function AppSettingsTab(props: {
|
|
|
979
979
|
|
|
980
980
|
<div className="space-y-6">
|
|
981
981
|
{props.status === "needs_setup" && (
|
|
982
|
-
<div className="rounded-
|
|
982
|
+
<div className="rounded-xl border border-brand-attention/10 bg-brand-attention/[0.03] p-4 sm:p-5">
|
|
983
983
|
<div className="flex items-start gap-3">
|
|
984
984
|
<HiMiniExclamationTriangle className="mt-0.5 h-5 w-5 shrink-0 text-brand-attention" />
|
|
985
985
|
<div>
|
|
@@ -1026,7 +1026,7 @@ function ActivitySparkline({ receipts }: { receipts: GuardReceipt[] }) {
|
|
|
1026
1026
|
const maxVal = Math.max(...data.map((d) => d.allowed + d.blocked), 1);
|
|
1027
1027
|
|
|
1028
1028
|
return (
|
|
1029
|
-
<div className="rounded-
|
|
1029
|
+
<div className="rounded-xl border border-slate-100 p-4 sm:p-5">
|
|
1030
1030
|
<div className="flex items-center justify-between">
|
|
1031
1031
|
<SectionLabel>Last 7 days</SectionLabel>
|
|
1032
1032
|
<HiMiniChartBar className="h-4 w-4 text-slate-400" aria-hidden="true" />
|
|
@@ -1125,7 +1125,7 @@ function CloudValueBanner({
|
|
|
1125
1125
|
cta: { label: string; href: string };
|
|
1126
1126
|
}) {
|
|
1127
1127
|
return (
|
|
1128
|
-
<div className="rounded-
|
|
1128
|
+
<div className="rounded-xl border border-brand-purple/10 bg-brand-purple/[0.03] p-4 sm:p-5">
|
|
1129
1129
|
<div className="flex items-start gap-3">
|
|
1130
1130
|
<div className="mt-0.5 shrink-0">{icon}</div>
|
|
1131
1131
|
<div className="flex-1">
|
|
@@ -0,0 +1,210 @@
|
|
|
1
|
+
import { useCallback } from "react";
|
|
2
|
+
import {
|
|
3
|
+
HiMiniCheckCircle,
|
|
4
|
+
HiMiniExclamationCircle,
|
|
5
|
+
HiMiniWrenchScrewdriver,
|
|
6
|
+
HiMiniXCircle,
|
|
7
|
+
HiMiniChevronRight,
|
|
8
|
+
} from "react-icons/hi2";
|
|
9
|
+
import {
|
|
10
|
+
ActionButton,
|
|
11
|
+
EmptyState,
|
|
12
|
+
SectionLabel,
|
|
13
|
+
Tag,
|
|
14
|
+
GuardHero,
|
|
15
|
+
ProofStrip,
|
|
16
|
+
} from "./approval-center-primitives";
|
|
17
|
+
import { harnessDisplayName } from "./approval-center-utils";
|
|
18
|
+
import type { GuardInventoryItem, GuardPolicyDecision, GuardReceipt, GuardRuntimeSnapshot } from "./guard-types";
|
|
19
|
+
|
|
20
|
+
type FleetWorkspaceProps = {
|
|
21
|
+
runtime: GuardRuntimeSnapshot;
|
|
22
|
+
policies: GuardPolicyDecision[];
|
|
23
|
+
inventory:
|
|
24
|
+
| { kind: "idle" }
|
|
25
|
+
| { kind: "loading" }
|
|
26
|
+
| { kind: "error"; message: string }
|
|
27
|
+
| { kind: "ready"; items: GuardInventoryItem[] };
|
|
28
|
+
onConnectHarness?: (harness: string) => void;
|
|
29
|
+
onTestHarness?: (harness: string) => void;
|
|
30
|
+
onRepairHarness?: (harness: string) => void;
|
|
31
|
+
onOpenAppDetail?: (harness: string) => void;
|
|
32
|
+
};
|
|
33
|
+
|
|
34
|
+
function collectHarnesses(snapshot: GuardRuntimeSnapshot): string[] {
|
|
35
|
+
const harnesses = new Set<string>();
|
|
36
|
+
for (const item of snapshot.items) harnesses.add(item.harness);
|
|
37
|
+
for (const receipt of snapshot.latest_receipts) harnesses.add(receipt.harness);
|
|
38
|
+
return Array.from(harnesses).sort((a, b) => a.localeCompare(b));
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
function renderReceiptContext(receipt: GuardReceipt): string {
|
|
42
|
+
return `${harnessDisplayName(receipt.harness)} · ${receipt.policy_decision.replace(/-/g, " ")}`;
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
type AppStatus = "protected" | "found_unprotected" | "needs_repair" | "not_found";
|
|
46
|
+
|
|
47
|
+
function resolveAppStatus(
|
|
48
|
+
install: { active?: boolean } | undefined,
|
|
49
|
+
hasInventory: boolean,
|
|
50
|
+
hasReceipts: boolean
|
|
51
|
+
): AppStatus {
|
|
52
|
+
if (install !== undefined) {
|
|
53
|
+
if (install.active) return "protected";
|
|
54
|
+
return "needs_repair";
|
|
55
|
+
}
|
|
56
|
+
if (!hasInventory && !hasReceipts) return "not_found";
|
|
57
|
+
return "found_unprotected";
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
function StatusIcon({ status }: { status: AppStatus }) {
|
|
61
|
+
if (status === "protected") return <HiMiniCheckCircle className="h-4 w-4 text-emerald-500" aria-hidden="true" />;
|
|
62
|
+
if (status === "found_unprotected") return <HiMiniExclamationCircle className="h-4 w-4 text-brand-attention" aria-hidden="true" />;
|
|
63
|
+
if (status === "needs_repair") return <HiMiniWrenchScrewdriver className="h-4 w-4 text-brand-purple" aria-hidden="true" />;
|
|
64
|
+
return <HiMiniXCircle className="h-4 w-4 text-slate-300" aria-hidden="true" />;
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
function StatusBadge({ status }: { status: AppStatus }) {
|
|
68
|
+
if (status === "protected") return <span className="text-xs font-medium text-emerald-600">Protected</span>;
|
|
69
|
+
if (status === "found_unprotected") return <span className="text-xs font-medium text-brand-attention">Found, not installed</span>;
|
|
70
|
+
if (status === "needs_repair") return <span className="text-xs font-medium text-brand-purple">Repair needed</span>;
|
|
71
|
+
return <span className="text-xs text-slate-400">Not found</span>;
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
export function FleetWorkspace(props: FleetWorkspaceProps) {
|
|
75
|
+
const harnesses = collectHarnesses(props.runtime);
|
|
76
|
+
const managedInstalls = props.runtime.managed_installs ?? [];
|
|
77
|
+
const activeInstalls = managedInstalls.filter((i) => i.active);
|
|
78
|
+
const inventory = props.inventory.kind === "ready" ? props.inventory.items : [];
|
|
79
|
+
const visibleHarnesses = Array.from(
|
|
80
|
+
new Set([
|
|
81
|
+
...managedInstalls.map((i) => i.harness),
|
|
82
|
+
...harnesses,
|
|
83
|
+
...inventory.map((i) => i.harness),
|
|
84
|
+
...props.policies.map((p) => p.harness),
|
|
85
|
+
])
|
|
86
|
+
).sort((a, b) => a.localeCompare(b));
|
|
87
|
+
const runtimeState = props.runtime.runtime_state;
|
|
88
|
+
const receiptHarnesses = new Set(props.runtime.latest_receipts.map((r) => r.harness));
|
|
89
|
+
|
|
90
|
+
return (
|
|
91
|
+
<div className="space-y-8">
|
|
92
|
+
<GuardHero
|
|
93
|
+
status={activeInstalls.length > 0 ? "clear" : "setup_gap"}
|
|
94
|
+
headline={activeInstalls.length > 0 ? "Your apps are covered" : "Connect an app to start"}
|
|
95
|
+
subheadline={
|
|
96
|
+
activeInstalls.length > 0
|
|
97
|
+
? "Confirm that Guard is running and protecting your local AI apps."
|
|
98
|
+
: "Guard works with Codex, Claude Code, Cursor, Hermes, OpenClaw, and more."
|
|
99
|
+
}
|
|
100
|
+
cta={<ActionButton href={props.runtime.fleet_url}>Open Cloud Devices</ActionButton>}
|
|
101
|
+
secondaryCta={
|
|
102
|
+
<ActionButton href={props.runtime.dashboard_url} variant="outline">
|
|
103
|
+
Open Home
|
|
104
|
+
</ActionButton>
|
|
105
|
+
}
|
|
106
|
+
/>
|
|
107
|
+
|
|
108
|
+
<ProofStrip
|
|
109
|
+
items={[
|
|
110
|
+
{ label: "Needs review", value: `${props.runtime.pending_count}`, tone: props.runtime.pending_count > 0 ? "blue" : "slate" },
|
|
111
|
+
{ label: "History", value: `${props.runtime.receipt_count}`, tone: "purple" },
|
|
112
|
+
{ label: "Watched apps", value: `${activeInstalls.length > 0 ? activeInstalls.length : visibleHarnesses.length}`, tone: activeInstalls.length > 0 ? "green" : "slate" },
|
|
113
|
+
{ label: "Runtime", value: runtimeState ? "active" : "offline", tone: runtimeState ? "green" : "slate" },
|
|
114
|
+
]}
|
|
115
|
+
/>
|
|
116
|
+
|
|
117
|
+
<div className="grid gap-8 lg:grid-cols-[minmax(0,1.4fr)_minmax(0,0.8fr)]">
|
|
118
|
+
<section>
|
|
119
|
+
<div className="mb-4">
|
|
120
|
+
<SectionLabel>App coverage</SectionLabel>
|
|
121
|
+
<p className="mt-1 text-sm text-slate-500">Which apps Guard is watching on this machine.</p>
|
|
122
|
+
</div>
|
|
123
|
+
|
|
124
|
+
{visibleHarnesses.length > 0 ? (
|
|
125
|
+
<div className="divide-y divide-slate-100 border-t border-slate-100">
|
|
126
|
+
{visibleHarnesses.map((harness) => {
|
|
127
|
+
const install = managedInstalls.find((i) => i.harness === harness);
|
|
128
|
+
const harnessInventory = inventory.filter((i) => i.harness === harness && i.present);
|
|
129
|
+
const harnessPolicies = props.policies.filter((p) => p.harness === harness);
|
|
130
|
+
const hasReceipts = receiptHarnesses.has(harness);
|
|
131
|
+
const status = resolveAppStatus(install, harnessInventory.length > 0, hasReceipts);
|
|
132
|
+
const isClickable = props.onOpenAppDetail !== undefined;
|
|
133
|
+
|
|
134
|
+
return (
|
|
135
|
+
<div
|
|
136
|
+
key={harness}
|
|
137
|
+
className={`flex items-center justify-between gap-3 py-3 transition-colors ${
|
|
138
|
+
isClickable ? "cursor-pointer hover:bg-slate-50/60" : ""
|
|
139
|
+
}`}
|
|
140
|
+
onClick={isClickable ? () => props.onOpenAppDetail?.(harness) : undefined}
|
|
141
|
+
role={isClickable ? "button" : undefined}
|
|
142
|
+
tabIndex={isClickable ? 0 : undefined}
|
|
143
|
+
onKeyDown={
|
|
144
|
+
isClickable
|
|
145
|
+
? (e) => {
|
|
146
|
+
if (e.key === "Enter" || e.key === " ") {
|
|
147
|
+
e.preventDefault();
|
|
148
|
+
props.onOpenAppDetail?.(harness);
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
: undefined
|
|
152
|
+
}
|
|
153
|
+
>
|
|
154
|
+
<div className="flex min-w-0 items-center gap-3">
|
|
155
|
+
<StatusIcon status={status} />
|
|
156
|
+
<div className="min-w-0">
|
|
157
|
+
<p className="text-sm font-medium text-brand-dark">{harnessDisplayName(harness)}</p>
|
|
158
|
+
<p className="text-xs text-slate-400">
|
|
159
|
+
{harnessInventory.length} actions · {harnessPolicies.length} decisions
|
|
160
|
+
</p>
|
|
161
|
+
</div>
|
|
162
|
+
</div>
|
|
163
|
+
<div className="flex items-center gap-2">
|
|
164
|
+
<StatusBadge status={status} />
|
|
165
|
+
{isClickable && <HiMiniChevronRight className="h-4 w-4 text-slate-300" aria-hidden="true" />}
|
|
166
|
+
</div>
|
|
167
|
+
</div>
|
|
168
|
+
);
|
|
169
|
+
})}
|
|
170
|
+
</div>
|
|
171
|
+
) : (
|
|
172
|
+
<EmptyState
|
|
173
|
+
title="No watched apps yet"
|
|
174
|
+
body="Run HOL Guard once with Codex, Claude Code, Cursor, Hermes, or another supported app and this machine will show coverage here."
|
|
175
|
+
tone="teach"
|
|
176
|
+
/>
|
|
177
|
+
)}
|
|
178
|
+
|
|
179
|
+
{props.inventory.kind === "error" ? (
|
|
180
|
+
<p className="mt-3 text-xs text-slate-500">{props.inventory.message}</p>
|
|
181
|
+
) : null}
|
|
182
|
+
</section>
|
|
183
|
+
|
|
184
|
+
<section>
|
|
185
|
+
<div className="mb-4">
|
|
186
|
+
<SectionLabel>Recent choices</SectionLabel>
|
|
187
|
+
<p className="mt-1 text-sm text-slate-500">What Guard decided recently.</p>
|
|
188
|
+
</div>
|
|
189
|
+
{props.runtime.latest_receipts.length > 0 ? (
|
|
190
|
+
<div className="space-y-0 divide-y divide-slate-100 border-t border-slate-100">
|
|
191
|
+
{props.runtime.latest_receipts.slice(0, 6).map((receipt) => (
|
|
192
|
+
<div key={receipt.receipt_id} className="py-2.5">
|
|
193
|
+
<p className="truncate text-sm font-medium text-brand-dark">
|
|
194
|
+
{receipt.artifact_name ?? receipt.artifact_id}
|
|
195
|
+
</p>
|
|
196
|
+
<p className="text-xs text-slate-400">{renderReceiptContext(receipt)}</p>
|
|
197
|
+
</div>
|
|
198
|
+
))}
|
|
199
|
+
</div>
|
|
200
|
+
) : (
|
|
201
|
+
<EmptyState
|
|
202
|
+
title="No choices yet"
|
|
203
|
+
body="Allow or block an action once and HOL Guard will start building local history for this machine."
|
|
204
|
+
/>
|
|
205
|
+
)}
|
|
206
|
+
</section>
|
|
207
|
+
</div>
|
|
208
|
+
</div>
|
|
209
|
+
);
|
|
210
|
+
}
|