@opencode-trace/viewer 0.0.2

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.
@@ -0,0 +1 @@
1
+ header[data-v-741ff62c]{display:flex;align-items:center;justify-content:space-between;padding:12px 24px;background:var(--bg-primary);border-bottom:1px solid var(--border);flex-shrink:0;min-height:48px}header h1[data-v-741ff62c]{font-size:16px;font-weight:700;display:flex;align-items:center;gap:8px;line-height:1}.header-actions[data-v-741ff62c]{display:flex;align-items:center;gap:12px}.dropdown[data-v-741ff62c]{position:relative}.dropdown-toggle[data-v-741ff62c]{padding:8px;border-radius:var(--radius);display:flex;align-items:center;justify-content:center;transition:background .1s}.dropdown-toggle[data-v-741ff62c]:hover{background:var(--bg-hover)}.dropdown-menu[data-v-741ff62c]{position:absolute;right:0;top:100%;margin-top:4px;background:var(--bg-secondary);border:1px solid var(--border);border-radius:var(--radius);min-width:200px;z-index:100;padding:4px}.dropdown-item[data-v-741ff62c]{display:flex;align-items:center;gap:8px;width:100%;padding:8px 12px;font-size:14px;font-weight:500;border-radius:var(--radius);transition:background .1s;text-align:left}.dropdown-item[data-v-741ff62c]:hover{background:var(--bg-hover)}.dropdown-item.danger[data-v-741ff62c]{color:var(--danger)}.dropdown-divider[data-v-741ff62c]{height:1px;background:var(--border);margin:4px 0}.breadcrumb[data-v-ddc4b5bf]{display:flex;align-items:center;gap:6px;font-size:13px;color:var(--text-secondary);padding:8px 24px;background:var(--bg-primary);border-bottom:1px solid var(--border);flex-shrink:0;flex-wrap:wrap}.breadcrumb a[data-v-ddc4b5bf]{color:var(--text-secondary);text-decoration:none}.breadcrumb a[data-v-ddc4b5bf]:hover{color:var(--accent);text-decoration:underline}.sep[data-v-ddc4b5bf]{color:var(--text-tertiary);margin:0 2px}.current[data-v-ddc4b5bf]{color:var(--text-primary);font-weight:500}.fab-container[data-v-730271da]{position:fixed;bottom:24px;right:24px;display:flex;flex-direction:column;gap:8px;z-index:50}.fab-scroll[data-v-730271da]{width:40px;height:40px;border-radius:var(--radius);background:var(--bg-secondary);border:1px solid var(--border);display:flex;align-items:center;justify-content:center;transition:background .1s}.fab-scroll[data-v-730271da]:hover{background:var(--bg-hover)}.toast-container[data-v-0871cfbd]{position:fixed;bottom:24px;left:50%;transform:translate(-50%);display:flex;flex-direction:column;gap:8px;z-index:200;pointer-events:none}.toast[data-v-0871cfbd]{display:flex;align-items:center;gap:12px;padding:10px 16px;border-radius:var(--radius);font-size:14px;font-weight:500;pointer-events:auto;animation:toast-in-0871cfbd .2s ease-out;border:1px solid}.toast.success[data-v-0871cfbd]{background:#30d1581f;border-color:var(--success);color:var(--success)}.toast.error[data-v-0871cfbd]{background:#ff3b301f;border-color:var(--danger);color:var(--danger)}.toast.info[data-v-0871cfbd]{background:var(--bg-secondary);border-color:var(--border);color:var(--text-primary)}.toast.warning[data-v-0871cfbd]{background:#ff9f0a1f;border-color:var(--warning);color:var(--warning)}.toast-action[data-v-0871cfbd]{padding:4px 10px;border-radius:var(--radius);font-size:12px;font-weight:700;background:var(--bg-tertiary);white-space:nowrap}@keyframes toast-in-0871cfbd{0%{opacity:0;transform:translateY(8px)}to{opacity:1;transform:translateY(0)}}.confirm-overlay[data-v-4c95e883]{position:fixed;top:0;right:0;bottom:0;left:0;background:#00000080;display:flex;align-items:center;justify-content:center;z-index:300}.confirm-dialog[data-v-4c95e883]{background:var(--bg-primary);border:1px solid var(--border);border-radius:var(--radius);padding:24px;max-width:420px;width:90%}.confirm-title[data-v-4c95e883]{font-size:16px;font-weight:700;margin-bottom:8px}.confirm-message[data-v-4c95e883]{font-size:14px;color:var(--text-secondary);margin-bottom:20px;line-height:1.5}.confirm-actions[data-v-4c95e883]{display:flex;gap:8px;justify-content:flex-end}.confirm-btn[data-v-4c95e883]{padding:8px 20px;border-radius:var(--radius);font-size:14px;font-weight:500;transition:background .1s}.confirm-btn.cancel[data-v-4c95e883]{background:var(--bg-tertiary);color:var(--text-primary)}.confirm-btn.cancel[data-v-4c95e883]:hover{background:var(--bg-hover)}.confirm-btn.danger[data-v-4c95e883]{background:var(--danger);color:var(--oc-light)}.confirm-btn.danger[data-v-4c95e883]:hover{background:var(--danger-hover)}.keyboard-help-overlay[data-v-0d74ddfc]{position:fixed;top:0;right:0;bottom:0;left:0;background:#00000080;display:flex;align-items:center;justify-content:center;z-index:300}.keyboard-help[data-v-0d74ddfc]{background:var(--bg-primary);border:1px solid var(--border);border-radius:var(--radius);padding:24px;max-width:380px;width:90%}.keyboard-help h3[data-v-0d74ddfc]{font-size:16px;font-weight:700;margin-bottom:16px}.shortcut-list[data-v-0d74ddfc]{display:flex;flex-direction:column;gap:10px;margin-bottom:20px}.shortcut[data-v-0d74ddfc]{display:flex;align-items:center;gap:10px;font-size:14px;color:var(--text-secondary)}.help-close[data-v-0d74ddfc]{width:100%;padding:8px 20px;border-radius:var(--radius);font-size:14px;font-weight:500;background:var(--bg-tertiary);transition:background .1s}.help-close[data-v-0d74ddfc]:hover{background:var(--bg-hover)}.modal-overlay[data-v-e865cf63]{position:fixed;top:0;right:0;bottom:0;left:0;background:#00000080;display:flex;align-items:center;justify-content:center;z-index:300}.modal-content[data-v-e865cf63]{background:var(--bg-primary);border:1px solid var(--border);border-radius:var(--radius);max-width:480px;width:90%}.modal-header[data-v-e865cf63]{display:flex;align-items:center;justify-content:space-between;padding:16px 20px;border-bottom:1px solid var(--border)}.modal-header h3[data-v-e865cf63]{font-size:16px;font-weight:700}.modal-close[data-v-e865cf63]{font-size:20px;padding:4px 8px;border-radius:var(--radius)}.modal-close[data-v-e865cf63]:hover{background:var(--bg-hover)}.modal-body[data-v-e865cf63]{padding:20px}.file-label[data-v-e865cf63]{display:flex;flex-direction:column;gap:8px;margin-bottom:16px}.file-label-text[data-v-e865cf63]{font-size:14px;font-weight:500}.file-input-wrapper[data-v-e865cf63]{position:relative;display:inline-block}.file-input-native[data-v-e865cf63]{position:absolute;top:0;left:0;width:100%;height:100%;opacity:0;cursor:pointer}.file-input-button[data-v-e865cf63]{display:inline-block;padding:10px 18px;background:var(--bg-tertiary);color:var(--text-primary);border-radius:var(--radius);font-size:14px;font-weight:500;cursor:pointer;transition:background .1s;border:1px solid var(--border)}.file-input-button[data-v-e865cf63]:hover{background:var(--bg-hover)}.file-input-native:focus-visible+.file-input-button[data-v-e865cf63]{outline:2px solid var(--accent);outline-offset:2px}.import-status[data-v-e865cf63]{min-height:24px}.status-msg[data-v-e865cf63]{font-size:14px;padding:8px 12px;border-radius:var(--radius)}.status-msg.success[data-v-e865cf63]{background:#30d1581f;color:var(--success)}.status-msg.error[data-v-e865cf63]{background:#ff3b301f;color:var(--danger)}.status-msg.warning[data-v-e865cf63]{background:#ff9f0a1f;color:var(--warning)}.conflict-panel[data-v-e865cf63]{margin-top:16px}.conflict-panel h4[data-v-e865cf63]{font-size:14px;font-weight:700;margin-bottom:8px}.conflict-list[data-v-e865cf63]{display:flex;flex-direction:column;gap:4px;margin-bottom:12px;max-height:200px;overflow-y:auto}.conflict-item[data-v-e865cf63]{display:flex;gap:8px;padding:6px 10px;background:var(--bg-tertiary);border-radius:var(--radius);font-size:13px}.conflict-id[data-v-e865cf63]{font-weight:500;color:var(--text-primary)}.conflict-title[data-v-e865cf63]{color:var(--text-secondary)}.conflict-actions[data-v-e865cf63]{display:flex;gap:8px}.conflict-actions button[data-v-e865cf63]{padding:6px 14px;border-radius:var(--radius);font-size:13px;font-weight:500}.action-rename[data-v-e865cf63]{background:var(--accent);color:var(--oc-light)}.action-rename[data-v-e865cf63]:hover{background:var(--accent-hover)}.action-skip[data-v-e865cf63]{background:var(--bg-tertiary)}.action-skip[data-v-e865cf63]:hover{background:var(--bg-hover)}.action-cancel[data-v-e865cf63]{background:var(--bg-tertiary)}.action-cancel[data-v-e865cf63]:hover{background:var(--bg-hover)}.sessions-controls-bar[data-v-1cb29492]{margin-top:16px;margin-bottom:16px}.view-controls[data-v-1cb29492]{display:flex;align-items:center;justify-content:flex-end;gap:12px;flex-wrap:wrap}.search-input-container[data-v-1cb29492]{position:relative;display:flex;align-items:center}.search-icon[data-v-1cb29492]{position:absolute;left:12px;color:var(--text-tertiary);pointer-events:none;display:flex}.search-input[data-v-1cb29492]{padding:8px 36px;border:1px solid var(--border);border-radius:var(--radius-input);background:var(--bg-tertiary);color:var(--text-primary);font-family:var(--font-family);font-size:14px;width:240px;outline:none;transition:border-color .1s}.search-input[data-v-1cb29492]:focus{border-color:var(--accent)}.search-clear[data-v-1cb29492]{position:absolute;right:4px;padding:4px;border-radius:2px;display:flex;align-items:center}.search-clear[data-v-1cb29492]:hover{background:var(--bg-hover)}.sort-label[data-v-1cb29492]{display:flex;align-items:center;gap:6px;font-size:13px;color:var(--text-secondary)}.sort-select[data-v-1cb29492]{padding:6px 32px 6px 12px;border:1px solid var(--border);border-radius:var(--radius);background:var(--bg-tertiary);color:var(--text-primary);font-family:var(--font-family);font-size:13px;outline:none;-webkit-appearance:none;-moz-appearance:none;appearance:none;cursor:pointer;position:relative}.sort-select[data-v-1cb29492]:after{content:"";position:absolute;right:10px;top:50%;transform:translateY(-50%);width:12px;height:12px;background-image:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='12' height='12' viewBox='0 0 24 24' fill='none' stroke='currentColor' stroke-width='2'%3E%3Cpath d='M6 9l6 6 6-6'/%3E%3C/svg%3E");background-repeat:no-repeat;background-position:center;color:var(--text-secondary);pointer-events:none}.sort-select option[data-v-1cb29492]{background:var(--bg-primary);color:var(--text-primary)}.folder-groups[data-v-1cb29492]{display:flex;flex-direction:column;gap:24px}.folder-group[data-v-1cb29492]{border:1px solid var(--border);border-radius:var(--radius);overflow:hidden}.folder-header[data-v-1cb29492]{display:flex;align-items:center;gap:12px;padding:12px 16px;background:var(--bg-tertiary);border-bottom:1px solid var(--border);flex-wrap:wrap}.folder-title[data-v-1cb29492]{font-size:14px;font-weight:700}.folder-path[data-v-1cb29492]{font-size:12px;color:var(--text-tertiary);flex:1}.folder-sessions[data-v-1cb29492]{padding:8px;display:flex;flex-direction:column;gap:4px}.session-card[data-v-1cb29492]{padding:12px 16px}.session-card-header[data-v-1cb29492]{display:flex;align-items:center;justify-content:space-between}.session-title-row[data-v-1cb29492]{flex:1;min-width:0}.session-card-content[data-v-1cb29492]{display:flex;flex-direction:column;gap:4px}.session-title[data-v-1cb29492]{font-size:14px;font-weight:700;word-break:break-word}.session-id[data-v-1cb29492]{font-size:12px;color:var(--text-tertiary);word-break:break-all}.session-meta[data-v-1cb29492]{display:flex;gap:6px;flex-wrap:wrap;margin-top:4px}.session-card-actions[data-v-1cb29492]{flex-shrink:0;margin-left:12px}.session-card-more[data-v-1cb29492]{position:relative}.session-more-btn[data-v-1cb29492]{padding:4px 8px;border-radius:var(--radius);font-size:16px;line-height:1}.session-more-btn[data-v-1cb29492]:hover{background:var(--bg-hover)}.session-more-menu[data-v-1cb29492]{position:absolute;right:0;top:100%;margin-top:4px;background:var(--bg-secondary);border:1px solid var(--border);border-radius:var(--radius);min-width:200px;z-index:50;padding:4px}.session-more-menu .dropdown-item[data-v-1cb29492]{display:flex;align-items:center;gap:8px;width:100%;padding:8px 12px;font-size:13px;font-weight:500;border-radius:var(--radius);text-align:left}.session-more-menu .dropdown-item[data-v-1cb29492]:hover{background:var(--bg-hover)}.session-more-menu .dropdown-item.danger[data-v-1cb29492]{color:var(--danger)}.subsession-arrow[data-v-1cb29492]{transition:transform .2s ease}.subsession-arrow.rotated[data-v-1cb29492]{transform:rotate(180deg)}.children-container[data-v-1cb29492]{margin-left:24px;margin-top:4px;margin-bottom:4px;display:flex;flex-direction:column;gap:4px}.child-card[data-v-1cb29492]{padding:10px 14px}.cs-group{display:inline-flex;align-items:center;gap:4px;margin-right:8px}.cs-label{font-size:11px;font-weight:700;padding:1px 5px;border-radius:2px}.cs-label.sys{background:#8b5cf626;color:var(--sys-color)}.cs-label.tool{background:#ff9f0a26;color:var(--warning)}.cs-label.msg{background:#3d8bff26;color:var(--accent)}.cs-types{display:inline-flex;gap:3px}.cs-type{font-size:11px;font-weight:500;color:var(--text-secondary)}.cs-plus{color:var(--success);margin-left:2px}.cs-minus{color:var(--danger);margin-left:2px}.metadata-header[data-v-3c80789b]{padding:12px 16px;background:var(--bg-tertiary);font-size:12px;font-weight:700;color:var(--text-tertiary);text-transform:uppercase;letter-spacing:.8px}.metadata-body[data-v-3c80789b]{padding:16px}.stat-section[data-v-3c80789b]{margin-bottom:12px}.stat-section[data-v-3c80789b]:last-child{margin-bottom:0}.stat-section-title[data-v-3c80789b]{font-size:11px;color:var(--text-tertiary);font-weight:600;margin-bottom:6px;text-transform:uppercase;letter-spacing:.5px}.stat-row-inline[data-v-3c80789b]{display:flex;gap:12px;flex-wrap:wrap}.stat-item[data-v-3c80789b]{display:flex;align-items:center;gap:4px;font-size:12px;padding:4px 10px;background:var(--bg-tertiary);border-radius:var(--radius)}.stat-key[data-v-3c80789b]{color:var(--text-tertiary);font-weight:500;font-size:10px;text-transform:uppercase;letter-spacing:.4px}.stat-key[data-v-3c80789b]:after{content:":"}.stat-val[data-v-3c80789b]{color:var(--text-primary);font-weight:600;font-family:var(--font-family);font-size:12px}.stat-item.highlight .stat-val[data-v-3c80789b]{color:var(--accent)}.stat-item.warning .stat-val[data-v-3c80789b]{color:var(--warning)}.stat-item.danger .stat-val[data-v-3c80789b]{color:var(--danger)}.stat-item.success .stat-val[data-v-3c80789b]{color:var(--success)}.stat-grid[data-v-3c80789b]{display:grid;grid-template-columns:repeat(auto-fill,minmax(140px,1fr));gap:8px}.stat-grid .stat-item[data-v-3c80789b]{flex-direction:column;align-items:flex-start;gap:2px;padding:8px;background:var(--bg-tertiary);border-radius:var(--radius)}.stat-grid .stat-key[data-v-3c80789b]{font-size:10px;text-transform:uppercase;letter-spacing:.5px}.stat-grid .stat-val[data-v-3c80789b]{font-size:14px}.stat-link[data-v-3c80789b]{color:var(--accent);text-decoration:underline;font-weight:500}.stat-link[data-v-3c80789b]:hover{color:var(--accent-hover)}.timeline-controls[data-v-a8f75860]{display:flex;justify-content:flex-end;margin-top:16px;margin-bottom:16px}.sort-label[data-v-a8f75860]{display:flex;align-items:center;gap:6px;font-size:13px;color:var(--text-secondary)}.sort-select[data-v-a8f75860]{padding:6px 32px 6px 12px;border:1px solid var(--border);border-radius:var(--radius);background:var(--bg-tertiary);color:var(--text-primary);font-family:var(--font-family);font-size:13px;outline:none;-webkit-appearance:none;-moz-appearance:none;appearance:none;cursor:pointer;position:relative}.sort-select[data-v-a8f75860]:after{content:"";position:absolute;right:10px;top:50%;transform:translateY(-50%);width:12px;height:12px;background-image:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='12' height='12' viewBox='0 0 24 24' fill='none' stroke='currentColor' stroke-width='2'%3E%3Cpath d='M6 9l6 6 6-6'/%3E%3C/svg%3E");background-repeat:no-repeat;background-position:center;color:var(--text-secondary);pointer-events:none}.sort-select option[data-v-a8f75860]{background:var(--bg-primary);color:var(--text-primary)}.timeline[data-v-a8f75860]{display:flex;flex-direction:column;gap:4px}.timeline-item[data-v-a8f75860]{margin-bottom:4px}.timeline-card[data-v-a8f75860]{padding:12px 16px}.timeline-card-header[data-v-a8f75860]{display:flex;align-items:center;justify-content:space-between;gap:12px;flex-wrap:wrap}.timeline-card-header .left[data-v-a8f75860]{display:flex;align-items:center;gap:8px;flex-wrap:wrap;flex:1;min-width:0}.timeline-card-header .right[data-v-a8f75860]{display:flex;align-items:center;gap:6px;flex-shrink:0}.req-num[data-v-a8f75860]{font-weight:700;font-size:14px;color:var(--text-primary)}.call-type[data-v-a8f75860]{font-size:11px;font-weight:700;padding:1px 6px;border-radius:2px}.call-type.user[data-v-a8f75860]{background:#3d8bff26;color:var(--accent)}.call-type.agent[data-v-a8f75860]{background:#8b5cf626;color:var(--sys-color)}.url[data-v-a8f75860]{font-size:13px;color:var(--text-secondary);overflow:hidden;text-overflow:ellipsis;white-space:nowrap;max-width:300px}.time-badge[data-v-a8f75860]{font-size:12px;font-weight:500;padding:2px 8px;border-radius:var(--radius);white-space:nowrap}.time-badge.dur[data-v-a8f75860]{background:#3d8bff1a;color:var(--accent)}.time-badge.gap[data-v-a8f75860]{background:#ff9f0a1a;color:var(--warning)}.time-badge.model[data-v-a8f75860]{background:#30d1581a;color:var(--success)}.time-badge.provider[data-v-a8f75860]{background:var(--bg-tertiary);color:var(--text-secondary)}.block-item[data-v-0b5e0f57]{margin-bottom:8px}.block-type-bar[data-v-0b5e0f57]{display:flex;align-items:center;justify-content:space-between;padding:6px 10px;background:var(--bg-tertiary);border-radius:var(--radius) var(--radius) 0 0;border:1px solid var(--border);border-bottom:none}.block-type-left[data-v-0b5e0f57]{display:flex;align-items:center;gap:8px;font-size:12px}.block-type-right[data-v-0b5e0f57]{display:flex;align-items:center;gap:4px}.block-type-tag[data-v-0b5e0f57]{font-weight:700;font-size:11px;padding:1px 6px;border-radius:2px}.block-type-tag.text[data-v-0b5e0f57]{background:#3d8bff26;color:var(--accent)}.block-type-tag.thinking[data-v-0b5e0f57]{background:#8b5cf626;color:var(--sys-color)}.block-type-tag.td[data-v-0b5e0f57]{background:#ff9f0a26;color:var(--warning)}.block-type-tag.tc[data-v-0b5e0f57],.block-type-tag.tr[data-v-0b5e0f57]{background:#30d15826;color:var(--success)}.block-type-tag.image[data-v-0b5e0f57]{background:#9a989826;color:var(--text-secondary)}.block-type-tag.json[data-v-0b5e0f57]{background:#a78bfa26;color:var(--json-key-color)}.block-type-tag.xml[data-v-0b5e0f57]{background:#ff9f0a26;color:var(--warning)}.block-type-tag.status[data-v-0b5e0f57]{background:#9a989826;color:var(--text-secondary)}.tool-meta-name[data-v-0b5e0f57]{font-weight:500;color:var(--text-primary)}.tool-meta-id[data-v-0b5e0f57]{color:var(--text-secondary)}.toggle-btn[data-v-0b5e0f57]{font-size:11px;font-weight:700;padding:2px 6px;border-radius:2px;color:var(--text-tertiary);transition:color .1s,background .1s}.toggle-btn[data-v-0b5e0f57]:hover,.toggle-btn.active[data-v-0b5e0f57]{color:var(--accent);background:#3d8bff1a}.block-content[data-v-0b5e0f57]{padding:10px 12px;background:var(--bg-primary);border:1px solid var(--border);border-radius:0 0 var(--radius) var(--radius);font-size:13px;line-height:1.6;overflow-x:auto;position:relative}.block-content.empty[data-v-0b5e0f57]{display:none}.block-content.raw[data-v-0b5e0f57]{white-space:pre-wrap;word-break:break-word}.copy-btn[data-v-0b5e0f57]{position:absolute;top:6px;right:6px;padding:4px;border-radius:2px;opacity:0;transition:opacity .1s;z-index:1}.block-content:hover .copy-btn[data-v-0b5e0f57]{opacity:1}.copy-btn[data-v-0b5e0f57]:hover{background:var(--bg-hover)}.content-muted[data-v-0b5e0f57]{color:var(--text-tertiary);font-style:italic}.tool-def-desc[data-v-0b5e0f57]{margin-bottom:8px;color:var(--text-secondary)}.tool-def-params[data-v-0b5e0f57]{margin-top:4px}.section-content[data-v-33730ba0]{margin-bottom:16px}.change-card[data-v-33730ba0]{border:1px solid var(--border);border-radius:var(--radius);margin-bottom:8px;overflow:hidden}.change-card.added[data-v-33730ba0]{border-left:3px solid var(--success)}.change-card.removed[data-v-33730ba0]{border-left:3px solid var(--danger)}.change-card-header[data-v-33730ba0]{display:flex;align-items:center;gap:8px;padding:8px 12px;background:var(--bg-tertiary);border-bottom:1px solid var(--border)}.change-card-body[data-v-33730ba0]{padding:8px}.cat-tag[data-v-33730ba0]{font-size:11px;font-weight:700;padding:1px 6px;border-radius:2px}.cat-tag.sys[data-v-33730ba0]{background:#8b5cf626;color:var(--sys-color)}.cat-tag.tool[data-v-33730ba0]{background:#ff9f0a26;color:var(--warning)}.cat-tag.msg[data-v-33730ba0]{background:#3d8bff26;color:var(--accent)}.action-tag[data-v-33730ba0]{font-size:11px;font-weight:700;padding:1px 6px;border-radius:2px}.action-tag.new[data-v-33730ba0]{background:#30d15826;color:var(--success)}.action-tag.del[data-v-33730ba0]{background:#ff3b3026;color:var(--danger)}.section-content[data-v-7b569ee4]{margin-bottom:16px}.msg[data-v-7b569ee4]{border:1px solid var(--border);border-radius:var(--radius);margin-bottom:8px;overflow:hidden}.msg-header[data-v-7b569ee4]{display:flex;align-items:center;gap:8px;padding:8px 12px;background:var(--bg-tertiary);border-bottom:1px solid var(--border)}.msg-body[data-v-7b569ee4]{padding:8px}.cat-tag[data-v-7b569ee4]{font-size:11px;font-weight:700;padding:1px 6px;border-radius:2px}.cat-tag.sys[data-v-7b569ee4]{background:#8b5cf626;color:var(--sys-color)}.cat-tag.tool[data-v-7b569ee4]{background:#ff9f0a26;color:var(--warning)}.cat-tag.msg[data-v-7b569ee4]{background:#3d8bff26;color:var(--accent)}.role-tag[data-v-7b569ee4]{font-size:12px;font-weight:500;color:var(--text-secondary)}.json-container[data-v-ec5b7cf7]{position:relative;background:var(--bg-primary);font-size:13px;line-height:1.6}.copy-btn[data-v-ec5b7cf7]{position:absolute;top:6px;right:6px;padding:4px;border-radius:2px;opacity:0;transition:opacity .1s;z-index:1}.json-container:hover .copy-btn[data-v-ec5b7cf7]{opacity:1}.copy-btn[data-v-ec5b7cf7]:hover{background:var(--bg-hover)}.json-line[data-v-ec5b7cf7]{display:flex;padding:0 12px;min-height:22px}.json-line[data-v-ec5b7cf7]:hover{background:var(--bg-hover)}.json-num[data-v-ec5b7cf7]{flex-shrink:0;width:40px;text-align:right;padding-right:12px;color:var(--text-tertiary);-webkit-user-select:none;user-select:none}.section-content[data-v-33c2a25e]{margin-bottom:16px}.json-block[data-v-33c2a25e]{border:1px solid var(--border);border-radius:var(--radius);overflow:hidden}.section-content[data-v-f86302b7]{margin-bottom:16px}.json-block[data-v-f86302b7]{border:1px solid var(--border);border-radius:var(--radius);overflow:hidden}.view-toggle[data-v-5c1b1583]{display:flex;gap:4px;margin-top:16px;margin-bottom:16px;padding:4px;background:var(--bg-tertiary);border-radius:var(--radius);width:fit-content}.view-toggle button[data-v-5c1b1583]{padding:6px 16px;border-radius:var(--radius);font-size:13px;font-weight:500;color:var(--text-secondary);transition:background .1s,color .1s}.view-toggle button[data-v-5c1b1583]:hover{color:var(--text-primary)}.view-toggle button.active[data-v-5c1b1583]{background:var(--bg-primary);color:var(--text-primary);font-weight:700}.record-view-content[data-v-5c1b1583]{margin-top:8px}*,*:before,*:after{box-sizing:border-box;margin:0;padding:0}:root{--oc-dark: #201d1d;--oc-light: #fdfcfc;--oc-mid: #9a9898;--oc-dark-surface: #302c2c;--oc-border-gray: #646262;--oc-light-surface: #f1eeee;--accent: #3d8bff;--accent-hover: #1a6fff;--accent-active: #0056b3;--danger: #ff3b30;--danger-hover: #d70015;--danger-active: #a50011;--success: #30d158;--warning: #ff9f0a;--warning-hover: #cc7f08;--warning-active: #995f06;--sys-color: #8b5cf6;--json-key-color: #a78bfa;--text-muted: #6e6e73;--text-secondary-light: #424245;--border-warm: rgba(15, 0, 0, .12);--border-tab: #9a9898;--border-outline: #646262;--input-bg: #f8f7f7;--bg-primary: var(--oc-dark);--bg-secondary: var(--oc-dark-surface);--bg-tertiary: rgba(253, 252, 252, .04);--bg-hover: rgba(253, 252, 252, .08);--text-primary: var(--oc-light);--text-secondary: var(--oc-mid);--text-tertiary: var(--text-muted);--border: rgba(100, 98, 98, .25);--added-bg: rgba(48, 209, 88, .08);--added-border: var(--success);--removed-bg: rgba(255, 59, 48, .08);--removed-border: var(--danger);--radius: 4px;--radius-input: 6px;--font-family: "Berkeley Mono", "IBM Plex Mono", ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace}[data-theme=light]{--bg-primary: var(--oc-light);--bg-secondary: var(--oc-light-surface);--bg-tertiary: rgba(32, 29, 29, .04);--bg-hover: rgba(32, 29, 29, .06);--text-primary: var(--oc-dark);--text-secondary: var(--text-secondary-light);--text-tertiary: var(--oc-mid);--border: var(--border-warm);--added-bg: rgba(48, 209, 88, .06);--added-border: #1a8a3d;--removed-bg: rgba(255, 59, 48, .06);--removed-border: #c41e3a}html,body{height:100%;font-family:var(--font-family);background:var(--bg-primary);color:var(--text-primary);line-height:1.5;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}a{color:var(--accent);text-decoration:underline;font-weight:500;transition:color .1s}a:hover{color:var(--accent-hover)}button{font-family:inherit;cursor:pointer;border:none;background:none;color:inherit;font-size:inherit}.skip-link{position:absolute;top:-100px;left:50%;transform:translate(-50%);padding:8px 16px;background:var(--accent);color:var(--oc-light);font-weight:500;border-radius:var(--radius);z-index:1000;transition:top .2s}.skip-link:focus{top:8px}.sr-only{position:absolute;width:1px;height:1px;padding:0;margin:-1px;overflow:hidden;clip:rect(0,0,0,0);white-space:nowrap;border:0}#app{display:flex;flex-direction:column;height:100vh;overflow:hidden}main{flex:1;overflow-y:auto;padding:24px;-webkit-overflow-scrolling:touch}.container{max-width:1200px;margin:0 auto}.spinner{width:20px;height:20px;border:2px solid var(--border);border-top-color:var(--accent);border-radius:50%;animation:spin .6s linear infinite;display:inline-block;vertical-align:middle;margin-right:8px}@keyframes spin{to{transform:rotate(360deg)}}.loading{display:flex;align-items:center;justify-content:center;padding:48px;color:var(--text-secondary);font-size:14px}.empty-state{text-align:center;padding:64px 24px;color:var(--text-secondary)}.empty-state .icon{font-size:48px;margin-bottom:16px}.empty-state p{font-size:16px;margin-bottom:8px}.error-banner{padding:16px 20px;background:var(--removed-bg);border:1px solid var(--removed-border);border-radius:var(--radius);color:var(--danger);font-size:14px}.page-title{font-size:38px;font-weight:700;line-height:1.5;margin-bottom:24px;display:flex;align-items:baseline;gap:12px;flex-wrap:wrap}.page-title .count{font-size:16px;font-weight:500;color:var(--text-secondary)}.page-title .subtitle{font-size:16px;font-weight:400;color:var(--text-tertiary)}.card{background:var(--bg-secondary);border:1px solid var(--border);border-radius:var(--radius)}.card-interactive{cursor:pointer;transition:background .1s,border-color .1s}.card-interactive:hover{background:var(--bg-hover)}.card-interactive:focus-visible{outline:2px solid var(--accent);outline-offset:2px}.badge{display:inline-flex;align-items:center;padding:2px 8px;font-size:12px;font-weight:500;color:var(--text-secondary);background:var(--bg-tertiary);border-radius:var(--radius);white-space:nowrap}code{background:var(--bg-tertiary);padding:2px 6px;border-radius:var(--radius);font-family:var(--font-family);font-size:13px}kbd{display:inline-block;padding:2px 8px;font-size:12px;font-family:var(--font-family);background:var(--bg-tertiary);border:1px solid var(--border);border-radius:var(--radius);min-width:24px;text-align:center}.section-title{display:flex;align-items:center;gap:8px;font-size:13px;font-weight:700;padding:8px 12px;background:var(--bg-tertiary);border-radius:var(--radius);margin-bottom:8px;cursor:pointer;-webkit-user-select:none;user-select:none;transition:background .1s}.section-title:hover{background:var(--bg-hover)}.section-arrow{transition:transform .2s ease;color:var(--text-secondary)}.section-title.expanded .section-arrow{transform:rotate(0)}.section-title:not(.expanded) .section-arrow{transform:rotate(-90deg)}
@@ -0,0 +1,30 @@
1
+ <!DOCTYPE html>
2
+ <html lang="en" data-theme="dark">
3
+
4
+ <head>
5
+ <meta charset="UTF-8" />
6
+ <meta name="viewport" content="width=device-width, initial-scale=1.0" />
7
+ <title>OpenCode Trace</title>
8
+ <link rel="icon" type="image/svg+xml" href="/assets/favicon-v9nnlQV_.svg" />
9
+ <style>
10
+ html,
11
+ body,
12
+ #app {
13
+ margin: 0;
14
+ padding: 0;
15
+ height: 100%;
16
+ }
17
+
18
+ body {
19
+ font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, sans-serif;
20
+ }
21
+ </style>
22
+ <script type="module" crossorigin src="/assets/index-DoL5ISmB.js"></script>
23
+ <link rel="stylesheet" crossorigin href="/assets/index-idry_N9R.css">
24
+ </head>
25
+
26
+ <body>
27
+ <div id="app"></div>
28
+ </body>
29
+
30
+ </html>
@@ -0,0 +1,12 @@
1
+ export interface ViewerOptions {
2
+ port?: number;
3
+ traceDir?: string;
4
+ open?: boolean;
5
+ corsOrigin?: string | string[] | RegExp | boolean;
6
+ }
7
+ export interface ViewerInstance {
8
+ url: string;
9
+ close: () => Promise<void>;
10
+ }
11
+ export declare function createViewer(options?: ViewerOptions): Promise<ViewerInstance>;
12
+ //# sourceMappingURL=server.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server.d.ts","sourceRoot":"","sources":["../src/server.ts"],"names":[],"mappings":"AAuCA,MAAM,WAAW,aAAa;IAC5B,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,IAAI,CAAC,EAAE,OAAO,CAAC;IACf,UAAU,CAAC,EAAE,MAAM,GAAG,MAAM,EAAE,GAAG,MAAM,GAAG,OAAO,CAAC;CACnD;AAED,MAAM,WAAW,cAAc;IAC7B,GAAG,EAAE,MAAM,CAAC;IACZ,KAAK,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;CAC5B;AAED,wBAAsB,YAAY,CAAC,OAAO,CAAC,EAAE,aAAa,GAAG,OAAO,CAAC,cAAc,CAAC,CAqWnF"}
package/dist/server.js ADDED
@@ -0,0 +1,348 @@
1
+ import { readFileSync, existsSync } from "node:fs";
2
+ import { join, dirname } from "node:path";
3
+ import { fileURLToPath } from "node:url";
4
+ import Fastify from "fastify";
5
+ import cors from "@fastify/cors";
6
+ import rateLimit from "@fastify/rate-limit";
7
+ import multipart from "@fastify/multipart";
8
+ import fastifyStatic from "@fastify/static";
9
+ import { store, parse, query, transform, record } from "@opencode-trace/core";
10
+ const __dirname = dirname(fileURLToPath(import.meta.url));
11
+ function validateSessionId(sessionId) {
12
+ return typeof sessionId === "string" && sessionId.length > 0 && sessionId.length <= 256 && /^[a-zA-Z0-9_-]+$/.test(sessionId);
13
+ }
14
+ function validateRecordId(recordId) {
15
+ const num = parseInt(recordId, 10);
16
+ return { valid: !isNaN(num) && num > 0 && num <= 999999, value: num };
17
+ }
18
+ function validateParams(reply, sessionId, recordId) {
19
+ if (!validateSessionId(sessionId)) {
20
+ reply.code(400);
21
+ reply.send({ error: "Invalid session ID format" });
22
+ return null;
23
+ }
24
+ if (recordId !== undefined) {
25
+ const result = validateRecordId(recordId);
26
+ if (!result.valid) {
27
+ reply.code(400);
28
+ reply.send({ error: "Invalid record ID format" });
29
+ return null;
30
+ }
31
+ return result.value;
32
+ }
33
+ return 0;
34
+ }
35
+ export async function createViewer(options) {
36
+ const port = options?.port ?? 3210;
37
+ const traceDir = options?.traceDir;
38
+ const storeOpts = traceDir ? { traceDir } : undefined;
39
+ const app = Fastify({ logger: false });
40
+ await app.register(cors, {
41
+ origin: options?.corsOrigin ?? [/^https?:\/\/localhost(:\d+)?$/],
42
+ });
43
+ await app.register(rateLimit, {
44
+ max: 1000,
45
+ timeWindow: "1 minute",
46
+ allowList: ["127.0.0.1", "::1"],
47
+ });
48
+ await app.register(multipart);
49
+ const publicDir = join(__dirname, "public");
50
+ if (existsSync(publicDir)) {
51
+ await app.register(fastifyStatic, {
52
+ root: publicDir,
53
+ prefix: "/",
54
+ });
55
+ }
56
+ app.setNotFoundHandler(async (_req, reply) => {
57
+ const indexPath = join(publicDir, "index.html");
58
+ if (existsSync(indexPath)) {
59
+ reply.type("text/html; charset=utf-8").send(readFileSync(indexPath, "utf-8"));
60
+ }
61
+ else {
62
+ reply.code(404).send({ error: "Not found" });
63
+ }
64
+ });
65
+ app.get("/api/sessions", async (_req, reply) => {
66
+ const sessions = store.listSessions(storeOpts);
67
+ return sessions;
68
+ });
69
+ app.get("/api/sessions/tree", async (_req, reply) => {
70
+ const tree = store.listSessionsTree(storeOpts);
71
+ return tree;
72
+ });
73
+ app.get("/api/sessions/:sessionId/timeline", async (req, reply) => {
74
+ const { sessionId } = req.params;
75
+ if (!validateSessionId(sessionId)) {
76
+ reply.code(400);
77
+ return { error: "Invalid session ID format" };
78
+ }
79
+ const records = store.getSessionRecords(sessionId, storeOpts);
80
+ const parsedRecords = records
81
+ .map((rec) => {
82
+ const parsed = parse.detectAndParse(rec);
83
+ const provider = parse.detectProvider(rec.request.url, rec.request.body);
84
+ let requestMsgs = parsed.msgs;
85
+ if (provider === "openai-chat") {
86
+ const reqParsed = parse.openaiChatParser.parseRequest(rec.request.body);
87
+ requestMsgs = reqParsed.msgs;
88
+ }
89
+ else if (provider === "openai-responses") {
90
+ const reqParsed = parse.openaiResponsesParser.parseRequest(rec.request.body);
91
+ requestMsgs = reqParsed.msgs;
92
+ }
93
+ else if (provider === "anthropic") {
94
+ const reqParsed = parse.anthropicParser.parseRequest(rec.request.body);
95
+ requestMsgs = reqParsed.msgs;
96
+ }
97
+ return {
98
+ id: rec.id,
99
+ requestAt: rec.requestAt,
100
+ requestMsgs,
101
+ parsed,
102
+ };
103
+ })
104
+ .filter((c) => c.parsed.provider !== "unknown" || c.parsed.msgs.length > 0);
105
+ const timeline = query.buildSessionTimeline(sessionId, parsedRecords);
106
+ const recordMeta = parsedRecords.map((r) => ({
107
+ id: r.id,
108
+ model: r.parsed.model,
109
+ provider: r.parsed.provider,
110
+ }));
111
+ return { ...timeline, recordMeta };
112
+ });
113
+ app.get("/api/sessions/:sessionId/metadata", async (req, reply) => {
114
+ const { sessionId } = req.params;
115
+ if (!validateSessionId(sessionId)) {
116
+ reply.code(400);
117
+ return { error: "Invalid session ID format" };
118
+ }
119
+ const records = store.getSessionRecords(sessionId, storeOpts);
120
+ const parsedRecords = records
121
+ .map((rec) => ({
122
+ id: rec.id,
123
+ record: rec,
124
+ parsed: parse.detectAndParse(rec),
125
+ }))
126
+ .filter((c) => c.parsed.provider !== "unknown" || c.parsed.msgs.length > 0);
127
+ const sessions = store.listSessions(storeOpts);
128
+ const sessionMeta = sessions.find((s) => s.id === sessionId);
129
+ const tree = store.listSessionsTree(storeOpts);
130
+ const node = tree.find((n) => n.id === sessionId);
131
+ const metadata = query.buildSessionMetadata(sessionId, parsedRecords, sessionMeta?.folderPath);
132
+ if (sessionMeta) {
133
+ metadata.createdAt = sessionMeta.createdAt;
134
+ metadata.updatedAt = sessionMeta.updatedAt;
135
+ metadata.subSessions = node?.children?.map(c => c.id) ?? [];
136
+ metadata.parentSession = sessionMeta.parentID ?? null;
137
+ }
138
+ return metadata;
139
+ });
140
+ app.get("/api/sessions/:sessionId/records/:recordId/parsed", async (req, reply) => {
141
+ const { sessionId, recordId } = req.params;
142
+ const rid = validateParams(reply, sessionId, recordId);
143
+ if (rid === null)
144
+ return;
145
+ const rec = store.getRecord(sessionId, rid, storeOpts);
146
+ if (!rec) {
147
+ reply.code(404);
148
+ return { error: "Record not found" };
149
+ }
150
+ const parsed = parse.detectAndParse(rec);
151
+ return parsed;
152
+ });
153
+ app.get("/api/sessions/:sessionId/records/:recordId/usage", async (req, reply) => {
154
+ const { sessionId, recordId } = req.params;
155
+ const rid = validateParams(reply, sessionId, recordId);
156
+ if (rid === null)
157
+ return;
158
+ const rec = store.getRecord(sessionId, rid, storeOpts);
159
+ if (!rec) {
160
+ reply.code(404);
161
+ return { error: "Record not found" };
162
+ }
163
+ const usage = parse.extractUsage(rec);
164
+ return usage;
165
+ });
166
+ app.get("/api/sessions/:sessionId/records/:recordId/latency", async (req, reply) => {
167
+ const { sessionId, recordId } = req.params;
168
+ const rid = validateParams(reply, sessionId, recordId);
169
+ if (rid === null)
170
+ return;
171
+ const rec = store.getRecord(sessionId, rid, storeOpts);
172
+ if (!rec) {
173
+ reply.code(404);
174
+ return { error: "Record not found" };
175
+ }
176
+ const latency = parse.extractLatency(rec);
177
+ return latency ?? { error: "No latency data available" };
178
+ });
179
+ app.get("/api/sessions/:sessionId/records/:recordId/sse", async (req, reply) => {
180
+ const { sessionId, recordId } = req.params;
181
+ const rid = validateParams(reply, sessionId, recordId);
182
+ if (rid === null)
183
+ return;
184
+ const sseData = store.getSSEStream(sessionId, rid, storeOpts);
185
+ if (!sseData) {
186
+ reply.code(404);
187
+ return { error: "No SSE data found" };
188
+ }
189
+ const rec = store.getRecord(sessionId, rid, storeOpts);
190
+ const provider = rec?.request
191
+ ? parse.detectProvider(rec.request.url, rec.request.body)
192
+ : null;
193
+ let messages;
194
+ if (provider === "anthropic") {
195
+ messages = transform.sseAnthropicToMessages(sseData);
196
+ }
197
+ else if (provider === "openai-responses") {
198
+ messages = transform.sseOpenaiResponsesToMessages(sseData);
199
+ }
200
+ else {
201
+ messages = transform.sseOpenaiChatToMessages(sseData);
202
+ }
203
+ return { raw: sseData, messages };
204
+ });
205
+ app.get("/api/sessions/:sessionId/records/:recordId", async (req, reply) => {
206
+ const { sessionId, recordId } = req.params;
207
+ const rid = validateParams(reply, sessionId, recordId);
208
+ if (rid === null)
209
+ return;
210
+ const rec = store.getRecord(sessionId, rid, storeOpts);
211
+ if (!rec) {
212
+ reply.code(404);
213
+ return { error: "Record not found" };
214
+ }
215
+ return rec;
216
+ });
217
+ app.get("/api/sessions/:sessionId", async (req, reply) => {
218
+ const { sessionId } = req.params;
219
+ if (!validateSessionId(sessionId)) {
220
+ reply.code(400);
221
+ return { error: "Invalid session ID format" };
222
+ }
223
+ const sessions = store.listSessions(storeOpts);
224
+ const sessionMeta = sessions.find((s) => s.id === sessionId);
225
+ const records = store.getSessionRecords(sessionId, storeOpts);
226
+ const enriched = records.map((rec) => ({
227
+ ...rec,
228
+ provider: rec?.request
229
+ ? parse.detectProvider(rec.request.url, rec.request.body)
230
+ : null,
231
+ }));
232
+ return {
233
+ session: sessionMeta ??
234
+ {
235
+ id: sessionId,
236
+ requestCount: records.length,
237
+ createdAt: null,
238
+ updatedAt: null,
239
+ },
240
+ records: enriched,
241
+ };
242
+ });
243
+ app.get("/api/trace/status", async (_req, reply) => {
244
+ const enabled = record.getGlobalTraceEnabled(traceDir);
245
+ return { globalEnabled: enabled };
246
+ });
247
+ app.get("/api/trace/enable", async (_req, reply) => {
248
+ record.setGlobalTraceEnabled(true, traceDir);
249
+ return { success: true, globalEnabled: true };
250
+ });
251
+ app.get("/api/trace/disable", async (_req, reply) => {
252
+ record.setGlobalTraceEnabled(false, traceDir);
253
+ return { success: true, globalEnabled: false };
254
+ });
255
+ app.get("/api/trace-dir", async (_req, reply) => {
256
+ return { traceDir: store.getTraceDir(storeOpts) };
257
+ });
258
+ app.post("/api/sessions/:sessionId/export", async (req, reply) => {
259
+ try {
260
+ const { sessionId } = req.params;
261
+ if (!validateSessionId(sessionId)) {
262
+ reply.code(400);
263
+ return { error: "Invalid session ID format" };
264
+ }
265
+ const stream = await store.exportSessionZip(sessionId, storeOpts);
266
+ reply
267
+ .type("application/zip")
268
+ .header("Content-Disposition", `attachment; filename="session-${sessionId}.zip"`)
269
+ .send(stream);
270
+ }
271
+ catch (e) {
272
+ const err = e;
273
+ if (err.message === "Session not found") {
274
+ reply.code(404);
275
+ return { error: "Session not found" };
276
+ }
277
+ reply.code(500);
278
+ return { error: "Export failed: " + err.message };
279
+ }
280
+ });
281
+ app.post("/api/sessions/import", async (req, reply) => {
282
+ try {
283
+ const data = await req.file();
284
+ if (!data) {
285
+ reply.code(400);
286
+ return { error: "No file in multipart data" };
287
+ }
288
+ const fileBuffer = await data.toBuffer();
289
+ const conflictStrategy = data.fields.conflictStrategy?.value ?? "prompt";
290
+ const validStrategies = ["prompt", "rename", "skip", "overwrite"];
291
+ if (!validStrategies.includes(conflictStrategy)) {
292
+ reply.code(400);
293
+ return { error: `Invalid conflict strategy: ${conflictStrategy}. Valid: ${validStrategies.join(", ")}` };
294
+ }
295
+ const result = await store.importSessionZip(fileBuffer, {
296
+ ...storeOpts,
297
+ conflictStrategy: conflictStrategy,
298
+ });
299
+ return result;
300
+ }
301
+ catch (e) {
302
+ const err = e;
303
+ if (err.message.includes("Invalid") || err.message.includes("No file")) {
304
+ reply.code(400);
305
+ return { error: err.message };
306
+ }
307
+ reply.code(500);
308
+ return { error: "Import failed: " + err.message };
309
+ }
310
+ });
311
+ app.post("/api/sessions/:sessionId/delete", async (req, reply) => {
312
+ try {
313
+ const { sessionId } = req.params;
314
+ if (!validateSessionId(sessionId)) {
315
+ reply.code(400);
316
+ return { error: "Invalid session ID format" };
317
+ }
318
+ await store.deleteSession(sessionId, storeOpts);
319
+ return { success: true, sessionId };
320
+ }
321
+ catch (e) {
322
+ const err = e;
323
+ if (err.message === "Session not found") {
324
+ reply.code(404);
325
+ return { error: "Session not found" };
326
+ }
327
+ reply.code(500);
328
+ return { error: "Delete failed: " + err.message };
329
+ }
330
+ });
331
+ await app.listen({ port, host: "0.0.0.0" });
332
+ const addr = `http://localhost:${port}`;
333
+ if (options?.open) {
334
+ import("node:child_process").then(({ exec }) => {
335
+ const cmd = process.platform === "darwin"
336
+ ? "open"
337
+ : process.platform === "win32"
338
+ ? "start"
339
+ : "xdg-open";
340
+ exec(`${cmd} ${addr}`);
341
+ });
342
+ }
343
+ return {
344
+ url: addr,
345
+ close: () => app.close(),
346
+ };
347
+ }
348
+ //# sourceMappingURL=server.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server.js","sourceRoot":"","sources":["../src/server.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=server.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server.test.d.ts","sourceRoot":"","sources":["../src/server.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,102 @@
1
+ import { describe, it, expect, vi, beforeEach, afterEach } from "vitest";
2
+ import Fastify from "fastify";
3
+ import cors from "@fastify/cors";
4
+ import multipart from "@fastify/multipart";
5
+ import { store, parse } from "@opencode-trace/core";
6
+ function buildTestApp() {
7
+ const app = Fastify({ logger: false });
8
+ app.register(cors, { origin: "*" });
9
+ app.register(multipart);
10
+ app.get("/", async (_req, reply) => {
11
+ reply.type("text/html; charset=utf-8").send("<html></html>");
12
+ });
13
+ app.get("/api/sessions", async (_req, reply) => {
14
+ return store.listSessions();
15
+ });
16
+ app.get("/api/sessions/tree", async (_req, reply) => {
17
+ return store.listSessionsTree();
18
+ });
19
+ app.get("/api/sessions/:sessionId/timeline", async (req, reply) => {
20
+ const { sessionId } = req.params;
21
+ const records = store.getSessionRecords(sessionId);
22
+ const parsedRecords = records.map((rec) => {
23
+ const parsed = parse.detectAndParse(rec);
24
+ const provider = parse.detectProvider(rec.request.url, rec.request.body);
25
+ let requestMsgs = parsed.msgs;
26
+ if (provider === "openai-chat") {
27
+ requestMsgs = parse.openaiChatParser.parseRequest(rec.request.body).msgs;
28
+ }
29
+ else if (provider === "openai-responses") {
30
+ requestMsgs = parse.openaiResponsesParser.parseRequest(rec.request.body).msgs;
31
+ }
32
+ else if (provider === "anthropic") {
33
+ requestMsgs = parse.anthropicParser.parseRequest(rec.request.body).msgs;
34
+ }
35
+ return { id: rec.id, requestAt: rec.requestAt, requestMsgs, parsed };
36
+ }).filter((c) => c.parsed.provider !== "unknown" || c.parsed.msgs.length > 0);
37
+ const timeline = { messages: [], recordMeta: [] };
38
+ return { ...timeline, recordMeta: parsedRecords.map((r) => ({ id: r.id, model: r.parsed.model, provider: r.parsed.provider })) };
39
+ });
40
+ app.get("/api/sessions/:sessionId/records/:recordId/latency", async (req, reply) => {
41
+ const { sessionId, recordId } = req.params;
42
+ const rid = parseInt(recordId, 10);
43
+ const rec = store.getRecord(sessionId, rid);
44
+ if (!rec) {
45
+ reply.code(404);
46
+ return { error: "Record not found" };
47
+ }
48
+ const latency = parse.extractLatency(rec);
49
+ return latency ?? { error: "No latency data available" };
50
+ });
51
+ return app;
52
+ }
53
+ describe("Server API", () => {
54
+ let server = null;
55
+ beforeEach(() => {
56
+ vi.clearAllMocks();
57
+ });
58
+ afterEach(async () => {
59
+ if (server) {
60
+ await server.close();
61
+ server = null;
62
+ }
63
+ });
64
+ describe("Latency endpoint", () => {
65
+ it("should return latency info for stream requests", async () => {
66
+ const mockRecord = {
67
+ id: 1,
68
+ purpose: "",
69
+ requestAt: "2026-04-29T00:00:00.000Z",
70
+ responseAt: "2026-04-29T00:00:01.000Z",
71
+ request: { method: "POST", url: "https://example.com", headers: {}, body: null },
72
+ response: null,
73
+ error: null,
74
+ requestSentAt: 1234567.89,
75
+ firstTokenAt: 1234570.12,
76
+ lastTokenAt: 1234590.34,
77
+ };
78
+ vi.spyOn(store, "getRecord").mockReturnValue(mockRecord);
79
+ const app = buildTestApp();
80
+ await app.ready();
81
+ const response = await app.inject({
82
+ method: "GET",
83
+ url: "/api/sessions/test/records/1/latency",
84
+ });
85
+ expect(response.statusCode).toBe(200);
86
+ const data = response.json();
87
+ expect(data.ttft).toBeDefined();
88
+ expect(data.totalDuration).toBeDefined();
89
+ });
90
+ it("should return 404 for non-existent record", async () => {
91
+ vi.spyOn(store, "getRecord").mockReturnValue(null);
92
+ const app = buildTestApp();
93
+ await app.ready();
94
+ const response = await app.inject({
95
+ method: "GET",
96
+ url: "/api/sessions/test/records/999/latency",
97
+ });
98
+ expect(response.statusCode).toBe(404);
99
+ });
100
+ });
101
+ });
102
+ //# sourceMappingURL=server.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server.test.js","sourceRoot":"","sources":["../src/server.test.ts"],"names":[],"mappings":"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"}