failproofai 0.0.11-beta.2 → 0.0.11-beta.4
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.
- package/.next/standalone/.next/BUILD_ID +1 -1
- package/.next/standalone/.next/app-path-routes-manifest.json +8 -1
- package/.next/standalone/.next/build-manifest.json +10 -10
- package/.next/standalone/.next/prerender-manifest.json +3 -32
- package/.next/standalone/.next/required-server-files.json +2 -1
- package/.next/standalone/.next/routes-manifest.json +45 -3
- package/.next/standalone/.next/server/app/_global-error/page/build-manifest.json +7 -7
- package/.next/standalone/.next/server/app/_global-error/page/server-reference-manifest.json +1 -1
- package/.next/standalone/.next/server/app/_global-error/page.js +6 -6
- package/.next/standalone/.next/server/app/_global-error/page.js.nft.json +1 -1
- package/.next/standalone/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
- package/.next/standalone/.next/server/app/_global-error.html +1 -1
- package/.next/standalone/.next/server/app/_global-error.rsc +7 -7
- package/.next/standalone/.next/server/app/_global-error.segments/__PAGE__.segment.rsc +2 -2
- package/.next/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +7 -7
- package/.next/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +3 -3
- package/.next/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +3 -3
- package/.next/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
- package/.next/standalone/.next/server/app/_not-found/page/build-manifest.json +7 -7
- package/.next/standalone/.next/server/app/_not-found/page/next-font-manifest.json +2 -6
- package/.next/standalone/.next/server/app/_not-found/page/server-reference-manifest.json +1 -1
- package/.next/standalone/.next/server/app/_not-found/page.js +12 -13
- package/.next/standalone/.next/server/app/_not-found/page.js.nft.json +1 -1
- package/.next/standalone/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
- package/.next/standalone/.next/server/app/_not-found.html +1 -1
- package/.next/standalone/.next/server/app/_not-found.rsc +16 -16
- package/.next/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +16 -16
- package/.next/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +4 -4
- package/.next/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +10 -10
- package/.next/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +2 -2
- package/.next/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +3 -3
- package/.next/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +2 -2
- package/.next/standalone/.next/server/app/api/audit/run/route/app-paths-manifest.json +3 -0
- package/.next/standalone/.next/server/app/api/audit/run/route/server-reference-manifest.json +4 -0
- package/.next/standalone/.next/server/app/api/audit/run/route.js +8 -0
- package/.next/standalone/.next/server/app/api/audit/run/route.js.nft.json +1 -0
- package/.next/standalone/.next/server/app/api/audit/run/route_client-reference-manifest.js +3 -0
- package/.next/standalone/.next/server/app/api/audit/status/route/app-paths-manifest.json +3 -0
- package/.next/standalone/.next/server/app/api/audit/status/route/build-manifest.json +9 -0
- package/.next/standalone/.next/server/app/api/audit/status/route/server-reference-manifest.json +4 -0
- package/.next/standalone/.next/server/app/api/audit/status/route.js +6 -0
- package/.next/standalone/.next/server/app/api/audit/status/route.js.map +5 -0
- package/.next/standalone/.next/server/app/api/audit/status/route.js.nft.json +1 -0
- package/.next/standalone/.next/server/app/api/audit/status/route_client-reference-manifest.js +3 -0
- package/.next/standalone/.next/server/app/api/auth/login-request/route/app-paths-manifest.json +3 -0
- package/.next/standalone/.next/server/app/api/auth/login-request/route/build-manifest.json +9 -0
- package/.next/standalone/.next/server/app/api/auth/login-request/route/server-reference-manifest.json +4 -0
- package/.next/standalone/.next/server/app/api/auth/login-request/route.js +6 -0
- package/.next/standalone/.next/server/app/api/auth/login-request/route.js.map +5 -0
- package/.next/standalone/.next/server/app/api/auth/login-request/route.js.nft.json +1 -0
- package/.next/standalone/.next/server/app/api/auth/login-request/route_client-reference-manifest.js +3 -0
- package/.next/standalone/.next/server/app/api/auth/login-verify/route/app-paths-manifest.json +3 -0
- package/.next/standalone/.next/server/app/api/auth/login-verify/route/build-manifest.json +9 -0
- package/.next/standalone/.next/server/app/api/auth/login-verify/route/server-reference-manifest.json +4 -0
- package/.next/standalone/.next/server/app/api/auth/login-verify/route.js +7 -0
- package/.next/standalone/.next/server/app/api/auth/login-verify/route.js.map +5 -0
- package/.next/standalone/.next/server/app/api/auth/login-verify/route.js.nft.json +1 -0
- package/.next/standalone/.next/server/app/api/auth/login-verify/route_client-reference-manifest.js +3 -0
- package/.next/standalone/.next/server/app/api/auth/logout/route/app-paths-manifest.json +3 -0
- package/.next/standalone/.next/server/app/api/auth/logout/route/build-manifest.json +9 -0
- package/.next/standalone/.next/server/app/api/auth/logout/route/server-reference-manifest.json +4 -0
- package/.next/standalone/.next/server/app/api/auth/logout/route.js +7 -0
- package/.next/standalone/.next/server/app/api/auth/logout/route.js.map +5 -0
- package/.next/standalone/.next/server/app/api/auth/logout/route.js.nft.json +1 -0
- package/.next/standalone/.next/server/app/api/auth/logout/route_client-reference-manifest.js +3 -0
- package/.next/standalone/.next/server/app/api/auth/reminder/route/app-paths-manifest.json +3 -0
- package/.next/standalone/.next/server/app/api/auth/reminder/route/build-manifest.json +9 -0
- package/.next/standalone/.next/server/app/api/auth/reminder/route/server-reference-manifest.json +4 -0
- package/.next/standalone/.next/server/app/api/auth/reminder/route.js +7 -0
- package/.next/standalone/.next/server/app/api/auth/reminder/route.js.map +5 -0
- package/.next/standalone/.next/server/app/api/auth/reminder/route.js.nft.json +1 -0
- package/.next/standalone/.next/server/app/api/auth/reminder/route_client-reference-manifest.js +3 -0
- package/.next/standalone/.next/server/app/api/auth/status/route/app-paths-manifest.json +3 -0
- package/.next/standalone/.next/server/app/api/auth/status/route/build-manifest.json +9 -0
- package/.next/standalone/.next/server/app/api/auth/status/route/server-reference-manifest.json +4 -0
- package/.next/standalone/.next/server/app/api/auth/status/route.js +7 -0
- package/.next/standalone/.next/server/app/api/auth/status/route.js.map +5 -0
- package/.next/standalone/.next/server/app/api/auth/status/route.js.nft.json +1 -0
- package/.next/standalone/.next/server/app/api/auth/status/route_client-reference-manifest.js +3 -0
- package/.next/standalone/.next/server/app/api/download/[project]/[session]/route.js +3 -3
- package/.next/standalone/.next/server/app/api/download/[project]/[session]/route.js.nft.json +1 -1
- package/.next/standalone/.next/server/app/audit/page/app-paths-manifest.json +3 -0
- package/.next/standalone/.next/server/app/audit/page/build-manifest.json +18 -0
- package/.next/standalone/.next/server/app/audit/page/next-font-manifest.json +6 -0
- package/.next/standalone/.next/server/app/audit/page/react-loadable-manifest.json +1 -0
- package/.next/standalone/.next/server/app/audit/page/server-reference-manifest.json +29 -0
- package/.next/standalone/.next/server/app/audit/page.js +17 -0
- package/.next/standalone/.next/server/app/audit/page.js.map +5 -0
- package/.next/standalone/.next/server/app/audit/page.js.nft.json +1 -0
- package/.next/standalone/.next/server/app/audit/page_client-reference-manifest.js +3 -0
- package/.next/standalone/.next/server/app/index.html +1 -1
- package/.next/standalone/.next/server/app/index.rsc +16 -17
- package/.next/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +2 -2
- package/.next/standalone/.next/server/app/index.segments/_full.segment.rsc +16 -17
- package/.next/standalone/.next/server/app/index.segments/_head.segment.rsc +4 -4
- package/.next/standalone/.next/server/app/index.segments/_index.segment.rsc +10 -10
- package/.next/standalone/.next/server/app/index.segments/_tree.segment.rsc +2 -3
- package/.next/standalone/.next/server/app/page/build-manifest.json +7 -7
- package/.next/standalone/.next/server/app/page/next-font-manifest.json +2 -6
- package/.next/standalone/.next/server/app/page/server-reference-manifest.json +1 -1
- package/.next/standalone/.next/server/app/page.js +14 -15
- package/.next/standalone/.next/server/app/page.js.nft.json +1 -1
- package/.next/standalone/.next/server/app/page_client-reference-manifest.js +1 -1
- package/.next/standalone/.next/server/app/policies/page/build-manifest.json +7 -7
- package/.next/standalone/.next/server/app/policies/page/next-font-manifest.json +2 -6
- package/.next/standalone/.next/server/app/policies/page/server-reference-manifest.json +8 -8
- package/.next/standalone/.next/server/app/policies/page.js +16 -16
- package/.next/standalone/.next/server/app/policies/page.js.nft.json +1 -1
- package/.next/standalone/.next/server/app/policies/page_client-reference-manifest.js +1 -1
- package/.next/standalone/.next/server/app/project/[name]/page/build-manifest.json +7 -7
- package/.next/standalone/.next/server/app/project/[name]/page/next-font-manifest.json +2 -6
- package/.next/standalone/.next/server/app/project/[name]/page/server-reference-manifest.json +1 -1
- package/.next/standalone/.next/server/app/project/[name]/page.js +16 -17
- package/.next/standalone/.next/server/app/project/[name]/page.js.nft.json +1 -1
- package/.next/standalone/.next/server/app/project/[name]/page_client-reference-manifest.js +1 -1
- package/.next/standalone/.next/server/app/project/[name]/session/[sessionId]/page/build-manifest.json +7 -7
- package/.next/standalone/.next/server/app/project/[name]/session/[sessionId]/page/next-font-manifest.json +2 -6
- package/.next/standalone/.next/server/app/project/[name]/session/[sessionId]/page/react-loadable-manifest.json +2 -2
- package/.next/standalone/.next/server/app/project/[name]/session/[sessionId]/page/server-reference-manifest.json +2 -2
- package/.next/standalone/.next/server/app/project/[name]/session/[sessionId]/page.js +19 -20
- package/.next/standalone/.next/server/app/project/[name]/session/[sessionId]/page.js.nft.json +1 -1
- package/.next/standalone/.next/server/app/project/[name]/session/[sessionId]/page_client-reference-manifest.js +1 -1
- package/.next/standalone/.next/server/app/projects/page/build-manifest.json +7 -7
- package/.next/standalone/.next/server/app/projects/page/next-font-manifest.json +2 -6
- package/.next/standalone/.next/server/app/projects/page/server-reference-manifest.json +1 -1
- package/.next/standalone/.next/server/app/projects/page.js +15 -16
- package/.next/standalone/.next/server/app/projects/page.js.nft.json +1 -1
- package/.next/standalone/.next/server/app/projects/page_client-reference-manifest.js +1 -1
- package/.next/standalone/.next/server/app-paths-manifest.json +8 -1
- package/.next/standalone/.next/server/chunks/[externals]__14odj07._.js +3 -0
- package/.next/standalone/.next/server/chunks/{[externals]__0z0j--b._.js → [externals]__1nl3dvw._.js} +1 -1
- package/.next/standalone/.next/server/chunks/{[externals]__0-p9.k~._.js → [externals]__1s61mel._.js} +1 -1
- package/.next/standalone/.next/server/chunks/{[externals]_node_os_06ur78j._.js → [externals]_node_os_0by37l-._.js} +1 -1
- package/.next/standalone/.next/server/chunks/[root-of-the-server]__07tgnzi._.js +3 -0
- package/.next/standalone/.next/server/chunks/{[root-of-the-server]__0kjo7d_._.js → [root-of-the-server]__0_0xu5z._.js} +2 -2
- package/.next/standalone/.next/server/chunks/[root-of-the-server]__0cag8qd._.js +3 -0
- package/.next/standalone/.next/server/chunks/{[root-of-the-server]__0z-180.._.js → [root-of-the-server]__0cycwg6._.js} +2 -2
- package/.next/standalone/.next/server/chunks/{[root-of-the-server]__08px0ym._.js → [root-of-the-server]__0f7mikp._.js} +1 -1
- package/.next/standalone/.next/server/chunks/[root-of-the-server]__0oeun7z._.js +3 -0
- package/.next/standalone/.next/server/chunks/[root-of-the-server]__0q-v9z2._.js +3 -0
- package/.next/standalone/.next/server/chunks/[root-of-the-server]__0rv7m0k._.js +3 -0
- package/.next/standalone/.next/server/chunks/{[root-of-the-server]__0g48iv.._.js → [root-of-the-server]__0sb_5m8._.js} +3 -3
- package/.next/standalone/.next/server/chunks/{[root-of-the-server]__0j8-xkl._.js → [root-of-the-server]__0xuaoik._.js} +2 -2
- package/.next/standalone/.next/server/chunks/[root-of-the-server]__12pit4m._.js +3 -0
- package/.next/standalone/.next/server/chunks/[root-of-the-server]__13h8pzr._.js +3 -0
- package/.next/standalone/.next/server/chunks/[root-of-the-server]__13ra2jq._.js +3 -0
- package/.next/standalone/.next/server/chunks/[root-of-the-server]__17g9wh7._.js +3 -0
- package/.next/standalone/.next/server/chunks/{[root-of-the-server]__0d_ob4n._.js → [root-of-the-server]__1_mqemn._.js} +2 -2
- package/.next/standalone/.next/server/chunks/[root-of-the-server]__1b9z5-i._.js +3 -0
- package/.next/standalone/.next/server/chunks/{[root-of-the-server]__0vlhtkc._.js → [root-of-the-server]__1hgv_75._.js} +1 -1
- package/.next/standalone/.next/server/chunks/[root-of-the-server]__1ixjiy8._.js +3 -0
- package/.next/standalone/.next/server/chunks/{[root-of-the-server]__0wu7fr7._.js → [root-of-the-server]__1jm9fw6._.js} +1 -1
- package/.next/standalone/.next/server/chunks/[root-of-the-server]__1legmza._.js +3 -0
- package/.next/standalone/.next/server/chunks/{[root-of-the-server]__0fwb7ao._.js → [root-of-the-server]__1m2_4t0._.js} +2 -2
- package/.next/standalone/.next/server/chunks/{[root-of-the-server]__0yfq1yr._.js → [root-of-the-server]__1mhmdzs._.js} +1 -1
- package/.next/standalone/.next/server/chunks/{[root-of-the-server]__0zso~62._.js → [root-of-the-server]__1ou2ehh._.js} +1 -1
- package/.next/standalone/.next/server/chunks/[root-of-the-server]__1qxztj-._.js +3 -0
- package/.next/standalone/.next/server/chunks/[root-of-the-server]__1rhmvod._.js +3 -0
- package/.next/standalone/.next/server/chunks/{[root-of-the-server]__0.~nmr9._.js → [root-of-the-server]__1w9zl9-._.js} +1 -1
- package/.next/standalone/.next/server/chunks/{_0ebx_lc._.js → _0p53ge1._.js} +2 -2
- package/.next/standalone/.next/server/chunks/_1-1804f._.js +3 -0
- package/.next/standalone/.next/server/chunks/_next-internal_server_app_api_audit_run_route_actions_1qgp9io.js +3 -0
- package/.next/standalone/.next/server/chunks/_next-internal_server_app_api_audit_status_route_actions_1f7pjof.js +3 -0
- package/.next/standalone/.next/server/chunks/_next-internal_server_app_api_auth_login-request_route_actions_1c49co0.js +3 -0
- package/.next/standalone/.next/server/chunks/_next-internal_server_app_api_auth_login-verify_route_actions_1r3slzk.js +3 -0
- package/.next/standalone/.next/server/chunks/_next-internal_server_app_api_auth_logout_route_actions_0regwyr.js +3 -0
- package/.next/standalone/.next/server/chunks/_next-internal_server_app_api_auth_reminder_route_actions_1kjgxf8.js +3 -0
- package/.next/standalone/.next/server/chunks/_next-internal_server_app_api_auth_status_route_actions_1aho9zu.js +3 -0
- package/.next/standalone/.next/server/chunks/{_next-internal_server_app_api_download_[project]_[session]_route_actions_0wb00i-.js → _next-internal_server_app_api_download_[project]_[session]_route_actions_1is7vs7.js} +1 -1
- package/.next/standalone/.next/server/chunks/{lib_logger_ts_047tt9f._.js → lib_logger_ts_07e65t5._.js} +1 -1
- package/.next/standalone/.next/server/chunks/node_modules_next_dist_esm_build_templates_app-route_17k9e3w.js +23 -0
- package/.next/standalone/.next/server/chunks/node_modules_posthog-node_dist_entrypoints_index_node_mjs_01r25oi._.js +3 -0
- package/.next/standalone/.next/server/chunks/node_modules_posthog-node_dist_entrypoints_index_node_mjs_09z9-p7._.js +3 -0
- package/.next/standalone/.next/server/chunks/{package_json_[json]_cjs_0z7w.hh._.js → package_json_[json]_cjs_1nxcc4v._.js} +2 -2
- package/.next/standalone/.next/server/chunks/ssr/{[externals]__12dv.x0._.js → [externals]__1_g_b3t._.js} +1 -1
- package/.next/standalone/.next/server/chunks/ssr/{[externals]_node_async_hooks_0v0ln8c._.js → [externals]_node_async_hooks_1gjz99j._.js} +1 -1
- package/.next/standalone/.next/server/chunks/ssr/{[root-of-the-server]__0ts150~._.js → [root-of-the-server]__0-wdurj._.js} +2 -2
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__00jkjmt._.js +4 -0
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__013du6r._.js +4 -0
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__06f-yix._.js +3 -0
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__0989_dx._.js +3 -0
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__0e85wxv._.js +4 -0
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__0gfxvb1._.js +3 -0
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__0h12me5._.js +3 -0
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__0l13qf2._.js +3 -0
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__1-scthx._.js +3 -0
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__11rtg6s._.js +3 -0
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__14dd6h8._.js +3 -0
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__1cd25c7._.js +4 -0
- package/.next/standalone/.next/server/chunks/ssr/{[root-of-the-server]__0t5l7a5._.js → [root-of-the-server]__1d8omgc._.js} +1 -1
- package/.next/standalone/.next/server/chunks/ssr/{[root-of-the-server]__0tcyn68._.js → [root-of-the-server]__1dky4g0._.js} +1 -1
- package/.next/standalone/.next/server/chunks/ssr/{[root-of-the-server]__098zro9._.js → [root-of-the-server]__1fax1sl._.js} +4 -4
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__1hlrq6y._.js +4 -0
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__1ihxdo5._.js +4 -0
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__1mt35_w._.js +221 -0
- package/.next/standalone/.next/server/chunks/ssr/{[root-of-the-server]__0-wn51s._.js → [root-of-the-server]__1pcxxwg._.js} +3 -3
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__1usf8v2._.js +3 -0
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__1vvfde2._.js +4 -0
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__212nf49._.js +3 -0
- package/.next/standalone/.next/server/chunks/ssr/{_03d7qyt._.js → _05whahf._.js} +1 -1
- package/.next/standalone/.next/server/chunks/ssr/_11_p9y8._.js +3 -0
- package/.next/standalone/.next/server/chunks/ssr/{_0xb8ngh._.js → _1kje4fm._.js} +1 -1
- package/.next/standalone/.next/server/chunks/ssr/{_0zx~s__._.js → _1p0-leb._.js} +1 -1
- package/.next/standalone/.next/server/chunks/ssr/{app_04qfs8z._.js → app_087bt9w._.js} +1 -1
- package/.next/standalone/.next/server/chunks/ssr/{app_0uosk1e._.js → app_1fvisnp._.js} +1 -1
- package/.next/standalone/.next/server/chunks/ssr/{app_13f0ohr._.js → app_209u41o._.js} +1 -1
- package/.next/standalone/.next/server/chunks/ssr/app_audit__components_audit-dashboard_tsx_0p9ud47._.js +43 -0
- package/.next/standalone/.next/server/chunks/ssr/app_audit_loading_tsx_1j1kc6j._.js +3 -0
- package/.next/standalone/.next/server/chunks/ssr/{app_error_tsx_11t4ysq._.js → app_error_tsx_1zds1ns._.js} +1 -1
- package/.next/standalone/.next/server/chunks/ssr/{app_global-error_tsx_0m9qisk._.js → app_global-error_tsx_113y3za._.js} +1 -1
- package/.next/standalone/.next/server/chunks/ssr/{app_global-error_tsx_0xerkr6._.js → app_global-error_tsx_1kp6l3x._.js} +2 -2
- package/.next/standalone/.next/server/chunks/ssr/app_policies_hooks-client_tsx_19dqvpc._.js +8 -0
- package/.next/standalone/.next/server/chunks/ssr/{app_project_[name]_error_tsx_0.9-fod._.js → app_project_[name]_error_tsx_1v02_5n._.js} +1 -1
- package/.next/standalone/.next/server/chunks/ssr/{app_project_[name]_loading_tsx_03g9xy0._.js → app_project_[name]_loading_tsx_05-l4uf._.js} +1 -1
- package/.next/standalone/.next/server/chunks/ssr/{app_project_[name]_session_[sessionId]_error_tsx_0ler-mr._.js → app_project_[name]_session_[sessionId]_error_tsx_0-lj3nd._.js} +1 -1
- package/.next/standalone/.next/server/chunks/ssr/{app_project_[name]_session_[sessionId]_loading_tsx_0c0e3yx._.js → app_project_[name]_session_[sessionId]_loading_tsx_0l4aixs._.js} +1 -1
- package/.next/standalone/.next/server/chunks/ssr/app_projects_loading_tsx_20-3u8b._.js +3 -0
- package/.next/standalone/.next/server/chunks/ssr/{lib_codex-projects_ts_0eosib~._.js → lib_codex-projects_ts_0pqlw37._.js} +1 -1
- package/.next/standalone/.next/server/chunks/ssr/{lib_copilot-projects_ts_0r8xkn8._.js → lib_copilot-projects_ts_19wl7tp._.js} +1 -1
- package/.next/standalone/.next/server/chunks/ssr/{lib_cursor-projects_ts_0qt1scg._.js → lib_cursor-projects_ts_18-iwyk._.js} +1 -1
- package/.next/standalone/.next/server/chunks/ssr/{lib_gemini-projects_ts_0sl~yqr._.js → lib_gemini-projects_ts_1c7bgx-._.js} +1 -1
- package/.next/standalone/.next/server/chunks/ssr/{lib_opencode-projects_ts_0op9gyp._.js → lib_opencode-projects_ts_15bjxkm._.js} +1 -1
- package/.next/standalone/.next/server/chunks/ssr/{lib_pi-projects_ts_103tsh1._.js → lib_pi-projects_ts_1wikofb._.js} +1 -1
- package/.next/standalone/.next/server/chunks/ssr/{lib_utils_ts_068jk73._.js → lib_utils_ts_0az0sfq._.js} +1 -1
- package/.next/standalone/.next/server/chunks/ssr/node_modules_1ynf7el._.js +3 -0
- package/.next/standalone/.next/server/chunks/ssr/node_modules_html2canvas_dist_html2canvas_esm_0hmsdod.js +3 -0
- package/.next/standalone/.next/server/chunks/ssr/node_modules_html2canvas_dist_html2canvas_esm_1n-0xws.js +3 -0
- package/.next/standalone/.next/server/chunks/ssr/{node_modules_next_0rd0oc-._.js → node_modules_next_1a1kch7._.js} +1 -1
- package/.next/standalone/.next/server/chunks/ssr/{node_modules_next_dist_0h9llsw._.js → node_modules_next_dist_0uboya6._.js} +2 -2
- package/.next/standalone/.next/server/chunks/ssr/{node_modules_next_dist_11dij6w._.js → node_modules_next_dist_1d_onnt._.js} +3 -3
- package/.next/standalone/.next/server/chunks/ssr/{node_modules_next_dist_client_components_0inhx6q._.js → node_modules_next_dist_client_components_0wpq8j3._.js} +1 -1
- package/.next/standalone/.next/server/chunks/ssr/{node_modules_next_dist_client_components_builtin_forbidden_0ghu-f7.js → node_modules_next_dist_client_components_builtin_forbidden_0symwr9.js} +1 -1
- package/.next/standalone/.next/server/chunks/ssr/{node_modules_next_dist_client_components_builtin_unauthorized_0cjv-23.js → node_modules_next_dist_client_components_builtin_unauthorized_0l_sp0x.js} +1 -1
- package/.next/standalone/.next/server/chunks/ssr/node_modules_next_dist_esm_build_templates_app-page_0-uvagv.js +4 -0
- package/.next/standalone/.next/server/chunks/ssr/node_modules_next_dist_esm_build_templates_app-page_03c7gi5.js +4 -0
- package/.next/standalone/.next/server/chunks/ssr/node_modules_next_dist_esm_build_templates_app-page_09p-8om.js +4 -0
- package/.next/standalone/.next/server/chunks/ssr/node_modules_next_dist_esm_build_templates_app-page_0v-kfiu.js +4 -0
- package/.next/standalone/.next/server/chunks/ssr/{node_modules_next_dist_esm_build_templates_app-page_0j79~gv.js → node_modules_next_dist_esm_build_templates_app-page_0xrgzyz.js} +1 -1
- package/.next/standalone/.next/server/chunks/ssr/node_modules_next_dist_esm_build_templates_app-page_1806lsc.js +4 -0
- package/.next/standalone/.next/server/chunks/ssr/node_modules_next_dist_esm_build_templates_app-page_1j6dd-e.js +4 -0
- package/.next/standalone/.next/server/chunks/ssr/node_modules_next_dist_esm_build_templates_app-page_1sa65r-.js +4 -0
- package/.next/standalone/.next/server/chunks/ssr/node_modules_posthog-node_dist_entrypoints_index_node_mjs_11bnuzn._.js +3 -0
- package/.next/standalone/.next/server/chunks/ssr/src_hooks_1ezd2jf._.js +5 -0
- package/.next/standalone/.next/server/chunks/ssr/src_hooks_1tnuifj._.js +5 -0
- package/.next/standalone/.next/server/functions-config-manifest.json +3 -0
- package/.next/standalone/.next/server/instrumentation.js +1 -1
- package/.next/standalone/.next/server/middleware-build-manifest.js +10 -10
- package/.next/standalone/.next/server/middleware.js +2 -2
- package/.next/standalone/.next/server/next-font-manifest.js +1 -1
- package/.next/standalone/.next/server/next-font-manifest.json +2 -21
- package/.next/standalone/.next/server/pages/404.html +1 -1
- package/.next/standalone/.next/server/pages/500.html +1 -1
- package/.next/standalone/.next/server/server-reference-manifest.js +1 -1
- package/.next/standalone/.next/server/server-reference-manifest.json +27 -9
- package/.next/standalone/.next/static/chunks/{07uz2g0_38qia.js → 03fmihek9n986.js} +1 -1
- package/.next/standalone/.next/static/chunks/09ueq8s1as8xs.css +2 -0
- package/.next/standalone/.next/static/chunks/0fkntmojmibda.js +6 -0
- package/.next/standalone/.next/static/chunks/{11w14gnqzprir.js → 0nbmu4_n1j2-c.js} +1 -1
- package/.next/standalone/.next/static/chunks/0qassxjx1ef04.js +1 -0
- package/.next/standalone/.next/static/chunks/0zbxssxh53n-3.js +1 -0
- package/.next/standalone/.next/static/chunks/13f1kmjea-0md.js +2 -0
- package/.next/standalone/.next/static/chunks/1btx2c49fk9xt.css +1 -0
- package/.next/standalone/.next/static/chunks/1bxdpd036n3l9.js +41 -0
- package/.next/standalone/.next/static/chunks/{0bke.~atnsbeb.js → 1e0o-6kjfnwzh.js} +1 -1
- package/.next/standalone/.next/static/chunks/20b5bz43so-1k.js +1 -0
- package/.next/standalone/.next/static/chunks/2zjavpea8glet.js +1 -0
- package/.next/standalone/.next/static/chunks/{0d3shmwh5_nmn.js → 33u59vf_8xpd-.js} +1 -1
- package/.next/standalone/.next/static/chunks/{0dvhi-prcsh3~.js → 36l507r2mbf5h.js} +1 -1
- package/.next/standalone/.next/static/chunks/3gti1qdk5epqn.js +1 -0
- package/.next/standalone/.next/static/chunks/{17mubwtqwijpu.js → 3w8d8k_dca5rp.js} +1 -1
- package/.next/standalone/.next/static/chunks/{0bv1oyxspkpkb.js → 3y3l184vijlfg.js} +1 -1
- package/.next/standalone/.next/static/chunks/41jjdgqqpl7em.js +1 -0
- package/.next/standalone/.next/static/chunks/{turbopack-0nh.aopesgj~5.js → turbopack-00qy7zfa7m--m.js} +1 -1
- package/.next/standalone/SECURITY.md +73 -0
- package/.next/standalone/app/actions/get-audit-result.ts +24 -0
- package/.next/standalone/app/api/audit/_state.ts +72 -0
- package/.next/standalone/app/api/audit/run/route.ts +89 -0
- package/.next/standalone/app/api/audit/status/route.ts +23 -0
- package/.next/standalone/app/api/auth/login-request/route.ts +91 -0
- package/.next/standalone/app/api/auth/login-verify/route.ts +98 -0
- package/.next/standalone/app/api/auth/logout/route.ts +48 -0
- package/.next/standalone/app/api/auth/reminder/route.ts +213 -0
- package/.next/standalone/app/api/auth/status/route.ts +42 -0
- package/.next/standalone/app/audit/_components/audit-dashboard.tsx +364 -0
- package/.next/standalone/app/audit/_components/auth-dialog.tsx +401 -0
- package/.next/standalone/app/audit/_components/empty-state.tsx +146 -0
- package/.next/standalone/app/audit/_components/findings-section.tsx +135 -0
- package/.next/standalone/app/audit/_components/identity-section.tsx +126 -0
- package/.next/standalone/app/audit/_components/policies-section.tsx +194 -0
- package/.next/standalone/app/audit/_components/report-footer.tsx +35 -0
- package/.next/standalone/app/audit/_components/rerun-button.tsx +81 -0
- package/.next/standalone/app/audit/_components/return-section.tsx +428 -0
- package/.next/standalone/app/audit/_components/run-progress.tsx +120 -0
- package/.next/standalone/app/audit/_components/score-section.tsx +179 -0
- package/.next/standalone/app/audit/_components/share-dock.tsx +265 -0
- package/.next/standalone/app/audit/_components/share-templates.ts +67 -0
- package/.next/standalone/app/audit/_components/show-off-cta.tsx +135 -0
- package/.next/standalone/app/audit/_components/sigil.tsx +93 -0
- package/.next/standalone/app/audit/_components/strengths-section.tsx +57 -0
- package/.next/standalone/app/audit/audit-styles.css +2066 -0
- package/.next/standalone/app/audit/loading.tsx +24 -0
- package/.next/standalone/app/audit/page.tsx +53 -0
- package/.next/standalone/app/globals.css +570 -137
- package/.next/standalone/app/layout.tsx +16 -9
- package/.next/standalone/app/policies/hooks-client.tsx +223 -129
- package/.next/standalone/app/project/[name]/page.tsx +89 -39
- package/.next/standalone/app/projects/loading.tsx +30 -8
- package/.next/standalone/app/projects/page.tsx +76 -18
- package/.next/standalone/assets/audit/Audit Report.html +22 -0
- package/.next/standalone/assets/audit/Show Off Your Agent.html +22 -0
- package/.next/standalone/assets/audit/archetypes.jsx +277 -0
- package/.next/standalone/assets/audit/assets/fonts/bitcount-prop-single.woff2 +0 -0
- package/.next/standalone/assets/audit/audit.jsx +825 -0
- package/.next/standalone/assets/audit/poster-styles.css +424 -0
- package/.next/standalone/assets/audit/poster.jsx +247 -0
- package/.next/standalone/assets/audit/screenshots/poster-optimist.png +0 -0
- package/.next/standalone/assets/audit/screenshots/poster-scrolled.png +0 -0
- package/.next/standalone/assets/audit/styles.css +1225 -0
- package/.next/standalone/assets/audit/tweaks-panel.jsx +425 -0
- package/.next/standalone/assets/logos/company/icon.svg +1 -0
- package/.next/standalone/assets/logos/company/logo.svg +1 -0
- package/.next/standalone/components/navbar.tsx +154 -65
- package/.next/standalone/components/reach-developers.tsx +37 -9
- package/.next/standalone/lib/atomic-write.ts +67 -0
- package/.next/standalone/lib/auth/api-server-client.ts +281 -0
- package/.next/standalone/lib/auth/auth-store.ts +250 -0
- package/.next/standalone/lib/client-telemetry.ts +2 -0
- package/.next/standalone/lib/fetch-with-timeout.ts +42 -0
- package/.next/standalone/lib/share-card.ts +144 -0
- package/.next/standalone/lib/telemetry.ts +12 -7
- package/.next/standalone/node_modules/@next/env/package.json +1 -1
- package/.next/standalone/node_modules/next/dist/build/swc/index.js +1 -1
- package/.next/standalone/node_modules/next/dist/client/dev/debug-channel.js +102 -2
- package/.next/standalone/node_modules/next/dist/compiled/next-server/app-page-turbo-experimental.runtime.prod.js +11 -11
- package/.next/standalone/node_modules/next/dist/compiled/next-server/app-page-turbo.runtime.prod.js +13 -13
- package/.next/standalone/node_modules/next/dist/compiled/next-server/app-route-turbo.runtime.prod.js +2 -2
- package/.next/standalone/node_modules/next/dist/compiled/next-server/pages-turbo.runtime.prod.js +1 -1
- package/.next/standalone/node_modules/next/dist/lib/patch-incorrect-lockfile.js +3 -3
- package/.next/standalone/node_modules/next/dist/server/app-render/action-handler.js +18 -8
- package/.next/standalone/node_modules/next/dist/server/config-schema.js +1 -0
- package/.next/standalone/node_modules/next/dist/server/config.js +1 -1
- package/.next/standalone/node_modules/next/dist/server/dev/hot-reloader-turbopack.js +2 -2
- package/.next/standalone/node_modules/next/dist/server/dev/hot-reloader-webpack.js +1 -1
- package/.next/standalone/node_modules/next/dist/server/lib/app-info-log.js +1 -1
- package/.next/standalone/node_modules/next/dist/server/lib/encode-cache-tag.js +45 -0
- package/.next/standalone/node_modules/next/dist/server/lib/implicit-tags.js +6 -3
- package/.next/standalone/node_modules/next/dist/server/lib/patch-fetch.js +5 -1
- package/.next/standalone/node_modules/next/dist/server/lib/start-server.js +1 -1
- package/.next/standalone/node_modules/next/dist/server/web/spec-extension/revalidate.js +4 -3
- package/.next/standalone/node_modules/next/dist/server/web/spec-extension/unstable-cache.js +6 -2
- package/.next/standalone/node_modules/next/dist/shared/lib/errors/canary-only-config-error.js +1 -1
- package/.next/standalone/node_modules/next/dist/telemetry/anonymous-meta.js +1 -1
- package/.next/standalone/node_modules/next/dist/telemetry/events/swc-load-failure.js +1 -1
- package/.next/standalone/node_modules/next/dist/telemetry/events/version.js +2 -2
- package/.next/standalone/node_modules/next/package.json +15 -15
- package/.next/standalone/node_modules/react/cjs/react.development.js +1 -1
- package/.next/standalone/node_modules/react/cjs/react.production.js +1 -1
- package/.next/standalone/node_modules/react/package.json +1 -1
- package/.next/standalone/node_modules/react-dom/cjs/react-dom-server-legacy.browser.production.js +1 -1
- package/.next/standalone/node_modules/react-dom/cjs/react-dom-server-legacy.node.production.js +1 -1
- package/.next/standalone/node_modules/react-dom/cjs/react-dom-server.browser.production.js +3 -3
- package/.next/standalone/node_modules/react-dom/cjs/react-dom-server.edge.production.js +3 -3
- package/.next/standalone/node_modules/react-dom/cjs/react-dom-server.node.production.js +3 -3
- package/.next/standalone/node_modules/react-dom/cjs/react-dom.production.js +1 -1
- package/.next/standalone/node_modules/react-dom/package.json +2 -2
- package/.next/standalone/osv-scanner.toml +17 -0
- package/.next/standalone/package.json +8 -3
- package/.next/standalone/public/audit/fonts/bitcount-prop-single.woff2 +0 -0
- package/.next/standalone/public/icon.svg +1 -0
- package/.next/standalone/public/logo.svg +1 -0
- package/.next/standalone/server.js +1 -1
- package/.next/standalone/templates/bitcount-font/README.md +42 -0
- package/.next/standalone/templates/bitcount-font/bitcount-prop-single.woff2 +0 -0
- package/.next/standalone/templates/bitcount-font/bitcount.css +49 -0
- package/.next/standalone/templates/bitcount-font/fonts.ts.example +23 -0
- package/README.md +2 -1
- package/bin/failproofai.mjs +165 -144
- package/dist/cli.mjs +607 -1858
- package/lib/atomic-write.ts +67 -0
- package/lib/auth/api-server-client.ts +281 -0
- package/lib/auth/auth-store.ts +250 -0
- package/lib/client-telemetry.ts +2 -0
- package/lib/fetch-with-timeout.ts +42 -0
- package/lib/share-card.ts +144 -0
- package/lib/telemetry.ts +12 -7
- package/package.json +8 -3
- package/scripts/install-telemetry.mjs +4 -0
- package/src/audit/archetypes.ts +924 -0
- package/src/audit/cache.ts +21 -2
- package/src/audit/dashboard-cache.ts +111 -0
- package/src/audit/features.ts +268 -0
- package/src/audit/findings.ts +298 -0
- package/src/audit/index.ts +39 -21
- package/src/audit/replay.ts +29 -3
- package/src/audit/scoring.ts +174 -0
- package/src/audit/strengths.ts +138 -0
- package/src/audit/types.ts +24 -1
- package/src/auth/cli.ts +359 -0
- package/src/hooks/builtin-policies.ts +2 -1
- package/src/hooks/hook-telemetry.ts +2 -2
- package/src/hooks/policy-registry.ts +20 -0
- package/src/posthog-key.ts +9 -0
- package/.next/standalone/.next/server/app/icon.png/route/app-paths-manifest.json +0 -3
- package/.next/standalone/.next/server/app/icon.png/route.js +0 -7
- package/.next/standalone/.next/server/app/icon.png/route.js.nft.json +0 -1
- package/.next/standalone/.next/server/app/icon.png.body +0 -0
- package/.next/standalone/.next/server/app/icon.png.meta +0 -1
- package/.next/standalone/.next/server/chunks/[externals]_next_dist_0sqmaqd._.js +0 -3
- package/.next/standalone/.next/server/chunks/[root-of-the-server]__06.arfm._.js +0 -3
- package/.next/standalone/.next/server/chunks/[root-of-the-server]__0__i0h0._.js +0 -3
- package/.next/standalone/.next/server/chunks/[root-of-the-server]__0fe7_q_._.js +0 -3
- package/.next/standalone/.next/server/chunks/[root-of-the-server]__0fw.e.h._.js +0 -3
- package/.next/standalone/.next/server/chunks/[root-of-the-server]__0pxn0e1._.js +0 -3
- package/.next/standalone/.next/server/chunks/[root-of-the-server]__0xv0jh2._.js +0 -3
- package/.next/standalone/.next/server/chunks/_next-internal_server_app_icon_png_route_actions_12.gv.r.js +0 -3
- package/.next/standalone/.next/server/chunks/node_modules_next_dist_esm_build_templates_app-route_0bdfoky.js +0 -3
- package/.next/standalone/.next/server/chunks/node_modules_posthog-node_dist_entrypoints_index_node_mjs_05pz9._._.js +0 -3
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__01as125._.js +0 -3
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__0370~qj._.js +0 -3
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__09v.ljl._.js +0 -3
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__0agrcb8._.js +0 -4
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__0b7hkr~._.js +0 -3
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__0ehh6vp._.js +0 -4
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__0g8l0tu._.js +0 -3
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__0j4l6hl._.js +0 -3
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__0k5n2kz._.js +0 -4
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__0lp08ll._.js +0 -3
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__0n0yaqw._.js +0 -4
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__0o21f.o._.js +0 -3
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__0t8juvy._.js +0 -4
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__0uylufv._.js +0 -4
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__0ymlddl._.js +0 -223
- package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__0~03grs._.js +0 -3
- package/.next/standalone/.next/server/chunks/ssr/app_0cdqd9w._.js +0 -3
- package/.next/standalone/.next/server/chunks/ssr/app_policies_hooks-client_tsx_0q-m0y-._.js +0 -8
- package/.next/standalone/.next/server/chunks/ssr/app_projects_loading_tsx_13veom4._.js +0 -3
- package/.next/standalone/.next/server/chunks/ssr/node_modules_0ttbz1~._.js +0 -3
- package/.next/standalone/.next/server/chunks/ssr/node_modules_next_dist_06u0kr8._.js +0 -3
- package/.next/standalone/.next/server/chunks/ssr/node_modules_next_dist_esm_build_templates_app-page_0a_7sdg.js +0 -4
- package/.next/standalone/.next/server/chunks/ssr/node_modules_next_dist_esm_build_templates_app-page_0ef3uwk.js +0 -4
- package/.next/standalone/.next/server/chunks/ssr/node_modules_next_dist_esm_build_templates_app-page_0pbja1x.js +0 -4
- package/.next/standalone/.next/server/chunks/ssr/node_modules_next_dist_esm_build_templates_app-page_0r6o0i2.js +0 -4
- package/.next/standalone/.next/server/chunks/ssr/node_modules_next_dist_esm_build_templates_app-page_11y81~_.js +0 -4
- package/.next/standalone/.next/server/chunks/ssr/node_modules_next_dist_esm_build_templates_app-page_12or2kf.js +0 -4
- package/.next/standalone/.next/server/chunks/ssr/node_modules_posthog-node_dist_entrypoints_index_node_mjs_0mebn66._.js +0 -3
- package/.next/standalone/.next/static/chunks/07kpqoo7kuckx.js +0 -6
- package/.next/standalone/.next/static/chunks/0azb~vy9ds_uy.js +0 -1
- package/.next/standalone/.next/static/chunks/0f5p9plm.aqlp.css +0 -2
- package/.next/standalone/.next/static/chunks/0ffvlbgzgnlw7.js +0 -2
- package/.next/standalone/.next/static/chunks/0spktq7xqab9h.js +0 -1
- package/.next/standalone/.next/static/chunks/118q3uljozd5z.js +0 -4
- package/.next/standalone/.next/static/chunks/12pt~2f.c1sha.js +0 -1
- package/.next/standalone/.next/static/media/4fa387ec64143e14-s.0.qu-9752pffj.woff2 +0 -0
- package/.next/standalone/.next/static/media/5ce348bf30bf5439-s.0ee55_hj9qcer.woff2 +0 -0
- package/.next/standalone/.next/static/media/6306c77e7c8268e4-s.0mao5jbfbduzp.woff2 +0 -0
- package/.next/standalone/.next/static/media/797e433ab948586e-s.p.09zddjkbdep5a.woff2 +0 -0
- package/.next/standalone/.next/static/media/7d817b4c03b0c5f1-s.0uzt.a6d44yda.woff2 +0 -0
- package/.next/standalone/.next/static/media/bbc41e54d2fcbd21-s.0mvwgmnhv29no.woff2 +0 -0
- package/.next/standalone/.next/static/media/icon.0a.gigb3_x5pd.png +0 -0
- package/.next/standalone/app/icon.png +0 -0
- package/src/audit/telemetry.ts +0 -113
- /package/.next/standalone/.next/server/app/{icon.png → api/audit/run}/route/build-manifest.json +0 -0
- /package/.next/standalone/.next/server/app/{icon.png → api/audit/run}/route.js.map +0 -0
- /package/.next/standalone/.next/static/chunks/{03~yq9q893hmn.js → 0cz1d0mv5g_q7.js} +0 -0
- /package/.next/standalone/.next/static/chunks/{0a40sy4tk8ioe.js → 0wwt5o04i4zwh.js} +0 -0
- /package/.next/standalone/.next/static/chunks/{0n1n67imq.udf.js → 1__i9af9g78vd.js} +0 -0
- /package/.next/standalone/.next/static/chunks/{0w6fzf.07a24u.js → 2so39wg7mjbi7.js} +0 -0
- /package/.next/standalone/.next/static/chunks/{0xbo5nl6w4lka.js → 2wbuxnsvux4di.js} +0 -0
- /package/.next/standalone/.next/static/chunks/{0_s0luks5tay-.js → 35fgpd_feci6x.js} +0 -0
- /package/.next/standalone/.next/static/chunks/{15fklyav5py5m.js → 3xpjn3cdgm-7m.js} +0 -0
- /package/.next/standalone/.next/static/chunks/{17.b3suj8zjjj.js → 4448_qq7bd963.js} +0 -0
- /package/.next/standalone/.next/static/{tGVQM5SE3NvbVu0gbAJm7 → t4nu8RTRp3Z2srol-9pVb}/_buildManifest.js +0 -0
- /package/.next/standalone/.next/static/{tGVQM5SE3NvbVu0gbAJm7 → t4nu8RTRp3Z2srol-9pVb}/_clientMiddlewareManifest.js +0 -0
- /package/.next/standalone/.next/static/{tGVQM5SE3NvbVu0gbAJm7 → t4nu8RTRp3Z2srol-9pVb}/_ssgManifest.js +0 -0
|
@@ -0,0 +1,401 @@
|
|
|
1
|
+
"use client";
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Auth dialog — modal overlay shown when an unauthenticated user clicks
|
|
5
|
+
* "[ set a reminder ]". Two-step flow:
|
|
6
|
+
*
|
|
7
|
+
* 1. Email entry → POST /api/auth/login-request
|
|
8
|
+
* 2. OTP entry → POST /api/auth/login-verify
|
|
9
|
+
*
|
|
10
|
+
* Styled to match the rest of the /audit page: pixel brackets, sharp pink
|
|
11
|
+
* accent, terminal-style frame. The dialog never sees the refresh token —
|
|
12
|
+
* the dashboard's API route writes it to ~/.failproofai/auth.json.
|
|
13
|
+
*/
|
|
14
|
+
|
|
15
|
+
import React, { useCallback, useEffect, useRef, useState } from "react";
|
|
16
|
+
import { usePostHog } from "@/contexts/PostHogContext";
|
|
17
|
+
import { fetchWithTimeout, isAbortError } from "@/lib/fetch-with-timeout";
|
|
18
|
+
|
|
19
|
+
export interface AuthedUser {
|
|
20
|
+
id: string;
|
|
21
|
+
email: string;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
interface Props {
|
|
25
|
+
open: boolean;
|
|
26
|
+
/** Copy shown above the title, e.g. "oops — you are unknown." */
|
|
27
|
+
headline?: string;
|
|
28
|
+
/** Copy under the title explaining why we need auth right now. */
|
|
29
|
+
reason?: string;
|
|
30
|
+
onClose: () => void;
|
|
31
|
+
/** Fired after successful verify. Caller decides what to do next. */
|
|
32
|
+
onAuthed: (user: AuthedUser) => void;
|
|
33
|
+
/** Telemetry tag identifying which CTA opened the dialog. Defaults to
|
|
34
|
+
* "unknown" so existing call sites continue to compile. */
|
|
35
|
+
source?: string;
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
type Step =
|
|
39
|
+
| { kind: "email"; error: string | null }
|
|
40
|
+
| { kind: "code"; email: string; error: string | null; expiresIn: number; resendIn: number }
|
|
41
|
+
| { kind: "done"; user: AuthedUser };
|
|
42
|
+
|
|
43
|
+
const EMAIL_RE = /^[^\s@]+@[^\s@]+\.[^\s@]+$/;
|
|
44
|
+
|
|
45
|
+
function describeFetchError(err: unknown): string {
|
|
46
|
+
if (isAbortError(err)) {
|
|
47
|
+
return "request timed out. check your network and try again.";
|
|
48
|
+
}
|
|
49
|
+
const message = err instanceof Error ? err.message : String(err);
|
|
50
|
+
return `network error: ${message}`;
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
export function AuthDialog({
|
|
54
|
+
open,
|
|
55
|
+
headline = "oops — you are unknown.",
|
|
56
|
+
reason = "verify yourself to continue.",
|
|
57
|
+
onClose,
|
|
58
|
+
onAuthed,
|
|
59
|
+
source = "unknown",
|
|
60
|
+
}: Props): React.ReactElement | null {
|
|
61
|
+
const { capture } = usePostHog();
|
|
62
|
+
const [step, setStep] = useState<Step>({ kind: "email", error: null });
|
|
63
|
+
const [busy, setBusy] = useState(false);
|
|
64
|
+
const succeededRef = useRef(false);
|
|
65
|
+
const emailInputRef = useRef<HTMLInputElement | null>(null);
|
|
66
|
+
const codeInputRef = useRef<HTMLInputElement | null>(null);
|
|
67
|
+
|
|
68
|
+
// Reset internal state every time the dialog opens. Also fire the
|
|
69
|
+
// funnel-opened event so we can measure dismissal vs verification rates.
|
|
70
|
+
useEffect(() => {
|
|
71
|
+
if (open) {
|
|
72
|
+
setStep({ kind: "email", error: null });
|
|
73
|
+
setBusy(false);
|
|
74
|
+
succeededRef.current = false;
|
|
75
|
+
capture("audit_auth_dialog_opened", { source });
|
|
76
|
+
}
|
|
77
|
+
}, [capture, open, source]);
|
|
78
|
+
|
|
79
|
+
// Fire dismissed when the dialog closes WITHOUT a successful verify.
|
|
80
|
+
// We piggyback on `open` flipping false instead of intercepting every
|
|
81
|
+
// close path so resend / step transitions don't double-count.
|
|
82
|
+
const wasOpenRef = useRef(false);
|
|
83
|
+
useEffect(() => {
|
|
84
|
+
if (open) {
|
|
85
|
+
wasOpenRef.current = true;
|
|
86
|
+
return;
|
|
87
|
+
}
|
|
88
|
+
if (wasOpenRef.current && !succeededRef.current) {
|
|
89
|
+
capture("audit_auth_dialog_dismissed", {
|
|
90
|
+
source,
|
|
91
|
+
step: step.kind,
|
|
92
|
+
});
|
|
93
|
+
}
|
|
94
|
+
wasOpenRef.current = false;
|
|
95
|
+
}, [capture, open, source, step.kind]);
|
|
96
|
+
|
|
97
|
+
// Autofocus the right input as the step changes.
|
|
98
|
+
useEffect(() => {
|
|
99
|
+
if (!open) return;
|
|
100
|
+
const t = setTimeout(() => {
|
|
101
|
+
if (step.kind === "email") emailInputRef.current?.focus();
|
|
102
|
+
else if (step.kind === "code") codeInputRef.current?.focus();
|
|
103
|
+
}, 50);
|
|
104
|
+
return () => clearTimeout(t);
|
|
105
|
+
}, [open, step.kind]);
|
|
106
|
+
|
|
107
|
+
// ESC to close.
|
|
108
|
+
useEffect(() => {
|
|
109
|
+
if (!open) return;
|
|
110
|
+
const onKey = (e: KeyboardEvent): void => {
|
|
111
|
+
if (e.key === "Escape" && !busy) onClose();
|
|
112
|
+
};
|
|
113
|
+
window.addEventListener("keydown", onKey);
|
|
114
|
+
return () => window.removeEventListener("keydown", onKey);
|
|
115
|
+
}, [open, busy, onClose]);
|
|
116
|
+
|
|
117
|
+
// Resend countdown ticker.
|
|
118
|
+
const resendActive = step.kind === "code" && step.resendIn > 0;
|
|
119
|
+
useEffect(() => {
|
|
120
|
+
if (!resendActive) return;
|
|
121
|
+
const id = setInterval(() => {
|
|
122
|
+
setStep((s) =>
|
|
123
|
+
s.kind === "code" ? { ...s, resendIn: Math.max(0, s.resendIn - 1) } : s,
|
|
124
|
+
);
|
|
125
|
+
}, 1000);
|
|
126
|
+
return () => clearInterval(id);
|
|
127
|
+
}, [resendActive]);
|
|
128
|
+
|
|
129
|
+
const requestCode = useCallback(
|
|
130
|
+
async (email: string, opts: { isResend?: boolean } = {}): Promise<void> => {
|
|
131
|
+
const { isResend = false } = opts;
|
|
132
|
+
// Show resend errors inline on the OTP step — the previously sent
|
|
133
|
+
// code is still usable. Only the first-send error path bounces back
|
|
134
|
+
// to the email step.
|
|
135
|
+
const setError = (msg: string) => {
|
|
136
|
+
if (isResend) {
|
|
137
|
+
setStep((s) => (s.kind === "code" ? { ...s, error: msg } : s));
|
|
138
|
+
} else {
|
|
139
|
+
setStep({ kind: "email", error: msg });
|
|
140
|
+
}
|
|
141
|
+
};
|
|
142
|
+
setBusy(true);
|
|
143
|
+
try {
|
|
144
|
+
const res = await fetchWithTimeout("/api/auth/login-request", {
|
|
145
|
+
method: "POST",
|
|
146
|
+
headers: { "content-type": "application/json" },
|
|
147
|
+
body: JSON.stringify({ email }),
|
|
148
|
+
});
|
|
149
|
+
const body = (await res.json().catch(() => ({}))) as {
|
|
150
|
+
code?: string;
|
|
151
|
+
message?: string;
|
|
152
|
+
expires_in?: number;
|
|
153
|
+
resend_available_in?: number;
|
|
154
|
+
retry_after_secs?: number;
|
|
155
|
+
};
|
|
156
|
+
if (!res.ok) {
|
|
157
|
+
let msg = body.message ?? "could not send code.";
|
|
158
|
+
if (body.code === "rate_limited" && body.retry_after_secs !== undefined) {
|
|
159
|
+
msg = `too many tries. wait ${body.retry_after_secs}s and try again.`;
|
|
160
|
+
} else if (body.code === "upstream_unreachable") {
|
|
161
|
+
msg = "api-server unreachable. check your network.";
|
|
162
|
+
}
|
|
163
|
+
setError(msg);
|
|
164
|
+
return;
|
|
165
|
+
}
|
|
166
|
+
setStep({
|
|
167
|
+
kind: "code",
|
|
168
|
+
email,
|
|
169
|
+
error: null,
|
|
170
|
+
expiresIn: body.expires_in ?? 600,
|
|
171
|
+
resendIn: body.resend_available_in ?? 30,
|
|
172
|
+
});
|
|
173
|
+
} catch (err) {
|
|
174
|
+
setError(describeFetchError(err));
|
|
175
|
+
} finally {
|
|
176
|
+
setBusy(false);
|
|
177
|
+
}
|
|
178
|
+
},
|
|
179
|
+
[],
|
|
180
|
+
);
|
|
181
|
+
|
|
182
|
+
const verifyCode = useCallback(
|
|
183
|
+
async (email: string, code: string): Promise<void> => {
|
|
184
|
+
setBusy(true);
|
|
185
|
+
try {
|
|
186
|
+
const res = await fetchWithTimeout("/api/auth/login-verify", {
|
|
187
|
+
method: "POST",
|
|
188
|
+
headers: { "content-type": "application/json" },
|
|
189
|
+
body: JSON.stringify({ email, code }),
|
|
190
|
+
});
|
|
191
|
+
const body = (await res.json().catch(() => ({}))) as {
|
|
192
|
+
authenticated?: boolean;
|
|
193
|
+
user?: AuthedUser;
|
|
194
|
+
code?: string;
|
|
195
|
+
message?: string;
|
|
196
|
+
};
|
|
197
|
+
if (!res.ok || !body.authenticated || !body.user) {
|
|
198
|
+
let msg = body.message ?? "invalid code.";
|
|
199
|
+
if (body.code === "invalid_code") msg = "wrong code, or it expired. try again.";
|
|
200
|
+
setStep((s) =>
|
|
201
|
+
s.kind === "code" ? { ...s, error: msg } : s,
|
|
202
|
+
);
|
|
203
|
+
return;
|
|
204
|
+
}
|
|
205
|
+
succeededRef.current = true;
|
|
206
|
+
capture("audit_auth_dialog_succeeded", { source });
|
|
207
|
+
setStep({ kind: "done", user: body.user });
|
|
208
|
+
onAuthed(body.user);
|
|
209
|
+
} catch (err) {
|
|
210
|
+
const msg = describeFetchError(err);
|
|
211
|
+
setStep((s) =>
|
|
212
|
+
s.kind === "code" ? { ...s, error: msg } : s,
|
|
213
|
+
);
|
|
214
|
+
} finally {
|
|
215
|
+
setBusy(false);
|
|
216
|
+
}
|
|
217
|
+
},
|
|
218
|
+
[capture, onAuthed, source],
|
|
219
|
+
);
|
|
220
|
+
|
|
221
|
+
const onEmailSubmit = useCallback(
|
|
222
|
+
async (e: React.FormEvent<HTMLFormElement>) => {
|
|
223
|
+
e.preventDefault();
|
|
224
|
+
if (busy || step.kind !== "email") return;
|
|
225
|
+
const fd = new FormData(e.currentTarget);
|
|
226
|
+
const email = String(fd.get("email") ?? "").trim().toLowerCase();
|
|
227
|
+
if (!EMAIL_RE.test(email)) {
|
|
228
|
+
setStep({ kind: "email", error: "that doesn't look like an email." });
|
|
229
|
+
return;
|
|
230
|
+
}
|
|
231
|
+
await requestCode(email);
|
|
232
|
+
},
|
|
233
|
+
[busy, step, requestCode],
|
|
234
|
+
);
|
|
235
|
+
|
|
236
|
+
const onCodeSubmit = useCallback(
|
|
237
|
+
async (e: React.FormEvent<HTMLFormElement>) => {
|
|
238
|
+
e.preventDefault();
|
|
239
|
+
if (busy || step.kind !== "code") return;
|
|
240
|
+
const fd = new FormData(e.currentTarget);
|
|
241
|
+
const code = String(fd.get("code") ?? "").trim();
|
|
242
|
+
if (code.length < 4 || code.length > 12) {
|
|
243
|
+
setStep((s) =>
|
|
244
|
+
s.kind === "code" ? { ...s, error: "code is 4–12 characters." } : s,
|
|
245
|
+
);
|
|
246
|
+
return;
|
|
247
|
+
}
|
|
248
|
+
await verifyCode(step.email, code);
|
|
249
|
+
},
|
|
250
|
+
[busy, step, verifyCode],
|
|
251
|
+
);
|
|
252
|
+
|
|
253
|
+
const onResend = useCallback(async () => {
|
|
254
|
+
if (step.kind !== "code" || step.resendIn > 0 || busy) return;
|
|
255
|
+
await requestCode(step.email, { isResend: true });
|
|
256
|
+
}, [step, busy, requestCode]);
|
|
257
|
+
|
|
258
|
+
if (!open) return null;
|
|
259
|
+
|
|
260
|
+
return (
|
|
261
|
+
<div
|
|
262
|
+
className="auth-dialog-backdrop"
|
|
263
|
+
role="dialog"
|
|
264
|
+
aria-modal="true"
|
|
265
|
+
aria-labelledby="auth-dialog-title"
|
|
266
|
+
onClick={(e) => {
|
|
267
|
+
if (!busy && e.target === e.currentTarget) onClose();
|
|
268
|
+
}}
|
|
269
|
+
>
|
|
270
|
+
<div className="auth-dialog">
|
|
271
|
+
<span className="corner tl">┌</span>
|
|
272
|
+
<span className="corner tr">┐</span>
|
|
273
|
+
<span className="corner bl">└</span>
|
|
274
|
+
<span className="corner br">┘</span>
|
|
275
|
+
|
|
276
|
+
<button
|
|
277
|
+
type="button"
|
|
278
|
+
className="auth-close"
|
|
279
|
+
onClick={onClose}
|
|
280
|
+
disabled={busy}
|
|
281
|
+
aria-label="close"
|
|
282
|
+
>
|
|
283
|
+
[ x ]
|
|
284
|
+
</button>
|
|
285
|
+
|
|
286
|
+
<div className="auth-label">━━ identity check</div>
|
|
287
|
+
<h2 id="auth-dialog-title" className="auth-headline">
|
|
288
|
+
{headline}
|
|
289
|
+
</h2>
|
|
290
|
+
|
|
291
|
+
{step.kind === "email" && (
|
|
292
|
+
<>
|
|
293
|
+
<p className="auth-sub">{reason}</p>
|
|
294
|
+
<form onSubmit={onEmailSubmit} className="auth-form">
|
|
295
|
+
<label className="auth-field-label" htmlFor="auth-dialog-email">
|
|
296
|
+
email
|
|
297
|
+
</label>
|
|
298
|
+
<input
|
|
299
|
+
ref={emailInputRef}
|
|
300
|
+
id="auth-dialog-email"
|
|
301
|
+
name="email"
|
|
302
|
+
type="email"
|
|
303
|
+
autoComplete="email"
|
|
304
|
+
inputMode="email"
|
|
305
|
+
spellCheck={false}
|
|
306
|
+
placeholder="you@yourdomain.com"
|
|
307
|
+
disabled={busy}
|
|
308
|
+
className="auth-input"
|
|
309
|
+
required
|
|
310
|
+
/>
|
|
311
|
+
{step.error && <div className="auth-error">{step.error}</div>}
|
|
312
|
+
<div className="auth-actions">
|
|
313
|
+
<button type="submit" className="auth-btn primary" disabled={busy}>
|
|
314
|
+
{busy ? "[ sending… ]" : "[ send code ]"}
|
|
315
|
+
</button>
|
|
316
|
+
<button
|
|
317
|
+
type="button"
|
|
318
|
+
className="auth-btn"
|
|
319
|
+
onClick={onClose}
|
|
320
|
+
disabled={busy}
|
|
321
|
+
>
|
|
322
|
+
[ cancel ]
|
|
323
|
+
</button>
|
|
324
|
+
</div>
|
|
325
|
+
</form>
|
|
326
|
+
</>
|
|
327
|
+
)}
|
|
328
|
+
|
|
329
|
+
{step.kind === "code" && (
|
|
330
|
+
<>
|
|
331
|
+
<p className="auth-sub">
|
|
332
|
+
we sent a code to <span className="auth-email">{step.email}</span>.
|
|
333
|
+
<br />
|
|
334
|
+
check your inbox — it expires in {Math.ceil(step.expiresIn / 60)} min.
|
|
335
|
+
</p>
|
|
336
|
+
<form onSubmit={onCodeSubmit} className="auth-form">
|
|
337
|
+
<label className="auth-field-label" htmlFor="auth-dialog-code">
|
|
338
|
+
one-time code
|
|
339
|
+
</label>
|
|
340
|
+
<input
|
|
341
|
+
ref={codeInputRef}
|
|
342
|
+
id="auth-dialog-code"
|
|
343
|
+
name="code"
|
|
344
|
+
type="text"
|
|
345
|
+
inputMode="numeric"
|
|
346
|
+
autoComplete="one-time-code"
|
|
347
|
+
spellCheck={false}
|
|
348
|
+
placeholder="123456"
|
|
349
|
+
disabled={busy}
|
|
350
|
+
className="auth-input auth-input-code"
|
|
351
|
+
maxLength={12}
|
|
352
|
+
required
|
|
353
|
+
/>
|
|
354
|
+
{step.error && <div className="auth-error">{step.error}</div>}
|
|
355
|
+
<div className="auth-actions">
|
|
356
|
+
<button type="submit" className="auth-btn primary" disabled={busy}>
|
|
357
|
+
{busy ? "[ verifying… ]" : "[ verify ]"}
|
|
358
|
+
</button>
|
|
359
|
+
<button
|
|
360
|
+
type="button"
|
|
361
|
+
className="auth-btn"
|
|
362
|
+
onClick={onResend}
|
|
363
|
+
disabled={busy || step.resendIn > 0}
|
|
364
|
+
>
|
|
365
|
+
{step.resendIn > 0
|
|
366
|
+
? `[ resend in ${step.resendIn}s ]`
|
|
367
|
+
: "[ resend code ]"}
|
|
368
|
+
</button>
|
|
369
|
+
</div>
|
|
370
|
+
<button
|
|
371
|
+
type="button"
|
|
372
|
+
className="auth-back"
|
|
373
|
+
onClick={() => setStep({ kind: "email", error: null })}
|
|
374
|
+
disabled={busy}
|
|
375
|
+
>
|
|
376
|
+
← use a different email
|
|
377
|
+
</button>
|
|
378
|
+
</form>
|
|
379
|
+
</>
|
|
380
|
+
)}
|
|
381
|
+
|
|
382
|
+
{step.kind === "done" && (
|
|
383
|
+
<>
|
|
384
|
+
<p className="auth-sub">
|
|
385
|
+
<span className="auth-ok">✓</span> you are{" "}
|
|
386
|
+
<span className="auth-email">{step.user.email}</span>.
|
|
387
|
+
</p>
|
|
388
|
+
<p className="auth-sub" style={{ marginTop: 8 }}>
|
|
389
|
+
session saved locally.
|
|
390
|
+
</p>
|
|
391
|
+
<div className="auth-actions">
|
|
392
|
+
<button type="button" className="auth-btn primary" onClick={onClose}>
|
|
393
|
+
[ continue ]
|
|
394
|
+
</button>
|
|
395
|
+
</div>
|
|
396
|
+
</>
|
|
397
|
+
)}
|
|
398
|
+
</div>
|
|
399
|
+
</div>
|
|
400
|
+
);
|
|
401
|
+
}
|
|
@@ -0,0 +1,146 @@
|
|
|
1
|
+
"use client";
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Two-mode empty state for /audit, styled to the audit pixel-craft system:
|
|
5
|
+
*
|
|
6
|
+
* - "no-cache" — first time the user visits /audit. CTA to run.
|
|
7
|
+
* - "zero-sessions" — ran a scan but no transcripts were found. Likely the
|
|
8
|
+
* user hasn't installed hooks for any CLI yet.
|
|
9
|
+
*
|
|
10
|
+
* Both modes use the shared `.panel` chrome with pink corner brackets, a
|
|
11
|
+
* green section eyebrow, a Bitcount Prop Single display headline, and a
|
|
12
|
+
* sharp `.btn-press` action button. Sized so it occupies the same vertical
|
|
13
|
+
* space as the loaded dashboard does on its hero — no more cramped popover.
|
|
14
|
+
*/
|
|
15
|
+
import React from "react";
|
|
16
|
+
import { triggerRun } from "./rerun-button";
|
|
17
|
+
import { usePostHog } from "@/contexts/PostHogContext";
|
|
18
|
+
|
|
19
|
+
interface Props {
|
|
20
|
+
mode: "no-cache" | "zero-sessions";
|
|
21
|
+
running: boolean;
|
|
22
|
+
onStarted: () => void;
|
|
23
|
+
onCompleted: () => Promise<void> | void;
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
export function EmptyState({ mode, running, onStarted, onCompleted }: Props) {
|
|
27
|
+
const { capture } = usePostHog();
|
|
28
|
+
const handleRun = async () => {
|
|
29
|
+
capture("audit_first_run_clicked", {
|
|
30
|
+
source: "empty_state",
|
|
31
|
+
mode,
|
|
32
|
+
});
|
|
33
|
+
onStarted();
|
|
34
|
+
try {
|
|
35
|
+
await triggerRun({ cli: [], since: "30d" });
|
|
36
|
+
} finally {
|
|
37
|
+
await onCompleted();
|
|
38
|
+
}
|
|
39
|
+
};
|
|
40
|
+
|
|
41
|
+
if (mode === "no-cache") {
|
|
42
|
+
return (
|
|
43
|
+
<section className="section empty-section" data-screen-label="00 Empty">
|
|
44
|
+
<div className="section-mast">
|
|
45
|
+
<div className="section-label">
|
|
46
|
+
<span className="glyph">━━</span> audit{" "}
|
|
47
|
+
<span style={{ color: "var(--dim)" }}>·</span> first run
|
|
48
|
+
</div>
|
|
49
|
+
<div className="section-meta">
|
|
50
|
+
<span style={{ color: "var(--dim)" }}>○</span> no cache yet
|
|
51
|
+
</div>
|
|
52
|
+
</div>
|
|
53
|
+
<h2 className="section-h">scan and see.</h2>
|
|
54
|
+
|
|
55
|
+
<div className="panel empty-panel">
|
|
56
|
+
<div className="empty-glyph" aria-hidden="true">
|
|
57
|
+
<div className="empty-glyph-grid">
|
|
58
|
+
{Array.from({ length: 36 }).map((_, i) => {
|
|
59
|
+
// Form the word "GO" inside an 8x6 grid using a sparse hard-coded
|
|
60
|
+
// mask. Pure decoration — the grid layout sells the pixel-craft.
|
|
61
|
+
const on = ["1", "5", "8", "10", "13", "17", "20", "23", "25", "30", "32"].includes(String(i));
|
|
62
|
+
return <span key={i} className={"px" + (on ? " on" : "")} />;
|
|
63
|
+
})}
|
|
64
|
+
</div>
|
|
65
|
+
<div className="empty-glyph-label">▮▮ no audit data yet</div>
|
|
66
|
+
</div>
|
|
67
|
+
|
|
68
|
+
<h3 className="empty-headline">run your first audit.</h3>
|
|
69
|
+
<p className="empty-sub">
|
|
70
|
+
we'll walk every transcript across your installed CLIs — Claude Code,
|
|
71
|
+
Codex, Copilot, Cursor, OpenCode, Pi, Gemini — and count every wasteful
|
|
72
|
+
or risky action. you'll get a tier, a score, and a punch-list.
|
|
73
|
+
</p>
|
|
74
|
+
|
|
75
|
+
<div className="empty-actions">
|
|
76
|
+
<button
|
|
77
|
+
type="button"
|
|
78
|
+
className="btn btn-primary btn-press empty-cta"
|
|
79
|
+
onClick={handleRun}
|
|
80
|
+
disabled={running}
|
|
81
|
+
>
|
|
82
|
+
{running ? "[ scanning… ]" : "[ run audit ]"}
|
|
83
|
+
</button>
|
|
84
|
+
<span className="empty-meta">
|
|
85
|
+
scans the last 30 days · all installed CLIs · 10–30s
|
|
86
|
+
</span>
|
|
87
|
+
</div>
|
|
88
|
+
</div>
|
|
89
|
+
</section>
|
|
90
|
+
);
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
// mode === "zero-sessions"
|
|
94
|
+
return (
|
|
95
|
+
<section className="section empty-section" data-screen-label="00 Empty">
|
|
96
|
+
<div className="section-mast">
|
|
97
|
+
<div className="section-label">
|
|
98
|
+
<span className="glyph">━━</span> audit{" "}
|
|
99
|
+
<span style={{ color: "var(--dim)" }}>·</span> zero transcripts
|
|
100
|
+
</div>
|
|
101
|
+
<div className="section-meta">
|
|
102
|
+
<span style={{ color: "var(--amber)" }}>●</span> hooks not installed
|
|
103
|
+
</div>
|
|
104
|
+
</div>
|
|
105
|
+
<h2 className="section-h">nothing to read.</h2>
|
|
106
|
+
|
|
107
|
+
<div className="panel empty-panel">
|
|
108
|
+
<div className="empty-glyph" aria-hidden="true">
|
|
109
|
+
<div className="empty-glyph-grid">
|
|
110
|
+
{Array.from({ length: 36 }).map((_, i) => {
|
|
111
|
+
const on = ["2", "4", "9", "11", "16", "18", "23", "25", "27", "30", "33"].includes(String(i));
|
|
112
|
+
return <span key={i} className={"px" + (on ? " on" : "")} />;
|
|
113
|
+
})}
|
|
114
|
+
</div>
|
|
115
|
+
<div className="empty-glyph-label">▮▮ no sessions found</div>
|
|
116
|
+
</div>
|
|
117
|
+
|
|
118
|
+
<h3 className="empty-headline">install hooks first.</h3>
|
|
119
|
+
<p className="empty-sub">
|
|
120
|
+
failproofai couldn't find any transcripts to scan on this machine.
|
|
121
|
+
install the hooks for at least one CLI and come back.
|
|
122
|
+
</p>
|
|
123
|
+
|
|
124
|
+
<div className="empty-actions">
|
|
125
|
+
<a
|
|
126
|
+
href="https://docs.befailproof.ai/getting-started"
|
|
127
|
+
target="_blank"
|
|
128
|
+
rel="noopener noreferrer"
|
|
129
|
+
className="btn btn-primary btn-press empty-cta"
|
|
130
|
+
onClick={() => {
|
|
131
|
+
capture("audit_install_guide_clicked", {
|
|
132
|
+
source: "empty_state",
|
|
133
|
+
mode,
|
|
134
|
+
});
|
|
135
|
+
}}
|
|
136
|
+
>
|
|
137
|
+
[ install guide → ]
|
|
138
|
+
</a>
|
|
139
|
+
<span className="empty-meta">
|
|
140
|
+
takes about 30 seconds · one command per CLI
|
|
141
|
+
</span>
|
|
142
|
+
</div>
|
|
143
|
+
</div>
|
|
144
|
+
</section>
|
|
145
|
+
);
|
|
146
|
+
}
|
|
@@ -0,0 +1,135 @@
|
|
|
1
|
+
"use client";
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Section 04 — FINDINGS. "your agent has some quirks."
|
|
5
|
+
*
|
|
6
|
+
* Per-finding cards with four blocks: what happened / what this costs /
|
|
7
|
+
* evidence sample / the fix. Data sourced from `src/audit/findings.ts`.
|
|
8
|
+
*/
|
|
9
|
+
import React, { useState } from "react";
|
|
10
|
+
import type { FindingCard } from "@/src/audit/findings";
|
|
11
|
+
import { usePostHog } from "@/contexts/PostHogContext";
|
|
12
|
+
|
|
13
|
+
interface Props {
|
|
14
|
+
findings: FindingCard[];
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
export function FindingsSection({ findings }: Props) {
|
|
18
|
+
if (findings.length === 0) return null;
|
|
19
|
+
|
|
20
|
+
return (
|
|
21
|
+
<section className="section" data-screen-label="04 Findings">
|
|
22
|
+
<div className="section-mast">
|
|
23
|
+
<div className="section-label">
|
|
24
|
+
<span className="glyph">━━</span> findings{" "}
|
|
25
|
+
<span style={{ color: "var(--dim)" }}>·</span> ranked by impact
|
|
26
|
+
</div>
|
|
27
|
+
<div className="section-meta">
|
|
28
|
+
<span className="p">●</span> {findings.length} detector{findings.length === 1 ? "" : "s"} triggered
|
|
29
|
+
</div>
|
|
30
|
+
</div>
|
|
31
|
+
<h2 className="section-h">your agent has some quirks.</h2>
|
|
32
|
+
|
|
33
|
+
<div className="findings-list">
|
|
34
|
+
{findings.map((f) => <Finding key={f.sourceSlug} f={f} />)}
|
|
35
|
+
</div>
|
|
36
|
+
</section>
|
|
37
|
+
);
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
function Finding({ f }: { f: FindingCard }) {
|
|
41
|
+
const { capture } = usePostHog();
|
|
42
|
+
const [copied, setCopied] = useState(false);
|
|
43
|
+
|
|
44
|
+
const handleCopy = async (e: React.MouseEvent) => {
|
|
45
|
+
e.stopPropagation();
|
|
46
|
+
try {
|
|
47
|
+
await navigator.clipboard.writeText(f.fix.install);
|
|
48
|
+
setCopied(true);
|
|
49
|
+
capture("audit_copy_clicked", {
|
|
50
|
+
source: "findings_section",
|
|
51
|
+
item_type: "single_policy_install_command",
|
|
52
|
+
policy_name: f.fix.slug,
|
|
53
|
+
finding_slug: f.sourceSlug,
|
|
54
|
+
});
|
|
55
|
+
setTimeout(() => setCopied(false), 1500);
|
|
56
|
+
} catch { /* ignore */ }
|
|
57
|
+
};
|
|
58
|
+
|
|
59
|
+
return (
|
|
60
|
+
<article className="finding">
|
|
61
|
+
<header className="finding-head">
|
|
62
|
+
<div className="finding-num">№{f.num}</div>
|
|
63
|
+
<div className="finding-title">{f.title}</div>
|
|
64
|
+
<div className="finding-count">
|
|
65
|
+
{f.count}×
|
|
66
|
+
<span className="label">occurrences</span>
|
|
67
|
+
</div>
|
|
68
|
+
</header>
|
|
69
|
+
<div className="finding-meta">
|
|
70
|
+
<span>
|
|
71
|
+
<span style={{ color: "var(--dim)" }}>policy</span>{" "}
|
|
72
|
+
<span className="policy">{f.policy}</span>
|
|
73
|
+
</span>
|
|
74
|
+
<span className="sep">·</span>
|
|
75
|
+
<span>{f.projects} {f.projects === 1 ? "project" : "projects"}</span>
|
|
76
|
+
<span className="sep">·</span>
|
|
77
|
+
<span>last seen {f.lastSeen}</span>
|
|
78
|
+
{f.alreadyEnabled && (
|
|
79
|
+
<>
|
|
80
|
+
<span className="sep">·</span>
|
|
81
|
+
<span style={{ color: "var(--accent-green)" }}>enforced</span>
|
|
82
|
+
</>
|
|
83
|
+
)}
|
|
84
|
+
</div>
|
|
85
|
+
<div className="finding-body">
|
|
86
|
+
<div className="finding-block">
|
|
87
|
+
<div className="fb-label">what happened</div>
|
|
88
|
+
<div className="fb-body">{f.body}</div>
|
|
89
|
+
</div>
|
|
90
|
+
<div className="finding-block">
|
|
91
|
+
<div className="fb-label cost">what this costs</div>
|
|
92
|
+
<div className="fb-body">{f.cost}</div>
|
|
93
|
+
</div>
|
|
94
|
+
<div className="finding-block">
|
|
95
|
+
<div className="fb-label">evidence · sample</div>
|
|
96
|
+
<div className="fb-evidence">
|
|
97
|
+
{f.evidence.map((e, i) => {
|
|
98
|
+
if (e.kind === "comment") {
|
|
99
|
+
return <div key={i} className="comment">{e.text}</div>;
|
|
100
|
+
}
|
|
101
|
+
if (e.kind === "err") {
|
|
102
|
+
return <div key={i} className="err">{e.text}</div>;
|
|
103
|
+
}
|
|
104
|
+
return (
|
|
105
|
+
<div key={i}>
|
|
106
|
+
<span className="arrow">→ </span>
|
|
107
|
+
<span>{e.text}</span>
|
|
108
|
+
</div>
|
|
109
|
+
);
|
|
110
|
+
})}
|
|
111
|
+
</div>
|
|
112
|
+
</div>
|
|
113
|
+
<div className="finding-block">
|
|
114
|
+
<div className="fb-label fix">the fix</div>
|
|
115
|
+
<div className="fb-fix">
|
|
116
|
+
<span className="slug">{f.fix.slug}</span>
|
|
117
|
+
<div style={{ color: "var(--ink-2)" }}>{f.fix.desc}</div>
|
|
118
|
+
{f.fix.alsoCoveredBy && (
|
|
119
|
+
<div style={{ color: "var(--dim)", fontSize: 11, marginTop: 4 }}>
|
|
120
|
+
also covered by{" "}
|
|
121
|
+
<span style={{ color: "var(--accent-green)" }}>{f.fix.alsoCoveredBy}</span>
|
|
122
|
+
</div>
|
|
123
|
+
)}
|
|
124
|
+
<code className="cmd" onClick={handleCopy} style={{ cursor: "pointer" }}>
|
|
125
|
+
<span className="prompt">$</span>{f.fix.install}{" "}
|
|
126
|
+
<span style={{ color: "var(--dim)", marginLeft: 8, fontSize: 10, letterSpacing: "0.15em", textTransform: "uppercase" }}>
|
|
127
|
+
{copied ? "copied" : "click to copy"}
|
|
128
|
+
</span>
|
|
129
|
+
</code>
|
|
130
|
+
</div>
|
|
131
|
+
</div>
|
|
132
|
+
</div>
|
|
133
|
+
</article>
|
|
134
|
+
);
|
|
135
|
+
}
|