Tracked orders — persist strategy positions in SQLite

Backend:
- New orders table (symbol, name, legs_json, entry_cost, created_at,
  closed_at, status, note) sharing the snapshots.db
- Endpoints:
  POST   /api/orders          save a position
  GET    /api/orders?symbol=  list (most recent first)
  GET    /api/orders/:id      single
  PATCH  /api/orders/:id      { status:'open'|'closed', note? }
  DELETE /api/orders/:id      remove

Strategy page:
- "Save to Tracker" button posts the currently-active legs (with name,
  net cost, side, qty, entry, IV, lock) as an order

Tracker page:
- New "Tracked Orders" section above the IV history charts: lists each
  saved position for the current symbol with strategy name, leg summary,
  entered cost, current value, P/L $ and %, opened date, status, and
  Close/Reopen + Remove actions. Live P/L uses /api/chain mids for each
  unique leg expiry.

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
ojy
2026-05-13 06:54:40 +00:00
parent 58f898b47d
commit c1520d7962
4 changed files with 422 additions and 1 deletions

View File

@@ -255,6 +255,54 @@
<div class="page-body">
<div class="container-xl">
<!-- Tracked Orders (persisted strategy positions) -->
<div class="card mb-3" x-show="orders.length > 0 || ordersLoading" x-cloak style="background:#161824; border:1px solid #2d3045;">
<div class="card-header d-flex align-items-center justify-content-between" style="border-bottom:1px solid #2d3045;">
<h3 class="card-title text-white mb-0">
Tracked Orders
<span class="text-secondary small fw-normal" x-show="symbol"><span x-text="symbol"></span></span>
</h3>
<div class="d-flex align-items-center gap-2">
<span x-show="ordersLoading" class="spinner-border spinner-border-sm text-secondary"></span>
<button class="btn btn-sm btn-outline-secondary" @click="loadOrders()" :disabled="ordersLoading">Refresh</button>
<span class="text-secondary small" x-text="orders.length + ' order' + (orders.length===1?'':'s')"></span>
</div>
</div>
<div class="table-responsive">
<table class="table table-sm table-hover mb-0" style="color:#d0d5e0;">
<thead style="background:#1a1c2e;color:#8b95a7;font-size:.72rem;text-transform:uppercase;letter-spacing:.05em;">
<tr>
<th>Strategy</th>
<th>Legs</th>
<th class="text-end">Entered</th>
<th class="text-end">Current</th>
<th class="text-end">P/L $</th>
<th class="text-end">P/L %</th>
<th>Opened</th>
<th>Status</th>
<th></th><th></th>
</tr>
</thead>
<tbody>
<template x-for="o in orders" :key="o.id">
<tr :style="o.status === 'closed' ? 'opacity:.55' : ''" style="font-size:.85rem;">
<td><span class="badge bg-purple-lt" x-text="o.name || ('Custom (' + o.legs.length + ' legs)')"></span></td>
<td class="text-secondary" style="font-family:'JetBrains Mono',monospace;font-size:.75rem;" x-text="legsSummary(o)"></td>
<td class="text-end" style="font-family:'JetBrains Mono',monospace;" x-text="fmtMoney(orderPL(o).entered)"></td>
<td class="text-end" style="font-family:'JetBrains Mono',monospace;" x-text="fmtMoney(orderPL(o).value)"></td>
<td class="text-end" style="font-family:'JetBrains Mono',monospace;" :class="orderPL(o).pl >= 0 ? 'text-success' : 'text-danger'" x-text="fmtMoney(orderPL(o).pl)"></td>
<td class="text-end" style="font-family:'JetBrains Mono',monospace;" :class="orderPL(o).pl >= 0 ? 'text-success' : 'text-danger'" x-text="orderPL(o).plPct.toFixed(1) + '%'"></td>
<td class="small text-secondary" x-text="(o.created_at||'').slice(0,10)"></td>
<td><span class="badge" :class="o.status === 'open' ? 'bg-success-lt text-success' : 'bg-secondary-lt text-secondary'" x-text="o.status"></span></td>
<td><button class="btn btn-sm" :class="o.status === 'open' ? 'btn-outline-warning' : 'btn-outline-success'" @click="toggleCloseOrder(o)" x-text="o.status === 'open' ? 'Close' : 'Reopen'"></button></td>
<td><button class="btn btn-sm btn-ghost-danger" @click="removeOrder(o.id)" aria-label="Remove order"></button></td>
</tr>
</template>
</tbody>
</table>
</div>
</div>
<!-- Empty state -->
<div x-show="!loading && filteredSnapshots.length === 0" x-cloak class="text-center py-5">
<svg xmlns="http://www.w3.org/2000/svg" width="48" height="48" viewBox="0 0 24 24"
@@ -612,6 +660,10 @@
loading: false,
error: '',
orders: [],
ordersLoading: false,
ordersChain: {}, // "SYMBOL@EXPIRY" -> { "strike@type": option }
_charts: { atmIv: null, rr25: null, fly25: null },
async init() {
@@ -624,6 +676,96 @@
this.snapshots = vs.snapshots ?? [];
if (this.snapshots.length) this.$nextTick(() => this.renderCharts());
}
if (this.symbol) this.loadOrders();
},
async loadOrders() {
if (!this.symbol) return;
this.ordersLoading = true;
try {
const r = await fetch('/api/orders?symbol=' + encodeURIComponent(this.symbol.trim().toUpperCase()));
const d = await r.json();
this.orders = d.data?.orders || [];
// fetch live chains for the unique expiries (one /api/chain call each)
const need = new Set();
for (const o of this.orders) for (const l of (o.legs||[])) if (l.expiry) need.add(o.symbol + '@' + l.expiry);
const cache = { ...this.ordersChain };
for (const key of need) {
if (cache[key]) continue;
const [sym, exp] = key.split('@');
try {
const cr = await fetch('/api/chain?symbol=' + encodeURIComponent(sym) + '&expiry=' + encodeURIComponent(exp));
if (!cr.ok) continue;
const ce = await cr.json();
const snap = ce.data?.snapshots?.[0];
if (!snap) continue;
const map = {};
for (const o of (snap.chain || [])) {
const t = (o.type || o.optionType || '').toLowerCase();
map[Number(o.strike) + '@' + t] = o;
}
cache[key] = map;
} catch {}
}
this.ordersChain = cache;
} catch (e) {
console.warn('loadOrders failed:', e);
} finally {
this.ordersLoading = false;
}
},
orderPL(o) {
let value = 0, entered = 0;
for (const l of (o.legs||[])) {
const sign = l.side === 'short' ? -1 : 1;
entered += sign * l.qty * 100 * (l.entryPrice || 0);
const m = this.ordersChain[o.symbol + '@' + l.expiry];
const oo = m && m[Number(l.strike) + '@' + l.type];
const mid = oo ? Number(oo.midPrice ?? oo.mid ?? oo.bsPrice ?? 0) : 0;
value += sign * l.qty * 100 * mid;
}
const baseCost = (o.entry_cost != null) ? Number(o.entry_cost) : entered;
const pl = value - baseCost;
const plPct = Math.abs(baseCost) > 1e-6 ? (pl / Math.abs(baseCost)) * 100 : 0;
return { value, entered: baseCost, pl, plPct };
},
async toggleCloseOrder(o) {
const newStatus = o.status === 'open' ? 'closed' : 'open';
try {
const r = await fetch('/api/orders/' + o.id, {
method:'PATCH', headers:{'Content-Type':'application/json'},
body: JSON.stringify({ status: newStatus }),
});
const d = await r.json();
if (d.ok) {
const idx = this.orders.findIndex(x => x.id === o.id);
if (idx >= 0) this.orders[idx] = d.data;
}
} catch (e) { /* silent */ }
},
async removeOrder(id) {
if (!confirm('Remove this tracked order?')) return;
try {
await fetch('/api/orders/' + id, { method:'DELETE' });
this.orders = this.orders.filter(o => o.id !== id);
} catch {}
},
legsSummary(o) {
return (o.legs||[]).map(l =>
(l.side === 'long' ? '+' : '-') + l.qty +
' ' + l.strike + (l.type === 'call' ? 'C' : 'P') +
'·' + (l.expiry || '').slice(5)
).join(' / ');
},
fmtMoney(v) {
if (v == null || !isFinite(v)) return '—';
const a = Math.abs(v), sign = v < 0 ? '-' : '';
return sign + '$' + a.toLocaleString('en-US', { maximumFractionDigits: a < 100 ? 2 : 0 });
},
_persist() {
@@ -670,6 +812,7 @@
}
this._persist();
this.$nextTick(() => this.renderCharts());
this.loadOrders();
} catch (e) {
this.error = e.message;
this.snapshots = [];