@ouro.bot/cli 0.1.0-alpha.5 → 0.1.0-alpha.500
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/README.md +226 -183
- package/SerpentGuide.ouro/agent.json +82 -0
- package/SerpentGuide.ouro/psyche/SOUL.md +25 -0
- package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/monty.md +2 -2
- package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-serpent.md +1 -1
- package/assets/ouroboros.png +0 -0
- package/changelog.json +3418 -0
- package/dist/arc/attention-types.js +8 -0
- package/dist/arc/cares.js +140 -0
- package/dist/arc/episodes.js +117 -0
- package/dist/arc/intentions.js +133 -0
- package/dist/arc/json-store.js +117 -0
- package/dist/arc/obligations.js +237 -0
- package/dist/arc/packets.js +193 -0
- package/dist/arc/presence.js +185 -0
- package/dist/arc/task-lifecycle.js +65 -0
- package/dist/heart/active-work.js +989 -0
- package/dist/heart/agent-entry.js +58 -3
- package/dist/heart/attachments/image-normalize.js +194 -0
- package/dist/heart/attachments/materialize.js +97 -0
- package/dist/heart/attachments/originals.js +88 -0
- package/dist/heart/attachments/render.js +29 -0
- package/dist/heart/attachments/sources/adapter.js +2 -0
- package/dist/heart/attachments/sources/bluebubbles.js +156 -0
- package/dist/heart/attachments/sources/cli-local-file.js +78 -0
- package/dist/heart/attachments/sources/index.js +16 -0
- package/dist/heart/attachments/store.js +103 -0
- package/dist/heart/attachments/types.js +93 -0
- package/dist/heart/auth/auth-flow.js +426 -0
- package/dist/heart/background-operations.js +281 -0
- package/dist/heart/bridges/manager.js +358 -0
- package/dist/heart/bridges/state-machine.js +135 -0
- package/dist/heart/bridges/store.js +123 -0
- package/dist/heart/bundle-state.js +168 -0
- package/dist/heart/commitments.js +111 -0
- package/dist/heart/config-registry.js +304 -0
- package/dist/heart/config.js +193 -130
- package/dist/heart/core.js +1010 -261
- package/dist/heart/cross-chat-delivery.js +131 -0
- package/dist/heart/daemon/agent-config-check.js +490 -0
- package/dist/heart/daemon/agent-discovery.js +157 -0
- package/dist/heart/daemon/agent-service.js +360 -0
- package/dist/heart/daemon/agentic-repair.js +216 -0
- package/dist/heart/daemon/bluebubbles-health-diagnostics.js +122 -0
- package/dist/heart/daemon/cadence.js +70 -0
- package/dist/heart/daemon/cli-defaults.js +640 -0
- package/dist/heart/daemon/cli-exec.js +7239 -0
- package/dist/heart/daemon/cli-help.js +493 -0
- package/dist/heart/daemon/cli-parse.js +1533 -0
- package/dist/heart/daemon/cli-render-doctor.js +57 -0
- package/dist/heart/daemon/cli-render.js +561 -0
- package/dist/heart/daemon/cli-types.js +8 -0
- package/dist/heart/daemon/connect-bay.js +323 -0
- package/dist/heart/daemon/daemon-cli.js +30 -697
- package/dist/heart/daemon/daemon-entry.js +359 -8
- package/dist/heart/daemon/daemon-health.js +141 -0
- package/dist/heart/daemon/daemon-runtime-sync.js +268 -0
- package/dist/heart/daemon/daemon-tombstone.js +236 -0
- package/dist/heart/daemon/daemon.js +813 -19
- package/dist/heart/daemon/dns-workflow.js +394 -0
- package/dist/heart/daemon/doctor-types.js +8 -0
- package/dist/heart/daemon/doctor.js +615 -0
- package/dist/heart/daemon/health-monitor.js +92 -1
- package/dist/heart/daemon/hooks/agent-config-v2.js +33 -0
- package/dist/heart/daemon/hooks/bundle-meta.js +206 -0
- package/dist/heart/daemon/http-health-probe.js +80 -0
- package/dist/heart/daemon/human-command-screens.js +234 -0
- package/dist/heart/daemon/human-readiness.js +114 -0
- package/dist/heart/daemon/inner-status.js +89 -0
- package/dist/heart/daemon/interactive-repair.js +394 -0
- package/dist/heart/daemon/launchd.js +171 -0
- package/dist/heart/daemon/log-tailer.js +82 -12
- package/dist/heart/daemon/logs-prune.js +110 -0
- package/dist/heart/daemon/message-router.js +17 -8
- package/dist/heart/daemon/os-cron-deps.js +134 -0
- package/dist/heart/daemon/ouro-bot-entry.js +4 -2
- package/dist/heart/daemon/ouro-entry.js +3 -1
- package/dist/heart/daemon/process-manager.js +215 -1
- package/dist/heart/daemon/provider-discovery.js +137 -0
- package/dist/heart/daemon/provider-ping-progress.js +83 -0
- package/dist/heart/daemon/pulse.js +475 -0
- package/dist/heart/daemon/readiness-repair.js +365 -0
- package/dist/heart/daemon/run-hooks.js +39 -0
- package/dist/heart/daemon/runtime-logging.js +67 -16
- package/dist/heart/daemon/runtime-metadata.js +191 -0
- package/dist/heart/daemon/runtime-mode.js +67 -0
- package/dist/heart/daemon/safe-mode.js +161 -0
- package/dist/heart/daemon/sense-manager.js +431 -0
- package/dist/heart/daemon/session-id-resolver.js +131 -0
- package/dist/heart/daemon/skill-management-installer.js +94 -0
- package/dist/heart/daemon/socket-client.js +307 -0
- package/dist/heart/daemon/stale-bundle-prune.js +96 -0
- package/dist/heart/daemon/startup-tui.js +264 -0
- package/dist/heart/daemon/task-scheduler.js +3 -25
- package/dist/heart/daemon/terminal-ui.js +499 -0
- package/dist/heart/daemon/thoughts.js +524 -0
- package/dist/heart/daemon/up-progress.js +366 -0
- package/dist/heart/daemon/vault-items.js +56 -0
- package/dist/heart/delegation.js +62 -0
- package/dist/heart/habits/habit-migration.js +189 -0
- package/dist/heart/habits/habit-parser.js +140 -0
- package/dist/heart/habits/habit-runtime-state.js +100 -0
- package/dist/heart/habits/habit-scheduler.js +372 -0
- package/dist/heart/{daemon → hatch}/hatch-animation.js +10 -3
- package/dist/heart/{daemon → hatch}/hatch-flow.js +54 -136
- package/dist/heart/{daemon → hatch}/hatch-specialist.js +3 -3
- package/dist/heart/hatch/specialist-orchestrator.js +129 -0
- package/dist/heart/hatch/specialist-prompt.js +102 -0
- package/dist/heart/hatch/specialist-tools.js +306 -0
- package/dist/heart/identity.js +274 -61
- package/dist/heart/kept-notes.js +357 -0
- package/dist/heart/kicks.js +2 -20
- package/dist/heart/machine-identity.js +161 -0
- package/dist/heart/mail-import-discovery.js +353 -0
- package/dist/heart/mcp/mcp-server.js +653 -0
- package/dist/heart/migrate-config.js +100 -0
- package/dist/heart/model-capabilities.js +59 -0
- package/dist/heart/outlook/outlook-http-hooks.js +66 -0
- package/dist/heart/outlook/outlook-http-response.js +7 -0
- package/dist/heart/outlook/outlook-http-routes.js +244 -0
- package/dist/heart/outlook/outlook-http-static.js +103 -0
- package/dist/heart/outlook/outlook-http-transport.js +116 -0
- package/dist/heart/outlook/outlook-http.js +99 -0
- package/dist/heart/outlook/outlook-read.js +31 -0
- package/dist/heart/outlook/outlook-types.js +27 -0
- package/dist/heart/outlook/outlook-view.js +195 -0
- package/dist/heart/outlook/readers/agent-machine.js +382 -0
- package/dist/heart/outlook/readers/continuity-readers.js +336 -0
- package/dist/heart/outlook/readers/mail.js +362 -0
- package/dist/heart/outlook/readers/runtime-readers.js +644 -0
- package/dist/heart/outlook/readers/sessions.js +232 -0
- package/dist/heart/outlook/readers/shared.js +111 -0
- package/dist/heart/platform.js +81 -0
- package/dist/heart/progress-story.js +42 -0
- package/dist/heart/provider-attempt.js +134 -0
- package/dist/heart/provider-binding-resolver.js +255 -0
- package/dist/heart/provider-credentials.js +424 -0
- package/dist/heart/provider-failover.js +301 -0
- package/dist/heart/provider-models.js +81 -0
- package/dist/heart/provider-ping.js +262 -0
- package/dist/heart/provider-state.js +216 -0
- package/dist/heart/provider-visibility.js +188 -0
- package/dist/heart/providers/anthropic-token.js +131 -0
- package/dist/heart/providers/anthropic.js +202 -50
- package/dist/heart/providers/azure.js +104 -13
- package/dist/heart/providers/error-classification.js +63 -0
- package/dist/heart/providers/github-copilot.js +145 -0
- package/dist/heart/providers/minimax-vlm.js +189 -0
- package/dist/heart/providers/minimax.js +29 -7
- package/dist/heart/providers/openai-codex.js +63 -39
- package/dist/heart/runtime-capability-check.js +170 -0
- package/dist/heart/runtime-credentials.js +260 -0
- package/dist/heart/sense-truth.js +68 -0
- package/dist/heart/session-activity.js +190 -0
- package/dist/heart/session-events.js +1089 -0
- package/dist/heart/session-playback-cli-main.js +5 -0
- package/dist/heart/session-playback-cli.js +36 -0
- package/dist/heart/session-playback.js +231 -0
- package/dist/heart/session-transcript.js +167 -0
- package/dist/heart/start-of-turn-packet.js +345 -0
- package/dist/heart/streaming.js +129 -34
- package/dist/heart/sync.js +332 -0
- package/dist/heart/target-resolution.js +127 -0
- package/dist/heart/tempo.js +93 -0
- package/dist/heart/temporal-view.js +41 -0
- package/dist/heart/tool-activity-callbacks.js +36 -0
- package/dist/heart/tool-description.js +135 -0
- package/dist/heart/tool-friction.js +55 -0
- package/dist/heart/tool-loop.js +200 -0
- package/dist/heart/turn-context.js +372 -0
- package/dist/heart/turn-coordinator.js +28 -0
- package/dist/heart/versioning/ouro-bot-global-installer.js +128 -0
- package/dist/heart/{daemon → versioning}/ouro-bot-wrapper.js +1 -1
- package/dist/heart/versioning/ouro-path-installer.js +425 -0
- package/dist/heart/{daemon → versioning}/ouro-uti.js +11 -2
- package/dist/heart/versioning/ouro-version-manager.js +295 -0
- package/dist/heart/versioning/staged-restart.js +146 -0
- package/dist/heart/versioning/update-checker.js +115 -0
- package/dist/heart/versioning/update-hooks.js +142 -0
- package/dist/heart/versioning/wrapper-publish-guard.js +86 -0
- package/dist/mailroom/attention.js +167 -0
- package/dist/mailroom/autonomy.js +209 -0
- package/dist/mailroom/blob-store.js +606 -0
- package/dist/mailroom/core.js +672 -0
- package/dist/mailroom/entry.js +160 -0
- package/dist/mailroom/file-store.js +426 -0
- package/dist/mailroom/mbox-import.js +382 -0
- package/dist/mailroom/outbound.js +380 -0
- package/dist/mailroom/policy.js +263 -0
- package/dist/mailroom/reader.js +219 -0
- package/dist/mailroom/search-cache.js +182 -0
- package/dist/mailroom/search-relevance.js +319 -0
- package/dist/mailroom/smtp-ingress.js +176 -0
- package/dist/mailroom/source-state.js +176 -0
- package/dist/mailroom/thread.js +109 -0
- package/dist/mailroom/travel-extract.js +89 -0
- package/dist/mind/bundle-manifest.js +77 -1
- package/dist/mind/context.js +173 -94
- package/dist/mind/diary-integrity.js +60 -0
- package/dist/mind/{memory.js → diary.js} +84 -96
- package/dist/mind/embedding-provider.js +60 -0
- package/dist/mind/file-state.js +179 -0
- package/dist/mind/first-impressions.js +16 -2
- package/dist/mind/friends/channel.js +73 -0
- package/dist/mind/friends/group-context.js +144 -0
- package/dist/mind/friends/resolver.js +54 -2
- package/dist/mind/friends/store-file.js +58 -3
- package/dist/mind/friends/trust-explanation.js +74 -0
- package/dist/mind/friends/types.js +10 -2
- package/dist/mind/journal-index.js +161 -0
- package/dist/mind/note-search.js +268 -0
- package/dist/mind/obligation-steering.js +221 -0
- package/dist/mind/pending.js +76 -9
- package/dist/mind/phrases.js +1 -0
- package/dist/mind/prompt-refresh.js +3 -2
- package/dist/mind/prompt.js +1144 -117
- package/dist/mind/provenance-trust.js +26 -0
- package/dist/mind/scrutiny.js +173 -0
- package/dist/mind/token-estimate.js +8 -12
- package/dist/nerves/cli-logging.js +22 -3
- package/dist/nerves/coverage/audit-rules.js +15 -6
- package/dist/nerves/coverage/audit.js +28 -2
- package/dist/nerves/coverage/cli.js +1 -1
- package/dist/nerves/coverage/contract.js +5 -5
- package/dist/nerves/coverage/file-completeness.js +101 -5
- package/dist/nerves/coverage/run-artifacts.js +1 -1
- package/dist/nerves/event-buffer.js +111 -0
- package/dist/nerves/index.js +224 -4
- package/dist/nerves/observation.js +20 -0
- package/dist/nerves/redact.js +79 -0
- package/dist/nerves/runtime.js +5 -1
- package/dist/outlook-ui/assets/index-BPr5vNuM.css +1 -0
- package/dist/outlook-ui/assets/index-Cm51CY9W.js +61 -0
- package/dist/outlook-ui/index.html +15 -0
- package/dist/repertoire/ado-client.js +17 -56
- package/dist/repertoire/ado-semantic.js +11 -10
- package/dist/repertoire/api-client.js +97 -0
- package/dist/repertoire/bitwarden-store.js +774 -0
- package/dist/repertoire/bundle-templates.js +72 -0
- package/dist/repertoire/bw-installer.js +180 -0
- package/dist/repertoire/coding/codex-jsonl.js +64 -0
- package/dist/repertoire/coding/context-pack.js +330 -0
- package/dist/repertoire/coding/feedback.js +301 -0
- package/dist/repertoire/coding/index.js +4 -1
- package/dist/repertoire/coding/manager.js +220 -13
- package/dist/repertoire/coding/spawner.js +58 -12
- package/dist/repertoire/coding/tools.js +209 -7
- package/dist/repertoire/commerce-errors.js +109 -0
- package/dist/repertoire/commerce-self-test.js +156 -0
- package/dist/repertoire/credential-access.js +111 -0
- package/dist/repertoire/data/ado-endpoints.json +188 -0
- package/dist/repertoire/duffel-client.js +185 -0
- package/dist/repertoire/github-client.js +14 -55
- package/dist/repertoire/graph-client.js +11 -52
- package/dist/repertoire/guardrails.js +396 -0
- package/dist/repertoire/mcp-client.js +255 -0
- package/dist/repertoire/mcp-manager.js +305 -0
- package/dist/repertoire/mcp-tools.js +63 -0
- package/dist/repertoire/shell-sessions.js +133 -0
- package/dist/repertoire/skills.js +15 -24
- package/dist/repertoire/stripe-client.js +131 -0
- package/dist/repertoire/tasks/board.js +43 -5
- package/dist/repertoire/tasks/fix.js +182 -0
- package/dist/repertoire/tasks/index.js +39 -13
- package/dist/repertoire/tasks/lifecycle.js +2 -2
- package/dist/repertoire/tasks/parser.js +3 -2
- package/dist/repertoire/tasks/scanner.js +194 -37
- package/dist/repertoire/tasks/transitions.js +16 -79
- package/dist/repertoire/tool-results.js +29 -0
- package/dist/repertoire/tools-attachments.js +317 -0
- package/dist/repertoire/tools-base.js +49 -707
- package/dist/repertoire/tools-bluebubbles.js +94 -0
- package/dist/repertoire/tools-bridge.js +141 -0
- package/dist/repertoire/tools-bundle.js +984 -0
- package/dist/repertoire/tools-config.js +185 -0
- package/dist/repertoire/tools-continuity.js +248 -0
- package/dist/repertoire/tools-credential.js +381 -0
- package/dist/repertoire/tools-files.js +342 -0
- package/dist/repertoire/tools-flight.js +224 -0
- package/dist/repertoire/tools-flow.js +105 -0
- package/dist/repertoire/tools-github.js +1 -7
- package/dist/repertoire/tools-mail.js +1377 -0
- package/dist/repertoire/tools-notes.js +376 -0
- package/dist/repertoire/tools-session.js +749 -0
- package/dist/repertoire/tools-shell.js +120 -0
- package/dist/repertoire/tools-stripe.js +180 -0
- package/dist/repertoire/tools-surface.js +243 -0
- package/dist/repertoire/tools-teams.js +64 -61
- package/dist/repertoire/tools-travel.js +125 -0
- package/dist/repertoire/tools-trip.js +356 -0
- package/dist/repertoire/tools-user-profile.js +144 -0
- package/dist/repertoire/tools-vault.js +40 -0
- package/dist/repertoire/tools.js +149 -98
- package/dist/repertoire/travel-api-client.js +360 -0
- package/dist/repertoire/user-profile.js +131 -0
- package/dist/repertoire/vault-setup.js +246 -0
- package/dist/repertoire/vault-unlock.js +561 -0
- package/dist/scripts/claude-code-hook.js +41 -0
- package/dist/scripts/claude-code-stop-hook.js +47 -0
- package/dist/senses/attention-queue.js +116 -0
- package/dist/senses/bluebubbles/attachment-cache.js +53 -0
- package/dist/senses/bluebubbles/attachment-download.js +137 -0
- package/dist/senses/bluebubbles/client.js +685 -0
- package/dist/senses/bluebubbles/entry.js +73 -0
- package/dist/senses/bluebubbles/inbound-log.js +126 -0
- package/dist/senses/bluebubbles/index.js +1881 -0
- package/dist/senses/bluebubbles/media.js +389 -0
- package/dist/senses/bluebubbles/model.js +282 -0
- package/dist/senses/bluebubbles/mutation-log.js +116 -0
- package/dist/senses/bluebubbles/processed-log.js +111 -0
- package/dist/senses/bluebubbles/replay.js +129 -0
- package/dist/senses/bluebubbles/runtime-state.js +109 -0
- package/dist/senses/bluebubbles/session-cleanup.js +72 -0
- package/dist/senses/cli/bracketed-paste.js +82 -0
- package/dist/senses/cli/image-paste.js +287 -0
- package/dist/senses/cli/image-ref-navigation.js +75 -0
- package/dist/senses/cli/ink-app.js +156 -0
- package/dist/senses/cli/inline-diff.js +64 -0
- package/dist/senses/cli/input-keys.js +174 -0
- package/dist/senses/cli/kill-ring.js +86 -0
- package/dist/senses/cli/message-list.js +51 -0
- package/dist/senses/cli/ouro-tui.js +605 -0
- package/dist/senses/cli/spinner-imperative.js +135 -0
- package/dist/senses/cli/spinner.js +101 -0
- package/dist/senses/cli/status-line.js +60 -0
- package/dist/senses/cli/streaming-markdown.js +526 -0
- package/dist/senses/cli/tool-display.js +83 -0
- package/dist/senses/cli/tool-render.js +85 -0
- package/dist/senses/cli/tui-store.js +240 -0
- package/dist/senses/cli/virtual-list.js +35 -0
- package/dist/senses/cli-entry.js +60 -8
- package/dist/senses/cli-layout.js +187 -0
- package/dist/senses/cli.js +768 -264
- package/dist/senses/commands.js +66 -3
- package/dist/senses/continuity.js +94 -0
- package/dist/senses/habit-turn-message.js +108 -0
- package/dist/senses/inner-dialog-worker.js +199 -16
- package/dist/senses/inner-dialog.js +640 -91
- package/dist/senses/mail-entry.js +66 -0
- package/dist/senses/mail.js +379 -0
- package/dist/senses/pipeline.js +665 -0
- package/dist/senses/proactive-content-guard.js +51 -0
- package/dist/senses/shared-turn.js +248 -0
- package/dist/senses/surface-tool.js +68 -0
- package/dist/senses/teams-entry.js +60 -8
- package/dist/senses/teams.js +844 -197
- package/dist/senses/trust-gate.js +207 -2
- package/dist/trips/core.js +138 -0
- package/dist/trips/store.js +146 -0
- package/package.json +47 -6
- package/skills/agent-commerce.md +106 -0
- package/skills/browser-navigation.md +117 -0
- package/skills/commerce-setup-guide.md +116 -0
- package/skills/commerce-setup.md +84 -0
- package/skills/configure-dev-tools.md +101 -0
- package/skills/travel-planning.md +138 -0
- package/AdoptionSpecialist.ouro/agent.json +0 -20
- package/AdoptionSpecialist.ouro/psyche/SOUL.md +0 -22
- package/dist/heart/daemon/specialist-orchestrator.js +0 -160
- package/dist/heart/daemon/specialist-prompt.js +0 -40
- package/dist/heart/daemon/specialist-session.js +0 -142
- package/dist/heart/daemon/specialist-tools.js +0 -128
- package/dist/heart/daemon/subagent-installer.js +0 -125
- package/dist/inner-worker-entry.js +0 -4
- package/dist/mind/associative-recall.js +0 -197
- package/subagents/README.md +0 -73
- package/subagents/work-doer.md +0 -233
- package/subagents/work-merger.md +0 -624
- package/subagents/work-planner.md +0 -373
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/basilisk.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jafar.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/jormungandr.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/kaa.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/medusa.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/nagini.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/ouroboros.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/python.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/quetzalcoatl.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/sir-hiss.md +0 -0
- /package/{AdoptionSpecialist.ouro → SerpentGuide.ouro}/psyche/identities/the-snake.md +0 -0
package/dist/repertoire/tools.js
CHANGED
|
@@ -1,29 +1,52 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.
|
|
3
|
+
exports.surfaceToolDef = exports.restTool = exports.ponderTool = exports.observeTool = exports.settleTool = exports.tools = void 0;
|
|
4
|
+
exports.resetMcpDefinitions = resetMcpDefinitions;
|
|
4
5
|
exports.getToolsForChannel = getToolsForChannel;
|
|
5
|
-
exports.isConfirmationRequired = isConfirmationRequired;
|
|
6
6
|
exports.execTool = execTool;
|
|
7
7
|
exports.summarizeArgs = summarizeArgs;
|
|
8
|
+
exports.buildToolResultSummary = buildToolResultSummary;
|
|
8
9
|
const tools_base_1 = require("./tools-base");
|
|
9
10
|
const tools_teams_1 = require("./tools-teams");
|
|
11
|
+
const tools_bluebubbles_1 = require("./tools-bluebubbles");
|
|
10
12
|
const ado_semantic_1 = require("./ado-semantic");
|
|
11
13
|
const tools_github_1 = require("./tools-github");
|
|
14
|
+
const tools_bundle_1 = require("./tools-bundle");
|
|
12
15
|
const runtime_1 = require("../nerves/runtime");
|
|
16
|
+
const guardrails_1 = require("./guardrails");
|
|
17
|
+
const identity_1 = require("../heart/identity");
|
|
18
|
+
const tools_surface_1 = require("./tools-surface");
|
|
19
|
+
const mcp_tools_1 = require("./mcp-tools");
|
|
20
|
+
function safeGetAgentRoot() {
|
|
21
|
+
try {
|
|
22
|
+
return (0, identity_1.getAgentRoot)();
|
|
23
|
+
}
|
|
24
|
+
catch {
|
|
25
|
+
return undefined;
|
|
26
|
+
}
|
|
27
|
+
}
|
|
13
28
|
// Re-export types and constants used by the rest of the codebase
|
|
14
29
|
var tools_base_2 = require("./tools-base");
|
|
15
30
|
Object.defineProperty(exports, "tools", { enumerable: true, get: function () { return tools_base_2.tools; } });
|
|
16
|
-
Object.defineProperty(exports, "
|
|
17
|
-
|
|
18
|
-
Object.defineProperty(exports, "
|
|
31
|
+
Object.defineProperty(exports, "settleTool", { enumerable: true, get: function () { return tools_base_2.settleTool; } });
|
|
32
|
+
Object.defineProperty(exports, "observeTool", { enumerable: true, get: function () { return tools_base_2.observeTool; } });
|
|
33
|
+
Object.defineProperty(exports, "ponderTool", { enumerable: true, get: function () { return tools_base_2.ponderTool; } });
|
|
34
|
+
Object.defineProperty(exports, "restTool", { enumerable: true, get: function () { return tools_base_2.restTool; } });
|
|
35
|
+
// Re-export surface tool schema for consumers (e.g. heart/core.ts)
|
|
36
|
+
var tools_surface_2 = require("./tools-surface");
|
|
37
|
+
Object.defineProperty(exports, "surfaceToolDef", { enumerable: true, get: function () { return tools_surface_2.surfaceToolDef; } });
|
|
19
38
|
// All tool definitions in a single registry
|
|
20
|
-
const allDefinitions = [...tools_base_1.baseToolDefinitions, ...tools_teams_1.teamsToolDefinitions, ...ado_semantic_1.adoSemanticToolDefinitions, ...tools_github_1.githubToolDefinitions];
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
39
|
+
const allDefinitions = [...tools_base_1.baseToolDefinitions, ...tools_bluebubbles_1.bluebubblesToolDefinitions, ...tools_teams_1.teamsToolDefinitions, ...ado_semantic_1.adoSemanticToolDefinitions, ...tools_github_1.githubToolDefinitions, ...tools_bundle_1.bundleToolDefinitions, tools_surface_1.surfaceToolDefinition];
|
|
40
|
+
// MCP tool definitions — populated each time getToolsForChannel() is called with an mcpManager.
|
|
41
|
+
// Kept separate from allDefinitions so execTool can find them.
|
|
42
|
+
let mcpDefinitions = [];
|
|
43
|
+
/** Exported for testing — reset the MCP definitions cache. */
|
|
44
|
+
function resetMcpDefinitions() {
|
|
45
|
+
mcpDefinitions = [];
|
|
46
|
+
}
|
|
47
|
+
function baseToolsForCapabilities() {
|
|
48
|
+
// Use baseToolDefinitions at call time so dynamically-added tools are included
|
|
49
|
+
return tools_base_1.baseToolDefinitions.map((d) => d.tool);
|
|
27
50
|
}
|
|
28
51
|
// Apply a single tool preference to a tool schema, returning a new object.
|
|
29
52
|
function applyPreference(tool, pref) {
|
|
@@ -35,48 +58,79 @@ function applyPreference(tool, pref) {
|
|
|
35
58
|
},
|
|
36
59
|
};
|
|
37
60
|
}
|
|
61
|
+
// Filter out tools whose requiredCapability is not in the provider's capability set.
|
|
62
|
+
// Uses baseToolDefinitions at call time so dynamically-added tools are included.
|
|
63
|
+
// Only base tools can have requiredCapability (integration tools do not).
|
|
64
|
+
function filterByCapability(toolList, providerCapabilities) {
|
|
65
|
+
return toolList.filter((tool) => {
|
|
66
|
+
const def = tools_base_1.baseToolDefinitions.find((d) => d.tool.function.name === tool.function.name);
|
|
67
|
+
if (!def?.requiredCapability)
|
|
68
|
+
return true;
|
|
69
|
+
return providerCapabilities?.has(def.requiredCapability) === true;
|
|
70
|
+
});
|
|
71
|
+
}
|
|
38
72
|
// Return the appropriate tools list based on channel capabilities.
|
|
39
73
|
// Base tools (no integration) are always included.
|
|
40
74
|
// Teams/integration tools are included only if their integration is in availableIntegrations.
|
|
41
75
|
// When toolPreferences is provided, matching preferences are appended to tool descriptions.
|
|
42
|
-
|
|
43
|
-
|
|
76
|
+
// When providerCapabilities is provided, tools with requiredCapability are filtered.
|
|
77
|
+
function getToolsForChannel(capabilities, toolPreferences, _context, providerCapabilities, mcpManager, _chatModel) {
|
|
78
|
+
const baseTools = baseToolsForCapabilities();
|
|
79
|
+
const bluebubblesTools = capabilities?.channel === "bluebubbles"
|
|
80
|
+
? tools_bluebubbles_1.bluebubblesToolDefinitions.map((d) => d.tool)
|
|
81
|
+
: [];
|
|
82
|
+
let result;
|
|
44
83
|
if (!capabilities || capabilities.availableIntegrations.length === 0) {
|
|
45
|
-
|
|
84
|
+
result = [...baseTools, ...bluebubblesTools];
|
|
46
85
|
}
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
86
|
+
else {
|
|
87
|
+
const available = new Set(capabilities.availableIntegrations);
|
|
88
|
+
const channelDefs = [...tools_teams_1.teamsToolDefinitions, ...ado_semantic_1.adoSemanticToolDefinitions, ...tools_github_1.githubToolDefinitions];
|
|
89
|
+
// Include tools whose integration is available, plus channel tools with no integration gate (e.g. teams_send_message)
|
|
90
|
+
const integrationDefs = channelDefs.filter((d) => d.integration ? available.has(d.integration) : capabilities.channel === "teams");
|
|
91
|
+
if (!toolPreferences || Object.keys(toolPreferences).length === 0) {
|
|
92
|
+
result = [...baseTools, ...bluebubblesTools, ...integrationDefs.map((d) => d.tool)];
|
|
93
|
+
}
|
|
94
|
+
else {
|
|
95
|
+
// Build a map of integration -> preference text for fast lookup
|
|
96
|
+
const prefMap = new Map();
|
|
97
|
+
for (const [key, value] of Object.entries(toolPreferences)) {
|
|
98
|
+
prefMap.set(key, value);
|
|
99
|
+
}
|
|
100
|
+
// Apply preferences to matching integration tools (new objects, no mutation)
|
|
101
|
+
// d.integration is guaranteed truthy -- integrationDefs are pre-filtered above
|
|
102
|
+
const enrichedIntegrationTools = integrationDefs.map((d) => {
|
|
103
|
+
const pref = prefMap.get(d.integration);
|
|
104
|
+
return pref ? applyPreference(d.tool, pref) : d.tool;
|
|
105
|
+
});
|
|
106
|
+
result = [...baseTools, ...bluebubblesTools, ...enrichedIntegrationTools];
|
|
107
|
+
}
|
|
51
108
|
}
|
|
52
|
-
//
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
109
|
+
// Append first-class MCP tools when mcpManager is provided
|
|
110
|
+
if (mcpManager) {
|
|
111
|
+
mcpDefinitions = (0, mcp_tools_1.mcpToolsAsDefinitions)(mcpManager);
|
|
112
|
+
const mcpSchemas = mcpDefinitions.map((d) => d.tool);
|
|
113
|
+
result = [...result, ...mcpSchemas];
|
|
56
114
|
}
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
return [...baseTools, ...enrichedIntegrationTools];
|
|
115
|
+
return filterByCapability(result, providerCapabilities);
|
|
116
|
+
}
|
|
117
|
+
// Look up a tool definition from the combined registry (native + MCP).
|
|
118
|
+
function findDefinition(toolName) {
|
|
119
|
+
return allDefinitions.find((d) => d.tool.function.name === toolName)
|
|
120
|
+
?? mcpDefinitions.find((d) => d.tool.function.name === toolName);
|
|
64
121
|
}
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
function isConfirmationRequired(toolName) {
|
|
68
|
-
const def = allDefinitions.find((d) => d.tool.function.name === toolName);
|
|
69
|
-
return def?.confirmationRequired === true;
|
|
122
|
+
function normalizeGuardArgs(_name, args) {
|
|
123
|
+
return args;
|
|
70
124
|
}
|
|
71
125
|
async function execTool(name, args, ctx) {
|
|
72
126
|
(0, runtime_1.emitNervesEvent)({
|
|
73
127
|
event: "tool.start",
|
|
74
128
|
component: "tools",
|
|
75
129
|
message: "tool execution started",
|
|
76
|
-
meta: { name },
|
|
130
|
+
meta: { name, ...(name === "shell" && args.command ? { command: args.command } : {}) },
|
|
77
131
|
});
|
|
78
|
-
// Look up from combined registry
|
|
79
|
-
const def =
|
|
132
|
+
// Look up from combined registry (native + MCP)
|
|
133
|
+
const def = findDefinition(name);
|
|
80
134
|
if (!def) {
|
|
81
135
|
(0, runtime_1.emitNervesEvent)({
|
|
82
136
|
level: "error",
|
|
@@ -87,17 +141,26 @@ async function execTool(name, args, ctx) {
|
|
|
87
141
|
});
|
|
88
142
|
return `unknown: ${name}`;
|
|
89
143
|
}
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
144
|
+
// Guardrail check: structural + trust-level
|
|
145
|
+
const mcpDef = mcpDefinitions.find((d) => d.tool.function.name === name);
|
|
146
|
+
const guardContext = {
|
|
147
|
+
readPaths: tools_base_1.editFileReadTracker,
|
|
148
|
+
trustLevel: ctx?.context?.friend?.trustLevel,
|
|
149
|
+
agentRoot: safeGetAgentRoot(),
|
|
150
|
+
...(mcpDef?.mcpServer ? { mcpServerName: mcpDef.mcpServer } : {}),
|
|
151
|
+
...(ctx?.context?.isGroupChat !== undefined ? { isGroupChat: (ctx?.context).isGroupChat } : {}),
|
|
152
|
+
};
|
|
153
|
+
const guardArgs = normalizeGuardArgs(name, args);
|
|
154
|
+
const guardResult = (0, guardrails_1.guardInvocation)(name, guardArgs, guardContext);
|
|
155
|
+
if (!guardResult.allowed) {
|
|
93
156
|
(0, runtime_1.emitNervesEvent)({
|
|
94
157
|
level: "warn",
|
|
95
|
-
event: "tool.
|
|
158
|
+
event: "tool.guardrail_block",
|
|
96
159
|
component: "tools",
|
|
97
|
-
message: "blocked
|
|
98
|
-
meta: { name,
|
|
160
|
+
message: "guardrail blocked tool execution",
|
|
161
|
+
meta: { name, reason: guardResult.reason },
|
|
99
162
|
});
|
|
100
|
-
return
|
|
163
|
+
return guardResult.reason;
|
|
101
164
|
}
|
|
102
165
|
try {
|
|
103
166
|
const result = await def.handler(args, ctx);
|
|
@@ -141,59 +204,47 @@ function summarizeUnknownArgs(args) {
|
|
|
141
204
|
return summarizeKeyValues(args, keys);
|
|
142
205
|
}
|
|
143
206
|
function summarizeArgs(name, args) {
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
return teamsSummary;
|
|
148
|
-
// Check github tools
|
|
149
|
-
const githubSummary = (0, tools_github_1.summarizeGithubArgs)(name, args);
|
|
150
|
-
if (githubSummary !== undefined)
|
|
151
|
-
return githubSummary;
|
|
152
|
-
// Base tools
|
|
153
|
-
if (name === "read_file" || name === "write_file")
|
|
154
|
-
return summarizeKeyValues(args, ["path"]);
|
|
155
|
-
if (name === "shell")
|
|
156
|
-
return summarizeKeyValues(args, ["command"]);
|
|
157
|
-
if (name === "list_directory")
|
|
158
|
-
return summarizeKeyValues(args, ["path"]);
|
|
159
|
-
if (name === "git_commit")
|
|
160
|
-
return summarizeKeyValues(args, ["message"]);
|
|
161
|
-
if (name === "gh_cli")
|
|
162
|
-
return summarizeKeyValues(args, ["command"]);
|
|
163
|
-
if (name === "load_skill")
|
|
164
|
-
return summarizeKeyValues(args, ["name"]);
|
|
165
|
-
if (name === "task_create")
|
|
166
|
-
return summarizeKeyValues(args, ["title", "type", "category"]);
|
|
167
|
-
if (name === "task_update_status")
|
|
168
|
-
return summarizeKeyValues(args, ["name", "status"]);
|
|
169
|
-
if (name === "task_board_status")
|
|
170
|
-
return summarizeKeyValues(args, ["status"]);
|
|
171
|
-
if (name === "task_board_action")
|
|
172
|
-
return summarizeKeyValues(args, ["scope"]);
|
|
173
|
-
if (name === "task_board" || name === "task_board_deps" || name === "task_board_sessions")
|
|
174
|
-
return "";
|
|
175
|
-
if (name === "coding_spawn")
|
|
176
|
-
return summarizeKeyValues(args, ["runner", "workdir", "taskRef"]);
|
|
177
|
-
if (name === "coding_status")
|
|
178
|
-
return summarizeKeyValues(args, ["sessionId"]);
|
|
179
|
-
if (name === "coding_send_input")
|
|
180
|
-
return summarizeKeyValues(args, ["sessionId", "input"]);
|
|
181
|
-
if (name === "coding_kill")
|
|
182
|
-
return summarizeKeyValues(args, ["sessionId"]);
|
|
183
|
-
if (name === "claude")
|
|
184
|
-
return summarizeKeyValues(args, ["prompt"]);
|
|
185
|
-
if (name === "web_search")
|
|
186
|
-
return summarizeKeyValues(args, ["query"]);
|
|
187
|
-
if (name === "memory_search")
|
|
188
|
-
return summarizeKeyValues(args, ["query"]);
|
|
189
|
-
if (name === "memory_save")
|
|
190
|
-
return summarizeKeyValues(args, ["text", "about"]);
|
|
191
|
-
if (name === "get_friend_note")
|
|
192
|
-
return summarizeKeyValues(args, ["friendId"]);
|
|
193
|
-
if (name === "save_friend_note") {
|
|
194
|
-
return summarizeKeyValues(args, ["type", "key", "content"]);
|
|
207
|
+
const def = findDefinition(name);
|
|
208
|
+
if (def && def.summaryKeys !== undefined) {
|
|
209
|
+
return summarizeKeyValues(args, def.summaryKeys);
|
|
195
210
|
}
|
|
196
|
-
if (name === "ado_backlog_list")
|
|
197
|
-
return summarizeKeyValues(args, ["organization", "project"]);
|
|
198
211
|
return summarizeUnknownArgs(args);
|
|
199
212
|
}
|
|
213
|
+
/**
|
|
214
|
+
* Build an enriched tool result summary for display on remote channels.
|
|
215
|
+
* For recognized tools, includes result-derived info (diff stats, exit codes, etc.).
|
|
216
|
+
* Falls back to arg-based summary for unrecognized tools.
|
|
217
|
+
*/
|
|
218
|
+
function buildToolResultSummary(name, args, _result, success) {
|
|
219
|
+
/* v8 ignore start -- branches: ?? fallbacks and ternaries in tool summary formatting @preserve */
|
|
220
|
+
switch (name) {
|
|
221
|
+
case "edit_file": {
|
|
222
|
+
if (!success)
|
|
223
|
+
return summarizeArgs(name, args);
|
|
224
|
+
const addedLines = (args.new_string ?? "").split("\n").length;
|
|
225
|
+
const removedLines = (args.old_string ?? "").split("\n").length;
|
|
226
|
+
return `+${addedLines} -${removedLines} lines in ${args.path ?? "unknown"}`;
|
|
227
|
+
}
|
|
228
|
+
case "shell": {
|
|
229
|
+
const cmd = args.command ?? "?";
|
|
230
|
+
const exitCode = success ? 0 : 1;
|
|
231
|
+
return `$ ${cmd} (exit ${exitCode})`;
|
|
232
|
+
}
|
|
233
|
+
case "read_file":
|
|
234
|
+
return `path=${args.path ?? "unknown"}`;
|
|
235
|
+
case "write_file":
|
|
236
|
+
return `path=${args.path ?? "unknown"}`;
|
|
237
|
+
case "glob":
|
|
238
|
+
return `pattern=${args.pattern ?? "?"} ${args.cwd ? `cwd=${args.cwd}` : ""}`.trim();
|
|
239
|
+
case "grep":
|
|
240
|
+
return `pattern=${args.pattern ?? "?"} ${args.path ? `path=${args.path}` : ""}`.trim();
|
|
241
|
+
case "coding_spawn": {
|
|
242
|
+
const taskRef = args.taskRef ?? "unknown";
|
|
243
|
+
const status = success ? "spawned" : "failed";
|
|
244
|
+
return `${taskRef} -> ${status}`;
|
|
245
|
+
}
|
|
246
|
+
default:
|
|
247
|
+
return summarizeArgs(name, args);
|
|
248
|
+
}
|
|
249
|
+
/* v8 ignore stop */
|
|
250
|
+
}
|
|
@@ -0,0 +1,360 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* Travel API client module.
|
|
4
|
+
*
|
|
5
|
+
* Provides HTTP clients for:
|
|
6
|
+
* - Open-Meteo weather forecast (no auth, WMO weather codes)
|
|
7
|
+
* - US State Dept Travel Advisories (no auth)
|
|
8
|
+
* - Nominatim Geocoding (no auth, requires User-Agent)
|
|
9
|
+
*/
|
|
10
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
11
|
+
exports.TRAVEL_ADVISORY_RSS_URL = exports.WMO_WEATHER_CODES = void 0;
|
|
12
|
+
exports.wmoWeatherDescription = wmoWeatherDescription;
|
|
13
|
+
exports.getWeather = getWeather;
|
|
14
|
+
exports.getWeatherByCity = getWeatherByCity;
|
|
15
|
+
exports.isoToCountryName = isoToCountryName;
|
|
16
|
+
exports.parseAdvisoryLevel = parseAdvisoryLevel;
|
|
17
|
+
exports.parseCountryName = parseCountryName;
|
|
18
|
+
exports.parseAdvisoryText = parseAdvisoryText;
|
|
19
|
+
exports.getTravelAdvisory = getTravelAdvisory;
|
|
20
|
+
exports.geocode = geocode;
|
|
21
|
+
exports.reverseGeocode = reverseGeocode;
|
|
22
|
+
exports.searchPOI = searchPOI;
|
|
23
|
+
const runtime_1 = require("../nerves/runtime");
|
|
24
|
+
// --- Open-Meteo Weather (zero auth) ---
|
|
25
|
+
const OPEN_METEO_FORECAST = "https://api.open-meteo.com/v1/forecast";
|
|
26
|
+
const OPEN_METEO_GEOCODING = "https://geocoding-api.open-meteo.com/v1/search";
|
|
27
|
+
const OPEN_METEO_HEADERS = {
|
|
28
|
+
"User-Agent": "Ouroboros/1.0 (https://github.com/ouroborosbot/ouroboros)",
|
|
29
|
+
Accept: "application/json",
|
|
30
|
+
};
|
|
31
|
+
/**
|
|
32
|
+
* WMO weather interpretation codes (WW) to human-readable descriptions.
|
|
33
|
+
* Spec: https://www.nodc.noaa.gov/archive/arc0021/0002199/1.1/data/0-data/HTML/WMO-CODE/WMO4677.HTM
|
|
34
|
+
*/
|
|
35
|
+
exports.WMO_WEATHER_CODES = {
|
|
36
|
+
0: "clear sky",
|
|
37
|
+
1: "mainly clear",
|
|
38
|
+
2: "partly cloudy",
|
|
39
|
+
3: "overcast",
|
|
40
|
+
45: "fog",
|
|
41
|
+
48: "depositing rime fog",
|
|
42
|
+
51: "light drizzle",
|
|
43
|
+
53: "moderate drizzle",
|
|
44
|
+
55: "dense drizzle",
|
|
45
|
+
56: "light freezing drizzle",
|
|
46
|
+
57: "dense freezing drizzle",
|
|
47
|
+
61: "slight rain",
|
|
48
|
+
63: "moderate rain",
|
|
49
|
+
65: "heavy rain",
|
|
50
|
+
66: "light freezing rain",
|
|
51
|
+
67: "heavy freezing rain",
|
|
52
|
+
71: "slight snowfall",
|
|
53
|
+
73: "moderate snowfall",
|
|
54
|
+
75: "heavy snowfall",
|
|
55
|
+
77: "snow grains",
|
|
56
|
+
80: "slight rain showers",
|
|
57
|
+
81: "moderate rain showers",
|
|
58
|
+
82: "violent rain showers",
|
|
59
|
+
85: "slight snow showers",
|
|
60
|
+
86: "heavy snow showers",
|
|
61
|
+
95: "thunderstorm",
|
|
62
|
+
96: "thunderstorm with slight hail",
|
|
63
|
+
99: "thunderstorm with heavy hail",
|
|
64
|
+
};
|
|
65
|
+
/** Convert a WMO weather code to a human-readable description. */
|
|
66
|
+
function wmoWeatherDescription(code) {
|
|
67
|
+
return exports.WMO_WEATHER_CODES[code] ?? "unknown";
|
|
68
|
+
}
|
|
69
|
+
async function getWeather(lat, lon) {
|
|
70
|
+
return withNervesEvents("getWeather", { lat, lon }, async () => {
|
|
71
|
+
const url = `${OPEN_METEO_FORECAST}?latitude=${lat}&longitude=${lon}` +
|
|
72
|
+
`¤t=temperature_2m,apparent_temperature,relative_humidity_2m,wind_speed_10m,weather_code` +
|
|
73
|
+
`&daily=temperature_2m_max,temperature_2m_min,weather_code` +
|
|
74
|
+
`&timezone=auto`;
|
|
75
|
+
const res = await fetch(url, { headers: OPEN_METEO_HEADERS });
|
|
76
|
+
if (!res.ok) {
|
|
77
|
+
throw new Error(`Open-Meteo API error: ${res.status} ${res.statusText}`);
|
|
78
|
+
}
|
|
79
|
+
const data = await res.json();
|
|
80
|
+
return parseOpenMeteoResponse(data, lat, lon);
|
|
81
|
+
});
|
|
82
|
+
}
|
|
83
|
+
async function getWeatherByCity(city) {
|
|
84
|
+
return withNervesEvents("getWeatherByCity", { city }, async () => {
|
|
85
|
+
// Step 1: geocode the city name via Open-Meteo geocoding
|
|
86
|
+
const geoUrl = `${OPEN_METEO_GEOCODING}?name=${encodeURIComponent(city)}&count=1&language=en`;
|
|
87
|
+
const geoRes = await fetch(geoUrl, { headers: OPEN_METEO_HEADERS });
|
|
88
|
+
if (!geoRes.ok) {
|
|
89
|
+
throw new Error(`Open-Meteo geocoding error: ${geoRes.status} ${geoRes.statusText}`);
|
|
90
|
+
}
|
|
91
|
+
const geoData = await geoRes.json();
|
|
92
|
+
const results = geoData.results;
|
|
93
|
+
if (!results || results.length === 0) {
|
|
94
|
+
throw new Error(`No location found for city "${city}"`);
|
|
95
|
+
}
|
|
96
|
+
const loc = results[0];
|
|
97
|
+
// Step 2: fetch forecast for the resolved coordinates
|
|
98
|
+
const url = `${OPEN_METEO_FORECAST}?latitude=${loc.latitude}&longitude=${loc.longitude}` +
|
|
99
|
+
`¤t=temperature_2m,apparent_temperature,relative_humidity_2m,wind_speed_10m,weather_code` +
|
|
100
|
+
`&daily=temperature_2m_max,temperature_2m_min,weather_code` +
|
|
101
|
+
`&timezone=auto`;
|
|
102
|
+
const res = await fetch(url, { headers: OPEN_METEO_HEADERS });
|
|
103
|
+
if (!res.ok) {
|
|
104
|
+
throw new Error(`Open-Meteo API error: ${res.status} ${res.statusText}`);
|
|
105
|
+
}
|
|
106
|
+
const data = await res.json();
|
|
107
|
+
return parseOpenMeteoResponse(data, loc.latitude, loc.longitude, loc.name, loc.country_code);
|
|
108
|
+
});
|
|
109
|
+
}
|
|
110
|
+
function parseOpenMeteoResponse(data, lat, lon, cityName, countryCode) {
|
|
111
|
+
const current = data.current ?? {};
|
|
112
|
+
const daily = data.daily ?? {};
|
|
113
|
+
return {
|
|
114
|
+
temperature: current.temperature_2m ?? 0,
|
|
115
|
+
feelsLike: current.apparent_temperature ?? 0,
|
|
116
|
+
description: wmoWeatherDescription(current.weather_code ?? -1),
|
|
117
|
+
humidity: current.relative_humidity_2m ?? 0,
|
|
118
|
+
windSpeed: current.wind_speed_10m ?? 0,
|
|
119
|
+
city: cityName ?? `${lat},${lon}`,
|
|
120
|
+
country: countryCode ?? "",
|
|
121
|
+
dailyHigh: daily.temperature_2m_max?.[0],
|
|
122
|
+
dailyLow: daily.temperature_2m_min?.[0],
|
|
123
|
+
};
|
|
124
|
+
}
|
|
125
|
+
// --- US State Dept Travel Advisories ---
|
|
126
|
+
/**
|
|
127
|
+
* Official US State Department travel advisories RSS feed.
|
|
128
|
+
* This is the canonical public feed maintained by travel.state.gov.
|
|
129
|
+
* Returns XML/RSS with per-country advisory items.
|
|
130
|
+
*/
|
|
131
|
+
exports.TRAVEL_ADVISORY_RSS_URL = "https://travel.state.gov/_res/rss/TAsTWs.xml";
|
|
132
|
+
/**
|
|
133
|
+
* Mapping of ISO 3166-1 alpha-2 codes to country names, ONLY for codes
|
|
134
|
+
* where ISO differs from FIPS 10-4 (which the State Dept RSS feed uses).
|
|
135
|
+
*
|
|
136
|
+
* The RSS feed Country-Tag uses FIPS codes. Callers typically pass ISO codes.
|
|
137
|
+
* When the codes match (e.g. AF for Afghanistan), no lookup is needed.
|
|
138
|
+
* This table handles the ~30 codes where they diverge.
|
|
139
|
+
*/
|
|
140
|
+
const ISO_FIPS_DIVERGENCE = {
|
|
141
|
+
AG: "Antigua and Barbuda", // FIPS: AC
|
|
142
|
+
AT: "Austria", // FIPS: AU
|
|
143
|
+
AU: "Australia", // FIPS: AS
|
|
144
|
+
BD: "Bangladesh", // FIPS: BG
|
|
145
|
+
BH: "Bahrain", // FIPS: BA
|
|
146
|
+
BN: "Brunei", // FIPS: BX
|
|
147
|
+
BO: "Bolivia", // FIPS: BL
|
|
148
|
+
BS: "Bahamas", // FIPS: BF
|
|
149
|
+
BW: "Botswana", // FIPS: BC
|
|
150
|
+
CD: "Congo (Kinshasa)", // FIPS: CG
|
|
151
|
+
CH: "Switzerland", // FIPS: SZ
|
|
152
|
+
CI: "Cote d'Ivoire", // FIPS: IV
|
|
153
|
+
CL: "Chile", // FIPS: CI
|
|
154
|
+
CN: "China", // FIPS: CH
|
|
155
|
+
CZ: "Czechia", // FIPS: EZ
|
|
156
|
+
DE: "Germany", // FIPS: GM
|
|
157
|
+
DK: "Denmark", // FIPS: DA
|
|
158
|
+
DO: "Dominican Republic", // FIPS: DR
|
|
159
|
+
DZ: "Algeria", // FIPS: AG
|
|
160
|
+
ES: "Spain", // FIPS: SP
|
|
161
|
+
FI: "Finland", // FIPS: FI → same, but title might say "Finland"
|
|
162
|
+
GQ: "Equatorial Guinea", // FIPS: EK
|
|
163
|
+
HR: "Croatia", // FIPS: HR → same
|
|
164
|
+
IE: "Ireland", // FIPS: EI
|
|
165
|
+
IL: "Israel", // FIPS: IS
|
|
166
|
+
JP: "Japan", // FIPS: JA
|
|
167
|
+
KP: "North Korea", // FIPS: KN
|
|
168
|
+
KR: "South Korea", // FIPS: KS
|
|
169
|
+
LT: "Lithuania", // FIPS: LH
|
|
170
|
+
MA: "Morocco", // FIPS: MO
|
|
171
|
+
MM: "Burma", // FIPS: BM (State Dept uses "Burma (Myanmar)")
|
|
172
|
+
NG: "Nigeria", // FIPS: NI
|
|
173
|
+
NO: "Norway", // FIPS: NO → same
|
|
174
|
+
PH: "Philippines", // FIPS: RP
|
|
175
|
+
PT: "Portugal", // FIPS: PO
|
|
176
|
+
RO: "Romania", // FIPS: RO → same
|
|
177
|
+
SE: "Sweden", // FIPS: SW
|
|
178
|
+
SG: "Singapore", // FIPS: SN
|
|
179
|
+
TL: "Timor-Leste", // FIPS: TT
|
|
180
|
+
TR: "Turkey", // FIPS: TU
|
|
181
|
+
TW: "Taiwan", // FIPS: TW → same
|
|
182
|
+
UA: "Ukraine", // FIPS: UP
|
|
183
|
+
VA: "Holy See", // FIPS: VT
|
|
184
|
+
VN: "Vietnam", // FIPS: VM
|
|
185
|
+
YE: "Yemen", // FIPS: YM
|
|
186
|
+
ZA: "South Africa", // FIPS: SF
|
|
187
|
+
};
|
|
188
|
+
/**
|
|
189
|
+
* Look up the expected country name for an ISO alpha-2 code,
|
|
190
|
+
* but ONLY when that code diverges from the FIPS code used in the RSS feed.
|
|
191
|
+
* Returns undefined for codes where ISO == FIPS (no title fallback needed).
|
|
192
|
+
*/
|
|
193
|
+
function isoToCountryName(isoCode) {
|
|
194
|
+
return ISO_FIPS_DIVERGENCE[isoCode.toUpperCase()];
|
|
195
|
+
}
|
|
196
|
+
/**
|
|
197
|
+
* Parse advisory level (1-4) from an RSS item title like
|
|
198
|
+
* "Afghanistan - Level 4: Do Not Travel"
|
|
199
|
+
*/
|
|
200
|
+
function parseAdvisoryLevel(title) {
|
|
201
|
+
const match = title.match(/Level\s+(\d)/);
|
|
202
|
+
return match ? parseInt(match[1], 10) : 0;
|
|
203
|
+
}
|
|
204
|
+
/**
|
|
205
|
+
* Parse country name from an RSS item title like
|
|
206
|
+
* "Afghanistan - Level 4: Do Not Travel"
|
|
207
|
+
*/
|
|
208
|
+
function parseCountryName(title) {
|
|
209
|
+
const dashIdx = title.indexOf(" - ");
|
|
210
|
+
return dashIdx > 0 ? title.slice(0, dashIdx).trim() : title.trim();
|
|
211
|
+
}
|
|
212
|
+
/**
|
|
213
|
+
* Extract advisory text from title (the part after "Level N: ").
|
|
214
|
+
*/
|
|
215
|
+
function parseAdvisoryText(title) {
|
|
216
|
+
const match = title.match(/Level\s+\d:\s*(.+)/);
|
|
217
|
+
return match ? match[1].trim() : title.trim();
|
|
218
|
+
}
|
|
219
|
+
async function getTravelAdvisory(countryCode) {
|
|
220
|
+
return withNervesEvents("getTravelAdvisory", { countryCode }, async () => {
|
|
221
|
+
const res = await fetch(exports.TRAVEL_ADVISORY_RSS_URL, {
|
|
222
|
+
headers: {
|
|
223
|
+
Accept: "application/xml, text/xml",
|
|
224
|
+
},
|
|
225
|
+
});
|
|
226
|
+
if (!res.ok) {
|
|
227
|
+
throw new Error(`Travel advisory API error: ${res.status} ${res.statusText}`);
|
|
228
|
+
}
|
|
229
|
+
const xml = await res.text();
|
|
230
|
+
const upperCode = countryCode.toUpperCase();
|
|
231
|
+
// When the caller's ISO code diverges from FIPS, we need a title-based fallback.
|
|
232
|
+
const expectedCountryName = isoToCountryName(upperCode);
|
|
233
|
+
// Parse RSS items via regex (no XML parser dependency needed).
|
|
234
|
+
// Each <item> contains <title>, <pubDate>, and <category domain="Country-Tag">.
|
|
235
|
+
const itemPattern = /<item>([\s\S]*?)<\/item>/g;
|
|
236
|
+
let titleFallbackMatch = null;
|
|
237
|
+
let match;
|
|
238
|
+
while ((match = itemPattern.exec(xml)) !== null) {
|
|
239
|
+
const block = match[1];
|
|
240
|
+
const titleMatch = block.match(/<title>(.*?)<\/title>/);
|
|
241
|
+
const title = titleMatch?.[1] ?? "";
|
|
242
|
+
const pubDateMatch = block.match(/<pubDate>(.*?)<\/pubDate>/);
|
|
243
|
+
const pubDate = pubDateMatch?.[1]?.trim() ?? "";
|
|
244
|
+
// Extract country tag from <category domain="Country-Tag">XX</category>
|
|
245
|
+
const tagMatch = block.match(/<category\s+domain="Country-Tag">(.*?)<\/category>/);
|
|
246
|
+
const tag = tagMatch?.[1]?.trim().toUpperCase();
|
|
247
|
+
// Primary match: FIPS tag matches the requested code directly.
|
|
248
|
+
// Skip this path when the caller's ISO code is known to diverge from FIPS —
|
|
249
|
+
// e.g. ISO "ES" = Spain, but FIPS "ES" = El Salvador. In that case, only
|
|
250
|
+
// the title-based fallback should match.
|
|
251
|
+
if (tag === upperCode && !expectedCountryName) {
|
|
252
|
+
return {
|
|
253
|
+
countryCode,
|
|
254
|
+
countryName: parseCountryName(title),
|
|
255
|
+
advisoryLevel: parseAdvisoryLevel(title),
|
|
256
|
+
advisoryText: parseAdvisoryText(title),
|
|
257
|
+
lastUpdated: pubDate,
|
|
258
|
+
};
|
|
259
|
+
}
|
|
260
|
+
// Title-based match: when ISO differs from FIPS, match by country name.
|
|
261
|
+
// Uses startsWith to handle names like "Burma (Myanmar)" matching "Burma".
|
|
262
|
+
if (expectedCountryName && !titleFallbackMatch) {
|
|
263
|
+
const titleCountry = parseCountryName(title);
|
|
264
|
+
if (titleCountry.toLowerCase().startsWith(expectedCountryName.toLowerCase())) {
|
|
265
|
+
titleFallbackMatch = {
|
|
266
|
+
countryCode,
|
|
267
|
+
countryName: titleCountry,
|
|
268
|
+
advisoryLevel: parseAdvisoryLevel(title),
|
|
269
|
+
advisoryText: parseAdvisoryText(title),
|
|
270
|
+
lastUpdated: pubDate,
|
|
271
|
+
};
|
|
272
|
+
}
|
|
273
|
+
}
|
|
274
|
+
}
|
|
275
|
+
if (titleFallbackMatch)
|
|
276
|
+
return titleFallbackMatch;
|
|
277
|
+
throw new Error(`No travel advisory found for country code "${countryCode}"`);
|
|
278
|
+
});
|
|
279
|
+
}
|
|
280
|
+
// --- Nominatim Geocoding ---
|
|
281
|
+
const NOMINATIM_BASE = "https://nominatim.openstreetmap.org";
|
|
282
|
+
const NOMINATIM_HEADERS = {
|
|
283
|
+
"User-Agent": "Ouroboros/1.0 (https://github.com/ouroborosbot/ouroboros)",
|
|
284
|
+
Accept: "application/json",
|
|
285
|
+
};
|
|
286
|
+
async function geocode(query) {
|
|
287
|
+
return withNervesEvents("geocode", { query }, async () => {
|
|
288
|
+
const url = `${NOMINATIM_BASE}/search?q=${encodeURIComponent(query)}&format=json`;
|
|
289
|
+
const res = await fetch(url, { headers: NOMINATIM_HEADERS });
|
|
290
|
+
if (!res.ok) {
|
|
291
|
+
throw new Error(`Nominatim geocode error: ${res.status} ${res.statusText}`);
|
|
292
|
+
}
|
|
293
|
+
const data = await res.json();
|
|
294
|
+
return data.map(parseGeoLocation);
|
|
295
|
+
});
|
|
296
|
+
}
|
|
297
|
+
async function reverseGeocode(lat, lon) {
|
|
298
|
+
return withNervesEvents("reverseGeocode", { lat, lon }, async () => {
|
|
299
|
+
const url = `${NOMINATIM_BASE}/reverse?lat=${lat}&lon=${lon}&format=json`;
|
|
300
|
+
const res = await fetch(url, { headers: NOMINATIM_HEADERS });
|
|
301
|
+
if (!res.ok) {
|
|
302
|
+
throw new Error(`Nominatim reverse geocode error: ${res.status} ${res.statusText}`);
|
|
303
|
+
}
|
|
304
|
+
const data = await res.json();
|
|
305
|
+
return parseGeoLocation(data);
|
|
306
|
+
});
|
|
307
|
+
}
|
|
308
|
+
async function searchPOI(query, lat, lon, radiusKm) {
|
|
309
|
+
return withNervesEvents("searchPOI", { query, lat, lon, radiusKm }, async () => {
|
|
310
|
+
const radius = radiusKm ?? 10;
|
|
311
|
+
// Nominatim uses viewbox for bounded search (approximate degrees from km)
|
|
312
|
+
const degOffset = radius / 111; // ~111 km per degree
|
|
313
|
+
const viewbox = `${lon - degOffset},${lat + degOffset},${lon + degOffset},${lat - degOffset}`;
|
|
314
|
+
const url = `${NOMINATIM_BASE}/search?q=${encodeURIComponent(query)}&format=json&viewbox=${viewbox}&bounded=1`;
|
|
315
|
+
const res = await fetch(url, { headers: NOMINATIM_HEADERS });
|
|
316
|
+
if (!res.ok) {
|
|
317
|
+
throw new Error(`Nominatim POI search error: ${res.status} ${res.statusText}`);
|
|
318
|
+
}
|
|
319
|
+
const data = await res.json();
|
|
320
|
+
return data.map(parseGeoLocation);
|
|
321
|
+
});
|
|
322
|
+
}
|
|
323
|
+
function parseGeoLocation(item) {
|
|
324
|
+
return {
|
|
325
|
+
lat: parseFloat(item.lat),
|
|
326
|
+
lon: parseFloat(item.lon),
|
|
327
|
+
displayName: item.display_name,
|
|
328
|
+
type: item.type ?? "unknown",
|
|
329
|
+
};
|
|
330
|
+
}
|
|
331
|
+
// --- Shared nerves event wrapper ---
|
|
332
|
+
async function withNervesEvents(operation, meta, fn) {
|
|
333
|
+
(0, runtime_1.emitNervesEvent)({
|
|
334
|
+
event: "client.request_start",
|
|
335
|
+
component: "clients",
|
|
336
|
+
message: `travel API ${operation} starting`,
|
|
337
|
+
meta: { operation, ...meta },
|
|
338
|
+
});
|
|
339
|
+
try {
|
|
340
|
+
const result = await fn();
|
|
341
|
+
(0, runtime_1.emitNervesEvent)({
|
|
342
|
+
event: "client.request_end",
|
|
343
|
+
component: "clients",
|
|
344
|
+
message: `travel API ${operation} complete`,
|
|
345
|
+
meta: { operation, ...meta },
|
|
346
|
+
});
|
|
347
|
+
return result;
|
|
348
|
+
}
|
|
349
|
+
catch (err) {
|
|
350
|
+
(0, runtime_1.emitNervesEvent)({
|
|
351
|
+
level: "error",
|
|
352
|
+
event: "client.error",
|
|
353
|
+
component: "clients",
|
|
354
|
+
message: `travel API ${operation} failed`,
|
|
355
|
+
/* v8 ignore next -- defensive: callers throw Error instances @preserve */
|
|
356
|
+
meta: { operation, reason: err instanceof Error ? err.message : String(err), ...meta },
|
|
357
|
+
});
|
|
358
|
+
throw err;
|
|
359
|
+
}
|
|
360
|
+
}
|