Skip to content

fix(web,server): show real platform connection status in Settings (closes #1031)#1061

Merged
Wirasm merged 1 commit into
coleam00:devfrom
liorfranko:fix/platform-connections-status
Apr 21, 2026
Merged

fix(web,server): show real platform connection status in Settings (closes #1031)#1061
Wirasm merged 1 commit into
coleam00:devfrom
liorfranko:fix/platform-connections-status

Conversation

@liorfranko

@liorfranko liorfranko commented Apr 10, 2026

Copy link
Copy Markdown
Contributor

Summary

  • Problem: The Settings page's "Platform Connections" section hardcoded every platform except Web to "Not configured", so users couldn't tell whether their Slack/Telegram/Discord/GitHub/Gitea/GitLab adapters had actually started.
  • Why it matters: Health/UX — config errors go undetected until a message arrives (or doesn't).
  • What changed:
    • Server: /api/health now returns an activePlatforms array populated live as each adapter's start() resolves. The live reference is passed into registerApiRoutes because Telegram starts after the HTTP listener is already accepting requests, so a snapshot would miss it.
    • Web: SettingsPage.PlatformConnectionsSection reads activePlatforms and looks each platform up in a Set. Also adds Gitea and GitLab to the list (they already ship as adapters).
  • What did not change (scope boundary): No changes to the Telegram adapter — the telegraf/grammY startup semantics fix that was part of the original PR is no longer needed since dev migrated to grammY and now uses the onStart callback in packages/adapters/src/chat/telegram/adapter.ts. That portion of the original PR has been dropped on rebase.

UX Journey

Before

  User                 Server                   Web UI (Settings)
  ────                 ──────                   ─────────────────
  opens Settings ────▶ GET /api/health
                       returns { adapter } ────▶ shows only Web
                                                 others hardcoded 'Not configured' ❌

After

  User                 Server                   Web UI (Settings)
  ────                 ──────                   ─────────────────
  opens Settings ────▶ GET /api/health
                       returns { adapter,
                              *activePlatforms*: ['Web','Slack',...] } ────▶ reflects real state ✅

Architecture Diagram

Before

startServer
├─ adapter.start() (each platform)
└─ (log-only) activePlatforms computed at end       registerApiRoutes
                                                    └─ GET /api/health → { adapter }

After

startServer
├─ [+] const activePlatforms: string[] = ['Web']
├─ adapter.start(); [+] activePlatforms.push(name)   registerApiRoutes(..., [+] activePlatforms)
└─ log activePlatforms (reuses the live array)       └─ GET /api/health → { adapter, [+] activePlatforms }

Connection inventory:

From To Status Notes
startServer registerApiRoutes(activePlatforms) modified Live array reference
each adapter start() activePlatforms.push(...) new Populated as they start
GET /api/health HealthResponse.activePlatforms new Optional field, defaults to ['Web']
SettingsPage health.activePlatforms modified Reads real data (was hardcoded false)

Label Snapshot

  • Risk: risk: low
  • Size: size: S
  • Scope: server, web
  • Module: server:index, server:routes/api, web:SettingsPage

Change Metadata

  • Change type: fix
  • Primary scope: multi (server + web)

Linked Issue

Validation Evidence (required)

bun run type-check              # ✅ 10 packages
bun run lint                    # ✅ 0 errors, 0 warnings
bun --filter @archon/server test  # ✅ 46 tests pass

Security Impact (required)

  • New permissions/capabilities? No
  • New external network calls? No
  • Secrets/tokens handling changed? No
  • File system access scope changed? No

activePlatforms exposes only platform names (e.g. "Slack") — no tokens or config values.

Compatibility / Migration

  • Backward compatible? YesactivePlatforms is an optional field; clients that ignore it see no change.
  • Config/env changes? No
  • Database migration needed? No

Human Verification (required)

  • Verified scenarios: type-check / lint / @archon/server test suite pass locally.
  • Edge cases checked: Telegram starts after registerApiRoutes runs — the live array reference ensures /api/health reflects its state once it pushes. getHealth() without platforms configured returns ['Web'] as before.
  • What was not verified: Live browser check of the Settings page rendering against a running server.

Side Effects / Blast Radius (required)

  • Affected subsystems: /api/health, Settings page.
  • Potential unintended effects: None. The array is read-only from the handler's perspective; a copy is emitted each response.
  • Guardrails: Existing adapter-start logs (*.bot_started, github_webhook_registered, etc.) remain the source of truth for startup observability.

Rollback Plan (required)

  • Fast rollback: git revert <commit> — single commit, 4 files, contained.
  • Feature flags or config toggles: N/A.
  • Observable failure symptoms: Settings page returns to showing everything as "Not configured".

Risks and Mitigations

  • Risk: Telegram adapter fails to start but earlier code path already pushes 'Telegram' to the array.
    • Mitigation: Push is inside the try block after await telegramAdapter.start(); failure skips the push (matches pre-existing telegram = null handling).

Rebased from @liorfranko's original PR onto current dev. The adapter changes were dropped because dev migrated from telegraf to grammY and already resolves the bot.launch() never-resolves issue via the onStart callback — so that fix is no longer needed. The server + web portions remain and are credited to @liorfranko.

Summary by CodeRabbit

  • New Features

    • Settings panel now displays real-time connection status for all active platforms, including Gitea and GitLab.
    • Health check endpoint reports which platforms are currently active.
  • Bug Fixes

    • Platform connection status now reflects actual running platforms instead of hardcoded defaults.

@coderabbitai

coderabbitai Bot commented Apr 10, 2026

Copy link
Copy Markdown

No actionable comments were generated in the recent review. 🎉

ℹ️ Recent review info
⚙️ Run configuration

Configuration used: defaults

Review profile: CHILL

Plan: Pro

Run ID: 5d393dad-6bdf-4796-88a1-cb103b926846

📥 Commits

Reviewing files that changed from the base of the PR and between 6a816a9 and c31e481.

📒 Files selected for processing (4)
  • packages/server/src/index.ts
  • packages/server/src/routes/api.ts
  • packages/web/src/lib/api.ts
  • packages/web/src/routes/SettingsPage.tsx
🚧 Files skipped from review as they are similar to previous changes (3)
  • packages/web/src/routes/SettingsPage.tsx
  • packages/server/src/routes/api.ts
  • packages/server/src/index.ts

📝 Walkthrough

Walkthrough

A mutable activePlatforms array is initialized at server startup and populated as each adapter successfully starts. The shared array is passed to registerApiRoutes where it's exposed via the /api/health endpoint. The web UI consumes this data to display real-time platform connection status.

Changes

Cohort / File(s) Summary
Server-side platform tracking
packages/server/src/index.ts, packages/server/src/routes/api.ts
Introduces activePlatforms array initialized to ['Web'], populated incrementally as adapters start, and passed to API routes. /api/health endpoint extended with optional activePlatforms field in response schema and handler logic.
Web platform status integration
packages/web/src/lib/api.ts, packages/web/src/routes/SettingsPage.tsx
HealthResponse type updated with optional activePlatforms field. PlatformConnectionsSection refactored to determine platform connection status by checking presence in activePlatforms set, expanding rendered platform list to include Gitea and GitLab.

Estimated code review effort

🎯 2 (Simple) | ⏱️ ~12 minutes

Possibly related issues

Poem

🐰 The platforms gather, one by one,
Through startup dance till all are done!
In health responses, bright and clear,
The web now knows which platforms appear! ✨

🚥 Pre-merge checks | ✅ 2 | ❌ 1

❌ Failed checks (1 warning)

Check name Status Explanation Resolution
Docstring Coverage ⚠️ Warning Docstring coverage is 14.29% which is insufficient. The required threshold is 80.00%. Write docstrings for the functions missing them to satisfy the coverage threshold.
✅ Passed checks (2 passed)
Check name Status Explanation
Title check ✅ Passed The title clearly and concisely summarizes the main fix: showing real platform connection status in the Settings page, referencing the closed issue.
Description check ✅ Passed The description comprehensively covers all required template sections including summary, UX journey, architecture diagram, validation evidence, security impact, compatibility, human verification, side effects, and rollback plan.

✏️ Tip: You can configure your own custom pre-merge checks in the settings.

✨ Finishing Touches
🧪 Generate unit tests (beta)
  • Create PR with unit tests

Thanks for using CodeRabbit! It's free for OSS, and your support helps us grow. If you like it, consider giving us a shout-out.

❤️ Share

Comment @coderabbitai help to get the list of available commands and usage tips.

@coderabbitai coderabbitai Bot left a comment

Copy link
Copy Markdown

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 3

🤖 Prompt for all review comments with AI agents
Verify each finding against the current code and only fix it if needed.

Inline comments:
In `@packages/adapters/src/chat/telegram/adapter.ts`:
- Around line 196-217: The promise currently resolves inside the telegraf
onLaunch callback (this.bot.launch(..., () => { ... resolve() })), but onLaunch
fires before polling/getUpdates begin so 409 errors surface after resolve and
bypass the retry try/catch; change the logic so the outer promise does not
resolve on onLaunch but only after polling is actually started or the launch()
promise resolves/rejects: e.g., remove resolving from the onLaunch callback and
instead await/attach to the Promise returned by this.bot.launch(), or listen for
an event that confirms startPolling (or handle 'polling_error') and resolve only
when polling is confirmed, ensuring .catch on the launch promise can reject and
trigger the existing retry loop (relate to this.bot.launch, onLaunch,
startPolling, getUpdates).

In `@packages/server/src/index.ts`:
- Around line 490-494: activePlatforms is initialized once but not updated when
other adapters finish starting, causing /api/health to report stale data; after
each adapter successfully starts (e.g., when calling start() on
Telegram/Discord/etc.), immediately push that adapter's platform identifier into
the existing activePlatforms array (use the same array passed into
registerApiRoutes) — for example, in the promise resolution or async/await right
after adapter.start() do activePlatforms.push('Telegram') or
activePlatforms.push(adapter.platformName) so the health endpoint reflects
adapters as soon as they are up (ensure this change is applied in each adapter
startup location that currently only starts the adapter without mutating
activePlatforms).

In `@packages/web/src/routes/SettingsPage.tsx`:
- Around line 615-619: The settings UI is missing the server-reported platforms
Gitea and GitLab, so update the static platforms list in SettingsPage (the array
that currently contains { name: 'Web', connected: active.has('Web') }, ... {
name: 'GitHub', connected: active.has('GitHub') }) to include { name: 'Gitea',
connected: active.has('Gitea') } and { name: 'GitLab', connected:
active.has('GitLab') }, or replace the hardcoded list with a dynamic map derived
from the server-provided active set (using the existing active variable and
active.has checks) so the UI reflects whatever platforms the server reports.
🪄 Autofix (Beta)

Fix all unresolved CodeRabbit comments on this PR:

  • Push a commit to this branch (recommended)
  • Create a new PR with the fixes

ℹ️ Review info
⚙️ Run configuration

Configuration used: defaults

Review profile: CHILL

Plan: Pro

Run ID: 4fe1e94b-bdd9-4deb-a021-a4c4d40092e2

📥 Commits

Reviewing files that changed from the base of the PR and between e8334b3 and 50be822.

📒 Files selected for processing (5)
  • packages/adapters/src/chat/telegram/adapter.ts
  • packages/server/src/index.ts
  • packages/server/src/routes/api.ts
  • packages/web/src/lib/api.ts
  • packages/web/src/routes/SettingsPage.tsx

Comment thread packages/adapters/src/chat/telegram/adapter.ts Outdated
Comment thread packages/server/src/index.ts Outdated
Comment thread packages/web/src/routes/SettingsPage.tsx

@coderabbitai coderabbitai Bot left a comment

Copy link
Copy Markdown

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 1

🧹 Nitpick comments (1)
packages/adapters/src/chat/telegram/adapter.test.ts (1)

247-258: Add one regression test for a 409 that happens after onLaunch.

These cases still only exercise immediate launch() rejection. The new production path is the opposite ordering: onLaunch fires, then polling startup fails during the grace window. Without that case, the core race this PR fixes is still unproven.

🧪 Suggested test shape
+    test('should retry when polling fails after onLaunch but before grace settles', async () => {
+      const adapter = new TelegramAdapter('fake-token-for-testing');
+      const conflictError = new Error('409: Conflict: terminated by other getUpdates request');
+      let callCount = 0;
+
+      const mockLaunch = mock((_config: unknown, onLaunch?: () => void): Promise<void> => {
+        callCount++;
+        if (callCount === 1) {
+          onLaunch?.();
+          return new Promise((_, reject) => setTimeout(() => reject(conflictError), 0));
+        }
+        onLaunch?.();
+        return new Promise(() => {});
+      });
+
+      (adapter.getBot() as unknown as { launch: typeof mockLaunch }).launch = mockLaunch;
+
+      await adapter.start({ retryDelayMs: 0, pollingGraceMs: 10 });
+      expect(mockLaunch).toHaveBeenCalledTimes(2);
+    });

As per coding guidelines "Prefer reproducible commands in CI-sensitive paths; keep tests deterministic without flaky timing or network dependence; ensure local validation commands map to CI expectations."

Also applies to: 286-296, 305-310

🤖 Prompt for AI Agents
Verify each finding against the current code and only fix it if needed.

In `@packages/adapters/src/chat/telegram/adapter.test.ts` around lines 247 - 258,
Add a deterministic regression test that simulates a 409 Conflict occurring
after the telegraf onLaunch callback (to exercise the race where polling fails
during the grace window) by updating the mocked launch used in the existing
test: have mockLaunch call onLaunch() immediately, then simulate a polling-start
failure that rejects with new Error('409: Conflict: terminated by other
getUpdates request') at a controlled time (within the pollingGraceMs window)
instead of rejecting immediately; use adapter.getBot().launch = mockLaunch and
call adapter.start({ retryDelayMs: 0, pollingGraceMs: 0 or small value }) and
assert the adapter handles the post-onLaunch 409 the same as the pre-onLaunch
case (retries/continues) so the race is exercised deterministically.
🤖 Prompt for all review comments with AI agents
Verify each finding against the current code and only fix it if needed.

Inline comments:
In `@packages/web/src/routes/SettingsPage.tsx`:
- Around line 608-623: The platforms list currently marks integrations as not
connected whenever activePlatforms is undefined; change the connected value
computation in PlatformConnectionsSection so it is tri-state
(true/false/undefined) by setting connected: activePlatforms ? active.has('Web')
: undefined (and likewise for each platform name) instead of active.has(...),
and update the component rendering logic that consumes platforms (the code
reading platforms[].connected) to display an "Unknown"/"Loading" state when
connected is undefined rather than showing "Not configured".

---

Nitpick comments:
In `@packages/adapters/src/chat/telegram/adapter.test.ts`:
- Around line 247-258: Add a deterministic regression test that simulates a 409
Conflict occurring after the telegraf onLaunch callback (to exercise the race
where polling fails during the grace window) by updating the mocked launch used
in the existing test: have mockLaunch call onLaunch() immediately, then simulate
a polling-start failure that rejects with new Error('409: Conflict: terminated
by other getUpdates request') at a controlled time (within the pollingGraceMs
window) instead of rejecting immediately; use adapter.getBot().launch =
mockLaunch and call adapter.start({ retryDelayMs: 0, pollingGraceMs: 0 or small
value }) and assert the adapter handles the post-onLaunch 409 the same as the
pre-onLaunch case (retries/continues) so the race is exercised
deterministically.
🪄 Autofix (Beta)

Fix all unresolved CodeRabbit comments on this PR:

  • Push a commit to this branch (recommended)
  • Create a new PR with the fixes

ℹ️ Review info
⚙️ Run configuration

Configuration used: defaults

Review profile: CHILL

Plan: Pro

Run ID: 2bfa8b72-4bd9-4f89-bf0c-c6d27687bf9f

📥 Commits

Reviewing files that changed from the base of the PR and between 50be822 and 6a816a9.

📒 Files selected for processing (4)
  • packages/adapters/src/chat/telegram/adapter.test.ts
  • packages/adapters/src/chat/telegram/adapter.ts
  • packages/server/src/index.ts
  • packages/web/src/routes/SettingsPage.tsx
🚧 Files skipped from review as they are similar to previous changes (1)
  • packages/server/src/index.ts

Comment on lines 608 to 623
function PlatformConnectionsSection({
adapter,
activePlatforms,
}: {
adapter: string | undefined;
activePlatforms: string[] | undefined;
}): React.ReactElement {
const active = new Set(activePlatforms ?? []);
const platforms = [
{ name: 'Web', connected: adapter === 'web' },
{ name: 'Slack', connected: false },
{ name: 'Telegram', connected: false },
{ name: 'Discord', connected: false },
{ name: 'GitHub', connected: false },
{ name: 'Web', connected: active.has('Web') },
{ name: 'Slack', connected: active.has('Slack') },
{ name: 'Telegram', connected: active.has('Telegram') },
{ name: 'Discord', connected: active.has('Discord') },
{ name: 'GitHub', connected: active.has('GitHub') },
{ name: 'Gitea', connected: active.has('Gitea') },
{ name: 'GitLab', connected: active.has('GitLab') },
];

Copy link
Copy Markdown

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

⚠️ Potential issue | 🟡 Minor

Avoid showing “Not configured” before health data exists.

When activePlatforms is still undefined (initial load or health fetch failure), this renders every integration as Not configured. That reintroduces a false status during the exact window this PR is trying to fix. Render an Unknown/Loading state until the health payload arrives.

💡 Suggested tweak
 function PlatformConnectionsSection({
   activePlatforms,
 }: {
   activePlatforms: string[] | undefined;
 }): React.ReactElement {
+  const hasHealthData = activePlatforms !== undefined;
   const active = new Set(activePlatforms ?? []);
   const platforms = [
     { name: 'Web', connected: active.has('Web') },
     { name: 'Slack', connected: active.has('Slack') },
     { name: 'Telegram', connected: active.has('Telegram') },
@@
           {platforms.map(p => (
             <div key={p.name} className="flex items-center justify-between text-sm">
               <span>{p.name}</span>
-              <Badge variant={p.connected ? 'default' : 'secondary'}>
-                {p.connected ? 'Connected' : 'Not configured'}
+              <Badge variant={hasHealthData && p.connected ? 'default' : 'secondary'}>
+                {!hasHealthData ? 'Unknown' : p.connected ? 'Connected' : 'Not configured'}
               </Badge>
             </div>
           ))}
🤖 Prompt for AI Agents
Verify each finding against the current code and only fix it if needed.

In `@packages/web/src/routes/SettingsPage.tsx` around lines 608 - 623, The
platforms list currently marks integrations as not connected whenever
activePlatforms is undefined; change the connected value computation in
PlatformConnectionsSection so it is tri-state (true/false/undefined) by setting
connected: activePlatforms ? active.has('Web') : undefined (and likewise for
each platform name) instead of active.has(...), and update the component
rendering logic that consumes platforms (the code reading platforms[].connected)
to display an "Unknown"/"Loading" state when connected is undefined rather than
showing "Not configured".

@Wirasm

Wirasm commented Apr 17, 2026

Copy link
Copy Markdown
Collaborator

This PR appears to fully address #1031. Consider adding Closes #1031 to the PR body so the issue auto-closes on merge.

The Settings page's Platform Connections section hardcoded every platform
except Web to 'Not configured', so users couldn't tell whether their Slack/
Telegram/Discord/GitHub/Gitea/GitLab adapters had actually started.

- Server: /api/health now returns an activePlatforms array populated live
  as each adapter's start() resolves. Passed into registerApiRoutes so the
  reference stays mutable — Telegram starts after the HTTP listener is
  already accepting requests, so a snapshot would miss it.
- Web: SettingsPage.PlatformConnectionsSection now reads activePlatforms
  from /api/health and looks each platform up in a Set. Also adds Gitea
  and GitLab to the list (they already ship as adapters).

Closes coleam00#1031
@Wirasm Wirasm force-pushed the fix/platform-connections-status branch from 6a816a9 to c31e481 Compare April 20, 2026 11:04
@Wirasm Wirasm changed the title fix(web,server,adapters): show real platform connection status fix(web,server): show real platform connection status in Settings (closes #1031) Apr 20, 2026
@Wirasm

Wirasm commented Apr 20, 2026

Copy link
Copy Markdown
Collaborator

Hi @liorfranko — thanks for this! Rebased onto current dev for you. The adapter.ts + adapter.test.ts Telegram changes were dropped on rebase because dev migrated from telegraf to grammY (landed after you opened this PR) and already solves the bot.launch() never-resolves issue via grammY's onStart callback — your fix was exactly right at the time, just no longer needed. The server + web portions (the actual #1031 fix) stay, credited to you.

Also added Closes #1031 to the body so it auto-closes on merge.

@Wirasm Wirasm merged commit 08de8ee into coleam00:dev Apr 21, 2026
4 checks passed
prospapledge88 added a commit to prospapledge88/Archon that referenced this pull request May 5, 2026
* fix(workflows): fail loudly on SDK isError results (coleam00#1208) (coleam00#1291)

Previously, `dag-executor` only failed nodes/iterations when the SDK
returned an `error_max_budget_usd` result. Every other `isError: true`
subtype — including `error_during_execution` — was silently `break`ed
out of the stream with whatever partial output had accumulated, letting
failed runs masquerade as successful ones with empty output.

This is the most likely explanation for the "5-second crash" symptom in
coleam00#1208: iterations finish instantly with empty text, the loop keeps
going, and only the `claude.result_is_error` log tips the user off.

Changes:
- Capture the SDK's `errors: string[]` detail on result messages
  (previously discarded) and surface it through `MessageChunk.errors`.
- Log `errors`, `stopReason` alongside `errorSubtype` in
  `claude.result_is_error` so users can see what actually failed.
- Throw from both the general node path and the loop iteration path
  on any `isError: true` result, including the subtype and SDK errors
  detail in the thrown message.

Note: this does not implement auto-retry. See PR comments on coleam00#1121 and
the analysis on coleam00#1208 — a retry-with-fresh-session approach for loop
iterations is not obviously correct until we see what
`error_during_execution` actually carries in the reporter's env.
This change is the observability + fail-loud step that has to come
first so that signal is no longer silent.

Co-authored-by: Claude Opus 4.7 <noreply@anthropic.com>
(cherry picked from commit 4c6ddd9)

* fix(db): throw on corrupt commands JSON instead of silent empty fallback (coleam00#1033)

* fix(db): throw on corrupt commands JSON instead of silent empty fallback (coleam00#967)

getCodebaseCommands() silently returned {} when the commands column
contained corrupt JSON. Callers had no way to distinguish 'no commands'
from 'unreadable data', violating fail-fast principles.

Now throws a descriptive error with the codebase ID and a recovery hint.
The error is still logged for observability before throwing.

Adds two test cases: corrupt JSON throws, valid JSON string parses.

* fix: include parse error in log for better diagnostics

(cherry picked from commit 39a05b7)

* fix(isolation): raise worktree git-operation timeout to 5m (coleam00#1306)

All 15 worktree git-subprocess timeouts in WorktreeProvider were hardcoded
at 30000ms. Repos with heavy post-checkout hooks (lint, dependency install,
submodule init) routinely exceed that budget and fail worktree creation.

Consolidate them onto a single GIT_OPERATION_TIMEOUT_MS constant at 5 min.
Generous enough to cover reported cases while still catching genuine hangs
(credential prompts in non-TTY, stalled fetches).

Chosen over the config-key approach in coleam00#1029 to avoid adding permanent
.archon/config.yaml surface for a problem a raised default solves cleanly.
If 5 min turns out to also be too tight for real-world use, we'll revisit.

Closes coleam00#1119
Supersedes coleam00#1029

Co-authored-by: Shay Elmualem <12733941+norbinsh@users.noreply.github.com>
(cherry picked from commit cc78071)

* fix(web,server): show real platform connection status in Settings (coleam00#1061)

The Settings page's Platform Connections section hardcoded every platform
except Web to 'Not configured', so users couldn't tell whether their Slack/
Telegram/Discord/GitHub/Gitea/GitLab adapters had actually started.

- Server: /api/health now returns an activePlatforms array populated live
  as each adapter's start() resolves. Passed into registerApiRoutes so the
  reference stays mutable — Telegram starts after the HTTP listener is
  already accepting requests, so a snapshot would miss it.
- Web: SettingsPage.PlatformConnectionsSection now reads activePlatforms
  from /api/health and looks each platform up in a Set. Also adds Gitea
  and GitLab to the list (they already ship as adapters).

Closes coleam00#1031

Co-authored-by: Lior Franko <liorfr@dreamgroup.com>
(cherry picked from commit 08de8ee)

* fix: initialize options.hooks before merging YAML node hooks (coleam00#1177)

When a workflow node defines hooks (PreToolUse/PostToolUse) in YAML but
no hooks exist yet on the options object, applyNodeConfig crashes with
"undefined is not an object" because it tries to assign properties on
the undefined options.hooks.

Initialize options.hooks to {} before the merge loop.

Reproduces with: archon workflow run archon-architect (which uses
per-node hooks extensively).

Co-authored-by: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
(cherry picked from commit 7ea3214)

* fix: detect completion signal in any XML tag, not just <promise> (coleam00#1126) (coleam00#1184)

* fix: detect completion signal in any XML tag, not just <promise> (coleam00#1126)

Loop nodes with `until:` reported max_iterations_reached when the AI wrapped
the completion signal in XML tags other than `<promise>` (e.g.,
`<COMPLETE>ALL_CLEAN</COMPLETE>`). The three existing regex patterns all missed
this format, causing the loop to exhaust iterations and fail.

Changes:
- Add generic XML-wrapped signal pattern to `detectCompletionSignal`
- Extend `stripCompletionTags` to strip matched XML-wrapped signals from output
- Pass `loop.until` to `stripCompletionTags` call site in dag-executor
- Add unit tests for detection and stripping of XML-wrapped signals
- Add integration test for loop completing on final iteration with XML tags

Fixes coleam00#1126

* fix: address review findings for completion signal detection

- Update detectCompletionSignal JSDoc to document all three detection formats
- Update stripCompletionTags JSDoc to mention the `until` parameter
- Remove superfluous `m` flag from xmlWrappedPattern (no anchors, no effect)
- Document that XML tag names are matched independently (intentional permissiveness)
- Add test: detects signal in mismatched XML tags (permissive behavior)
- Add test: strips both <promise> and XML-tagged signal in same chunk
- Add assertion in DAG integration test that raw XML tags don't appear in sent messages

* simplify: reduce complexity in changed files

* fix: require matching XML tag names in completion-signal detection

Follow-up to the initial broadening in this PR. The first version of the
regex accepted mismatched open/close tags (e.g. `<COMPLETE>X</done>`)
which was a small false-positive surface when the AI interleaves tags
in prose. Tightens both detectCompletionSignal and stripCompletionTags
to capture the tag name and enforce it on the close via \1
backreference. Case-insensitivity on the tag name is preserved.

Test updates:
- Flip the "permissive mismatch" case to assert strict rejection with a
  comment explaining the guard.
- Add a case-insensitive matching case to lock that behavior in.

No behavior change for workflows that use matching tags (the
overwhelming common case) or for <promise>...</promise>. Behavior change
is limited to the narrow "open tag and close tag disagree" case, which
only happens when the AI is confused — in which case we'd rather report
max_iterations_reached and let the author inspect than silently call
the loop complete.

(cherry picked from commit bc25dee)

* fix(web): allow deleting nodes from Workflow Builder (coleam00#971) (coleam00#1113)

* fix(web): allow deleting nodes from Workflow Builder (coleam00#971)

Three independent gaps prevented users from deleting nodes added to the
Workflow Builder canvas: dropped nodes were never auto-selected so
keyboard shortcuts silently no-oped, no right-click context menu
existed, and the Delete Node button was buried in the Advanced tab
(hidden below the viewport for Prompt/Command, completely absent for
Bash since bash nodes have no Advanced tab).

Fixes coleam00#971.

* fix(web): push undo snapshot before adding nodes on canvas

Call onPushSnapshot() before setNodes() in both onDrop and quick-add
handlers so that node additions are captured by undo/redo history.

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>

* fix(web): address PR coleam00#1113 review feedback

- Hold nodes/edges in refs so handleNodeDeleteById and onPushSnapshot
  can't capture stale pre-drop state (fixes undo-stack correctness).
- Clamp context-menu x/y to viewport so right-click near edges stays
  fully on-screen.
- Drop non-conformant role=menu/menuitem from the single-item context
  menu; rely on the native button for accessibility.
- Extend isInputTarget() to cover ARIA combobox/textbox/searchbox so
  Backspace in Radix/shadcn widgets never nukes a node.
- Extract handleBuilderKeydown as a pure function and add tests
  covering the Delete/Backspace + isInputTarget invariant.
- Remove issue-number references from code comments per CLAUDE.md.
- Document the new delete affordances in the Workflow Builder docs.
- Inline context-menu dismissal, rename pointer handler, drop unused
  deps in keyboardActions useMemo.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

---------

Co-authored-by: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
(cherry picked from commit d7f36b2)

* fix(workflows): make archon-adversarial-dev sed replacement macOS-safe (coleam00#1155)

* fix(workflows): make adversarial init sed portable on macOS

* chore: regenerate bundled-defaults after adversarial-dev sed fix

Sync generated bundle with the new temp-file sed pattern in
archon-adversarial-dev.yaml so check:bundled passes and binary
distributions ship the macOS-safe version.

---------

Co-authored-by: laplace young <yangqk12@whu.edu.cn>
Co-authored-by: Rasmus Widing <rasmus.widing@gmail.com>
(cherry picked from commit 817186d)

* fix(deps): override transitive axios to ^1.15.0 for CVE-2025-62718 (coleam00#1330)

axios <1.15.0 can be coerced to bypass NO_PROXY rules via hostname
normalization, enabling SSRF in the right network shape. Archon pulls
axios transitively through @slack/bolt (^1.12.0) and @slack/web-api
(^1.13.5); before this change bun.lock resolved axios@1.13.6 — within
the vulnerable range.

Adding "axios": "^1.15.0" to the root package.json overrides bumps the
transitive resolution to axios@1.15.1 (latest compatible 1.x). Both
Slack range specs accept it without API surface changes — no downstream
code touches axios directly.

Supersedes coleam00#1153. Credits @stefans71 for identifying and reporting the
vulnerability; their PR was stale on the lockfile (0.3.5 → 0.3.6 drift
on dev), so this is a fresh one-line re-do on current dev.

Closes coleam00#1053.

Co-authored-by: Stefans71 <stefans71@users.noreply.github.com>
(cherry picked from commit ae2d936)

* fix(cli): surface stale-workspace registration error instead of fake "not a git repo" (coleam00#1332)

* fix(cli): surface stale-workspace registration error instead of fake "not a git repo"

When workflowRunCommand auto-registers an unregistered repo, a stale
~/.archon/workspaces/<owner>/<repo>/source symlink (pointing to an old
checkout) causes createProjectSourceSymlink() in @archon/paths to throw:

  Source symlink at <linkPath> already points to <existing>, expected <target>

The CLI caught that in a try/catch, logged it at warn level, continued
with `codebase = null`, and then the isolation / resume branches hit
their "codebase missing" fallback and threw the generic:

  Cannot create worktree: not in a git repository.

That message is false — the repo is valid; the Archon workspace entry
is stale. It sends users down the wrong diagnostic path (checking git
config, permissions, etc.) instead of pointing at the workspace dir.

Fix: preserve the registration error on a new `codebaseRegistrationError`
local, and at both fallback sites (resume + worktree-creation) check it
before the generic "not a git repo" branch. When set, throw a truthful:

  Cannot {create worktree,resume}: repository registration failed.
  Error: <original message>
  Hint: Remove the stale workspace entry at <dir> and retry, or
        use --no-worktree to skip isolation.

The hint's exact path comes from a small parser that extracts the
workspace directory from the known "Source symlink at …" format; when
the message shape doesn't match (future error text changes), the parser
returns null and we fall back to a generic "check registration under
<archon-home>/workspaces" hint — safe degradation.

Regression test in workflow.test.ts asserts the new error message and
negatively asserts the old "not in a git repository" string is gone.

Supersedes coleam00#1157 — that PR was draft + CONFLICTING against current dev,
and also mentioned Windows test-compat changes that weren't in the diff
(pruned scope). This is a fresh re-do focused strictly on coleam00#1146.

Closes coleam00#1146.

Co-authored-by: Bortlesboat <Bortlesboat@users.noreply.github.com>

* review: add resume-path test, null-fallback test, update troubleshooting docs

Addresses multi-agent review feedback on this PR:

- Add regression test for the --resume fallback site (the worktree-create
  site was already covered; the resume site had identical wiring but zero
  test coverage).
- Add test for the unrecognized-error-shape branch of
  buildRegistrationFailureError so the generic workspace hint is pinned
  (prevents accidental inversion of the stale-entry vs generic-hint
  ternary).
- Update the troubleshooting page to key on the new
  "Cannot create worktree: repository registration failed." message.
  Users hitting the new error won't find the page under the old heading,
  and the "In the future..." note is obsolete now that the error itself
  contains the cleanup path.
- Trim both new docblocks: keep the load-bearing cross-package error
  string contract in extractStaleWorkspaceEntry, drop narration of what
  the code already shows. Drop the "Before this helper existed..."
  paragraph from buildRegistrationFailureError — that's CHANGELOG
  material. Drop PR-reference suffix from the test section divider.

* review: guard getArchonHome in hint + export parser for direct tests

Two follow-up fixes to the multi-agent review commit (f32f002):

CodeRabbit finding — unguarded getArchonHome() in the fallback hint.
If getArchonHome() ever throws (misconfigured env vars, permission issues
on the resolution path), the registration-failure Error would never get
constructed: we'd throw a secondary home-resolution error that masks the
root cause. Wrap the fallback branch in try/catch — prefer losing the
exact path in the hint over replacing the actionable registration error.
A safe generic hint ("Check your Archon workspace registration and retry")
takes over when getArchonHome() throws. The original error.message is
always embedded verbatim in the re-thrown Error.

S2 — export extractStaleWorkspaceEntry for direct table tests. The parser
is where the cross-package string contract with @archon/paths actually
lives; direct tests against it are cheaper than end-to-end CLI tests and
pin the edge cases:

- POSIX path with forward slashes (typical unix user)
- Windows path with backslashes (verifies Math.max(lastIndexOf / , lastIndexOf \))
- Unrelated error message (no prefix) → null
- Prefix matches but delimiter missing → null
- Source path without any separator → null (guards against returning
  empty string, which would produce a nonsense "Remove the stale
  workspace entry at " hint)
- Empty string → null

Six new cases in the test file. The claim of Windows support in the
PR description is now actually verified.

* fix(test): make generic-hint assertion path-separator agnostic

Windows test runner (CI) hit:
  Expected to contain: "Check your Archon workspace registration under /home/test/.archon/workspaces"
  Received: "... under \home\test\.archon\workspaces and retry, ..."

path.join normalizes to `\` on Windows and `/` on POSIX. The test hardcoded
forward slashes in the expected substring. Split into two separator-agnostic
asserts: the prefix up to "under", then `/workspaces\b/` regex for the final
path segment. Behavior doesn't change — the hint still gets the full
path.join'd workspaces dir on either platform.

---------

Co-authored-by: Bortlesboat <Bortlesboat@users.noreply.github.com>
(cherry picked from commit 056707d)

* fix(server,web,workflows): web approval gates auto-resume + reject-with-reason dialog (coleam00#1329)

* fix(server,web,workflows): web approval gates auto-resume + reject-with-reason dialog

Fixes three tightly-coupled bugs that made web approval gates unusable:

1. orchestrator-agent did not pass parentConversationId to executeWorkflow
   for any web-dispatched foreground / interactive / resumable run. Without
   that field, findResumableRunByParentConversation (the machinery the CLI
   relies on for resume) couldn't find the paused run from the same
   conversation on a follow-up message, and the approve/reject API handlers
   had no conversation to dispatch back to.

2. POST /api/workflows/runs/:runId/{approve,reject} recorded the decision
   and returned "Send a message to continue the workflow." — the workflow
   never actually resumed. Added tryAutoResumeAfterGate() that mirrors what
   workflowApproveCommand / workflowRejectCommand already do on the CLI:
   look up the parent conversation, dispatch `/workflow run <name>
   <userMessage>` back through dispatchToOrchestrator. Failures are
   non-fatal — the user can still send a manual message as a fallback.

3. The during-streaming cancel-check in dag-executor aborted any streaming
   node whenever the run status left 'running', including the legitimate
   transition to 'paused' that an approval node performs. A concurrent AI
   node in the same DAG layer now tolerates 'paused' and finishes its own
   stream; only truly terminal / unknown states (null, cancelled, failed,
   completed) abort the in-flight stream.

Web UI: ConfirmRunActionDialog gains an optional reasonInput prop (label +
placeholder) that renders a textarea and passes the trimmed value to
onConfirm. WorkflowRunCard (dashboard) and WorkflowProgressCard (chat)
both use it for Reject now — the chat card was still on window.confirm,
which was both inconsistent with the dashboard and couldn't collect a
reason. The trimmed reason threads through to $REJECTION_REASON in the
workflow's on_reject prompt.

Supersedes coleam00#1147. @jonasvanderhaegen surfaced the root cause and shape of
the fix; that PR was 87 commits stale and pre-dated the reject-UX upgrade
(coleam00#1261 area), so this is a fresh re-do on current dev.

Tests:
- packages/server/src/routes/api.workflow-runs.test.ts — 5 new cases:
  approve with parent dispatches; approve without parent returns "Send a
  message"; approve with deleted parent conversation skips safely; reject
  dispatches on-reject flows; reject that cancels (no on_reject) does NOT
  dispatch.
- packages/core/src/orchestrator/orchestrator.test.ts — updated the two
  synthesizedPrompt-dispatch tests for the new executeWorkflow arity.

Closes coleam00#1131.

Co-authored-by: Jonas Vanderhaegen <7755555+jonasvanderhaegen@users.noreply.github.com>

* fix: address multi-agent review findings for web approval auto-resume

C1 (critical) — cross-adapter misrouting guard
  tryAutoResumeAfterGate now checks parentConv.platform_type === 'web'
  before dispatching. Non-web parents (Slack/Telegram/GitHub/Discord)
  being approved from the dashboard skip auto-resume rather than
  dispatching a Slack thread_ts or Telegram chat_id through the web
  adapter's lock manager.

C2 (critical) — fire-and-forget dispatch replaced with await
  void dispatchToOrchestrator() meant the "Resuming workflow." response
  fired before async work completed, and the outer try/catch couldn't
  observe dispatch failures. Changed to await; response now accurately
  reflects dispatch outcome.

I1 — replaced logPrefix string-template (which produced 3-segment
  api.workflow_*.dispatched event names violating {domain}.{action}_{state})
  with literal event names per action, branched inside the helper.
  Accepts action: 'approve' | 'reject' instead.

I2 — corrected misleading "foreground/interactive" qualifier in the
  approve-endpoint comment; background web dispatches also set
  parent_conversation_id via the pre-created run, so they auto-resume too.

I3 — extracted shouldContinueStreamingForStatus() as a small exported
  policy and added 7 unit tests covering running/paused/null/cancelled/
  failed/completed/unknown. Full-integration coverage of the paused-
  tolerance invariant would require manipulating the 10s
  CANCEL_CHECK_INTERVAL_MS, which is flaky-prone; unit test of the
  policy function captures the same invariant deterministically.

I4 — updated approval-nodes.md and authoring-workflows.md to reflect
  that Web UI approve/reject now auto-resumes (no "send a follow-up
  message" copy), documented the reject-with-reason dialog and
  $REJECTION_REASON flow, and called out the cross-platform caveat.

S1 — rewrote streaming status check as positive shouldContinue safe-list
  via the extracted policy function, matching the inline comment.

S2 — inlined handleReject on the dashboard rather than squeezing
  rejectWorkflowRun through runAction with a closure; keeps runAction
  narrow for the single-arg lifecycle actions.

S5 — new regression test covering the non-web-parent skip path
  (slack-platform parent → dispatch skipped → response falls back to
  "Send a message to continue").

S6 — removed stale reference to runAction in ConfirmRunActionDialog's
  onConfirm JSDoc (no longer accurate now that WorkflowProgressCard
  calls the dialog without runAction).

S7 — fixed misleading "user can resume manually by sending any message"
  docstring (resume is triggered by re-running the workflow command,
  not by an arbitrary message).

Skipped as out-of-scope:
  S3 — cancelWorkflowRun rowCount check (pre-existing defect; separate PR)
  S4 — tightening expect.anything() to UUID regex (deferred)
  S8 — 12-positional-arg executeWorkflow → options-bag refactor
    (tracked follow-up)

bun run validate green locally; 68 tests in api.workflow-runs.test.ts
(up from 67), 173 in dag-executor.test.ts (up from 166).

* review: close I1/I2/I3/I4/I6 — paused tolerance in loop + emitter, resume test, useId

I1 (loop inter-iteration check) — dag-executor.ts:1715
  Used `!== 'running'` in the loop node's between-iteration status check.
  A sibling approval node pausing the run in the same topological layer
  would abort the loop mid-iteration with "Loop node '<id>' stopped at
  iteration N (paused)". Switched to the shared shouldContinueStreamingForStatus
  helper so paused is tolerated — same semantics the streaming check got.
  Extended inline comment explains the sibling-layer concurrency reason.

I2 (skipIfStatusChanged emitter unregister) — dag-executor.ts:2886
  At DAG-finalization writes the helper correctly skipped writing on any
  non-running state (paused included — don't mark a paused run complete),
  but it *also* called getWorkflowEventEmitter().unregisterRun() which
  broke SSE observability for a run that's still live (waiting for user
  approval). Split the two responsibilities: skip the write for all
  non-running states, but only unregister the emitter for terminal states
  (cancelled / deleted / completed / failed). `paused` keeps the emitter
  registered so resume stays visible on the dashboard.

I3 (foreground_resume_detected branch untested) — orchestrator-agent.test.ts
  That branch was modified as part of the original fix (added
  parentConversationId as 11th positional arg) but no existing test
  configured mockFindResumableRunByParentConversation to return non-null.
  A positional mistake (e.g. accidentally swapping issueContext and
  parentConversationId) would silently break auto-resume with no failing
  test. New regression test configures the mock, asserts both the cwd
  comes from the resumable run's working_path AND parentConversationId
  is passed correctly at position 10.

I4 (null-parent log level) — api.ts tryAutoResumeAfterGate
  `getConversationById` returning null is a data-integrity signal (the
  parent conversation was deleted while the run was paused) — worth
  surfacing at info level so operators notice, not hiding at debug.
  Missing platform_conversation_id on an existing row would be an unusual
  DB state and stays at debug. Added `parentDeleted: boolean` to the log
  context so the two cases are distinguishable in observability.

I6 (hardcoded DOM id) — ConfirmRunActionDialog.tsx
  `id="confirm-run-action-reason"` collided when multiple dialog instances
  share the same page (Radix portals mitigate in practice but the code
  was fragile). Switched to React.useId() so each instance gets a unique
  id — htmlFor/id wiring preserved.

S11 (arity-only assertion) — orchestrator-agent.test.ts:1092 area
  The interactive-workflow-on-web test asserted mockExecuteWorkflow was
  called, but nothing about the args. Added a specific assertion that
  position 10 (parentConversationId) equals 'conv-1' (the caller
  conversation id) — pins the wiring that I1/I2 depend on being correct.

Deferred (from review S1-S10, I5, I7):
  - S1 (ExecuteWorkflowOptions bag) — tracked as standalone follow-up;
    12 positional args with 2 adjacent optionals is a real maintenance
    hazard but the refactor deserves its own PR.
  - S7 (WHY comment on non-web else branch) — review text says the branch
    "correctly omits" parentConversationId but the code passes it; the
    combination with the web-parent guard in tryAutoResumeAfterGate is
    intentional. Not adding a justify-what-we-don't-do comment.
  - S2/S3/S4/S5/S8/S9/S10 — pure polish (event-map ternary, platformConvId
    inlining, shared constant for REJECTION_REASON_INPUT, onChange arrow
    shorthand, discriminated union, docblock trim, suffix comment drop)
  - I5 (soften "Resuming workflow." to "— check the dashboard for progress")
    — users clicking from the dashboard are already on the dashboard; the
    current text is accurate (enqueue completed) and concise.
  - I7 (test dispatch-throws path) — covered implicitly by the try/catch
    branch of tryAutoResumeAfterGate returning false; a direct test would
    require mocking handleMessage to throw and would couple to
    dispatchToOrchestrator internals.

bun run validate green; 189 dag-executor tests, 98 orchestrator-agent
tests, 68 api.workflow-runs tests — all the new cases pass.

---------

Co-authored-by: Jonas Vanderhaegen <7755555+jonasvanderhaegen@users.noreply.github.com>
(cherry picked from commit d5c1cd9)

* feat(providers): autodetect canonical binary install paths for Claude and Codex (coleam00#1361)

Both binary resolvers previously stopped at env-var + explicit config and
threw a "not found" error when neither was set. Users who followed the
upstream-recommended install flow (Anthropic's `curl install.sh` for
Claude, `npm install -g @openai/codex`) still had to manually set either
`CLAUDE_BIN_PATH` / `CODEX_BIN_PATH` or the corresponding config field
before any workflow could run.

Add a tier-N autodetect step between the explicit config tier and the
install-instructions throw. Purely additive: env and config still win
when set (precedence covered by new tests). On autodetect miss, the same
install-instructions error fires as before.

Claude probe list (verified against docs.claude.com "Uninstall Claude
Code → Native installation" section):
  - $HOME/.local/bin/claude            (mac/linux native installer)
  - $USERPROFILE\.local\bin\claude.exe (Windows native installer)

Codex probe list (verified against openai/codex README; npm global-
install puts the binary at `{npm_prefix}/bin/<name>` on POSIX,
`{npm_prefix}\<name>.cmd` on Windows):
  - $HOME/.npm-global/bin/codex   (user-set `npm config set prefix`)
  - /opt/homebrew/bin/codex       (mac arm64 with homebrew-node)
  - /usr/local/bin/codex          (mac intel / linux system node)
  - %APPDATA%\npm\codex.cmd       (Windows npm global default)
  - $HOME\.npm-global\codex.cmd   (Windows user-set prefix)

Not probed (explicit override still required):
  - Custom npm prefixes — `npm root -g` would need a subprocess per
    resolve, too much surface for a probe helper
  - `brew install --cask codex` — cask layout isn't a PATH binary
  - Manual GitHub Releases extracts — placement is user-determined
  - `~/.bun/bin/codex` — not documented in openai/codex README

Pi provider intentionally has no equivalent change: the Pi SDK is
bundled into the archon binary (no subprocess), so there's no "binary"
to resolve. Pi auth lives at `~/.pi/agent/auth.json` which the SDK
already finds by default, and the PR A shim (`PI_PACKAGE_DIR`) handles
the package-dir case via Pi's own documented escape hatch.

E2E verified: removed both config entries from ~/.archon/config.yaml,
rebuilt compiled binary, ran `archon workflow run archon-assist` and a
Codex workflow. Logs showed `source: 'autodetect'` for both, responses
returned cleanly.

(cherry picked from commit b99cee4)

* fix(providers/test): use os.homedir() instead of $HOME in claude binary autodetect test

The native-installer autodetect test computed its expected path from
process.env.HOME, but the implementation uses node:os homedir(). On
Windows, HOME is typically unset (Windows uses USERPROFILE), so the
test fell back to '/Users/test' while the resolver returned the real
home dir — making the spy's path-equality check fail and breaking CI
on windows-latest.

Mirror the implementation by importing homedir() from node:os and
joining with node:path so the expected path matches the actual
platform-resolved home and separator.

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
(cherry picked from commit f9f8775)

* fix(server): contain Discord login failure so it doesn't kill the server (coleam00#1365)

Reported in coleam00#1365: a user running `archon serve` with DISCORD_BOT_TOKEN
set but the "Message Content Intent" toggle disabled in the Discord
Developer Portal saw the entire server crash with `Used disallowed
intents`. Discord rejects the gateway connection (close code 4014) when
a privileged intent is requested without being enabled, and the
unguarded `await discord.start()` propagated the error all the way up,
taking the web UI down with it.

Wrap discord.start() in try/catch — log the failure with an actionable
hint (special-cased for the disallowed-intent error) and continue
running. Other adapters and the web UI come up regardless. The shutdown
handler already uses optional chaining (`discord?.stop()`) so nulling
discord after a failed start is safe.

Other adapters (Telegram, Slack, GitHub, Gitea, GitLab) have the same
unguarded-start pattern but are out of scope for this fix — addressing
them is tracked separately.

Also expanded the Discord setup docs with a caution callout that names
the exact error string and the new log event so users can grep for
both.

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
(cherry picked from commit 5957c6e)

---------

Co-authored-by: Cole Medin <cole@dynamous.ai>
Co-authored-by: Claude Opus 4.7 <noreply@anthropic.com>
Co-authored-by: Kagura <kagura.chen28@gmail.com>
Co-authored-by: Rasmus Widing <152263317+Wirasm@users.noreply.github.com>
Co-authored-by: Shay Elmualem <12733941+norbinsh@users.noreply.github.com>
Co-authored-by: Lior Franko <lior.franko@ironsrc.com>
Co-authored-by: Lior Franko <liorfr@dreamgroup.com>
Co-authored-by: Alex Siri <alexsiri7@gmail.com>
Co-authored-by: Ahmed <44034059+medevs@users.noreply.github.com>
Co-authored-by: CauchYoung <2024302072042@whu.edu.cn>
Co-authored-by: laplace young <yangqk12@whu.edu.cn>
Co-authored-by: Rasmus Widing <rasmus.widing@gmail.com>
Co-authored-by: Stefans71 <stefans71@users.noreply.github.com>
Co-authored-by: Bortlesboat <Bortlesboat@users.noreply.github.com>
Co-authored-by: Jonas Vanderhaegen <7755555+jonasvanderhaegen@users.noreply.github.com>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

None yet

Projects

None yet

Development

Successfully merging this pull request may close these issues.

Settings page shows all platform adapters as 'Not configured' even when running

2 participants