botrun-horse 2.30.0 → 2.34.0
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/bin/bh.mjs +37 -170
- package/bin/commands/help.mjs +30 -30
- package/{lib/flows → modules/_core}/draft-writing.mjs +1 -1
- package/{lib/flows → modules/_core}/gemini-ask.mjs +1 -1
- package/{lib/flows → modules/_core}/legal-ask.mjs +4 -4
- package/{lib/flows → modules/_core}/openai-agent.mjs +2 -2
- package/{lib/flows → modules/_core}/openrouter-ask.mjs +1 -1
- package/{bin/commands/dag-cmd.mjs → modules/dag/command.mjs} +2 -2
- package/modules/dag/index.mjs +30 -0
- package/{bin/commands/db-cmd.mjs → modules/db/command.mjs} +3 -3
- package/modules/db/index.mjs +26 -0
- package/modules/discover.mjs +43 -0
- package/{bin/commands/doc.mjs → modules/doc/command.mjs} +5 -5
- package/modules/doc/index.mjs +55 -0
- package/{lib/doc/index.mjs → modules/doc/ingest.mjs} +1 -1
- package/{lib → modules}/doc/office2text.mjs +1 -1
- package/{lib → modules}/doc/pdf2text.mjs +1 -1
- package/{lib → modules}/doc/split.mjs +1 -1
- package/{bin/commands/gemini.mjs → modules/gemini/command.mjs} +3 -3
- package/modules/gemini/index.mjs +41 -0
- package/modules/imagen/command.mjs +117 -0
- package/modules/imagen/imagen-client.mjs +239 -0
- package/modules/imagen/index.mjs +60 -0
- package/modules/imagen/server.mjs +133 -0
- package/{bin/commands/legal.mjs → modules/legal/command.mjs} +5 -5
- package/modules/legal/index.mjs +56 -0
- package/{bin/commands/nchc.mjs → modules/nchc/command.mjs} +4 -4
- package/modules/nchc/index.mjs +48 -0
- package/modules/ocr/index.mjs +17 -0
- package/{bin/commands/openrouter.mjs → modules/openrouter/command.mjs} +5 -5
- package/modules/openrouter/index.mjs +49 -0
- package/modules/portal/index.mjs +17 -0
- package/{bin/commands/search.mjs → modules/search/command.mjs} +3 -3
- package/modules/search/index.mjs +23 -0
- package/{lib/search/index.mjs → modules/search/search-lib.mjs} +1 -1
- package/{bin/commands/skill.mjs → modules/skill/command.mjs} +46 -9
- package/modules/skill/index.mjs +35 -0
- package/{lib/prompt → modules/skill}/prompt-search.mjs +1 -1
- package/{lib/prompt → modules/skill}/prompt-store.mjs +28 -11
- package/modules/skill/prompts/zero-framework/address-matching.md +22 -0
- package/modules/skill/prompts/zero-framework/data-integrity.md +15 -0
- package/modules/skill/prompts/zero-framework/data-reasonability.md +17 -0
- package/modules/skill/prompts/zero-framework/gemini-image-gen.md +20 -0
- package/modules/skill/prompts/zero-framework/gemini-search-cli.md +10 -0
- package/modules/skill/prompts/zero-framework/html-report.md +15 -0
- package/modules/skill/prompts/zero-framework/smart-cache.md +10 -0
- package/modules/skill/prompts/zero-framework/trigram-search.md +12 -0
- package/modules/skill/prompts/zero-framework/tts-format.md +6 -0
- package/modules/skill/prompts/zero-framework/watermelon-sqlite.md +8 -0
- package/{bin/commands/writing.mjs → modules/writing/command.mjs} +3 -3
- package/{lib → modules}/writing/generate.mjs +2 -2
- package/modules/writing/index.mjs +38 -0
- package/package.json +3 -1
- package/parallel-dag-todo-list/imagen-module.md +51 -0
- package/bin/commands/schema.mjs +0 -254
- package/botrun-c/.claude/skills/DOCX/351/226/261/350/256/200/346/212/200/350/203/275/SKILL.md +0 -103
- package/botrun-c/.claude/skills/DOCX/351/226/261/350/256/200/346/212/200/350/203/275/scripts/docx-to-markdown.mjs +0 -206
- package/botrun-c/.claude/skills/GitHub/346/212/200/350/203/275/SKILL.md +0 -1093
- package/botrun-c/.claude/skills/GitHub/346/212/200/350/203/275/scripts/github-branch.mjs +0 -73
- package/botrun-c/.claude/skills/GitHub/346/212/200/350/203/275/scripts/github-branches.mjs +0 -77
- package/botrun-c/.claude/skills/GitHub/346/212/200/350/203/275/scripts/github-checkout.mjs +0 -72
- package/botrun-c/.claude/skills/GitHub/346/212/200/350/203/275/scripts/github-clone.mjs +0 -72
- package/botrun-c/.claude/skills/GitHub/346/212/200/350/203/275/scripts/github-commit.mjs +0 -75
- package/botrun-c/.claude/skills/GitHub/346/212/200/350/203/275/scripts/github-delete-branch.mjs +0 -75
- package/botrun-c/.claude/skills/GitHub/346/212/200/350/203/275/scripts/github-delete-file.mjs +0 -72
- package/botrun-c/.claude/skills/GitHub/346/212/200/350/203/275/scripts/github-diff.mjs +0 -80
- package/botrun-c/.claude/skills/GitHub/346/212/200/350/203/275/scripts/github-list-tree.mjs +0 -336
- package/botrun-c/.claude/skills/GitHub/346/212/200/350/203/275/scripts/github-list.mjs +0 -199
- package/botrun-c/.claude/skills/GitHub/346/212/200/350/203/275/scripts/github-merge.mjs +0 -86
- package/botrun-c/.claude/skills/GitHub/346/212/200/350/203/275/scripts/github-move.mjs +0 -75
- package/botrun-c/.claude/skills/GitHub/346/212/200/350/203/275/scripts/github-pr-create.mjs +0 -81
- package/botrun-c/.claude/skills/GitHub/346/212/200/350/203/275/scripts/github-pr-list.mjs +0 -74
- package/botrun-c/.claude/skills/GitHub/346/212/200/350/203/275/scripts/github-pr-merge.mjs +0 -83
- package/botrun-c/.claude/skills/GitHub/346/212/200/350/203/275/scripts/github-pr-view.mjs +0 -71
- package/botrun-c/.claude/skills/GitHub/346/212/200/350/203/275/scripts/github-push.mjs +0 -71
- package/botrun-c/.claude/skills/GitHub/346/212/200/350/203/275/scripts/github-read.mjs +0 -277
- package/botrun-c/.claude/skills/GitHub/346/212/200/350/203/275/scripts/github-search.mjs +0 -370
- package/botrun-c/.claude/skills/GitHub/346/212/200/350/203/275/scripts/github-stash.mjs +0 -116
- package/botrun-c/.claude/skills/GitHub/346/212/200/350/203/275/scripts/github-sync.mjs +0 -71
- package/botrun-c/.claude/skills/GitHub/346/212/200/350/203/275/scripts/github-write.mjs +0 -74
- package/botrun-c/.claude/skills/GitHub/346/212/200/350/203/275/scripts/lib/path-validator.mjs +0 -167
- package/botrun-c/.claude/skills/GoogleDrive/346/212/200/350/203/275/SKILL.md +0 -605
- package/botrun-c/.claude/skills/GoogleDrive/346/212/200/350/203/275/scripts/gdrive-create.mjs +0 -127
- package/botrun-c/.claude/skills/GoogleDrive/346/212/200/350/203/275/scripts/gdrive-delete.mjs +0 -77
- package/botrun-c/.claude/skills/GoogleDrive/346/212/200/350/203/275/scripts/gdrive-diff.mjs +0 -87
- package/botrun-c/.claude/skills/GoogleDrive/346/212/200/350/203/275/scripts/gdrive-history.mjs +0 -99
- package/botrun-c/.claude/skills/GoogleDrive/346/212/200/350/203/275/scripts/gdrive-list.mjs +0 -174
- package/botrun-c/.claude/skills/GoogleDrive/346/212/200/350/203/275/scripts/gdrive-read.mjs +0 -214
- package/botrun-c/.claude/skills/GoogleDrive/346/212/200/350/203/275/scripts/gdrive-restore.mjs +0 -75
- package/botrun-c/.claude/skills/GoogleDrive/346/212/200/350/203/275/scripts/gdrive-search.mjs +0 -270
- package/botrun-c/.claude/skills/GoogleDrive/346/212/200/350/203/275/scripts/gdrive-sync-back.mjs +0 -155
- package/botrun-c/.claude/skills/GoogleDrive/346/212/200/350/203/275/scripts/gdrive-update.mjs +0 -100
- package/botrun-c/.claude/skills/HTML/347/224/237/346/210/220/346/212/200/350/203/275/SKILL.md +0 -414
- package/botrun-c/.claude/skills/HTML/347/224/237/346/210/220/346/212/200/350/203/275/scripts/create-project.mjs +0 -91
- package/botrun-c/.claude/skills/HTML/347/224/237/346/210/220/346/212/200/350/203/275/scripts/finalize-project.mjs +0 -162
- package/botrun-c/.claude/skills/PDF/346/212/200/350/203/275/SKILL.md +0 -206
- package/botrun-c/.claude/skills/PDF/346/212/200/350/203/275/scripts/package.json +0 -19
- package/botrun-c/.claude/skills/PDF/346/212/200/350/203/275/scripts/pdf-analyze.mjs +0 -309
- package/botrun-c/.claude/skills/PDF/346/212/200/350/203/275/scripts/pdf-compress.mjs +0 -315
- package/botrun-c/.claude/skills/PDF/346/212/200/350/203/275/scripts/pdf-split.mjs +0 -275
- package/botrun-c/.claude/skills/PDF/346/212/200/350/203/275/scripts/pdf-to-text.mjs +0 -336
- package/botrun-c/.claude/skills/PDF/346/212/200/350/203/275/scripts/test-pdf-scripts.mjs +0 -491
- package/botrun-c/.claude/skills/docx-to-markdown/SKILL.md +0 -76
- package/botrun-c/.claude/skills/docx-to-markdown/scripts/convert_docx.py +0 -183
- package/botrun-c/.claude/skills/gcp-coord-ml-ocr/SKILL.md +0 -133
- package/botrun-c/.claude/skills/gcp-coord-ml-ocr/scripts/ocr_processor.py +0 -381
- package/botrun-c/.claude/skills/gemini-transcribe/SKILL.md +0 -115
- package/botrun-c/.claude/skills/gemini-transcribe/scripts/transcribe.py +0 -499
- package/botrun-c/.claude/skills/nchc-transcribe/SKILL.md +0 -131
- package/botrun-c/.claude/skills/nchc-transcribe/scripts/transcribe.py +0 -522
- package/botrun-c/.claude/skills/p320-moj-review/SKILL.md +0 -200
- package/botrun-c/.claude/skills/p320-moj-review/references/guideline.md +0 -118
- package/botrun-c/.claude/skills/pdf-multimodal-processor/SKILL.md +0 -186
- package/botrun-c/.claude/skills/pdf-multimodal-processor/scripts/process_pdf.py +0 -515
- package/botrun-c/.claude/skills/ripgrep/346/220/234/345/260/213/346/212/200/350/203/275/SKILL.md +0 -81
- package/botrun-c/.claude/skills/ripgrep/346/220/234/345/260/213/346/212/200/350/203/275/scripts/package.json +0 -13
- package/botrun-c/.claude/skills/ripgrep/346/220/234/345/260/213/346/212/200/350/203/275/scripts/secure-search.mjs +0 -232
- package/botrun-c/.claude/skills/top100/351/240/230/350/242/226/351/240/220/346/270/254/346/212/200/350/203/275/SKILL.md +0 -518
- package/botrun-c/.claude/skills//345/216/273/350/255/230/345/210/245/346/212/200/350/203/275/SKILL.md +0 -125
- package/botrun-c/.claude/skills//345/233/236/346/206/266/346/212/200/350/203/275/SKILL.md +0 -123
- package/botrun-c/.claude/skills//345/233/236/346/206/266/346/212/200/350/203/275/scripts/recall.mjs +0 -339
- package/botrun-c/.claude/skills//345/234/226/347/211/207/347/224/237/346/210/220/346/212/200/350/203/275/SKILL.md +0 -156
- package/botrun-c/.claude/skills//345/234/226/347/211/207/347/224/237/346/210/220/346/212/200/350/203/275/scripts/__tests__/utils.test.mjs +0 -139
- package/botrun-c/.claude/skills//345/234/226/347/211/207/347/224/237/346/210/220/346/212/200/350/203/275/scripts/constants.mjs +0 -40
- package/botrun-c/.claude/skills//345/234/226/347/211/207/347/224/237/346/210/220/346/212/200/350/203/275/scripts/gcs-uploader.mjs +0 -195
- package/botrun-c/.claude/skills//345/234/226/347/211/207/347/224/237/346/210/220/346/212/200/350/203/275/scripts/gemini-image-client.mjs +0 -307
- package/botrun-c/.claude/skills//345/234/226/347/211/207/347/224/237/346/210/220/346/212/200/350/203/275/scripts/generate-image.mjs +0 -103
- package/botrun-c/.claude/skills//345/234/226/347/211/207/347/224/237/346/210/220/346/212/200/350/203/275/scripts/image-session-manager.mjs +0 -219
- package/botrun-c/.claude/skills//345/234/226/347/211/207/347/224/237/346/210/220/346/212/200/350/203/275/scripts/output-formatter.mjs +0 -209
- package/botrun-c/.claude/skills//345/234/226/347/211/207/347/224/237/346/210/220/346/212/200/350/203/275/scripts/package.json +0 -20
- package/botrun-c/.claude/skills//345/234/226/347/211/207/347/224/237/346/210/220/346/212/200/350/203/275/scripts/utils.mjs +0 -115
- package/botrun-c/.claude/skills//345/244/232/346/250/241/346/205/213/351/226/261/350/256/200/346/212/200/350/203/275/SKILL.md +0 -86
- package/botrun-c/.claude/skills//345/244/232/346/250/241/346/205/213/351/226/261/350/256/200/346/212/200/350/203/275/scripts/multimodal-read.mjs +0 -304
- package/botrun-c/.claude/skills//345/244/232/346/250/241/346/205/213/351/226/261/350/256/200/346/212/200/350/203/275/scripts/package.json +0 -17
- package/botrun-c/.claude/skills//345/255/265/345/214/226/346/212/200/350/203/275/SKILL.md +0 -131
- package/botrun-c/.claude/skills//345/255/265/345/214/226/346/212/200/350/203/275/scripts/skill-manager.mjs +0 -542
- package/botrun-c/.claude/skills//346/234/203/350/255/260/350/250/230/351/214/204/346/212/200/350/203/275/SKILL.md +0 -127
- package/botrun-c/.claude/skills//347/233/256/351/214/204/345/210/227/350/241/250/346/212/200/350/203/275/SKILL.md +0 -53
- package/botrun-c/.claude/skills//347/233/256/351/214/204/345/210/227/350/241/250/346/212/200/350/203/275/scripts/package.json +0 -13
- package/botrun-c/.claude/skills//347/233/256/351/214/204/345/210/227/350/241/250/346/212/200/350/203/275/scripts/tree-view.mjs +0 -149
- package/botrun-c/.claude/skills//347/264/224/346/226/207/345/255/227/345/244/232/346/252/224/346/241/210/350/256/200/345/217/226/346/212/200/350/203/275/SKILL.md +0 -82
- package/botrun-c/.claude/skills//347/264/224/346/226/207/345/255/227/345/244/232/346/252/224/346/241/210/350/256/200/345/217/226/346/212/200/350/203/275/scripts/package.json +0 -16
- package/botrun-c/.claude/skills//347/264/224/346/226/207/345/255/227/345/244/232/346/252/224/346/241/210/350/256/200/345/217/226/346/212/200/350/203/275/scripts/secure-read.mjs +0 -260
- package/botrun-c/.claude/skills//347/266/262/350/267/257/346/220/234/345/260/213/346/212/200/350/203/275/SKILL.md +0 -156
- package/botrun-c/.claude/skills//347/266/262/350/267/257/346/220/234/345/260/213/346/212/200/350/203/275/scripts/package.json +0 -16
- package/botrun-c/.claude/skills//347/266/262/350/267/257/346/220/234/345/260/213/346/212/200/350/203/275/scripts/search.js +0 -139
- package/botrun-c/.claude/skills//347/266/262/350/267/257/346/220/234/345/260/213/346/212/200/350/203/275/scripts/search.mjs +0 -272
- package/botrun-c/.claude/skills//347/266/262/350/267/257/346/220/234/345/260/213/346/212/200/350/203/275/scripts/search.ts +0 -166
- package/botrun-c/.claude/skills//350/250/230/346/206/266/346/212/200/350/203/275/SKILL.md +0 -114
- package/botrun-c/.claude/skills//350/250/230/346/206/266/346/212/200/350/203/275/scripts/remember.mjs +0 -159
- package/botrun-c/.claude/skills//350/250/230/346/206/266/346/212/200/350/203/275/scripts/test-advanced.mjs +0 -342
- package/botrun-c/.claude/skills//350/250/230/346/206/266/346/212/200/350/203/275/scripts/test.mjs +0 -430
- package/botrun-c/.claude/skills//350/252/236/351/237/263/350/275/211/346/226/207/345/255/227/346/212/200/350/203/275/SKILL.md +0 -30
- package/botrun-c/.claude/skills//350/252/236/351/237/263/350/275/211/346/226/207/345/255/227/346/212/200/350/203/275/scripts/package.json +0 -13
- package/botrun-c/.claude/skills//350/252/236/351/237/263/350/275/211/346/226/207/345/255/227/346/212/200/350/203/275/scripts/transcribe.mjs +0 -294
- package/botrun-c/.claude/skills//351/233/266/345/271/273/350/246/272/350/255/211/346/230/216/346/212/200/350/203/275/SKILL.md +0 -338
- package/botrun-c/.dockerignore +0 -131
- package/botrun-c/.dockeroptimize +0 -13
- package/botrun-c/.firebase/hosting.b3V0.cache +0 -30
- package/botrun-c/.firebaserc +0 -5
- package/botrun-c/.gcloudignore +0 -56
- package/botrun-c/.ruby-version +0 -1
- package/botrun-c/CHANGELOG.md +0 -224
- package/botrun-c/Dockerfile.gcp +0 -365
- package/botrun-c/Dockerfile.slim +0 -195
- package/botrun-c/Gemfile +0 -8
- package/botrun-c/Gemfile.lock +0 -233
- package/botrun-c/LOCAL-DEVELOPMENT-GUIDE.md +0 -393
- package/botrun-c/PLAN.md +0 -131
- package/botrun-c/README.md +0 -129
- package/botrun-c/TODO-CAPACITOR-INTEGRATION.md +0 -482
- package/botrun-c/VERIFICATION.md +0 -121
- package/botrun-c/admin/RATE_LIMIT_MANAGEMENT.md +0 -312
- package/botrun-c/admin/README.md +0 -338
- package/botrun-c/admin/botrun-billing-report-1765235244038.csv +0 -22
- package/botrun-c/admin/botrun-billing-report-1765235323089.csv +0 -22
- package/botrun-c/admin/botrun-dashboard-1765235244039.html +0 -278
- package/botrun-c/admin/botrun-dashboard-1765235323089.html +0 -278
- package/botrun-c/admin/botrun-timeseries-074xGzIKWyfKTBt1NMIj9lxi5mO2-1765235244039.html +0 -161
- package/botrun-c/admin/botrun-timeseries-074xGzIKWyfKTBt1NMIj9lxi5mO2-1765235323090.html +0 -161
- package/botrun-c/admin/check-billing.ts +0 -48
- package/botrun-c/admin/count-all-users.ts +0 -65
- package/botrun-c/admin/generate-analytics-report.ts +0 -188
- package/botrun-c/admin/jest.config.ts +0 -43
- package/botrun-c/admin/lib/__tests__/formatters.test.ts +0 -264
- package/botrun-c/admin/lib/__tests__/mocks/firebase.mock.ts +0 -112
- package/botrun-c/admin/lib/__tests__/services/UserService.test.ts +0 -83
- package/botrun-c/admin/lib/__tests__/user-resolver.test.ts +0 -439
- package/botrun-c/admin/lib/analyzers/BillingAnalyticsService.ts +0 -242
- package/botrun-c/admin/lib/analyzers/CsvExporter.ts +0 -263
- package/botrun-c/admin/lib/analyzers/HtmlChartGenerator.ts +0 -530
- package/botrun-c/admin/lib/analyzers/SpikeAnalyzer.ts +0 -608
- package/botrun-c/admin/lib/analyzers/TimeSeriesAnalyzer.ts +0 -344
- package/botrun-c/admin/lib/cli/CLIBootstrapper.ts +0 -40
- package/botrun-c/admin/lib/commands/AbstractCommand.ts +0 -69
- package/botrun-c/admin/lib/commands/AnalyticsCommand.ts +0 -215
- package/botrun-c/admin/lib/commands/ListCommand.ts +0 -85
- package/botrun-c/admin/lib/commands/QueryCommand.ts +0 -81
- package/botrun-c/admin/lib/commands/SpikeCommand.ts +0 -83
- package/botrun-c/admin/lib/domain/User.ts +0 -38
- package/botrun-c/admin/lib/firebase.ts +0 -92
- package/botrun-c/admin/lib/firestore-safe.ts +0 -105
- package/botrun-c/admin/lib/formatters.ts +0 -154
- package/botrun-c/admin/lib/rate-limit-detailed.ts +0 -355
- package/botrun-c/admin/lib/rate-limit.ts +0 -286
- package/botrun-c/admin/lib/repositories/UserRepository.ts +0 -104
- package/botrun-c/admin/lib/services/UserService.ts +0 -108
- package/botrun-c/admin/lib/types.ts +0 -56
- package/botrun-c/admin/lib/user-resolver.ts +0 -275
- package/botrun-c/admin/manage-auth-domains.js +0 -178
- package/botrun-c/admin/migrate-plan-free.sh +0 -64
- package/botrun-c/admin/package-lock.json +0 -5656
- package/botrun-c/admin/package.json +0 -28
- package/botrun-c/admin/rate-limit.sh +0 -55
- package/botrun-c/admin/remove-custom-limits.ts +0 -75
- package/botrun-c/admin/reset-rate.sh +0 -70
- package/botrun-c/admin/reset-trial.sh +0 -68
- package/botrun-c/admin/seed-trial-data.sh +0 -38
- package/botrun-c/admin/trial.sh +0 -55
- package/botrun-c/admin/tsconfig.json +0 -20
- package/botrun-c/admin/users-cli.py +0 -298
- package/botrun-c/admin/users.sh +0 -12
- package/botrun-c/admin/users.ts +0 -1321
- package/botrun-c/api/admin-tools/check-whitelist.mjs.migration-bak +0 -74
- package/botrun-c/api/admin-tools/create-user.mjs +0 -92
- package/botrun-c/api/admin-tools/create-user.mjs.migration-bak +0 -92
- package/botrun-c/api/admin-tools/firebase-auth-domains-v2.py +0 -287
- package/botrun-c/api/admin-tools/firebase-auth-domains-v2.py.migration-bak +0 -287
- package/botrun-c/api/admin-tools/init-firestore.mjs +0 -75
- package/botrun-c/api/admin-tools/init-firestore.mjs.migration-bak +0 -101
- package/botrun-c/api/admin-tools/list-users.mjs +0 -43
- package/botrun-c/api/admin-tools/list-users.mjs.migration-bak +0 -43
- package/botrun-c/api/admin-tools/manage-authorized-domains.ts +0 -255
- package/botrun-c/api/admin-tools/manage-authorized-domains.ts.migration-bak +0 -255
- package/botrun-c/api/admin-tools/setup-custom-claims.ts.migration-bak +0 -155
- package/botrun-c/api/admin-tools/test-claims.mjs +0 -69
- package/botrun-c/api/admin-tools/test-claims.mjs.migration-bak +0 -69
- package/botrun-c/api/admin-tools/update-service-urls-gcloud.mjs +0 -72
- package/botrun-c/api/admin-tools/update-service-urls-gcloud.mjs.migration-bak +0 -72
- package/botrun-c/api/admin-tools/update-service-urls-to-lb.mjs +0 -159
- package/botrun-c/api/admin-tools/update-service-urls-to-lb.mjs.migration-bak +0 -159
- package/botrun-c/api/admin-tools/update-service-urls.mjs +0 -82
- package/botrun-c/api/admin-tools/update-service-urls.mjs.migration-bak +0 -82
- package/botrun-c/api/package-lock.json +0 -5031
- package/botrun-c/api/package.json +0 -63
- package/botrun-c/api/pnpm-lock.yaml +0 -4157
- package/botrun-c/api/rate-limit-export-1763620678737.csv +0 -79
- package/botrun-c/api/scripts/README-migrate-plan-free.md +0 -197
- package/botrun-c/api/scripts/check-users.ts +0 -45
- package/botrun-c/api/scripts/delete-firestore-docs.js +0 -43
- package/botrun-c/api/scripts/dump-subscriptions.ts +0 -395
- package/botrun-c/api/scripts/list-all-users.ts +0 -65
- package/botrun-c/api/scripts/migrate-plan-free-rpd-16to32.ts +0 -262
- package/botrun-c/api/scripts/migrate-plan-names-to-lite-max.ts +0 -269
- package/botrun-c/api/scripts/query-user-info.ts +0 -304
- package/botrun-c/api/scripts/reset-rate-limit.ts +0 -206
- package/botrun-c/api/scripts/reset-trial-subscription.ts +0 -166
- package/botrun-c/api/scripts/rollback-plan-names-from-lite-max.ts +0 -139
- package/botrun-c/api/scripts/seed-rate-limit-test-data.ts +0 -114
- package/botrun-c/api/scripts/seed-subscription-test-data.ts +0 -173
- package/botrun-c/api/scripts/test-rate-limit-message.ts +0 -108
- package/botrun-c/api/test-billing-write.ts +0 -103
- package/botrun-c/api/test-billing.ts +0 -81
- package/botrun-c/api/test-chat-upload.sh +0 -162
- package/botrun-c/api/test-nchc-whisper.ts +0 -87
- package/botrun-c/api/test-skills-loading.js +0 -77
- package/botrun-c/api/test-skills-symlink.js +0 -118
- package/botrun-c/api/test-upload-multimodal.sh +0 -167
- package/botrun-c/api/tsconfig.json +0 -27
- package/botrun-c/api/verify-form-2-1.mjs +0 -229
- package/botrun-c/api/vitest.config.ts +0 -9
- package/botrun-c/appium-capabilities.json +0 -20
- package/botrun-c/cors.json +0 -26
- package/botrun-c/delete-old.sh +0 -44
- package/botrun-c/design/google-drive/01-architecture.md +0 -186
- package/botrun-c/design/google-drive/02-api-design.md +0 -314
- package/botrun-c/design/google-drive/03-frontend-design.md +0 -278
- package/botrun-c/design/google-drive/04-skill-design.md +0 -384
- package/botrun-c/design/google-drive/05-test-plan.md +0 -507
- package/botrun-c/design/google-drive/README.md +0 -32
- package/botrun-c/dev/mcp-token-calculator/README.md +0 -92
- package/botrun-c/dev/mcp-token-calculator/RESULTS.md +0 -188
- package/botrun-c/dev/mcp-token-calculator/calculate-current-tokens.ts +0 -369
- package/botrun-c/dev/mcp-token-calculator/calculate-mcp-tokens.ts +0 -464
- package/botrun-c/dev/mcp-token-calculator/optimization-proposals.ts +0 -316
- package/botrun-c/dev/mcp-token-calculator/package-lock.json +0 -599
- package/botrun-c/dev/mcp-token-calculator/package.json +0 -22
- package/botrun-c/dev/mcp-token-calculator/test-all-schemas.ts +0 -314
- package/botrun-c/dev/mcp-token-calculator/test-schema-correctness.ts +0 -221
- package/botrun-c/dev/mcp-token-calculator/verify-optimization.ts +0 -159
- package/botrun-c/fastlane/ANDROID-DEPLOYMENT-SETUP.md +0 -825
- package/botrun-c/fastlane/IOS-DEPLOYMENT-SETUP.md +0 -431
- package/botrun-c/fastlane/Pluginfile +0 -5
- package/botrun-c/fastlane/QUICK-START.md +0 -133
- package/botrun-c/fastlane/README.md +0 -424
- package/botrun-c/fastlane/RUBY-SETUP.md +0 -449
- package/botrun-c/fastlane/android/Appfile +0 -11
- package/botrun-c/fastlane/android/Fastfile +0 -240
- package/botrun-c/firebase.json +0 -46
- package/botrun-c/firestore-init-data.json +0 -40
- package/botrun-c/firestore.indexes.json +0 -55
- package/botrun-c/firestore.rules +0 -53
- package/botrun-c/fix-codex-permissions.sh +0 -56
- package/botrun-c/gcs-lifecycle-30day-retention.json +0 -12
- package/botrun-c/html-architecture/architecture/deployment.html +0 -664
- package/botrun-c/html-architecture/architecture/index.html +0 -309
- package/botrun-c/html-architecture/architecture/styles.css +0 -500
- package/botrun-c/html-architecture/architecture/system-components.html +0 -538
- package/botrun-c/html-architecture/architecture/user-flow.html +0 -370
- package/botrun-c/mobile-android/FIREBASE-AUTH-SETUP.md +0 -302
- package/botrun-c/mobile-android/WSL-ANDROID-SETUP.md +0 -252
- package/botrun-c/mobile-android/android/app/build.gradle +0 -75
- package/botrun-c/mobile-android/android/app/capacitor.build.gradle +0 -31
- package/botrun-c/mobile-android/android/app/proguard-rules.pro +0 -21
- package/botrun-c/mobile-android/android/build.gradle +0 -29
- package/botrun-c/mobile-android/android/capacitor.settings.gradle +0 -42
- package/botrun-c/mobile-android/android/gradle/wrapper/gradle-wrapper.jar +0 -0
- package/botrun-c/mobile-android/android/gradle/wrapper/gradle-wrapper.properties +0 -7
- package/botrun-c/mobile-android/android/gradle.properties +0 -22
- package/botrun-c/mobile-android/android/gradlew +0 -252
- package/botrun-c/mobile-android/android/gradlew.bat +0 -94
- package/botrun-c/mobile-android/android/settings.gradle +0 -5
- package/botrun-c/mobile-android/android/variables.gradle +0 -16
- package/botrun-c/mobile-android/capacitor.config.ts +0 -44
- package/botrun-c/mobile-android/fastlane/Appfile +0 -11
- package/botrun-c/mobile-android/fastlane/Fastfile +0 -202
- package/botrun-c/mobile-android/fastlane/README.md +0 -96
- package/botrun-c/mobile-android/google-services.json.template +0 -39
- package/botrun-c/mobile-android/package.json +0 -32
- package/botrun-c/mobile-android/resources/icon.png +0 -0
- package/botrun-c/mobile-android/tsconfig.json +0 -15
- package/botrun-c/mobile-ios/.ruby-version +0 -1
- package/botrun-c/mobile-ios/BUILD-LOG.md +0 -82
- package/botrun-c/mobile-ios/Gemfile +0 -4
- package/botrun-c/mobile-ios/Gemfile.lock +0 -299
- package/botrun-c/mobile-ios/GoogleService-Info.plist.template +0 -34
- package/botrun-c/mobile-ios/IOS-PERMISSIONS.md +0 -80
- package/botrun-c/mobile-ios/QUICK-START.md +0 -189
- package/botrun-c/mobile-ios/README.md +0 -231
- package/botrun-c/mobile-ios/capacitor.config.ts +0 -46
- package/botrun-c/mobile-ios/fastlane/Fastfile +0 -326
- package/botrun-c/mobile-ios/fastlane/README.md +0 -88
- package/botrun-c/mobile-ios/ios/App/App/App.entitlements +0 -10
- package/botrun-c/mobile-ios/ios/App/App/AppDelegate.swift +0 -49
- package/botrun-c/mobile-ios/ios/App/App/Assets.xcassets/AppIcon.appiconset/AppIcon-512@2x.png +0 -0
- package/botrun-c/mobile-ios/ios/App/App/Assets.xcassets/AppIcon.appiconset/Contents.json +0 -14
- package/botrun-c/mobile-ios/ios/App/App/Assets.xcassets/Contents.json +0 -6
- package/botrun-c/mobile-ios/ios/App/App/Assets.xcassets/Splash.imageset/Contents.json +0 -56
- package/botrun-c/mobile-ios/ios/App/App/Assets.xcassets/Splash.imageset/Default@1x~universal~anyany-dark.png +0 -0
- package/botrun-c/mobile-ios/ios/App/App/Assets.xcassets/Splash.imageset/Default@1x~universal~anyany.png +0 -0
- package/botrun-c/mobile-ios/ios/App/App/Assets.xcassets/Splash.imageset/Default@2x~universal~anyany-dark.png +0 -0
- package/botrun-c/mobile-ios/ios/App/App/Assets.xcassets/Splash.imageset/Default@2x~universal~anyany.png +0 -0
- package/botrun-c/mobile-ios/ios/App/App/Assets.xcassets/Splash.imageset/Default@3x~universal~anyany-dark.png +0 -0
- package/botrun-c/mobile-ios/ios/App/App/Assets.xcassets/Splash.imageset/Default@3x~universal~anyany.png +0 -0
- package/botrun-c/mobile-ios/ios/App/App/Assets.xcassets/Splash.imageset/splash-2732x2732-1.png +0 -0
- package/botrun-c/mobile-ios/ios/App/App/Assets.xcassets/Splash.imageset/splash-2732x2732-2.png +0 -0
- package/botrun-c/mobile-ios/ios/App/App/Assets.xcassets/Splash.imageset/splash-2732x2732.png +0 -0
- package/botrun-c/mobile-ios/ios/App/App/Base.lproj/LaunchScreen.storyboard +0 -32
- package/botrun-c/mobile-ios/ios/App/App/Base.lproj/Main.storyboard +0 -19
- package/botrun-c/mobile-ios/ios/App/App/Info.plist +0 -72
- package/botrun-c/mobile-ios/ios/App/App.xcodeproj/project.pbxproj +0 -446
- package/botrun-c/mobile-ios/ios/App/App.xcodeproj/xcshareddata/xcschemes/App.xcscheme +0 -78
- package/botrun-c/mobile-ios/ios/App/App.xcworkspace/contents.xcworkspacedata +0 -10
- package/botrun-c/mobile-ios/ios/App/App.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist +0 -8
- package/botrun-c/mobile-ios/ios/App/Podfile +0 -52
- package/botrun-c/mobile-ios/ios/App/Podfile.lock +0 -205
- package/botrun-c/mobile-ios/ios/App/add_plist.rb +0 -30
- package/botrun-c/mobile-ios/ios/App/fix_plist_path.rb +0 -22
- package/botrun-c/mobile-ios/package.json +0 -29
- package/botrun-c/mobile-ios/resources/icon.png +0 -0
- package/botrun-c/mobile-ios/run-simulator.sh +0 -34
- package/botrun-c/mobile-ios/screenshot-after-clean-rebuild.png +0 -0
- package/botrun-c/mobile-ios/screenshot-final-version.png +0 -0
- package/botrun-c/mobile-ios/screenshot-simulator.png +0 -0
- package/botrun-c/mobile-ios/screenshot-version-1009.png +0 -0
- package/botrun-c/mobile-ios/screenshot-with-version.png +0 -0
- package/botrun-c/mobile-ios/screenshot-working.png +0 -0
- package/botrun-c/mobile-ios/setup-ios.sh +0 -106
- package/botrun-c/mobile-ios/tsconfig.json +0 -16
- package/botrun-c/org-policy-allow-all-users.yaml +0 -3
- package/botrun-c/package.json +0 -68
- package/botrun-c/pnpm-lock.yaml +0 -10198
- package/botrun-c/pnpm-workspace.yaml +0 -6
- package/botrun-c/prompts/BOTRUN-1.md +0 -13
- package/botrun-c/prompts/BOTRUN-2.md +0 -70
- package/botrun-c/prompts/BOTRUN-3.md +0 -139
- package/botrun-c/prototypes/firestore-gax-debug/package-lock.json +0 -1204
- package/botrun-c/prototypes/firestore-gax-debug/package.json +0 -12
- package/botrun-c/prototypes/firestore-gax-debug/pnpm-workspace.yaml +0 -2
- package/botrun-c/prototypes/firestore-gax-debug/test.js +0 -98
- package/botrun-c/prototypes/taiwan-haiku/README.md +0 -93
- package/botrun-c/prototypes/taiwan-haiku/index.ts +0 -139
- package/botrun-c/prototypes/taiwan-haiku/package.json +0 -23
- package/botrun-c/prototypes/taiwan-haiku/pnpm-lock.yaml +0 -730
- package/botrun-c/prototypes/taiwan-haiku/test-vertex-direct.ts +0 -83
- package/botrun-c/prototypes/taiwan-haiku/tsconfig.json +0 -18
- package/botrun-c/prototypes/web-search/FINDINGS-STAGE-1.md +0 -48
- package/botrun-c/prototypes/web-search/README.md +0 -46
- package/botrun-c/prototypes/web-search/VERIFICATION-REPORT.md +0 -310
- package/botrun-c/prototypes/web-search/package-lock.json +0 -595
- package/botrun-c/prototypes/web-search/package.json +0 -18
- package/botrun-c/prototypes/web-search/stage-1.1-basic-connection.js +0 -123
- package/botrun-c/prototypes/web-search/stage-1.2-with-websearch.js +0 -154
- package/botrun-c/prototypes/web-search/stage-1.3-search-query.js +0 -192
- package/botrun-c/prototypes/web-search/stage-2-agent-sdk-websearch.js +0 -265
- package/botrun-c/runtime-scripts/env-detect.sh +0 -64
- package/botrun-c/scripts/github-collaborator-manager.sh +0 -297
- package/botrun-c/scripts/github-collaborator-manager.ts +0 -325
- package/botrun-c/scripts/manage-collaborators.sh +0 -91
- package/botrun-c/scripts/setup-secrets.sh +0 -202
- package/botrun-c/scripts/stress-test-17-users.sh +0 -113
- package/botrun-c/scripts/stress-test-api.sh +0 -119
- package/botrun-c/scripts/stress-test-full-report.sh +0 -385
- package/botrun-c/specs/infra-vm/design.md +0 -0
- package/botrun-c/specs/share-skill/design.md +0 -1146
- package/botrun-c/specs/share-skill/tasks.md +0 -444
- package/botrun-c/stress-test-results/20260121-162630/config.json +0 -7
- package/botrun-c/stress-test-results/20260121-162630/user-1-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162630/user-1-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162630/user-1.json +0 -2
- package/botrun-c/stress-test-results/20260121-162646/config.json +0 -7
- package/botrun-c/stress-test-results/20260121-162646/user-1-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-1-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-1.json +0 -2
- package/botrun-c/stress-test-results/20260121-162646/user-10-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-10-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-10.json +0 -2
- package/botrun-c/stress-test-results/20260121-162646/user-11-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-11-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-11.json +0 -2
- package/botrun-c/stress-test-results/20260121-162646/user-12-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-12-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-12.json +0 -2
- package/botrun-c/stress-test-results/20260121-162646/user-13-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-13-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-13.json +0 -2
- package/botrun-c/stress-test-results/20260121-162646/user-14-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-14-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-14.json +0 -2
- package/botrun-c/stress-test-results/20260121-162646/user-15-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-15-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-15.json +0 -2
- package/botrun-c/stress-test-results/20260121-162646/user-16-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-16-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-16.json +0 -2
- package/botrun-c/stress-test-results/20260121-162646/user-17-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-17-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-17.json +0 -2
- package/botrun-c/stress-test-results/20260121-162646/user-2-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-2-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-2.json +0 -2
- package/botrun-c/stress-test-results/20260121-162646/user-3-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-3-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-3.json +0 -2
- package/botrun-c/stress-test-results/20260121-162646/user-4-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-4-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-4.json +0 -2
- package/botrun-c/stress-test-results/20260121-162646/user-5-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-5-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-5.json +0 -2
- package/botrun-c/stress-test-results/20260121-162646/user-6-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-6-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-6.json +0 -2
- package/botrun-c/stress-test-results/20260121-162646/user-7-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-7-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-7.json +0 -2
- package/botrun-c/stress-test-results/20260121-162646/user-8-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-8-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-8.json +0 -2
- package/botrun-c/stress-test-results/20260121-162646/user-9-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-9-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-162646/user-9.json +0 -2
- package/botrun-c/stress-test-results/20260121-163148/REPORT.html +0 -344
- package/botrun-c/stress-test-results/20260121-163148/REPORT.md +0 -93
- package/botrun-c/stress-test-results/20260121-163148/REPORT.pdf +0 -0
- package/botrun-c/stress-test-results/20260121-163148/config.json +0 -7
- package/botrun-c/stress-test-results/20260121-163148/summary.json +0 -26
- package/botrun-c/stress-test-results/20260121-163148/user-1-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-1-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-1.json +0 -206
- package/botrun-c/stress-test-results/20260121-163148/user-10-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-10-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-10.json +0 -110
- package/botrun-c/stress-test-results/20260121-163148/user-11-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-11-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-11.json +0 -104
- package/botrun-c/stress-test-results/20260121-163148/user-12-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-12-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-12.json +0 -98
- package/botrun-c/stress-test-results/20260121-163148/user-13-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-13-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-13.json +0 -68
- package/botrun-c/stress-test-results/20260121-163148/user-14-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-14-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-14.json +0 -116
- package/botrun-c/stress-test-results/20260121-163148/user-15-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-15-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-15.json +0 -131
- package/botrun-c/stress-test-results/20260121-163148/user-16-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-16-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-16.json +0 -110
- package/botrun-c/stress-test-results/20260121-163148/user-17-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-17-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-17.json +0 -116
- package/botrun-c/stress-test-results/20260121-163148/user-2-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-2-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-2.json +0 -98
- package/botrun-c/stress-test-results/20260121-163148/user-3-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-3-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-3.json +0 -62
- package/botrun-c/stress-test-results/20260121-163148/user-4-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-4-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-4.json +0 -113
- package/botrun-c/stress-test-results/20260121-163148/user-5-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-5-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-5.json +0 -83
- package/botrun-c/stress-test-results/20260121-163148/user-6-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-6-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-6.json +0 -125
- package/botrun-c/stress-test-results/20260121-163148/user-7-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-7-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-7.json +0 -62
- package/botrun-c/stress-test-results/20260121-163148/user-8-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-8-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-8.json +0 -41
- package/botrun-c/stress-test-results/20260121-163148/user-9-status.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-9-timing.txt +0 -1
- package/botrun-c/stress-test-results/20260121-163148/user-9.json +0 -53
- package/botrun-c/test-merge.md +0 -1
- package/botrun-c/version.txt +0 -1
- package/botrun-c/web/.version +0 -1
- package/botrun-c/web/ALLOW-PUBLIC-ACCESS.md +0 -57
- package/botrun-c/web/FINAL-SETUP-STEPS.md +0 -71
- package/botrun-c/web/FIREBASE-MANUAL-STEPS.md +0 -41
- package/botrun-c/web/QUICK-SETUP-HELPER.md +0 -79
- package/botrun-c/web/README.md +0 -162
- package/botrun-c/web/__tests__/image-extractor.test.ts +0 -147
- package/botrun-c/web/__tests__/useGitHub.test.ts +0 -245
- package/botrun-c/web/app/favicon.ico +0 -0
- package/botrun-c/web/app/globals.css +0 -215
- package/botrun-c/web/app/image-preview/page.tsx +0 -213
- package/botrun-c/web/app/layout.tsx +0 -46
- package/botrun-c/web/app/page.tsx +0 -45
- package/botrun-c/web/components/CapacitorProvider.tsx +0 -43
- package/botrun-c/web/components/ChatPage.tsx +0 -1736
- package/botrun-c/web/components/ConversationList.tsx +0 -237
- package/botrun-c/web/components/ConversationListItem.tsx +0 -121
- package/botrun-c/web/components/LoginPage.tsx +0 -212
- package/botrun-c/web/components/Providers.tsx +0 -12
- package/botrun-c/web/components/StatusBarProvider.tsx +0 -16
- package/botrun-c/web/components/attachment-preview.tsx +0 -193
- package/botrun-c/web/components/botrun-incubator/BotrunIncubator.tsx +0 -266
- package/botrun-c/web/components/botrun-incubator/index.ts +0 -1
- package/botrun-c/web/components/chat-input.tsx +0 -1251
- package/botrun-c/web/components/chat-message.tsx +0 -598
- package/botrun-c/web/components/drop-zone.tsx +0 -143
- package/botrun-c/web/components/gdrive/gdrive-add-dialog.tsx +0 -254
- package/botrun-c/web/components/gdrive/gdrive-drawer.tsx +0 -236
- package/botrun-c/web/components/gdrive/gdrive-selector.tsx +0 -226
- package/botrun-c/web/components/gdrive/index.ts +0 -7
- package/botrun-c/web/components/generated-image-card.tsx +0 -363
- package/botrun-c/web/components/github/GitHubConnect.tsx +0 -366
- package/botrun-c/web/components/github/index.ts +0 -2
- package/botrun-c/web/components/import-skill-dialog.tsx +0 -208
- package/botrun-c/web/components/scroll-to-bottom-button.tsx +0 -90
- package/botrun-c/web/components/share-skill-dialog.tsx +0 -175
- package/botrun-c/web/components/skills-editor.tsx +0 -443
- package/botrun-c/web/components/skills-manager-drawer.tsx +0 -586
- package/botrun-c/web/components/smart-button-group.tsx +0 -132
- package/botrun-c/web/config/smart-buttons.ts +0 -295
- package/botrun-c/web/config/welcome-messages.ts +0 -29
- package/botrun-c/web/contexts/AuthContext.tsx +0 -220
- package/botrun-c/web/hooks/use-toast.ts +0 -187
- package/botrun-c/web/hooks/useAppStoreUpdate.ts +0 -356
- package/botrun-c/web/hooks/useAppVersion.ts +0 -113
- package/botrun-c/web/hooks/useAttachments.ts +0 -269
- package/botrun-c/web/hooks/useAuthToken.ts +0 -59
- package/botrun-c/web/hooks/useAutoWelcome.ts +0 -89
- package/botrun-c/web/hooks/useCapacitorPlatform.ts +0 -113
- package/botrun-c/web/hooks/useCapawesomeLiveUpdate.ts +0 -149
- package/botrun-c/web/hooks/useClaudeModel.ts +0 -118
- package/botrun-c/web/hooks/useConversation.ts +0 -115
- package/botrun-c/web/hooks/useConversations.ts +0 -235
- package/botrun-c/web/hooks/useGdriveFolders.ts +0 -199
- package/botrun-c/web/hooks/useGdriveSync.ts +0 -107
- package/botrun-c/web/hooks/useGitHub.ts +0 -409
- package/botrun-c/web/hooks/useScrollToBottom.ts +0 -150
- package/botrun-c/web/hooks/useStatusBar.ts +0 -42
- package/botrun-c/web/hooks/useUserPlan.ts +0 -153
- package/botrun-c/web/lib/api-config.ts +0 -162
- package/botrun-c/web/lib/api-url.ts +0 -62
- package/botrun-c/web/lib/api.ts +0 -105
- package/botrun-c/web/lib/attachment-utils.ts +0 -168
- package/botrun-c/web/lib/build-version.ts +0 -11
- package/botrun-c/web/lib/clipboard-utils.ts +0 -199
- package/botrun-c/web/lib/constants.ts +0 -62
- package/botrun-c/web/lib/conversation-storage.ts +0 -324
- package/botrun-c/web/lib/firebase-capacitor.ts +0 -135
- package/botrun-c/web/lib/firebase.ts +0 -137
- package/botrun-c/web/lib/firestore-auth.ts +0 -61
- package/botrun-c/web/lib/firestore-auth.ts.migration-bak +0 -319
- package/botrun-c/web/lib/image-extractor.ts +0 -156
- package/botrun-c/web/lib/path-utils.ts +0 -48
- package/botrun-c/web/lib/rehype-del-to-mark.ts +0 -19
- package/botrun-c/web/lib/upload-service.ts +0 -326
- package/botrun-c/web/lib/utils.ts +0 -6
- package/botrun-c/web/next.config.js +0 -49
- package/botrun-c/web/package-lock.json +0 -6711
- package/botrun-c/web/package.json +0 -63
- package/botrun-c/web/postcss.config.js +0 -6
- package/botrun-c/web/public/apple_icon.png +0 -0
- package/botrun-c/web/public/google_icon.png +0 -0
- package/botrun-c/web/public/logo-48-small.png +0 -0
- package/botrun-c/web/public/logo-48.png +0 -0
- package/botrun-c/web/scripts/generate-version.js +0 -36
- package/botrun-c/web/scripts/update-build-version.js +0 -41
- package/botrun-c/web/tailwind.config.ts +0 -80
- package/botrun-c/web/test-welcome-message.mjs +0 -115
- package/botrun-c/web/tsconfig.json +0 -42
- package/botrun-c/web/types/agent.ts +0 -6
- package/botrun-c/web/types/attachment.ts +0 -39
- package/botrun-c/web/types/conversation.ts +0 -68
- package/botrun-c/web/types/flutter.d.ts +0 -6
- package/botrun-c/web/types/project.ts +0 -41
- package/botrun-c/web/types/skills.ts +0 -13
- package/botrun-c/web//350/250/272/346/226/267-Skills-/345/211/215/347/253/257/351/241/257/347/244/272/345/225/217/351/241/214.md +0 -83
- package/lib/writing/index.mjs +0 -5
- /package/{lib/core → modules/_core}/adapters/base.mjs +0 -0
- /package/{lib/core → modules/_core}/adapters/claude.mjs +0 -0
- /package/{lib/core → modules/_core}/adapters/gemini-api.mjs +0 -0
- /package/{lib/core → modules/_core}/adapters/gemini-shared.mjs +0 -0
- /package/{lib/core → modules/_core}/adapters/gemini-vertex.mjs +0 -0
- /package/{lib/core → modules/_core}/adapters/local.mjs +0 -0
- /package/{lib/core → modules/_core}/adapters/nchc.mjs +0 -0
- /package/{lib/core → modules/_core}/adapters/openai-shared.mjs +0 -0
- /package/{lib/core → modules/_core}/adapters/openrouter.mjs +0 -0
- /package/{lib/core → modules/_core}/ai-cache.mjs +0 -0
- /package/{lib/core → modules/_core}/ai-router.mjs +0 -0
- /package/{lib/core → modules/_core}/cli-utils.mjs +0 -0
- /package/{lib/core → modules/_core}/dag.mjs +0 -0
- /package/{lib/core → modules/_core}/db.mjs +0 -0
- /package/{lib/core → modules/_core}/env.mjs +0 -0
- /package/{lib/flows → modules/_core}/hatch-portal.mjs +0 -0
- /package/{lib/core → modules/_core}/llm.mjs +0 -0
- /package/{lib/flows → modules/_core}/opencode-agent.mjs +0 -0
- /package/{lib/core → modules/_core}/paths.mjs +0 -0
- /package/{lib/core → modules/_core}/proxy.mjs +0 -0
- /package/{lib/flows → modules/_core}/review-doc.mjs +0 -0
- /package/{lib → modules/_core}/tools/fs-tools.mjs +0 -0
- /package/{lib → modules/_core}/tools/index.mjs +0 -0
- /package/{lib/core → modules/_core}/watermelon.mjs +0 -0
- /package/{lib/ocr/index.mjs → modules/ocr/ocr-lib.mjs} +0 -0
- /package/{lib → modules}/portal/hatch.mjs +0 -0
- /package/{lib/portal/index.mjs → modules/portal/portal-lib.mjs} +0 -0
- /package/{lib → modules}/search/crawler.mjs +0 -0
- /package/{lib/prompt → modules/skill}/prompts/zero-framework/coding.md +0 -0
- /package/{lib/prompt → modules/skill}/prompts/zero-framework/fullstack.md +0 -0
- /package/{lib/prompt → modules/skill}/prompts/zero-framework/search.md +0 -0
- /package/{lib/prompt → modules/skill}/prompts/zero-framework/segment.md +0 -0
- /package/{lib/prompt → modules/skill}/prompts/zero-framework/slice.md +0 -0
- /package/{lib → modules}/writing/generators/nstc-generators.mjs +0 -0
- /package/{lib → modules}/writing/generators/nstc-top5.mjs +0 -0
- /package/{lib → modules}/writing/layouts/nstc-layout.mjs +0 -0
- /package/{lib → modules}/writing/renderer.mjs +0 -0
|
@@ -1,439 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* 實質性保護傘測試:user-resolver.ts
|
|
3
|
-
* 測試現有的使用者解析功能,確保重構前功能完整
|
|
4
|
-
*/
|
|
5
|
-
|
|
6
|
-
import { createMockAuth, createMockUser } from './mocks/firebase.mock';
|
|
7
|
-
import * as userResolverModule from '../user-resolver';
|
|
8
|
-
|
|
9
|
-
// Mock Firebase 模組
|
|
10
|
-
jest.mock('../firebase.js', () => ({
|
|
11
|
-
getAuth: jest.fn(),
|
|
12
|
-
getFirestore: jest.fn(),
|
|
13
|
-
}));
|
|
14
|
-
|
|
15
|
-
describe('user-resolver.ts - 實質性保護傘測試', () => {
|
|
16
|
-
let mockAuth: any;
|
|
17
|
-
let mockFirestore: any;
|
|
18
|
-
let getAuthMock: jest.Mock;
|
|
19
|
-
let getFirestoreMock: jest.Mock;
|
|
20
|
-
|
|
21
|
-
beforeEach(() => {
|
|
22
|
-
mockAuth = createMockAuth();
|
|
23
|
-
mockFirestore = {
|
|
24
|
-
collection: jest.fn(() => ({
|
|
25
|
-
doc: jest.fn(() => ({
|
|
26
|
-
set: jest.fn().mockResolvedValue(undefined),
|
|
27
|
-
get: jest.fn().mockResolvedValue({ exists: false }),
|
|
28
|
-
update: jest.fn().mockResolvedValue(undefined),
|
|
29
|
-
})),
|
|
30
|
-
})),
|
|
31
|
-
};
|
|
32
|
-
|
|
33
|
-
const firebaseModule = require('../firebase.js');
|
|
34
|
-
getAuthMock = firebaseModule.getAuth as jest.Mock;
|
|
35
|
-
getFirestoreMock = firebaseModule.getFirestore as jest.Mock;
|
|
36
|
-
getAuthMock.mockReturnValue(mockAuth);
|
|
37
|
-
getFirestoreMock.mockReturnValue(mockFirestore);
|
|
38
|
-
});
|
|
39
|
-
|
|
40
|
-
describe('resolveUser()', () => {
|
|
41
|
-
test('應該透過 Email 解析使用者', async () => {
|
|
42
|
-
mockAuth.getUserByEmail = jest.fn().mockResolvedValue(createMockUser());
|
|
43
|
-
|
|
44
|
-
const user = await userResolverModule.resolveUser('test@example.com');
|
|
45
|
-
|
|
46
|
-
expect(mockAuth.getUserByEmail).toHaveBeenCalledWith('test@example.com');
|
|
47
|
-
expect(user).toEqual({
|
|
48
|
-
uid: 'test-uid-123',
|
|
49
|
-
email: 'test@example.com',
|
|
50
|
-
displayName: 'Test User',
|
|
51
|
-
createdAt: expect.any(Date),
|
|
52
|
-
lastSignInTime: expect.any(Date),
|
|
53
|
-
});
|
|
54
|
-
});
|
|
55
|
-
|
|
56
|
-
test('應該透過 UID 解析使用者', async () => {
|
|
57
|
-
mockAuth.getUser = jest.fn().mockResolvedValue(createMockUser());
|
|
58
|
-
|
|
59
|
-
const user = await userResolverModule.resolveUser('test-uid-123');
|
|
60
|
-
|
|
61
|
-
expect(mockAuth.getUser).toHaveBeenCalledWith('test-uid-123');
|
|
62
|
-
expect(user.uid).toBe('test-uid-123');
|
|
63
|
-
});
|
|
64
|
-
|
|
65
|
-
test('Email 和 UID 應該由 @ 符號區分', async () => {
|
|
66
|
-
mockAuth.getUserByEmail = jest.fn().mockResolvedValue(createMockUser());
|
|
67
|
-
mockAuth.getUser = jest.fn().mockResolvedValue(createMockUser());
|
|
68
|
-
|
|
69
|
-
// 包含 @ 的應該用 email 方式
|
|
70
|
-
await userResolverModule.resolveUser('user@example.com');
|
|
71
|
-
expect(mockAuth.getUserByEmail).toHaveBeenCalled();
|
|
72
|
-
expect(mockAuth.getUser).not.toHaveBeenCalled();
|
|
73
|
-
|
|
74
|
-
// 清空 mock
|
|
75
|
-
jest.clearAllMocks();
|
|
76
|
-
getAuthMock.mockReturnValue(mockAuth);
|
|
77
|
-
|
|
78
|
-
// 不包含 @ 的應該用 UID 方式
|
|
79
|
-
await userResolverModule.resolveUser('uid-without-at');
|
|
80
|
-
expect(mockAuth.getUser).toHaveBeenCalled();
|
|
81
|
-
expect(mockAuth.getUserByEmail).not.toHaveBeenCalled();
|
|
82
|
-
});
|
|
83
|
-
|
|
84
|
-
test('找不到使用者時應該拋出錯誤', async () => {
|
|
85
|
-
mockAuth.getUserByEmail = jest.fn().mockRejectedValue({
|
|
86
|
-
code: 'auth/user-not-found',
|
|
87
|
-
});
|
|
88
|
-
|
|
89
|
-
await expect(
|
|
90
|
-
userResolverModule.resolveUser('notfound@example.com')
|
|
91
|
-
).rejects.toThrow('找不到使用者');
|
|
92
|
-
});
|
|
93
|
-
|
|
94
|
-
test('應該返回完整的使用者資訊(包含時間戳記)', async () => {
|
|
95
|
-
const mockUserData = createMockUser({
|
|
96
|
-
metadata: {
|
|
97
|
-
creationTime: new Date('2025-01-01'),
|
|
98
|
-
lastSignInTime: new Date('2025-12-09'),
|
|
99
|
-
},
|
|
100
|
-
});
|
|
101
|
-
mockAuth.getUserByEmail = jest.fn().mockResolvedValue(mockUserData);
|
|
102
|
-
|
|
103
|
-
const user = await userResolverModule.resolveUser('test@example.com');
|
|
104
|
-
|
|
105
|
-
expect(user).toHaveProperty('uid');
|
|
106
|
-
expect(user).toHaveProperty('email');
|
|
107
|
-
expect(user).toHaveProperty('displayName');
|
|
108
|
-
expect(user).toHaveProperty('createdAt');
|
|
109
|
-
expect(user.createdAt).toBeInstanceOf(Date);
|
|
110
|
-
});
|
|
111
|
-
|
|
112
|
-
test('沒有 lastSignInTime 時應該為 undefined', async () => {
|
|
113
|
-
mockAuth.getUserByEmail = jest.fn().mockResolvedValue(
|
|
114
|
-
createMockUser({
|
|
115
|
-
metadata: {
|
|
116
|
-
creationTime: new Date('2025-01-01'),
|
|
117
|
-
lastSignInTime: null,
|
|
118
|
-
},
|
|
119
|
-
})
|
|
120
|
-
);
|
|
121
|
-
|
|
122
|
-
const user = await userResolverModule.resolveUser('test@example.com');
|
|
123
|
-
expect(user.lastSignInTime).toBeUndefined();
|
|
124
|
-
});
|
|
125
|
-
});
|
|
126
|
-
|
|
127
|
-
describe('resolveUsers()', () => {
|
|
128
|
-
test('應該批次解析多個使用者', async () => {
|
|
129
|
-
mockAuth.getUserByEmail = jest.fn()
|
|
130
|
-
.mockResolvedValueOnce(createMockUser({ email: 'user1@example.com' }))
|
|
131
|
-
.mockResolvedValueOnce(createMockUser({ email: 'user2@example.com' }));
|
|
132
|
-
|
|
133
|
-
const users = await userResolverModule.resolveUsers([
|
|
134
|
-
'user1@example.com',
|
|
135
|
-
'user2@example.com',
|
|
136
|
-
]);
|
|
137
|
-
|
|
138
|
-
expect(users).toHaveLength(2);
|
|
139
|
-
expect(users[0].email).toBe('user1@example.com');
|
|
140
|
-
expect(users[1].email).toBe('user2@example.com');
|
|
141
|
-
});
|
|
142
|
-
|
|
143
|
-
test('單個失敗不應該中斷整個批次', async () => {
|
|
144
|
-
mockAuth.getUserByEmail = jest.fn()
|
|
145
|
-
.mockResolvedValueOnce(createMockUser({ email: 'user1@example.com' }))
|
|
146
|
-
.mockRejectedValueOnce({ code: 'auth/user-not-found' })
|
|
147
|
-
.mockResolvedValueOnce(createMockUser({ email: 'user3@example.com' }));
|
|
148
|
-
|
|
149
|
-
const users = await userResolverModule.resolveUsers([
|
|
150
|
-
'user1@example.com',
|
|
151
|
-
'notfound@example.com',
|
|
152
|
-
'user3@example.com',
|
|
153
|
-
]);
|
|
154
|
-
|
|
155
|
-
// 應該返回成功的 2 個
|
|
156
|
-
expect(users).toHaveLength(2);
|
|
157
|
-
expect(users[0].email).toBe('user1@example.com');
|
|
158
|
-
expect(users[1].email).toBe('user3@example.com');
|
|
159
|
-
});
|
|
160
|
-
|
|
161
|
-
test('空清單應該返回空陣列', async () => {
|
|
162
|
-
const users = await userResolverModule.resolveUsers([]);
|
|
163
|
-
expect(users).toEqual([]);
|
|
164
|
-
});
|
|
165
|
-
});
|
|
166
|
-
|
|
167
|
-
describe('resolveOrCreateUser()', () => {
|
|
168
|
-
test('現有使用者應該直接返回,created=false', async () => {
|
|
169
|
-
mockAuth.getUserByEmail = jest.fn().mockResolvedValue(createMockUser());
|
|
170
|
-
|
|
171
|
-
const result = await userResolverModule.resolveOrCreateUser('test@example.com');
|
|
172
|
-
|
|
173
|
-
expect(result.created).toBe(false);
|
|
174
|
-
expect(result.user.email).toBe('test@example.com');
|
|
175
|
-
expect(result.user.uid).toBe('test-uid-123');
|
|
176
|
-
});
|
|
177
|
-
|
|
178
|
-
test('不存在的使用者應該建立新使用者(createIfNotExists=true)', async () => {
|
|
179
|
-
mockAuth.getUserByEmail = jest.fn()
|
|
180
|
-
.mockRejectedValueOnce({ code: 'auth/user-not-found' });
|
|
181
|
-
|
|
182
|
-
mockAuth.createUser = jest.fn().mockResolvedValue({
|
|
183
|
-
uid: 'new-uid-456',
|
|
184
|
-
email: 'newuser@example.com',
|
|
185
|
-
displayName: 'newuser',
|
|
186
|
-
});
|
|
187
|
-
|
|
188
|
-
const result = await userResolverModule.resolveOrCreateUser(
|
|
189
|
-
'newuser@example.com',
|
|
190
|
-
{ createIfNotExists: true, defaultPlan: 'Lite' }
|
|
191
|
-
);
|
|
192
|
-
|
|
193
|
-
expect(result.created).toBe(true);
|
|
194
|
-
expect(result.user.uid).toBe('new-uid-456');
|
|
195
|
-
expect(mockAuth.createUser).toHaveBeenCalledWith({
|
|
196
|
-
email: 'newuser@example.com',
|
|
197
|
-
displayName: 'newuser',
|
|
198
|
-
});
|
|
199
|
-
});
|
|
200
|
-
|
|
201
|
-
test('不存在且 createIfNotExists=false 應該拋出錯誤', async () => {
|
|
202
|
-
mockAuth.getUserByEmail = jest.fn()
|
|
203
|
-
.mockRejectedValueOnce({ code: 'auth/user-not-found' });
|
|
204
|
-
|
|
205
|
-
await expect(
|
|
206
|
-
userResolverModule.resolveOrCreateUser('notfound@example.com', {
|
|
207
|
-
createIfNotExists: false,
|
|
208
|
-
})
|
|
209
|
-
).rejects.toThrow('找不到使用者');
|
|
210
|
-
});
|
|
211
|
-
|
|
212
|
-
test('必須是 Email 格式(不能是 UID)', async () => {
|
|
213
|
-
await expect(
|
|
214
|
-
userResolverModule.resolveOrCreateUser('uid-without-at', {
|
|
215
|
-
createIfNotExists: true,
|
|
216
|
-
})
|
|
217
|
-
).rejects.toThrow('必須提供 Email 格式');
|
|
218
|
-
});
|
|
219
|
-
|
|
220
|
-
test('建立使用者時應該設定 Firestore users 文件', async () => {
|
|
221
|
-
mockAuth.getUserByEmail = jest.fn()
|
|
222
|
-
.mockRejectedValueOnce({ code: 'auth/user-not-found' });
|
|
223
|
-
|
|
224
|
-
mockAuth.createUser = jest.fn().mockResolvedValue({
|
|
225
|
-
uid: 'new-uid-456',
|
|
226
|
-
email: 'newuser@example.com',
|
|
227
|
-
displayName: 'newuser',
|
|
228
|
-
});
|
|
229
|
-
|
|
230
|
-
const mockSetFn = jest.fn().mockResolvedValue(undefined);
|
|
231
|
-
const mockDocFn = jest.fn(() => ({ set: mockSetFn }));
|
|
232
|
-
const mockCollectionFn = jest.fn(() => ({ doc: mockDocFn }));
|
|
233
|
-
mockFirestore.collection = mockCollectionFn;
|
|
234
|
-
|
|
235
|
-
await userResolverModule.resolveOrCreateUser('newuser@example.com', {
|
|
236
|
-
createIfNotExists: true,
|
|
237
|
-
defaultPlan: 'Lite',
|
|
238
|
-
});
|
|
239
|
-
|
|
240
|
-
expect(mockCollectionFn).toHaveBeenCalledWith('users');
|
|
241
|
-
expect(mockDocFn).toHaveBeenCalledWith('new-uid-456');
|
|
242
|
-
expect(mockSetFn).toHaveBeenCalledWith(
|
|
243
|
-
expect.objectContaining({
|
|
244
|
-
email: 'newuser@example.com',
|
|
245
|
-
displayName: 'newuser',
|
|
246
|
-
plan: 'Lite',
|
|
247
|
-
})
|
|
248
|
-
);
|
|
249
|
-
});
|
|
250
|
-
|
|
251
|
-
test('建立使用者時應該設定訂閱資訊', async () => {
|
|
252
|
-
mockAuth.getUserByEmail = jest.fn()
|
|
253
|
-
.mockRejectedValueOnce({ code: 'auth/user-not-found' });
|
|
254
|
-
|
|
255
|
-
mockAuth.createUser = jest.fn().mockResolvedValue({
|
|
256
|
-
uid: 'new-uid-456',
|
|
257
|
-
email: 'newuser@example.com',
|
|
258
|
-
displayName: 'newuser',
|
|
259
|
-
});
|
|
260
|
-
|
|
261
|
-
const mockSetFn = jest.fn().mockResolvedValue(undefined);
|
|
262
|
-
const mockDocFn = jest.fn(() => ({ set: mockSetFn }));
|
|
263
|
-
const mockCollectionFn = jest.fn(() => ({ doc: mockDocFn }));
|
|
264
|
-
mockFirestore.collection = mockCollectionFn;
|
|
265
|
-
|
|
266
|
-
const endDate = new Date('2026-01-01');
|
|
267
|
-
|
|
268
|
-
await userResolverModule.resolveOrCreateUser('newuser@example.com', {
|
|
269
|
-
createIfNotExists: true,
|
|
270
|
-
defaultPlan: 'Max-1',
|
|
271
|
-
subscriptionEndDate: endDate,
|
|
272
|
-
});
|
|
273
|
-
|
|
274
|
-
// 應該設定兩個集合:users 和 user_subscriptions
|
|
275
|
-
expect(mockCollectionFn).toHaveBeenCalledWith('users');
|
|
276
|
-
expect(mockCollectionFn).toHaveBeenCalledWith('user_subscriptions');
|
|
277
|
-
});
|
|
278
|
-
|
|
279
|
-
test('displayName 應該從 email 前綴產生', async () => {
|
|
280
|
-
mockAuth.getUserByEmail = jest.fn()
|
|
281
|
-
.mockRejectedValueOnce({ code: 'auth/user-not-found' });
|
|
282
|
-
|
|
283
|
-
mockAuth.createUser = jest.fn().mockResolvedValue({
|
|
284
|
-
uid: 'new-uid-456',
|
|
285
|
-
email: 'john.doe@example.com',
|
|
286
|
-
displayName: 'john.doe',
|
|
287
|
-
});
|
|
288
|
-
|
|
289
|
-
await userResolverModule.resolveOrCreateUser('john.doe@example.com', {
|
|
290
|
-
createIfNotExists: true,
|
|
291
|
-
});
|
|
292
|
-
|
|
293
|
-
expect(mockAuth.createUser).toHaveBeenCalledWith(
|
|
294
|
-
expect.objectContaining({
|
|
295
|
-
displayName: 'john.doe',
|
|
296
|
-
})
|
|
297
|
-
);
|
|
298
|
-
});
|
|
299
|
-
});
|
|
300
|
-
|
|
301
|
-
describe('batchProvisionUsers()', () => {
|
|
302
|
-
test('應該批次建立使用者', async () => {
|
|
303
|
-
const endDate = new Date('2026-01-01');
|
|
304
|
-
|
|
305
|
-
mockAuth.getUserByEmail = jest.fn()
|
|
306
|
-
.mockRejectedValueOnce({ code: 'auth/user-not-found' })
|
|
307
|
-
.mockRejectedValueOnce({ code: 'auth/user-not-found' });
|
|
308
|
-
|
|
309
|
-
mockAuth.createUser = jest.fn()
|
|
310
|
-
.mockResolvedValueOnce({ uid: 'uid-1', email: 'user1@example.com' })
|
|
311
|
-
.mockResolvedValueOnce({ uid: 'uid-2', email: 'user2@example.com' });
|
|
312
|
-
|
|
313
|
-
const results = await userResolverModule.batchProvisionUsers(
|
|
314
|
-
['user1@example.com', 'user2@example.com'],
|
|
315
|
-
'Lite',
|
|
316
|
-
endDate
|
|
317
|
-
);
|
|
318
|
-
|
|
319
|
-
expect(results).toHaveLength(2);
|
|
320
|
-
expect(results[0].success).toBe(true);
|
|
321
|
-
expect(results[0].created).toBe(true);
|
|
322
|
-
expect(results[1].success).toBe(true);
|
|
323
|
-
expect(results[1].created).toBe(true);
|
|
324
|
-
});
|
|
325
|
-
|
|
326
|
-
test('應該區別建立和更新的使用者', async () => {
|
|
327
|
-
const endDate = new Date('2026-01-01');
|
|
328
|
-
|
|
329
|
-
mockAuth.getUserByEmail = jest.fn()
|
|
330
|
-
.mockResolvedValueOnce(createMockUser({ email: 'existing@example.com' }))
|
|
331
|
-
.mockRejectedValueOnce({ code: 'auth/user-not-found' });
|
|
332
|
-
|
|
333
|
-
mockAuth.createUser = jest.fn().mockResolvedValue({
|
|
334
|
-
uid: 'new-uid',
|
|
335
|
-
email: 'new@example.com',
|
|
336
|
-
});
|
|
337
|
-
|
|
338
|
-
const mockUpdateFn = jest.fn().mockResolvedValue(undefined);
|
|
339
|
-
const mockDocFn = jest.fn(() => ({
|
|
340
|
-
set: jest.fn().mockResolvedValue(undefined),
|
|
341
|
-
get: jest.fn().mockResolvedValue({ exists: true }),
|
|
342
|
-
update: mockUpdateFn,
|
|
343
|
-
}));
|
|
344
|
-
const mockCollectionFn = jest.fn(() => ({ doc: mockDocFn }));
|
|
345
|
-
mockFirestore.collection = mockCollectionFn;
|
|
346
|
-
|
|
347
|
-
const results = await userResolverModule.batchProvisionUsers(
|
|
348
|
-
['existing@example.com', 'new@example.com'],
|
|
349
|
-
'Max-1',
|
|
350
|
-
endDate
|
|
351
|
-
);
|
|
352
|
-
|
|
353
|
-
expect(results[0].created).toBe(false); // 既有使用者
|
|
354
|
-
expect(results[1].created).toBe(true); // 新建立的使用者
|
|
355
|
-
});
|
|
356
|
-
|
|
357
|
-
test('無效的 Email 應該返回錯誤', async () => {
|
|
358
|
-
const endDate = new Date('2026-01-01');
|
|
359
|
-
|
|
360
|
-
const results = await userResolverModule.batchProvisionUsers(
|
|
361
|
-
['invalid-email', 'valid@example.com'],
|
|
362
|
-
'Lite',
|
|
363
|
-
endDate
|
|
364
|
-
);
|
|
365
|
-
|
|
366
|
-
expect(results[0].success).toBe(false);
|
|
367
|
-
expect(results[0].error).toContain('無效的 Email 格式');
|
|
368
|
-
});
|
|
369
|
-
|
|
370
|
-
test('空 Email 應該被跳過', async () => {
|
|
371
|
-
mockAuth.getUserByEmail = jest.fn();
|
|
372
|
-
mockAuth.createUser = jest.fn();
|
|
373
|
-
|
|
374
|
-
const results = await userResolverModule.batchProvisionUsers(
|
|
375
|
-
['', ' ', 'valid@example.com'],
|
|
376
|
-
'Lite',
|
|
377
|
-
new Date('2026-01-01')
|
|
378
|
-
);
|
|
379
|
-
|
|
380
|
-
expect(results[0].success).toBe(false);
|
|
381
|
-
expect(results[1].success).toBe(false);
|
|
382
|
-
});
|
|
383
|
-
|
|
384
|
-
test('應該更新既有使用者的方案和訂閱', async () => {
|
|
385
|
-
mockAuth.getUserByEmail = jest.fn()
|
|
386
|
-
.mockResolvedValue(createMockUser({ email: 'existing@example.com' }));
|
|
387
|
-
|
|
388
|
-
const mockUpdateFn = jest.fn().mockResolvedValue(undefined);
|
|
389
|
-
const mockDocFn = jest.fn(() => ({
|
|
390
|
-
get: jest.fn().mockResolvedValue({ exists: true }),
|
|
391
|
-
update: mockUpdateFn,
|
|
392
|
-
}));
|
|
393
|
-
const mockCollectionFn = jest.fn(() => ({ doc: mockDocFn }));
|
|
394
|
-
mockFirestore.collection = mockCollectionFn;
|
|
395
|
-
|
|
396
|
-
const endDate = new Date('2026-01-01');
|
|
397
|
-
|
|
398
|
-
await userResolverModule.batchProvisionUsers(
|
|
399
|
-
['existing@example.com'],
|
|
400
|
-
'Max-2',
|
|
401
|
-
endDate
|
|
402
|
-
);
|
|
403
|
-
|
|
404
|
-
// 應該更新 user_subscriptions
|
|
405
|
-
expect(mockUpdateFn).toHaveBeenCalledWith(
|
|
406
|
-
expect.objectContaining({
|
|
407
|
-
plan: 'Max-2',
|
|
408
|
-
status: 'active',
|
|
409
|
-
subscriptionEndDate: endDate,
|
|
410
|
-
})
|
|
411
|
-
);
|
|
412
|
-
});
|
|
413
|
-
|
|
414
|
-
test('應該統計成功、建立和失敗的數量', async () => {
|
|
415
|
-
mockAuth.getUserByEmail = jest.fn()
|
|
416
|
-
.mockResolvedValueOnce(createMockUser())
|
|
417
|
-
.mockRejectedValueOnce({ code: 'auth/user-not-found' });
|
|
418
|
-
|
|
419
|
-
mockAuth.createUser = jest.fn().mockResolvedValue({
|
|
420
|
-
uid: 'new-uid',
|
|
421
|
-
email: 'new@example.com',
|
|
422
|
-
});
|
|
423
|
-
|
|
424
|
-
const results = await userResolverModule.batchProvisionUsers(
|
|
425
|
-
['existing@example.com', 'new@example.com', 'invalid-email'],
|
|
426
|
-
'Lite',
|
|
427
|
-
new Date('2026-01-01')
|
|
428
|
-
);
|
|
429
|
-
|
|
430
|
-
const successful = results.filter((r) => r.success);
|
|
431
|
-
const created = results.filter((r) => r.created);
|
|
432
|
-
const failed = results.filter((r) => !r.success);
|
|
433
|
-
|
|
434
|
-
expect(successful.length).toBe(2); // 既有 + 新建立
|
|
435
|
-
expect(created.length).toBe(1); // 只有新建立
|
|
436
|
-
expect(failed.length).toBe(1); // 無效的 Email
|
|
437
|
-
});
|
|
438
|
-
});
|
|
439
|
-
});
|
|
@@ -1,242 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* 成本分析服務
|
|
3
|
-
* 計算使用者的成本、ROI、使用趨勢等商業指標
|
|
4
|
-
*/
|
|
5
|
-
|
|
6
|
-
import { TimeSeriesAnalyzer, TimeSeriesAnalysisResult } from './TimeSeriesAnalyzer.js';
|
|
7
|
-
|
|
8
|
-
export interface PricingTier {
|
|
9
|
-
plan: string;
|
|
10
|
-
rpmLimit: number;
|
|
11
|
-
tpmLimit: number;
|
|
12
|
-
rpdLimit: number;
|
|
13
|
-
monthlyCost: number;
|
|
14
|
-
costPerMToken?: number;
|
|
15
|
-
costPerRequest?: number;
|
|
16
|
-
}
|
|
17
|
-
|
|
18
|
-
export interface BillingMetrics {
|
|
19
|
-
userId: string;
|
|
20
|
-
email: string;
|
|
21
|
-
plan: string;
|
|
22
|
-
analysisDate: Date;
|
|
23
|
-
analysisDays: number;
|
|
24
|
-
|
|
25
|
-
// 使用量指標(分別統計輸入/輸出)
|
|
26
|
-
totalRequests: number;
|
|
27
|
-
totalTokens: number;
|
|
28
|
-
inputTokens: number; // 輸入 tokens
|
|
29
|
-
outputTokens: number; // 輸出 tokens
|
|
30
|
-
avgDailyTokens: number;
|
|
31
|
-
avgDailyRequests: number;
|
|
32
|
-
|
|
33
|
-
// 成本指標(分別計算輸入/輸出成本)
|
|
34
|
-
inputCost: number; // 輸入成本 ($1/1M)
|
|
35
|
-
outputCost: number; // 輸出成本 ($5/1M)
|
|
36
|
-
totalCost: number; // 總成本 = inputCost + outputCost
|
|
37
|
-
monthlyPlanCost: number;
|
|
38
|
-
estimatedMonthlyCost: number;
|
|
39
|
-
estimatedAnnualCost: number;
|
|
40
|
-
costPerToken: number;
|
|
41
|
-
costPerRequest: number;
|
|
42
|
-
|
|
43
|
-
// 使用率指標
|
|
44
|
-
tpmUtilization: number; // %
|
|
45
|
-
rpmUtilization: number; // %
|
|
46
|
-
rpdUtilization: number; // %
|
|
47
|
-
|
|
48
|
-
// 排行資訊
|
|
49
|
-
rank?: number;
|
|
50
|
-
percentileRank?: number; // 0-100
|
|
51
|
-
}
|
|
52
|
-
|
|
53
|
-
export interface BillingReport {
|
|
54
|
-
generatedAt: Date;
|
|
55
|
-
analysisPeriod: {
|
|
56
|
-
startDate: Date;
|
|
57
|
-
endDate: Date;
|
|
58
|
-
days: number;
|
|
59
|
-
};
|
|
60
|
-
totalUsers: number;
|
|
61
|
-
totalTokensAcrossUsers: number;
|
|
62
|
-
totalCostAcrossUsers: number;
|
|
63
|
-
metrics: BillingMetrics[];
|
|
64
|
-
topSpenders: BillingMetrics[];
|
|
65
|
-
costBreakdown: {
|
|
66
|
-
plan: string;
|
|
67
|
-
userCount: number;
|
|
68
|
-
totalCost: number;
|
|
69
|
-
percentage: number;
|
|
70
|
-
}[];
|
|
71
|
-
}
|
|
72
|
-
|
|
73
|
-
// Claude Haiku 4.5 定價(2025)
|
|
74
|
-
// 來源:https://docs.claude.com/en/docs/about-claude/pricing
|
|
75
|
-
export const COST_PER_MILLION_INPUT_TOKENS = 1; // $1 / 1M input tokens
|
|
76
|
-
export const COST_PER_MILLION_OUTPUT_TOKENS = 5; // $5 / 1M output tokens
|
|
77
|
-
|
|
78
|
-
export const PRICING_TIERS: Record<string, PricingTier> = {
|
|
79
|
-
'Lite': {
|
|
80
|
-
plan: 'Lite',
|
|
81
|
-
rpmLimit: 6,
|
|
82
|
-
tpmLimit: 200000,
|
|
83
|
-
rpdLimit: 32,
|
|
84
|
-
monthlyCost: 0, // 固定費用為 0,成本完全基於 Token 使用量
|
|
85
|
-
costPerMToken: COST_PER_MILLION_OUTPUT_TOKENS, // 預設使用輸出價格
|
|
86
|
-
costPerRequest: 0
|
|
87
|
-
},
|
|
88
|
-
'Max-1': {
|
|
89
|
-
plan: 'Max-1',
|
|
90
|
-
rpmLimit: 6,
|
|
91
|
-
tpmLimit: 600000,
|
|
92
|
-
rpdLimit: 111,
|
|
93
|
-
monthlyCost: 0, // 固定費用為 0,成本完全基於 Token 使用量
|
|
94
|
-
costPerMToken: COST_PER_MILLION_OUTPUT_TOKENS,
|
|
95
|
-
costPerRequest: 0
|
|
96
|
-
},
|
|
97
|
-
'Max-2': {
|
|
98
|
-
plan: 'Max-2',
|
|
99
|
-
rpmLimit: 6,
|
|
100
|
-
tpmLimit: 600000,
|
|
101
|
-
rpdLimit: 450,
|
|
102
|
-
monthlyCost: 0, // 固定費用為 0,成本完全基於 Token 使用量
|
|
103
|
-
costPerMToken: COST_PER_MILLION_OUTPUT_TOKENS,
|
|
104
|
-
costPerRequest: 0
|
|
105
|
-
}
|
|
106
|
-
};
|
|
107
|
-
|
|
108
|
-
export class BillingAnalyticsService {
|
|
109
|
-
private timeSeriesAnalyzer = new TimeSeriesAnalyzer();
|
|
110
|
-
|
|
111
|
-
/**
|
|
112
|
-
* 計算單一使用者的成本指標
|
|
113
|
-
* 成本純粹基於實際 Token 使用量:
|
|
114
|
-
* - 輸入 tokens: $1 / 1M tokens
|
|
115
|
-
* - 輸出 tokens: $5 / 1M tokens
|
|
116
|
-
*/
|
|
117
|
-
async calculateUserMetrics(
|
|
118
|
-
userId: string,
|
|
119
|
-
userEmail: string,
|
|
120
|
-
plan: string = 'Lite',
|
|
121
|
-
days: number = 30
|
|
122
|
-
): Promise<BillingMetrics> {
|
|
123
|
-
const analysis = await this.timeSeriesAnalyzer.analyzeUser(userId, userEmail, days);
|
|
124
|
-
const pricing = PRICING_TIERS[plan] || PRICING_TIERS['Lite'];
|
|
125
|
-
|
|
126
|
-
const totalRequests = analysis.totalRequests;
|
|
127
|
-
const totalTokens = analysis.totalTokens;
|
|
128
|
-
const inputTokens = analysis.inputTokens;
|
|
129
|
-
const outputTokens = analysis.outputTokens;
|
|
130
|
-
const dailyCount = days || 1;
|
|
131
|
-
|
|
132
|
-
// 計算成本 - 分別計算輸入/輸出 Token 成本
|
|
133
|
-
// 公式:inputCost = (inputTokens / 1M) * $1, outputCost = (outputTokens / 1M) * $5
|
|
134
|
-
const inputCost = (inputTokens / 1000000) * COST_PER_MILLION_INPUT_TOKENS;
|
|
135
|
-
const outputCost = (outputTokens / 1000000) * COST_PER_MILLION_OUTPUT_TOKENS;
|
|
136
|
-
const totalCost = inputCost + outputCost;
|
|
137
|
-
|
|
138
|
-
const avgDailyTokens = totalTokens / dailyCount;
|
|
139
|
-
const avgDailyCost = totalCost / dailyCount;
|
|
140
|
-
|
|
141
|
-
// 預估月度成本(基於分析期間的日均用量)
|
|
142
|
-
const estimatedMonthlyCost = avgDailyCost * 30;
|
|
143
|
-
const estimatedAnnualCost = estimatedMonthlyCost * 12;
|
|
144
|
-
|
|
145
|
-
// 計算使用率
|
|
146
|
-
const tpmUtilization = pricing.tpmLimit > 0 ? (avgDailyTokens / pricing.tpmLimit) * 100 : 0;
|
|
147
|
-
const avgDailyRequests = totalRequests / dailyCount;
|
|
148
|
-
const rpmUtilization = pricing.rpmLimit > 0 ? (avgDailyRequests / pricing.rpmLimit) * 100 : 0;
|
|
149
|
-
const rpdUtilization = pricing.rpdLimit > 0 ? (avgDailyRequests / pricing.rpdLimit) * 100 : 0;
|
|
150
|
-
|
|
151
|
-
// 每 Token / 每 Request 成本
|
|
152
|
-
const costPerToken = totalTokens > 0 ? totalCost / totalTokens : 0;
|
|
153
|
-
const costPerRequest = totalRequests > 0 ? totalCost / totalRequests : 0;
|
|
154
|
-
|
|
155
|
-
return {
|
|
156
|
-
userId,
|
|
157
|
-
email: userEmail,
|
|
158
|
-
plan,
|
|
159
|
-
analysisDate: new Date(),
|
|
160
|
-
analysisDays: days,
|
|
161
|
-
totalRequests,
|
|
162
|
-
totalTokens,
|
|
163
|
-
inputTokens,
|
|
164
|
-
outputTokens,
|
|
165
|
-
avgDailyTokens,
|
|
166
|
-
avgDailyRequests,
|
|
167
|
-
inputCost,
|
|
168
|
-
outputCost,
|
|
169
|
-
totalCost,
|
|
170
|
-
monthlyPlanCost: 0, // 無固定月費,成本基於使用量
|
|
171
|
-
estimatedMonthlyCost,
|
|
172
|
-
estimatedAnnualCost,
|
|
173
|
-
costPerToken,
|
|
174
|
-
costPerRequest,
|
|
175
|
-
tpmUtilization,
|
|
176
|
-
rpmUtilization,
|
|
177
|
-
rpdUtilization
|
|
178
|
-
};
|
|
179
|
-
}
|
|
180
|
-
|
|
181
|
-
/**
|
|
182
|
-
* 生成完整的成本分析報告
|
|
183
|
-
*/
|
|
184
|
-
async generateBillingReport(
|
|
185
|
-
users: Array<{ id: string; email: string; plan: string }>,
|
|
186
|
-
days: number = 30
|
|
187
|
-
): Promise<BillingReport> {
|
|
188
|
-
const metrics: BillingMetrics[] = [];
|
|
189
|
-
|
|
190
|
-
for (const user of users) {
|
|
191
|
-
try {
|
|
192
|
-
const metric = await this.calculateUserMetrics(user.id, user.email, user.plan, days);
|
|
193
|
-
metrics.push(metric);
|
|
194
|
-
} catch (error) {
|
|
195
|
-
console.warn(`⚠️ 無法計算 ${user.email} 的成本指標`);
|
|
196
|
-
}
|
|
197
|
-
}
|
|
198
|
-
|
|
199
|
-
// 添加排行資訊
|
|
200
|
-
const sortedBySpend = [...metrics].sort((a, b) => b.estimatedMonthlyCost - a.estimatedMonthlyCost);
|
|
201
|
-
metrics.forEach((metric, index) => {
|
|
202
|
-
metric.rank = sortedBySpend.findIndex(m => m.userId === metric.userId) + 1;
|
|
203
|
-
metric.percentileRank = (metric.rank / metrics.length) * 100;
|
|
204
|
-
});
|
|
205
|
-
|
|
206
|
-
// 計算總成本
|
|
207
|
-
const totalCost = metrics.reduce((sum, m) => sum + m.estimatedMonthlyCost, 0);
|
|
208
|
-
|
|
209
|
-
// 按方案分組
|
|
210
|
-
const planBreakdown = new Map<string, { count: number; cost: number }>();
|
|
211
|
-
metrics.forEach(metric => {
|
|
212
|
-
const existing = planBreakdown.get(metric.plan) || { count: 0, cost: 0 };
|
|
213
|
-
existing.count++;
|
|
214
|
-
existing.cost += metric.estimatedMonthlyCost;
|
|
215
|
-
planBreakdown.set(metric.plan, existing);
|
|
216
|
-
});
|
|
217
|
-
|
|
218
|
-
const costBreakdown = Array.from(planBreakdown.entries())
|
|
219
|
-
.map(([plan, data]) => ({
|
|
220
|
-
plan,
|
|
221
|
-
userCount: data.count,
|
|
222
|
-
totalCost: data.cost,
|
|
223
|
-
percentage: (data.cost / totalCost) * 100
|
|
224
|
-
}))
|
|
225
|
-
.sort((a, b) => b.totalCost - a.totalCost);
|
|
226
|
-
|
|
227
|
-
return {
|
|
228
|
-
generatedAt: new Date(),
|
|
229
|
-
analysisPeriod: {
|
|
230
|
-
startDate: new Date(Date.now() - days * 24 * 60 * 60 * 1000),
|
|
231
|
-
endDate: new Date(),
|
|
232
|
-
days
|
|
233
|
-
},
|
|
234
|
-
totalUsers: metrics.length,
|
|
235
|
-
totalTokensAcrossUsers: metrics.reduce((sum, m) => sum + m.totalTokens, 0),
|
|
236
|
-
totalCostAcrossUsers: totalCost,
|
|
237
|
-
metrics,
|
|
238
|
-
topSpenders: sortedBySpend.slice(0, 10),
|
|
239
|
-
costBreakdown
|
|
240
|
-
};
|
|
241
|
-
}
|
|
242
|
-
}
|