@poolzin/pool-bot 2026.2.17 → 2026.2.19
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/CHANGELOG.md +25 -0
- package/dist/agents/agent-scope.js +4 -0
- package/dist/agents/announce-idempotency.js +14 -0
- package/dist/agents/auth-profiles.resolve-auth-profile-order.fixtures.js +23 -0
- package/dist/agents/bash-tools.exec-runtime.js +438 -0
- package/dist/agents/bash-tools.shared.js +6 -0
- package/dist/agents/cli-runner/reliability.js +61 -0
- package/dist/agents/cli-watchdog-defaults.js +11 -0
- package/dist/agents/command-poll-backoff.js +63 -0
- package/dist/agents/current-time.js +16 -0
- package/dist/agents/model-alias-lines.js +18 -0
- package/dist/agents/model-auth-label.js +61 -0
- package/dist/agents/models-config.e2e-harness.js +115 -0
- package/dist/agents/ollama-stream.js +11 -3
- package/dist/agents/openclaw-tools.js +135 -0
- package/dist/agents/pi-auth-json.js +118 -0
- package/dist/agents/pi-embedded-runner/run.overflow-compaction.mocks.shared.js +147 -0
- package/dist/agents/pi-embedded-subscribe.e2e-harness.js +90 -0
- package/dist/agents/pi-embedded-subscribe.handlers.compaction.js +63 -0
- package/dist/agents/pi-embedded-subscribe.handlers.tools.media.test-helpers.js +30 -0
- package/dist/agents/pi-extensions/session-manager-runtime-registry.js +23 -0
- package/dist/agents/pi-tools.js +2 -0
- package/dist/agents/queued-file-writer.js +22 -0
- package/dist/agents/sandbox/docker.js +133 -40
- package/dist/agents/sandbox/fs-bridge.js +146 -0
- package/dist/agents/sandbox/fs-paths.js +205 -0
- package/dist/agents/sandbox/hash.js +4 -0
- package/dist/agents/sandbox-paths.js +3 -0
- package/dist/agents/session-dirs.js +20 -0
- package/dist/agents/skills/filter.js +24 -0
- package/dist/agents/skills/tools-dir.js +9 -0
- package/dist/agents/skills-install-download.js +290 -0
- package/dist/agents/skills-install-output.js +30 -0
- package/dist/agents/skills-install.download-test-utils.js +36 -0
- package/dist/agents/skills.e2e-test-helpers.js +13 -0
- package/dist/agents/subagent-announce-queue.js +59 -15
- package/dist/agents/subagent-depth.js +137 -0
- package/dist/agents/subagent-registry.js +448 -96
- package/dist/agents/subagent-spawn.js +262 -0
- package/dist/agents/test-helpers/fast-tool-stubs.js +18 -0
- package/dist/agents/test-helpers/host-sandbox-fs-bridge.js +74 -0
- package/dist/agents/tool-display-common.js +782 -0
- package/dist/agents/tools/image-tool.js +1 -1
- package/dist/agents/tools/sessions-access.js +178 -0
- package/dist/agents/tools/sessions-resolution.js +206 -0
- package/dist/agents/tools/subagents-tool.js +616 -0
- package/dist/agents/workspace-dir.js +18 -0
- package/dist/agents/workspace-dirs.js +14 -0
- package/dist/agents/workspace.js +70 -0
- package/dist/auto-reply/heartbeat-reply-payload.js +18 -0
- package/dist/auto-reply/reply/commands-export-session.js +163 -0
- package/dist/auto-reply/reply/commands-mesh.js +245 -0
- package/dist/auto-reply/reply/commands-setunset.js +28 -0
- package/dist/auto-reply/reply/commands-slash-parse.js +31 -0
- package/dist/auto-reply/reply/commands-system-prompt.js +117 -0
- package/dist/auto-reply/reply/directive-handling.levels.js +17 -0
- package/dist/auto-reply/reply/directive-handling.params.js +1 -0
- package/dist/auto-reply/reply/directive-parsing.js +36 -0
- package/dist/auto-reply/reply/dispatcher-registry.js +43 -0
- package/dist/auto-reply/reply/elevated-unavailable.js +20 -0
- package/dist/auto-reply/reply/reply-delivery.js +92 -0
- package/dist/auto-reply/reply/session-reset-prompt.js +1 -0
- package/dist/auto-reply/reply/session-run-accounting.js +33 -0
- package/dist/auto-reply/reply.directive.directive-behavior.e2e-harness.js +115 -0
- package/dist/auto-reply/reply.directive.directive-behavior.e2e-mocks.js +12 -0
- package/dist/browser/bridge-auth-registry.js +26 -0
- package/dist/browser/client-actions-url.js +10 -0
- package/dist/browser/control-auth.js +73 -0
- package/dist/browser/csrf.js +64 -0
- package/dist/browser/http-auth.js +52 -0
- package/dist/browser/paths.js +37 -0
- package/dist/browser/proxy-files.js +32 -0
- package/dist/browser/pw-ai-state.js +7 -0
- package/dist/browser/resolved-config-refresh.js +42 -0
- package/dist/browser/routes/path-output.js +1 -0
- package/dist/browser/server-context.chrome-test-harness.js +20 -0
- package/dist/browser/server-middleware.js +31 -0
- package/dist/browser/test-port.js +16 -0
- package/dist/build-info.json +3 -3
- package/dist/canvas-host/file-resolver.js +43 -0
- package/dist/channels/account-summary.js +19 -0
- package/dist/channels/draft-stream-loop.js +77 -0
- package/dist/channels/plugins/account-helpers.js +26 -0
- package/dist/channels/telegram/allow-from.js +10 -0
- package/dist/cli/browser-cli-resize.js +22 -0
- package/dist/cli/browser-cli-shared.js +8 -0
- package/dist/cli/clawbot-cli.js +5 -0
- package/dist/cli/completion-cli.js +566 -0
- package/dist/cli/config-cli.js +63 -5
- package/dist/cli/daemon-cli/lifecycle-core.js +256 -0
- package/dist/cli/daemon-cli/register-service-commands.js +60 -0
- package/dist/cli/daemon-cli-compat.js +80 -0
- package/dist/cli/nodes-cli/pairing-render.js +26 -0
- package/dist/cli/program/action-reparse.js +17 -0
- package/dist/cli/program/command-registry.js +17 -0
- package/dist/cli/program/program-context.js +8 -0
- package/dist/cli/program/register.subclis.js +7 -0
- package/dist/cli/program/routes.js +233 -0
- package/dist/cli/qr-cli.js +132 -0
- package/dist/cli/requirements-test-fixtures.js +17 -0
- package/dist/cli/respawn-policy.js +4 -0
- package/dist/cli/shared/parse-port.js +18 -0
- package/dist/cli/skills-cli.format.js +241 -0
- package/dist/cli/update-cli/progress.js +121 -0
- package/dist/cli/update-cli/restart-helper.js +108 -0
- package/dist/cli/update-cli/shared.js +196 -0
- package/dist/cli/update-cli/status.js +97 -0
- package/dist/cli/update-cli/suppress-deprecations.js +17 -0
- package/dist/cli/update-cli/update-command.js +506 -0
- package/dist/cli/update-cli/wizard.js +130 -0
- package/dist/cli/update-cli.js +3 -9
- package/dist/cli/windows-argv.js +69 -0
- package/dist/commands/auth-choice-legacy.js +20 -0
- package/dist/commands/auth-choice.apply-helpers.js +8 -0
- package/dist/commands/channel-test-helpers.js +19 -0
- package/dist/commands/cleanup-plan.js +10 -0
- package/dist/commands/cleanup-utils.js +7 -0
- package/dist/commands/config-validation.js +15 -0
- package/dist/commands/doctor-completion.js +112 -0
- package/dist/commands/doctor-memory-search.js +119 -0
- package/dist/commands/doctor-session-locks.js +73 -0
- package/dist/commands/doctor.e2e-harness.js +364 -0
- package/dist/commands/gateway-presence.js +19 -0
- package/dist/commands/model-default.js +35 -0
- package/dist/commands/models/fallbacks-shared.js +102 -0
- package/dist/commands/models/shared.js +24 -0
- package/dist/commands/onboard-auth.config-gateways.js +64 -0
- package/dist/commands/onboard-auth.config-litellm.js +45 -0
- package/dist/commands/onboard-auth.config-shared.js +116 -0
- package/dist/commands/onboard-config.js +16 -0
- package/dist/commands/onboard-non-interactive.test-helpers.js +31 -0
- package/dist/commands/onboard-provider-auth-flags.js +136 -0
- package/dist/commands/openai-codex-oauth.js +40 -0
- package/dist/commands/test-runtime-config-helpers.js +21 -0
- package/dist/commands/test-wizard-helpers.js +68 -0
- package/dist/commands/vllm-setup.js +66 -0
- package/dist/compat/legacy-names.js +2 -0
- package/dist/config/backup-rotation.js +19 -0
- package/dist/config/env-preserve.js +122 -0
- package/dist/config/includes-scan.js +78 -0
- package/dist/config/plugins-allowlist.js +13 -0
- package/dist/config/schema.help.js +256 -0
- package/dist/config/schema.hints.js +189 -0
- package/dist/config/schema.irc.js +20 -0
- package/dist/config/schema.labels.js +317 -0
- package/dist/config/sessions/delivery-info.js +40 -0
- package/dist/config/types.irc.js +1 -0
- package/dist/config/zod-schema.agent-model.js +10 -0
- package/dist/config/zod-schema.allowdeny.js +35 -0
- package/dist/config/zod-schema.sensitive.js +4 -0
- package/dist/control-ui/assets/index-HRr1grwl.js.map +1 -1
- package/dist/cron/isolated-agent/skills-snapshot.js +26 -0
- package/dist/cron/isolated-agent/subagent-followup.js +127 -0
- package/dist/cron/isolated-agent.mocks.js +12 -0
- package/dist/cron/isolated-agent.test-setup.js +22 -0
- package/dist/cron/legacy-delivery.js +43 -0
- package/dist/cron/webhook-url.js +22 -0
- package/dist/daemon/arg-split.js +40 -0
- package/dist/daemon/exec-file.js +23 -0
- package/dist/daemon/output.js +6 -0
- package/dist/daemon/runtime-format.js +31 -0
- package/dist/daemon/schtasks-exec.js +4 -0
- package/dist/daemon/service-audit.js +22 -0
- package/dist/discord/client.js +41 -0
- package/dist/discord/components-registry.js +57 -0
- package/dist/discord/components.js +816 -0
- package/dist/discord/guilds.js +12 -0
- package/dist/discord/monitor/gateway-plugin.js +48 -0
- package/dist/discord/monitor/presence.js +30 -0
- package/dist/discord/send.components.js +115 -0
- package/dist/discord/send.shared.js +4 -0
- package/dist/discord/ui.js +26 -0
- package/dist/discord/voice-message.js +254 -0
- package/dist/gateway/agent-event-assistant-text.js +5 -0
- package/dist/gateway/agent-prompt.js +33 -0
- package/dist/gateway/auth-rate-limit.js +136 -0
- package/dist/gateway/channel-health-monitor.js +114 -0
- package/dist/gateway/control-ui-contract.js +1 -0
- package/dist/gateway/control-ui-csp.js +15 -0
- package/dist/gateway/gateway-config-prompts.shared.js +25 -0
- package/dist/gateway/http-auth-helpers.js +18 -0
- package/dist/gateway/http-common.js +18 -0
- package/dist/gateway/http-endpoint-helpers.js +27 -0
- package/dist/gateway/node-invoke-sanitize.js +11 -0
- package/dist/gateway/node-invoke-system-run-approval.js +205 -0
- package/dist/gateway/probe-auth.js +21 -0
- package/dist/gateway/protocol/index.js +7 -2
- package/dist/gateway/protocol/schema/mesh.js +54 -0
- package/dist/gateway/protocol/schema/protocol-schemas.js +7 -0
- package/dist/gateway/protocol/schema.js +1 -0
- package/dist/gateway/server/ws-connection/auth-messages.js +54 -0
- package/dist/gateway/server-channels.js +11 -0
- package/dist/gateway/server-methods/attachment-normalize.js +16 -0
- package/dist/gateway/server-methods/base-hash.js +8 -0
- package/dist/gateway/server-methods/mesh.js +700 -0
- package/dist/gateway/server-methods/nodes.handlers.invoke-result.js +55 -0
- package/dist/gateway/server-methods/restart-request.js +13 -0
- package/dist/gateway/server-methods/validation.js +8 -0
- package/dist/gateway/server.agent.gateway-server-agent.mocks.js +35 -0
- package/dist/gateway/server.e2e-registry-helpers.js +1 -0
- package/dist/gateway/server.e2e-ws-harness.js +20 -0
- package/dist/gateway/test-helpers.js +2 -0
- package/dist/gateway/test-helpers.server.js +3 -1
- package/dist/gateway/test-http-response.js +12 -0
- package/dist/gateway/test-openai-responses-model.js +20 -0
- package/dist/gateway/test-temp-config.js +30 -0
- package/dist/gateway/test-with-server.js +32 -0
- package/dist/hooks/bundled/bootstrap-extra-files/handler.js +46 -0
- package/dist/imessage/monitor/abort-handler.js +23 -0
- package/dist/imessage/monitor/inbound-processing.js +346 -0
- package/dist/imessage/monitor/parse-notification.js +64 -0
- package/dist/imessage/target-parsing-helpers.js +92 -0
- package/dist/infra/archive.js +244 -20
- package/dist/infra/detect-package-manager.js +26 -0
- package/dist/infra/exec-approvals-allowlist.js +257 -0
- package/dist/infra/exec-approvals-analysis.js +770 -0
- package/dist/infra/exec-approvals.js +13 -0
- package/dist/infra/file-lock.js +1 -0
- package/dist/infra/gemini-auth.js +39 -0
- package/dist/infra/heartbeat-active-hours.js +85 -0
- package/dist/infra/heartbeat-events-filter.js +50 -0
- package/dist/infra/heartbeat-runner.test-utils.js +39 -0
- package/dist/infra/http-body.js +265 -0
- package/dist/infra/install-package-dir.js +50 -0
- package/dist/infra/install-safe-path.js +49 -0
- package/dist/infra/json-files.js +49 -0
- package/dist/infra/jsonl-socket.js +52 -0
- package/dist/infra/map-size.js +14 -0
- package/dist/infra/net/hostname.js +7 -0
- package/dist/infra/npm-registry-spec.js +39 -0
- package/dist/infra/openclaw-root.js +109 -0
- package/dist/infra/outbound/delivery-queue.js +214 -0
- package/dist/infra/outbound/identity.js +23 -0
- package/dist/infra/outbound/message-action-params.js +307 -0
- package/dist/infra/outbound/tool-payload.js +21 -0
- package/dist/infra/package-json.js +23 -0
- package/dist/infra/pairing-files.js +19 -0
- package/dist/infra/pairing-token.js +9 -0
- package/dist/infra/path-prepend.js +51 -0
- package/dist/infra/process-respawn.js +49 -0
- package/dist/infra/runtime-status.js +16 -0
- package/dist/infra/session-cost-usage.types.js +1 -0
- package/dist/infra/session-maintenance-warning.js +89 -0
- package/dist/infra/system-run-command.js +78 -0
- package/dist/infra/tmp-openclaw-dir.js +81 -0
- package/dist/infra/tmp-poolbot-dir.js +2 -0
- package/dist/infra/update-channels.js +19 -0
- package/dist/line/actions.js +45 -0
- package/dist/line/channel-access-token.js +9 -0
- package/dist/line/flex-templates/basic-cards.js +332 -0
- package/dist/line/flex-templates/common.js +18 -0
- package/dist/line/flex-templates/media-control-cards.js +453 -0
- package/dist/line/flex-templates/message.js +10 -0
- package/dist/line/flex-templates/schedule-cards.js +399 -0
- package/dist/line/flex-templates/types.js +1 -0
- package/dist/line/webhook-node.js +100 -0
- package/dist/line/webhook-utils.js +11 -0
- package/dist/logging/timestamps.js +14 -0
- package/dist/markdown/whatsapp.js +62 -0
- package/dist/media/base64.js +34 -0
- package/dist/media/local-roots.js +32 -0
- package/dist/media/outbound-attachment.js +10 -0
- package/dist/media/read-response-with-limit.js +41 -0
- package/dist/media/sniff-mime-from-base64.js +19 -0
- package/dist/media-understanding/audio-preflight.js +67 -0
- package/dist/media-understanding/fs.js +13 -0
- package/dist/media-understanding/output-extract.js +26 -0
- package/dist/media-understanding/providers/audio.test-helpers.js +34 -0
- package/dist/media-understanding/providers/google/inline-data.js +64 -0
- package/dist/media-understanding/providers/shared.js +7 -0
- package/dist/media-understanding/runner.entries.js +459 -0
- package/dist/memory/batch-error-utils.js +11 -0
- package/dist/memory/batch-http.js +27 -0
- package/dist/memory/batch-output.js +29 -0
- package/dist/memory/batch-runner.js +22 -0
- package/dist/memory/batch-upload.js +23 -0
- package/dist/memory/batch-utils.js +26 -0
- package/dist/memory/embeddings-debug.js +11 -0
- package/dist/memory/embeddings-remote-client.js +22 -0
- package/dist/memory/embeddings-remote-fetch.js +14 -0
- package/dist/memory/manager-embedding-ops.js +616 -0
- package/dist/memory/manager-sync-ops.js +953 -0
- package/dist/memory/qmd-manager.js +1061 -0
- package/dist/memory/qmd-query-parser.js +107 -0
- package/dist/memory/qmd-scope.js +93 -0
- package/dist/memory/search-manager.js +0 -1
- package/dist/memory/sync-index.js +21 -0
- package/dist/memory/sync-progress.js +22 -0
- package/dist/memory/sync-stale.js +30 -0
- package/dist/memory/test-embeddings-mock.js +16 -0
- package/dist/memory/test-manager-helpers.js +14 -0
- package/dist/memory/test-runtime-mocks.js +11 -0
- package/dist/node-host/invoke-browser.js +177 -0
- package/dist/node-host/invoke.js +685 -0
- package/dist/pairing/setup-code.js +285 -0
- package/dist/plugin-sdk/account-id.js +1 -0
- package/dist/plugin-sdk/agent-media-payload.js +13 -0
- package/dist/plugin-sdk/allow-from.js +47 -0
- package/dist/plugin-sdk/command-auth.js +23 -0
- package/dist/plugin-sdk/config-paths.js +9 -0
- package/dist/plugin-sdk/file-lock.js +116 -0
- package/dist/plugin-sdk/json-store.js +31 -0
- package/dist/plugin-sdk/onboarding.js +28 -0
- package/dist/plugin-sdk/provider-auth-result.js +29 -0
- package/dist/plugin-sdk/slack-message-actions.js +133 -0
- package/dist/plugin-sdk/status-helpers.js +35 -0
- package/dist/plugin-sdk/text-chunking.js +31 -0
- package/dist/plugin-sdk/tool-send.js +12 -0
- package/dist/plugin-sdk/webhook-path.js +27 -0
- package/dist/plugin-sdk/webhook-targets.js +34 -0
- package/dist/plugins/hooks.test-helpers.js +21 -0
- package/dist/plugins/uninstall.js +171 -0
- package/dist/process/supervisor/adapters/child.js +143 -0
- package/dist/process/supervisor/adapters/env.js +13 -0
- package/dist/process/supervisor/adapters/pty.js +148 -0
- package/dist/process/supervisor/index.js +10 -0
- package/dist/process/supervisor/registry.js +117 -0
- package/dist/process/supervisor/supervisor.js +244 -0
- package/dist/process/supervisor/types.js +1 -0
- package/dist/providers/google-shared.test-helpers.js +75 -0
- package/dist/security/audit-channel.js +419 -0
- package/dist/security/audit-tool-policy.js +1 -0
- package/dist/security/scan-paths.js +12 -0
- package/dist/sessions/input-provenance.js +55 -0
- package/dist/sessions/session-key-utils.js +7 -0
- package/dist/shared/chat-content.js +31 -0
- package/dist/shared/chat-envelope.js +45 -0
- package/dist/shared/config-eval.js +117 -0
- package/dist/shared/device-auth.js +16 -0
- package/dist/shared/entry-metadata.js +9 -0
- package/dist/shared/entry-status.js +25 -0
- package/dist/shared/frontmatter.js +98 -0
- package/dist/shared/model-param-b.js +19 -0
- package/dist/shared/net/ipv4.js +17 -0
- package/dist/shared/node-match.js +53 -0
- package/dist/shared/requirements.js +128 -0
- package/dist/shared/subagents-format.js +84 -0
- package/dist/shared/usage-aggregates.js +28 -0
- package/dist/signal/monitor/mentions.js +45 -0
- package/dist/signal/rpc-context.js +19 -0
- package/dist/slack/blocks-fallback.js +76 -0
- package/dist/slack/blocks-input.js +40 -0
- package/dist/slack/draft-stream.js +106 -0
- package/dist/slack/message-actions.js +51 -0
- package/dist/slack/modal-metadata.js +32 -0
- package/dist/slack/monitor/events/interactions.js +462 -0
- package/dist/slack/monitor/room-context.js +17 -0
- package/dist/slack/stream-mode.js +41 -0
- package/dist/telegram/bot-native-command-menu.js +64 -0
- package/dist/telegram/bot.media.e2e-harness.js +81 -0
- package/dist/telegram/button-types.js +1 -0
- package/dist/telegram/group-access.js +65 -0
- package/dist/telegram/outbound-params.js +21 -0
- package/dist/telegram/poll-vote-cache.js +21 -0
- package/dist/terminal/health-style.js +36 -0
- package/dist/test-utils/chunk-test-helpers.js +21 -0
- package/dist/test-utils/env.js +72 -0
- package/dist/test-utils/exec-assertions.js +12 -0
- package/dist/test-utils/imessage-test-plugin.js +54 -0
- package/dist/test-utils/mock-http-response.js +17 -0
- package/dist/test-utils/vitest-mock-fn.js +1 -0
- package/dist/tts/tts-core.js +550 -0
- package/dist/utils/chunk-items.js +10 -0
- package/dist/utils/reaction-level.js +52 -0
- package/dist/utils/safe-json.js +22 -0
- package/dist/utils/with-timeout.js +14 -0
- package/dist/web/media.js +17 -5
- package/dist/whatsapp/resolve-outbound-target.js +42 -0
- package/dist/wizard/onboarding.completion.js +74 -0
- package/extensions/bluebubbles/src/account-resolve.ts +29 -0
- package/extensions/bluebubbles/src/monitor-normalize.ts +796 -0
- package/extensions/bluebubbles/src/monitor-processing.ts +1007 -0
- package/extensions/bluebubbles/src/monitor-reply-cache.ts +185 -0
- package/extensions/bluebubbles/src/monitor-shared.ts +51 -0
- package/extensions/bluebubbles/src/multipart.ts +32 -0
- package/extensions/bluebubbles/src/send-helpers.ts +53 -0
- package/extensions/bluebubbles/src/test-harness.ts +50 -0
- package/extensions/bluebubbles/src/test-mocks.ts +11 -0
- package/extensions/device-pair/index.ts +554 -0
- package/extensions/device-pair/poolbot.plugin.json +20 -0
- package/extensions/discord/src/channel.js +366 -0
- package/extensions/discord/src/runtime.js +10 -0
- package/extensions/feishu/index.ts +63 -0
- package/extensions/feishu/package.json +37 -0
- package/extensions/feishu/poolbot.plugin.json +10 -0
- package/extensions/feishu/skills/feishu-doc/SKILL.md +105 -0
- package/extensions/feishu/skills/feishu-doc/references/block-types.md +103 -0
- package/extensions/feishu/skills/feishu-drive/SKILL.md +97 -0
- package/extensions/feishu/skills/feishu-perm/SKILL.md +119 -0
- package/extensions/feishu/skills/feishu-wiki/SKILL.md +111 -0
- package/extensions/feishu/src/accounts.ts +114 -0
- package/extensions/feishu/src/bitable.ts +739 -0
- package/extensions/feishu/src/bot.ts +965 -0
- package/extensions/feishu/src/channel.ts +351 -0
- package/extensions/feishu/src/client.ts +118 -0
- package/extensions/feishu/src/config-schema.ts +206 -0
- package/extensions/feishu/src/dedup.ts +33 -0
- package/extensions/feishu/src/directory.ts +177 -0
- package/extensions/feishu/src/doc-schema.ts +47 -0
- package/extensions/feishu/src/docx.ts +536 -0
- package/extensions/feishu/src/drive-schema.ts +46 -0
- package/extensions/feishu/src/drive.ts +227 -0
- package/extensions/feishu/src/dynamic-agent.ts +131 -0
- package/extensions/feishu/src/media.ts +449 -0
- package/extensions/feishu/src/mention.ts +126 -0
- package/extensions/feishu/src/monitor.ts +330 -0
- package/extensions/feishu/src/onboarding.ts +359 -0
- package/extensions/feishu/src/outbound.ts +55 -0
- package/extensions/feishu/src/perm-schema.ts +52 -0
- package/extensions/feishu/src/perm.ts +173 -0
- package/extensions/feishu/src/policy.ts +84 -0
- package/extensions/feishu/src/probe.ts +44 -0
- package/extensions/feishu/src/reactions.ts +160 -0
- package/extensions/feishu/src/reply-dispatcher.ts +239 -0
- package/extensions/feishu/src/runtime.ts +14 -0
- package/extensions/feishu/src/send-result.ts +29 -0
- package/extensions/feishu/src/send.ts +335 -0
- package/extensions/feishu/src/streaming-card.ts +223 -0
- package/extensions/feishu/src/targets.ts +78 -0
- package/extensions/feishu/src/tools-config.ts +21 -0
- package/extensions/feishu/src/types.ts +81 -0
- package/extensions/feishu/src/typing.ts +80 -0
- package/extensions/feishu/src/wiki-schema.ts +55 -0
- package/extensions/feishu/src/wiki.ts +232 -0
- package/extensions/imessage/src/channel.js +253 -0
- package/extensions/imessage/src/runtime.js +10 -0
- package/extensions/irc/index.ts +17 -0
- package/extensions/irc/package.json +14 -0
- package/extensions/irc/poolbot.plugin.json +9 -0
- package/extensions/irc/src/accounts.ts +268 -0
- package/extensions/irc/src/channel.ts +367 -0
- package/extensions/irc/src/client.ts +439 -0
- package/extensions/irc/src/config-schema.ts +97 -0
- package/extensions/irc/src/connect-options.ts +30 -0
- package/extensions/irc/src/control-chars.ts +22 -0
- package/extensions/irc/src/inbound.ts +334 -0
- package/extensions/irc/src/monitor.ts +147 -0
- package/extensions/irc/src/normalize.ts +117 -0
- package/extensions/irc/src/onboarding.ts +479 -0
- package/extensions/irc/src/policy.ts +157 -0
- package/extensions/irc/src/probe.ts +53 -0
- package/extensions/irc/src/protocol.ts +169 -0
- package/extensions/irc/src/runtime.ts +14 -0
- package/extensions/irc/src/send.ts +88 -0
- package/extensions/irc/src/types.ts +93 -0
- package/extensions/matrix/src/matrix/client-bootstrap.ts +39 -0
- package/extensions/mattermost/src/mattermost/monitor-onchar.ts +25 -0
- package/extensions/mattermost/src/mattermost/monitor-websocket.ts +221 -0
- package/extensions/mattermost/src/mattermost/reactions.ts +130 -0
- package/extensions/mattermost/src/mattermost/reconnect.ts +103 -0
- package/extensions/minimax-portal-auth/README.md +33 -0
- package/extensions/minimax-portal-auth/index.ts +161 -0
- package/extensions/minimax-portal-auth/oauth.ts +247 -0
- package/extensions/minimax-portal-auth/package.json +15 -0
- package/extensions/minimax-portal-auth/poolbot.plugin.json +9 -0
- package/extensions/msteams/src/file-lock.ts +1 -0
- package/extensions/msteams/src/graph.ts +92 -0
- package/extensions/msteams/src/mentions.ts +114 -0
- package/extensions/msteams/src/test-runtime.ts +16 -0
- package/extensions/openai-codex-auth/README.md +82 -0
- package/extensions/openai-codex-auth/index.ts +177 -0
- package/extensions/openai-codex-auth/package.json +15 -0
- package/extensions/openai-codex-auth/poolbot.plugin.json +9 -0
- package/extensions/phone-control/index.ts +421 -0
- package/extensions/phone-control/poolbot.plugin.json +10 -0
- package/extensions/shared/resolve-target-test-helpers.ts +66 -0
- package/extensions/signal/src/channel.js +273 -0
- package/extensions/signal/src/runtime.js +10 -0
- package/extensions/slack/src/channel.js +489 -0
- package/extensions/slack/src/runtime.js +10 -0
- package/extensions/talk-voice/index.ts +150 -0
- package/extensions/talk-voice/poolbot.plugin.json +10 -0
- package/extensions/telegram/src/channel.js +424 -0
- package/extensions/telegram/src/runtime.js +10 -0
- package/extensions/thread-ownership/index.ts +133 -0
- package/extensions/thread-ownership/poolbot.plugin.json +28 -0
- package/extensions/tlon/src/account-fields.ts +25 -0
- package/extensions/tlon/src/urbit/base-url.ts +57 -0
- package/extensions/tlon/src/urbit/channel-client.ts +157 -0
- package/extensions/tlon/src/urbit/channel-ops.ts +164 -0
- package/extensions/tlon/src/urbit/context.ts +47 -0
- package/extensions/tlon/src/urbit/errors.ts +51 -0
- package/extensions/tlon/src/urbit/fetch.ts +39 -0
- package/extensions/twitch/src/test-fixtures.ts +30 -0
- package/extensions/voice-call/src/allowlist.ts +19 -0
- package/extensions/whatsapp/src/channel.js +429 -0
- package/extensions/whatsapp/src/runtime.js +10 -0
- package/package.json +1 -1
|
@@ -0,0 +1,130 @@
|
|
|
1
|
+
import type { PoolBotConfig } from "poolbot/plugin-sdk";
|
|
2
|
+
import { resolveMattermostAccount } from "./accounts.js";
|
|
3
|
+
import { createMattermostClient, fetchMattermostMe, type MattermostClient } from "./client.js";
|
|
4
|
+
|
|
5
|
+
type Result = { ok: true } | { ok: false; error: string };
|
|
6
|
+
|
|
7
|
+
const BOT_USER_CACHE_TTL_MS = 10 * 60_000;
|
|
8
|
+
const botUserIdCache = new Map<string, { userId: string; expiresAt: number }>();
|
|
9
|
+
|
|
10
|
+
async function resolveBotUserId(
|
|
11
|
+
client: MattermostClient,
|
|
12
|
+
cacheKey: string,
|
|
13
|
+
): Promise<string | null> {
|
|
14
|
+
const cached = botUserIdCache.get(cacheKey);
|
|
15
|
+
if (cached && cached.expiresAt > Date.now()) {
|
|
16
|
+
return cached.userId;
|
|
17
|
+
}
|
|
18
|
+
const me = await fetchMattermostMe(client);
|
|
19
|
+
const userId = me?.id?.trim();
|
|
20
|
+
if (!userId) {
|
|
21
|
+
return null;
|
|
22
|
+
}
|
|
23
|
+
botUserIdCache.set(cacheKey, { userId, expiresAt: Date.now() + BOT_USER_CACHE_TTL_MS });
|
|
24
|
+
return userId;
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
export async function addMattermostReaction(params: {
|
|
28
|
+
cfg: PoolBotConfig;
|
|
29
|
+
postId: string;
|
|
30
|
+
emojiName: string;
|
|
31
|
+
accountId?: string | null;
|
|
32
|
+
fetchImpl?: typeof fetch;
|
|
33
|
+
}): Promise<Result> {
|
|
34
|
+
const resolved = resolveMattermostAccount({ cfg: params.cfg, accountId: params.accountId });
|
|
35
|
+
const baseUrl = resolved.baseUrl?.trim();
|
|
36
|
+
const botToken = resolved.botToken?.trim();
|
|
37
|
+
if (!baseUrl || !botToken) {
|
|
38
|
+
return { ok: false, error: "Mattermost botToken/baseUrl missing." };
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
const client = createMattermostClient({
|
|
42
|
+
baseUrl,
|
|
43
|
+
botToken,
|
|
44
|
+
fetchImpl: params.fetchImpl,
|
|
45
|
+
});
|
|
46
|
+
|
|
47
|
+
const cacheKey = `${baseUrl}:${botToken}`;
|
|
48
|
+
const userId = await resolveBotUserId(client, cacheKey);
|
|
49
|
+
if (!userId) {
|
|
50
|
+
return { ok: false, error: "Mattermost reactions failed: could not resolve bot user id." };
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
try {
|
|
54
|
+
await createReaction(client, {
|
|
55
|
+
userId,
|
|
56
|
+
postId: params.postId,
|
|
57
|
+
emojiName: params.emojiName,
|
|
58
|
+
});
|
|
59
|
+
} catch (err) {
|
|
60
|
+
return { ok: false, error: `Mattermost add reaction failed: ${String(err)}` };
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
return { ok: true };
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
export async function removeMattermostReaction(params: {
|
|
67
|
+
cfg: PoolBotConfig;
|
|
68
|
+
postId: string;
|
|
69
|
+
emojiName: string;
|
|
70
|
+
accountId?: string | null;
|
|
71
|
+
fetchImpl?: typeof fetch;
|
|
72
|
+
}): Promise<Result> {
|
|
73
|
+
const resolved = resolveMattermostAccount({ cfg: params.cfg, accountId: params.accountId });
|
|
74
|
+
const baseUrl = resolved.baseUrl?.trim();
|
|
75
|
+
const botToken = resolved.botToken?.trim();
|
|
76
|
+
if (!baseUrl || !botToken) {
|
|
77
|
+
return { ok: false, error: "Mattermost botToken/baseUrl missing." };
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
const client = createMattermostClient({
|
|
81
|
+
baseUrl,
|
|
82
|
+
botToken,
|
|
83
|
+
fetchImpl: params.fetchImpl,
|
|
84
|
+
});
|
|
85
|
+
|
|
86
|
+
const cacheKey = `${baseUrl}:${botToken}`;
|
|
87
|
+
const userId = await resolveBotUserId(client, cacheKey);
|
|
88
|
+
if (!userId) {
|
|
89
|
+
return { ok: false, error: "Mattermost reactions failed: could not resolve bot user id." };
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
try {
|
|
93
|
+
await deleteReaction(client, {
|
|
94
|
+
userId,
|
|
95
|
+
postId: params.postId,
|
|
96
|
+
emojiName: params.emojiName,
|
|
97
|
+
});
|
|
98
|
+
} catch (err) {
|
|
99
|
+
return { ok: false, error: `Mattermost remove reaction failed: ${String(err)}` };
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
return { ok: true };
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
async function createReaction(
|
|
106
|
+
client: MattermostClient,
|
|
107
|
+
params: { userId: string; postId: string; emojiName: string },
|
|
108
|
+
): Promise<void> {
|
|
109
|
+
await client.request<Record<string, unknown>>("/reactions", {
|
|
110
|
+
method: "POST",
|
|
111
|
+
body: JSON.stringify({
|
|
112
|
+
user_id: params.userId,
|
|
113
|
+
post_id: params.postId,
|
|
114
|
+
emoji_name: params.emojiName,
|
|
115
|
+
}),
|
|
116
|
+
});
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
async function deleteReaction(
|
|
120
|
+
client: MattermostClient,
|
|
121
|
+
params: { userId: string; postId: string; emojiName: string },
|
|
122
|
+
): Promise<void> {
|
|
123
|
+
const emoji = encodeURIComponent(params.emojiName);
|
|
124
|
+
await client.request<unknown>(
|
|
125
|
+
`/users/${params.userId}/posts/${params.postId}/reactions/${emoji}`,
|
|
126
|
+
{
|
|
127
|
+
method: "DELETE",
|
|
128
|
+
},
|
|
129
|
+
);
|
|
130
|
+
}
|
|
@@ -0,0 +1,103 @@
|
|
|
1
|
+
export type ReconnectOutcome = "resolved" | "rejected";
|
|
2
|
+
|
|
3
|
+
export type ShouldReconnectParams = {
|
|
4
|
+
attempt: number;
|
|
5
|
+
delayMs: number;
|
|
6
|
+
outcome: ReconnectOutcome;
|
|
7
|
+
error?: unknown;
|
|
8
|
+
};
|
|
9
|
+
|
|
10
|
+
export type RunWithReconnectOpts = {
|
|
11
|
+
abortSignal?: AbortSignal;
|
|
12
|
+
onError?: (err: unknown) => void;
|
|
13
|
+
onReconnect?: (delayMs: number) => void;
|
|
14
|
+
initialDelayMs?: number;
|
|
15
|
+
maxDelayMs?: number;
|
|
16
|
+
jitterRatio?: number;
|
|
17
|
+
random?: () => number;
|
|
18
|
+
shouldReconnect?: (params: ShouldReconnectParams) => boolean;
|
|
19
|
+
};
|
|
20
|
+
|
|
21
|
+
/**
|
|
22
|
+
* Reconnection loop with exponential backoff.
|
|
23
|
+
*
|
|
24
|
+
* Calls `connectFn` in a while loop. On normal resolve (connection closed),
|
|
25
|
+
* the backoff resets. On thrown error (connection failed), the current delay is
|
|
26
|
+
* used, then doubled for the next retry.
|
|
27
|
+
* The loop exits when `abortSignal` fires.
|
|
28
|
+
*/
|
|
29
|
+
export async function runWithReconnect(
|
|
30
|
+
connectFn: () => Promise<void>,
|
|
31
|
+
opts: RunWithReconnectOpts = {},
|
|
32
|
+
): Promise<void> {
|
|
33
|
+
const { initialDelayMs = 2000, maxDelayMs = 60_000 } = opts;
|
|
34
|
+
const jitterRatio = Math.max(0, opts.jitterRatio ?? 0);
|
|
35
|
+
const random = opts.random ?? Math.random;
|
|
36
|
+
let retryDelay = initialDelayMs;
|
|
37
|
+
let attempt = 0;
|
|
38
|
+
|
|
39
|
+
while (!opts.abortSignal?.aborted) {
|
|
40
|
+
let shouldIncreaseDelay = false;
|
|
41
|
+
let outcome: ReconnectOutcome = "resolved";
|
|
42
|
+
let error: unknown;
|
|
43
|
+
try {
|
|
44
|
+
await connectFn();
|
|
45
|
+
retryDelay = initialDelayMs;
|
|
46
|
+
} catch (err) {
|
|
47
|
+
if (opts.abortSignal?.aborted) {
|
|
48
|
+
return;
|
|
49
|
+
}
|
|
50
|
+
outcome = "rejected";
|
|
51
|
+
error = err;
|
|
52
|
+
opts.onError?.(err);
|
|
53
|
+
shouldIncreaseDelay = true;
|
|
54
|
+
}
|
|
55
|
+
if (opts.abortSignal?.aborted) {
|
|
56
|
+
return;
|
|
57
|
+
}
|
|
58
|
+
const delayMs = withJitter(retryDelay, jitterRatio, random);
|
|
59
|
+
const shouldReconnect =
|
|
60
|
+
opts.shouldReconnect?.({
|
|
61
|
+
attempt,
|
|
62
|
+
delayMs,
|
|
63
|
+
outcome,
|
|
64
|
+
error,
|
|
65
|
+
}) ?? true;
|
|
66
|
+
if (!shouldReconnect) {
|
|
67
|
+
return;
|
|
68
|
+
}
|
|
69
|
+
opts.onReconnect?.(delayMs);
|
|
70
|
+
await sleepAbortable(delayMs, opts.abortSignal);
|
|
71
|
+
if (shouldIncreaseDelay) {
|
|
72
|
+
retryDelay = Math.min(retryDelay * 2, maxDelayMs);
|
|
73
|
+
}
|
|
74
|
+
attempt++;
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
function withJitter(baseMs: number, jitterRatio: number, random: () => number): number {
|
|
79
|
+
if (jitterRatio <= 0) {
|
|
80
|
+
return baseMs;
|
|
81
|
+
}
|
|
82
|
+
const normalized = Math.max(0, Math.min(1, random()));
|
|
83
|
+
const spread = baseMs * jitterRatio;
|
|
84
|
+
return Math.max(1, Math.round(baseMs - spread + normalized * spread * 2));
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
function sleepAbortable(ms: number, signal?: AbortSignal): Promise<void> {
|
|
88
|
+
return new Promise((resolve) => {
|
|
89
|
+
if (signal?.aborted) {
|
|
90
|
+
resolve();
|
|
91
|
+
return;
|
|
92
|
+
}
|
|
93
|
+
const onAbort = () => {
|
|
94
|
+
clearTimeout(timer);
|
|
95
|
+
resolve();
|
|
96
|
+
};
|
|
97
|
+
const timer = setTimeout(() => {
|
|
98
|
+
signal?.removeEventListener("abort", onAbort);
|
|
99
|
+
resolve();
|
|
100
|
+
}, ms);
|
|
101
|
+
signal?.addEventListener("abort", onAbort, { once: true });
|
|
102
|
+
});
|
|
103
|
+
}
|
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
# MiniMax OAuth (Pool Bot plugin)
|
|
2
|
+
|
|
3
|
+
OAuth provider plugin for **MiniMax** (OAuth).
|
|
4
|
+
|
|
5
|
+
## Enable
|
|
6
|
+
|
|
7
|
+
Bundled plugins are disabled by default. Enable this one:
|
|
8
|
+
|
|
9
|
+
```bash
|
|
10
|
+
poolbot plugins enable minimax-portal-auth
|
|
11
|
+
```
|
|
12
|
+
|
|
13
|
+
Restart the Gateway after enabling.
|
|
14
|
+
|
|
15
|
+
```bash
|
|
16
|
+
poolbot gateway restart
|
|
17
|
+
```
|
|
18
|
+
|
|
19
|
+
## Authenticate
|
|
20
|
+
|
|
21
|
+
```bash
|
|
22
|
+
poolbot models auth login --provider minimax-portal --set-default
|
|
23
|
+
```
|
|
24
|
+
|
|
25
|
+
You will be prompted to select an endpoint:
|
|
26
|
+
|
|
27
|
+
- **Global** - International users, optimized for overseas access (`api.minimax.io`)
|
|
28
|
+
- **China** - Optimized for users in China (`api.minimaxi.com`)
|
|
29
|
+
|
|
30
|
+
## Notes
|
|
31
|
+
|
|
32
|
+
- MiniMax OAuth uses a user-code login flow.
|
|
33
|
+
- Currently, OAuth login is supported only for the Coding plan
|
|
@@ -0,0 +1,161 @@
|
|
|
1
|
+
import {
|
|
2
|
+
emptyPluginConfigSchema,
|
|
3
|
+
type PoolBotPluginApi,
|
|
4
|
+
type ProviderAuthContext,
|
|
5
|
+
type ProviderAuthResult,
|
|
6
|
+
} from "poolbot/plugin-sdk";
|
|
7
|
+
import { loginMiniMaxPortalOAuth, type MiniMaxRegion } from "./oauth.js";
|
|
8
|
+
|
|
9
|
+
const PROVIDER_ID = "minimax-portal";
|
|
10
|
+
const PROVIDER_LABEL = "MiniMax";
|
|
11
|
+
const DEFAULT_MODEL = "MiniMax-M2.5";
|
|
12
|
+
const DEFAULT_BASE_URL_CN = "https://api.minimaxi.com/anthropic";
|
|
13
|
+
const DEFAULT_BASE_URL_GLOBAL = "https://api.minimax.io/anthropic";
|
|
14
|
+
const DEFAULT_CONTEXT_WINDOW = 200000;
|
|
15
|
+
const DEFAULT_MAX_TOKENS = 8192;
|
|
16
|
+
const OAUTH_PLACEHOLDER = "minimax-oauth";
|
|
17
|
+
|
|
18
|
+
function getDefaultBaseUrl(region: MiniMaxRegion): string {
|
|
19
|
+
return region === "cn" ? DEFAULT_BASE_URL_CN : DEFAULT_BASE_URL_GLOBAL;
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
function modelRef(modelId: string): string {
|
|
23
|
+
return `${PROVIDER_ID}/${modelId}`;
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
function buildModelDefinition(params: {
|
|
27
|
+
id: string;
|
|
28
|
+
name: string;
|
|
29
|
+
input: Array<"text" | "image">;
|
|
30
|
+
reasoning?: boolean;
|
|
31
|
+
}) {
|
|
32
|
+
return {
|
|
33
|
+
id: params.id,
|
|
34
|
+
name: params.name,
|
|
35
|
+
reasoning: params.reasoning ?? false,
|
|
36
|
+
input: params.input,
|
|
37
|
+
cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 },
|
|
38
|
+
contextWindow: DEFAULT_CONTEXT_WINDOW,
|
|
39
|
+
maxTokens: DEFAULT_MAX_TOKENS,
|
|
40
|
+
};
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
function createOAuthHandler(region: MiniMaxRegion) {
|
|
44
|
+
const defaultBaseUrl = getDefaultBaseUrl(region);
|
|
45
|
+
const regionLabel = region === "cn" ? "CN" : "Global";
|
|
46
|
+
|
|
47
|
+
return async (ctx: ProviderAuthContext): Promise<ProviderAuthResult> => {
|
|
48
|
+
const progress = ctx.prompter.progress(`Starting MiniMax OAuth (${regionLabel})…`);
|
|
49
|
+
try {
|
|
50
|
+
const result = await loginMiniMaxPortalOAuth({
|
|
51
|
+
openUrl: ctx.openUrl,
|
|
52
|
+
note: ctx.prompter.note,
|
|
53
|
+
progress,
|
|
54
|
+
region,
|
|
55
|
+
});
|
|
56
|
+
|
|
57
|
+
progress.stop("MiniMax OAuth complete");
|
|
58
|
+
|
|
59
|
+
if (result.notification_message) {
|
|
60
|
+
await ctx.prompter.note(result.notification_message, "MiniMax OAuth");
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
const profileId = `${PROVIDER_ID}:default`;
|
|
64
|
+
const baseUrl = result.resourceUrl || defaultBaseUrl;
|
|
65
|
+
|
|
66
|
+
return {
|
|
67
|
+
profiles: [
|
|
68
|
+
{
|
|
69
|
+
profileId,
|
|
70
|
+
credential: {
|
|
71
|
+
type: "oauth" as const,
|
|
72
|
+
provider: PROVIDER_ID,
|
|
73
|
+
access: result.access,
|
|
74
|
+
refresh: result.refresh,
|
|
75
|
+
expires: result.expires,
|
|
76
|
+
},
|
|
77
|
+
},
|
|
78
|
+
],
|
|
79
|
+
configPatch: {
|
|
80
|
+
models: {
|
|
81
|
+
providers: {
|
|
82
|
+
[PROVIDER_ID]: {
|
|
83
|
+
baseUrl,
|
|
84
|
+
apiKey: OAUTH_PLACEHOLDER,
|
|
85
|
+
api: "anthropic-messages",
|
|
86
|
+
models: [
|
|
87
|
+
buildModelDefinition({
|
|
88
|
+
id: "MiniMax-M2.1",
|
|
89
|
+
name: "MiniMax M2.1",
|
|
90
|
+
input: ["text"],
|
|
91
|
+
}),
|
|
92
|
+
buildModelDefinition({
|
|
93
|
+
id: "MiniMax-M2.5",
|
|
94
|
+
name: "MiniMax M2.5",
|
|
95
|
+
input: ["text"],
|
|
96
|
+
reasoning: true,
|
|
97
|
+
}),
|
|
98
|
+
],
|
|
99
|
+
},
|
|
100
|
+
},
|
|
101
|
+
},
|
|
102
|
+
agents: {
|
|
103
|
+
defaults: {
|
|
104
|
+
models: {
|
|
105
|
+
[modelRef("MiniMax-M2.1")]: { alias: "minimax-m2.1" },
|
|
106
|
+
[modelRef("MiniMax-M2.5")]: { alias: "minimax-m2.5" },
|
|
107
|
+
},
|
|
108
|
+
},
|
|
109
|
+
},
|
|
110
|
+
},
|
|
111
|
+
defaultModel: modelRef(DEFAULT_MODEL),
|
|
112
|
+
notes: [
|
|
113
|
+
"MiniMax OAuth tokens auto-refresh. Re-run login if refresh fails or access is revoked.",
|
|
114
|
+
`Base URL defaults to ${defaultBaseUrl}. Override models.providers.${PROVIDER_ID}.baseUrl if needed.`,
|
|
115
|
+
...(result.notification_message ? [result.notification_message] : []),
|
|
116
|
+
],
|
|
117
|
+
};
|
|
118
|
+
} catch (err) {
|
|
119
|
+
const errorMsg = err instanceof Error ? err.message : String(err);
|
|
120
|
+
progress.stop(`MiniMax OAuth failed: ${errorMsg}`);
|
|
121
|
+
await ctx.prompter.note(
|
|
122
|
+
"If OAuth fails, verify your MiniMax account has portal access and try again.",
|
|
123
|
+
"MiniMax OAuth",
|
|
124
|
+
);
|
|
125
|
+
throw err;
|
|
126
|
+
}
|
|
127
|
+
};
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
const minimaxPortalPlugin = {
|
|
131
|
+
id: "minimax-portal-auth",
|
|
132
|
+
name: "MiniMax OAuth",
|
|
133
|
+
description: "OAuth flow for MiniMax models",
|
|
134
|
+
configSchema: emptyPluginConfigSchema(),
|
|
135
|
+
register(api: PoolBotPluginApi) {
|
|
136
|
+
api.registerProvider({
|
|
137
|
+
id: PROVIDER_ID,
|
|
138
|
+
label: PROVIDER_LABEL,
|
|
139
|
+
docsPath: "/providers/minimax",
|
|
140
|
+
aliases: ["minimax"],
|
|
141
|
+
auth: [
|
|
142
|
+
{
|
|
143
|
+
id: "oauth",
|
|
144
|
+
label: "MiniMax OAuth (Global)",
|
|
145
|
+
hint: "Global endpoint - api.minimax.io",
|
|
146
|
+
kind: "device_code",
|
|
147
|
+
run: createOAuthHandler("global"),
|
|
148
|
+
},
|
|
149
|
+
{
|
|
150
|
+
id: "oauth-cn",
|
|
151
|
+
label: "MiniMax OAuth (CN)",
|
|
152
|
+
hint: "CN endpoint - api.minimaxi.com",
|
|
153
|
+
kind: "device_code",
|
|
154
|
+
run: createOAuthHandler("cn"),
|
|
155
|
+
},
|
|
156
|
+
],
|
|
157
|
+
});
|
|
158
|
+
},
|
|
159
|
+
};
|
|
160
|
+
|
|
161
|
+
export default minimaxPortalPlugin;
|
|
@@ -0,0 +1,247 @@
|
|
|
1
|
+
import { createHash, randomBytes, randomUUID } from "node:crypto";
|
|
2
|
+
|
|
3
|
+
export type MiniMaxRegion = "cn" | "global";
|
|
4
|
+
|
|
5
|
+
const MINIMAX_OAUTH_CONFIG = {
|
|
6
|
+
cn: {
|
|
7
|
+
baseUrl: "https://api.minimaxi.com",
|
|
8
|
+
clientId: "78257093-7e40-4613-99e0-527b14b39113",
|
|
9
|
+
},
|
|
10
|
+
global: {
|
|
11
|
+
baseUrl: "https://api.minimax.io",
|
|
12
|
+
clientId: "78257093-7e40-4613-99e0-527b14b39113",
|
|
13
|
+
},
|
|
14
|
+
} as const;
|
|
15
|
+
|
|
16
|
+
const MINIMAX_OAUTH_SCOPE = "group_id profile model.completion";
|
|
17
|
+
const MINIMAX_OAUTH_GRANT_TYPE = "urn:ietf:params:oauth:grant-type:user_code";
|
|
18
|
+
|
|
19
|
+
function getOAuthEndpoints(region: MiniMaxRegion) {
|
|
20
|
+
const config = MINIMAX_OAUTH_CONFIG[region];
|
|
21
|
+
return {
|
|
22
|
+
codeEndpoint: `${config.baseUrl}/oauth/code`,
|
|
23
|
+
tokenEndpoint: `${config.baseUrl}/oauth/token`,
|
|
24
|
+
clientId: config.clientId,
|
|
25
|
+
baseUrl: config.baseUrl,
|
|
26
|
+
};
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
export type MiniMaxOAuthAuthorization = {
|
|
30
|
+
user_code: string;
|
|
31
|
+
verification_uri: string;
|
|
32
|
+
expired_in: number;
|
|
33
|
+
interval?: number;
|
|
34
|
+
state: string;
|
|
35
|
+
};
|
|
36
|
+
|
|
37
|
+
export type MiniMaxOAuthToken = {
|
|
38
|
+
access: string;
|
|
39
|
+
refresh: string;
|
|
40
|
+
expires: number;
|
|
41
|
+
resourceUrl?: string;
|
|
42
|
+
notification_message?: string;
|
|
43
|
+
};
|
|
44
|
+
|
|
45
|
+
type TokenPending = { status: "pending"; message?: string };
|
|
46
|
+
|
|
47
|
+
type TokenResult =
|
|
48
|
+
| { status: "success"; token: MiniMaxOAuthToken }
|
|
49
|
+
| TokenPending
|
|
50
|
+
| { status: "error"; message: string };
|
|
51
|
+
|
|
52
|
+
function toFormUrlEncoded(data: Record<string, string>): string {
|
|
53
|
+
return Object.entries(data)
|
|
54
|
+
.map(([key, value]) => `${encodeURIComponent(key)}=${encodeURIComponent(value)}`)
|
|
55
|
+
.join("&");
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
function generatePkce(): { verifier: string; challenge: string; state: string } {
|
|
59
|
+
const verifier = randomBytes(32).toString("base64url");
|
|
60
|
+
const challenge = createHash("sha256").update(verifier).digest("base64url");
|
|
61
|
+
const state = randomBytes(16).toString("base64url");
|
|
62
|
+
return { verifier, challenge, state };
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
async function requestOAuthCode(params: {
|
|
66
|
+
challenge: string;
|
|
67
|
+
state: string;
|
|
68
|
+
region: MiniMaxRegion;
|
|
69
|
+
}): Promise<MiniMaxOAuthAuthorization> {
|
|
70
|
+
const endpoints = getOAuthEndpoints(params.region);
|
|
71
|
+
const response = await fetch(endpoints.codeEndpoint, {
|
|
72
|
+
method: "POST",
|
|
73
|
+
headers: {
|
|
74
|
+
"Content-Type": "application/x-www-form-urlencoded",
|
|
75
|
+
Accept: "application/json",
|
|
76
|
+
"x-request-id": randomUUID(),
|
|
77
|
+
},
|
|
78
|
+
body: toFormUrlEncoded({
|
|
79
|
+
response_type: "code",
|
|
80
|
+
client_id: endpoints.clientId,
|
|
81
|
+
scope: MINIMAX_OAUTH_SCOPE,
|
|
82
|
+
code_challenge: params.challenge,
|
|
83
|
+
code_challenge_method: "S256",
|
|
84
|
+
state: params.state,
|
|
85
|
+
}),
|
|
86
|
+
});
|
|
87
|
+
|
|
88
|
+
if (!response.ok) {
|
|
89
|
+
const text = await response.text();
|
|
90
|
+
throw new Error(`MiniMax OAuth authorization failed: ${text || response.statusText}`);
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
const payload = (await response.json()) as MiniMaxOAuthAuthorization & { error?: string };
|
|
94
|
+
if (!payload.user_code || !payload.verification_uri) {
|
|
95
|
+
throw new Error(
|
|
96
|
+
payload.error ??
|
|
97
|
+
"MiniMax OAuth authorization returned an incomplete payload (missing user_code or verification_uri).",
|
|
98
|
+
);
|
|
99
|
+
}
|
|
100
|
+
if (payload.state !== params.state) {
|
|
101
|
+
throw new Error("MiniMax OAuth state mismatch: possible CSRF attack or session corruption.");
|
|
102
|
+
}
|
|
103
|
+
return payload;
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
async function pollOAuthToken(params: {
|
|
107
|
+
userCode: string;
|
|
108
|
+
verifier: string;
|
|
109
|
+
region: MiniMaxRegion;
|
|
110
|
+
}): Promise<TokenResult> {
|
|
111
|
+
const endpoints = getOAuthEndpoints(params.region);
|
|
112
|
+
const response = await fetch(endpoints.tokenEndpoint, {
|
|
113
|
+
method: "POST",
|
|
114
|
+
headers: {
|
|
115
|
+
"Content-Type": "application/x-www-form-urlencoded",
|
|
116
|
+
Accept: "application/json",
|
|
117
|
+
},
|
|
118
|
+
body: toFormUrlEncoded({
|
|
119
|
+
grant_type: MINIMAX_OAUTH_GRANT_TYPE,
|
|
120
|
+
client_id: endpoints.clientId,
|
|
121
|
+
user_code: params.userCode,
|
|
122
|
+
code_verifier: params.verifier,
|
|
123
|
+
}),
|
|
124
|
+
});
|
|
125
|
+
|
|
126
|
+
const text = await response.text();
|
|
127
|
+
let payload:
|
|
128
|
+
| {
|
|
129
|
+
status?: string;
|
|
130
|
+
base_resp?: { status_code?: number; status_msg?: string };
|
|
131
|
+
}
|
|
132
|
+
| undefined;
|
|
133
|
+
if (text) {
|
|
134
|
+
try {
|
|
135
|
+
payload = JSON.parse(text) as typeof payload;
|
|
136
|
+
} catch {
|
|
137
|
+
payload = undefined;
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
if (!response.ok) {
|
|
142
|
+
return {
|
|
143
|
+
status: "error",
|
|
144
|
+
message:
|
|
145
|
+
(payload?.base_resp?.status_msg ?? text) || "MiniMax OAuth failed to parse response.",
|
|
146
|
+
};
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
if (!payload) {
|
|
150
|
+
return { status: "error", message: "MiniMax OAuth failed to parse response." };
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
const tokenPayload = payload as {
|
|
154
|
+
status: string;
|
|
155
|
+
access_token?: string | null;
|
|
156
|
+
refresh_token?: string | null;
|
|
157
|
+
expired_in?: number | null;
|
|
158
|
+
token_type?: string;
|
|
159
|
+
resource_url?: string;
|
|
160
|
+
notification_message?: string;
|
|
161
|
+
};
|
|
162
|
+
|
|
163
|
+
if (tokenPayload.status === "error") {
|
|
164
|
+
return { status: "error", message: "An error occurred. Please try again later" };
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
if (tokenPayload.status != "success") {
|
|
168
|
+
return { status: "pending", message: "current user code is not authorized" };
|
|
169
|
+
}
|
|
170
|
+
|
|
171
|
+
if (!tokenPayload.access_token || !tokenPayload.refresh_token || !tokenPayload.expired_in) {
|
|
172
|
+
return { status: "error", message: "MiniMax OAuth returned incomplete token payload." };
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
return {
|
|
176
|
+
status: "success",
|
|
177
|
+
token: {
|
|
178
|
+
access: tokenPayload.access_token,
|
|
179
|
+
refresh: tokenPayload.refresh_token,
|
|
180
|
+
expires: tokenPayload.expired_in,
|
|
181
|
+
resourceUrl: tokenPayload.resource_url,
|
|
182
|
+
notification_message: tokenPayload.notification_message,
|
|
183
|
+
},
|
|
184
|
+
};
|
|
185
|
+
}
|
|
186
|
+
|
|
187
|
+
export async function loginMiniMaxPortalOAuth(params: {
|
|
188
|
+
openUrl: (url: string) => Promise<void>;
|
|
189
|
+
note: (message: string, title?: string) => Promise<void>;
|
|
190
|
+
progress: { update: (message: string) => void; stop: (message?: string) => void };
|
|
191
|
+
region?: MiniMaxRegion;
|
|
192
|
+
}): Promise<MiniMaxOAuthToken> {
|
|
193
|
+
const region = params.region ?? "global";
|
|
194
|
+
const { verifier, challenge, state } = generatePkce();
|
|
195
|
+
const oauth = await requestOAuthCode({ challenge, state, region });
|
|
196
|
+
const verificationUrl = oauth.verification_uri;
|
|
197
|
+
|
|
198
|
+
const noteLines = [
|
|
199
|
+
`Open ${verificationUrl} to approve access.`,
|
|
200
|
+
`If prompted, enter the code ${oauth.user_code}.`,
|
|
201
|
+
`Interval: ${oauth.interval ?? "default (2000ms)"}, Expires at: ${oauth.expired_in} unix timestamp`,
|
|
202
|
+
];
|
|
203
|
+
await params.note(noteLines.join("\n"), "MiniMax OAuth");
|
|
204
|
+
|
|
205
|
+
try {
|
|
206
|
+
await params.openUrl(verificationUrl);
|
|
207
|
+
} catch {
|
|
208
|
+
// Fall back to manual copy/paste if browser open fails.
|
|
209
|
+
}
|
|
210
|
+
|
|
211
|
+
let pollIntervalMs = oauth.interval ? oauth.interval : 2000;
|
|
212
|
+
const expireTimeMs = oauth.expired_in;
|
|
213
|
+
|
|
214
|
+
while (Date.now() < expireTimeMs) {
|
|
215
|
+
params.progress.update("Waiting for MiniMax OAuth approval…");
|
|
216
|
+
const result = await pollOAuthToken({
|
|
217
|
+
userCode: oauth.user_code,
|
|
218
|
+
verifier,
|
|
219
|
+
region,
|
|
220
|
+
});
|
|
221
|
+
|
|
222
|
+
// // Debug: print poll result
|
|
223
|
+
// await params.note(
|
|
224
|
+
// `status: ${result.status}` +
|
|
225
|
+
// (result.status === "success" ? `\ntoken: ${JSON.stringify(result.token, null, 2)}` : "") +
|
|
226
|
+
// (result.status === "error" ? `\nmessage: ${result.message}` : "") +
|
|
227
|
+
// (result.status === "pending" && result.message ? `\nmessage: ${result.message}` : ""),
|
|
228
|
+
// "MiniMax OAuth Poll Result",
|
|
229
|
+
// );
|
|
230
|
+
|
|
231
|
+
if (result.status === "success") {
|
|
232
|
+
return result.token;
|
|
233
|
+
}
|
|
234
|
+
|
|
235
|
+
if (result.status === "error") {
|
|
236
|
+
throw new Error(`MiniMax OAuth failed: ${result.message}`);
|
|
237
|
+
}
|
|
238
|
+
|
|
239
|
+
if (result.status === "pending") {
|
|
240
|
+
pollIntervalMs = Math.min(pollIntervalMs * 1.5, 10000);
|
|
241
|
+
}
|
|
242
|
+
|
|
243
|
+
await new Promise((resolve) => setTimeout(resolve, pollIntervalMs));
|
|
244
|
+
}
|
|
245
|
+
|
|
246
|
+
throw new Error("MiniMax OAuth timed out waiting for authorization.");
|
|
247
|
+
}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "@poolzin/minimax-portal-auth",
|
|
3
|
+
"version": "2026.2.18",
|
|
4
|
+
"private": true,
|
|
5
|
+
"description": "Pool Bot MiniMax Portal OAuth provider plugin",
|
|
6
|
+
"type": "module",
|
|
7
|
+
"devDependencies": {
|
|
8
|
+
"poolbot": "workspace:*"
|
|
9
|
+
},
|
|
10
|
+
"poolbot": {
|
|
11
|
+
"extensions": [
|
|
12
|
+
"./index.ts"
|
|
13
|
+
]
|
|
14
|
+
}
|
|
15
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export { withFileLock } from "poolbot/plugin-sdk";
|