Commit e0b211d
authored
(kiloclaw-admin): Add Destroy Machine functionality to the KiloClaw admin dash (#1550)
## Summary
Adds a "Destroy Machine" button to the KiloClaw admin instance detail
page that directly calls the Fly Machines API to force-destroy a machine
— equivalent to `fly machine destroy <id> -a <app> --force`.
**Why this exists:** The normal instance destroy flow goes through the
DO's teardown pipeline (machine → volume → cleanup). This button is an
admin escape hatch for when you need to delete the Fly machine directly
without triggering the full teardown.
**What it does NOT do:** This does not tear down the KiloClaw instance,
delete the Fly volume, or mark the instance as destroyed in the
database. Only the Fly machine is destroyed.
### Implementation
- **Platform route** (`POST /api/platform/destroy-fly-machine`): Calls
`DELETE /v1/apps/{app}/machines/{id}?force=true` on the Fly Machines API
using the worker's `FLY_API_TOKEN`. After a successful destroy, triggers
`forceRetryRecovery` on the DO to schedule an immediate reconcile alarm
— the DO will discover the machine is destroyed and clear its
`flyMachineId` state.
- **tRPC mutation** (`destroyFlyMachine`): Admin-only mutation that
proxies to the platform route and writes an audit log
(`kiloclaw.machine.destroy_fly`).
- **UI**: Orange "Destroy Machine" button in Machine Controls with a
confirmation dialog showing the machine ID and app name. After success,
the button shows a green "Machine Destroyed" state and is disabled. An
error toast is shown if `flyAppName` is missing from the DO state.
- **Observability**: `writeEvent` calls for success/failure (visible in
the DO & Reconcile Events table on the admin page), plus
`createKiloClawAdminAuditLog` for the audit trail.
### Reconciler fix: handle `"destroyed"` Fly state
`syncStatusWithFly` previously had no branch for the `"destroyed"` Fly
machine state. When a machine is force-destroyed via the Fly API, Fly
returns the machine with `state: "destroyed"` before it eventually
starts returning 404. Without this fix, the reconciler would no-op on
the `"destroyed"` state, leaving the stale `flyMachineId` in DO storage
until a later alarm cycle when Fly finally returns 404 and
`handleMachineGone` fires. The new `"destroyed"` branch mirrors
`handleMachineGone` — clears `flyMachineId`, sets status to `"stopped"`,
and persists immediately.
### New audit action
`kiloclaw.machine.destroy_fly` added to `KiloClawAdminAuditAction` enum.
## Verification
- [x] TypeScript compiles cleanly (`npx tsc --noEmit`)
- [x] Tested locally: destroy succeeds, Fly machine removed, DO
reconciles after alarm fires
- [x] 4 rounds of adversarial code review (code reviewer, silent failure
hunter, security auditor) with verification agents — converged to zero
new actionable findings
- [x] E2E manual testing
## Loom — Kilo team only
https://www.loom.com/share/7e898ec288da4e9a83d2554780622dd9
## Reviewer Notes
- The `forceRetryRecovery` call after destroy is fire-and-forget (caught
and warned). If it fails, the DO will still reconcile on its next
regular alarm. The response returns `ok: true` regardless — this was a
deliberate scoping decision to avoid threading a `reconcileTriggered`
flag through three layers.
- Until the reconcile task following machine destruction finishes
(should be fairly quick in the deployed env), the DO will log errors as
it has lost connection with the controller. The noise should be
relatively minimal and, as the reconcile alarm is set to _now_, it
should be resolved quickly.
- `isFlyMachineDestroyed` comes from `useMutation`'s `isSuccess` which
resets on component re-mount. A second click after re-mount would get a
Fly 404 error toast — not harmful, just redundant.File tree
9 files changed
+768
-1
lines changed- kiloclaw/src
- durable-objects
- kiloclaw-instance
- routes
- packages/db/src
- src
- app/admin/components/KiloclawInstances
- lib/kiloclaw
- routers
9 files changed
+768
-1
lines changed| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
5122 | 5122 | | |
5123 | 5123 | | |
5124 | 5124 | | |
| 5125 | + | |
| 5126 | + | |
| 5127 | + | |
| 5128 | + | |
| 5129 | + | |
| 5130 | + | |
| 5131 | + | |
| 5132 | + | |
| 5133 | + | |
| 5134 | + | |
| 5135 | + | |
| 5136 | + | |
| 5137 | + | |
| 5138 | + | |
| 5139 | + | |
| 5140 | + | |
| 5141 | + | |
| 5142 | + | |
| 5143 | + | |
| 5144 | + | |
| 5145 | + | |
| 5146 | + | |
| 5147 | + | |
| 5148 | + | |
| 5149 | + | |
| 5150 | + | |
| 5151 | + | |
| 5152 | + | |
| 5153 | + | |
| 5154 | + | |
| 5155 | + | |
| 5156 | + | |
| 5157 | + | |
| 5158 | + | |
5125 | 5159 | | |
5126 | 5160 | | |
5127 | 5161 | | |
| |||
Lines changed: 24 additions & 0 deletions
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
777 | 777 | | |
778 | 778 | | |
779 | 779 | | |
| 780 | + | |
| 781 | + | |
| 782 | + | |
| 783 | + | |
| 784 | + | |
| 785 | + | |
| 786 | + | |
| 787 | + | |
| 788 | + | |
| 789 | + | |
| 790 | + | |
| 791 | + | |
| 792 | + | |
| 793 | + | |
| 794 | + | |
| 795 | + | |
| 796 | + | |
| 797 | + | |
| 798 | + | |
| 799 | + | |
| 800 | + | |
| 801 | + | |
| 802 | + | |
| 803 | + | |
780 | 804 | | |
781 | 805 | | |
782 | 806 | | |
| |||
Lines changed: 221 additions & 0 deletions
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
| 1 | + | |
| 2 | + | |
| 3 | + | |
| 4 | + | |
| 5 | + | |
| 6 | + | |
| 7 | + | |
| 8 | + | |
| 9 | + | |
| 10 | + | |
| 11 | + | |
| 12 | + | |
| 13 | + | |
| 14 | + | |
| 15 | + | |
| 16 | + | |
| 17 | + | |
| 18 | + | |
| 19 | + | |
| 20 | + | |
| 21 | + | |
| 22 | + | |
| 23 | + | |
| 24 | + | |
| 25 | + | |
| 26 | + | |
| 27 | + | |
| 28 | + | |
| 29 | + | |
| 30 | + | |
| 31 | + | |
| 32 | + | |
| 33 | + | |
| 34 | + | |
| 35 | + | |
| 36 | + | |
| 37 | + | |
| 38 | + | |
| 39 | + | |
| 40 | + | |
| 41 | + | |
| 42 | + | |
| 43 | + | |
| 44 | + | |
| 45 | + | |
| 46 | + | |
| 47 | + | |
| 48 | + | |
| 49 | + | |
| 50 | + | |
| 51 | + | |
| 52 | + | |
| 53 | + | |
| 54 | + | |
| 55 | + | |
| 56 | + | |
| 57 | + | |
| 58 | + | |
| 59 | + | |
| 60 | + | |
| 61 | + | |
| 62 | + | |
| 63 | + | |
| 64 | + | |
| 65 | + | |
| 66 | + | |
| 67 | + | |
| 68 | + | |
| 69 | + | |
| 70 | + | |
| 71 | + | |
| 72 | + | |
| 73 | + | |
| 74 | + | |
| 75 | + | |
| 76 | + | |
| 77 | + | |
| 78 | + | |
| 79 | + | |
| 80 | + | |
| 81 | + | |
| 82 | + | |
| 83 | + | |
| 84 | + | |
| 85 | + | |
| 86 | + | |
| 87 | + | |
| 88 | + | |
| 89 | + | |
| 90 | + | |
| 91 | + | |
| 92 | + | |
| 93 | + | |
| 94 | + | |
| 95 | + | |
| 96 | + | |
| 97 | + | |
| 98 | + | |
| 99 | + | |
| 100 | + | |
| 101 | + | |
| 102 | + | |
| 103 | + | |
| 104 | + | |
| 105 | + | |
| 106 | + | |
| 107 | + | |
| 108 | + | |
| 109 | + | |
| 110 | + | |
| 111 | + | |
| 112 | + | |
| 113 | + | |
| 114 | + | |
| 115 | + | |
| 116 | + | |
| 117 | + | |
| 118 | + | |
| 119 | + | |
| 120 | + | |
| 121 | + | |
| 122 | + | |
| 123 | + | |
| 124 | + | |
| 125 | + | |
| 126 | + | |
| 127 | + | |
| 128 | + | |
| 129 | + | |
| 130 | + | |
| 131 | + | |
| 132 | + | |
| 133 | + | |
| 134 | + | |
| 135 | + | |
| 136 | + | |
| 137 | + | |
| 138 | + | |
| 139 | + | |
| 140 | + | |
| 141 | + | |
| 142 | + | |
| 143 | + | |
| 144 | + | |
| 145 | + | |
| 146 | + | |
| 147 | + | |
| 148 | + | |
| 149 | + | |
| 150 | + | |
| 151 | + | |
| 152 | + | |
| 153 | + | |
| 154 | + | |
| 155 | + | |
| 156 | + | |
| 157 | + | |
| 158 | + | |
| 159 | + | |
| 160 | + | |
| 161 | + | |
| 162 | + | |
| 163 | + | |
| 164 | + | |
| 165 | + | |
| 166 | + | |
| 167 | + | |
| 168 | + | |
| 169 | + | |
| 170 | + | |
| 171 | + | |
| 172 | + | |
| 173 | + | |
| 174 | + | |
| 175 | + | |
| 176 | + | |
| 177 | + | |
| 178 | + | |
| 179 | + | |
| 180 | + | |
| 181 | + | |
| 182 | + | |
| 183 | + | |
| 184 | + | |
| 185 | + | |
| 186 | + | |
| 187 | + | |
| 188 | + | |
| 189 | + | |
| 190 | + | |
| 191 | + | |
| 192 | + | |
| 193 | + | |
| 194 | + | |
| 195 | + | |
| 196 | + | |
| 197 | + | |
| 198 | + | |
| 199 | + | |
| 200 | + | |
| 201 | + | |
| 202 | + | |
| 203 | + | |
| 204 | + | |
| 205 | + | |
| 206 | + | |
| 207 | + | |
| 208 | + | |
| 209 | + | |
| 210 | + | |
| 211 | + | |
| 212 | + | |
| 213 | + | |
| 214 | + | |
| 215 | + | |
| 216 | + | |
| 217 | + | |
| 218 | + | |
| 219 | + | |
| 220 | + | |
| 221 | + | |
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
1378 | 1378 | | |
1379 | 1379 | | |
1380 | 1380 | | |
| 1381 | + | |
| 1382 | + | |
| 1383 | + | |
| 1384 | + | |
| 1385 | + | |
| 1386 | + | |
| 1387 | + | |
| 1388 | + | |
| 1389 | + | |
| 1390 | + | |
| 1391 | + | |
| 1392 | + | |
| 1393 | + | |
| 1394 | + | |
| 1395 | + | |
| 1396 | + | |
| 1397 | + | |
| 1398 | + | |
| 1399 | + | |
| 1400 | + | |
| 1401 | + | |
| 1402 | + | |
| 1403 | + | |
| 1404 | + | |
| 1405 | + | |
| 1406 | + | |
| 1407 | + | |
| 1408 | + | |
| 1409 | + | |
| 1410 | + | |
| 1411 | + | |
| 1412 | + | |
| 1413 | + | |
| 1414 | + | |
| 1415 | + | |
| 1416 | + | |
| 1417 | + | |
| 1418 | + | |
| 1419 | + | |
| 1420 | + | |
| 1421 | + | |
| 1422 | + | |
| 1423 | + | |
| 1424 | + | |
| 1425 | + | |
| 1426 | + | |
| 1427 | + | |
| 1428 | + | |
| 1429 | + | |
| 1430 | + | |
| 1431 | + | |
| 1432 | + | |
| 1433 | + | |
| 1434 | + | |
| 1435 | + | |
| 1436 | + | |
| 1437 | + | |
| 1438 | + | |
| 1439 | + | |
| 1440 | + | |
| 1441 | + | |
| 1442 | + | |
| 1443 | + | |
| 1444 | + | |
| 1445 | + | |
| 1446 | + | |
1381 | 1447 | | |
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
162 | 162 | | |
163 | 163 | | |
164 | 164 | | |
| 165 | + | |
165 | 166 | | |
166 | 167 | | |
167 | 168 | | |
| |||
0 commit comments