@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.
- package/README.md +94 -0
- package/dist/cli.d.ts +3 -0
- package/dist/cli.d.ts.map +1 -0
- package/dist/cli.js +21 -0
- package/dist/cli.js.map +1 -0
- package/dist/index.d.ts +3 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +2 -0
- package/dist/index.js.map +1 -0
- package/dist/public/assets/favicon-v9nnlQV_.svg +53 -0
- package/dist/public/assets/index-DoL5ISmB.js +91 -0
- package/dist/public/assets/index-idry_N9R.css +1 -0
- package/dist/public/index.html +30 -0
- package/dist/server.d.ts +12 -0
- package/dist/server.d.ts.map +1 -0
- package/dist/server.js +348 -0
- package/dist/server.js.map +1 -0
- package/dist/server.test.d.ts +2 -0
- package/dist/server.test.d.ts.map +1 -0
- package/dist/server.test.js +102 -0
- package/dist/server.test.js.map +1 -0
- package/package.json +53 -0
|
@@ -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>
|
package/dist/server.d.ts
ADDED
|
@@ -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":"AAAA,OAAO,EAAE,YAAY,EAAE,UAAU,EAAE,MAAM,SAAS,CAAC;AACnD,OAAO,EAAE,IAAI,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAC1C,OAAO,EAAE,aAAa,EAAE,MAAM,UAAU,CAAC;AACzC,OAAO,OAAO,MAAM,SAAS,CAAC;AAC9B,OAAO,IAAI,MAAM,eAAe,CAAC;AACjC,OAAO,SAAS,MAAM,qBAAqB,CAAC;AAC5C,OAAO,SAAS,MAAM,oBAAoB,CAAC;AAC3C,OAAO,aAAa,MAAM,iBAAiB,CAAC;AAC5C,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,KAAK,EAAE,SAAS,EAAE,MAAM,EAAE,MAAM,sBAAsB,CAAC;AAE9E,MAAM,SAAS,GAAG,OAAO,CAAC,aAAa,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC;AAE1D,SAAS,iBAAiB,CAAC,SAAiB;IAC1C,OAAO,OAAO,SAAS,KAAK,QAAQ,IAAI,SAAS,CAAC,MAAM,GAAG,CAAC,IAAI,SAAS,CAAC,MAAM,IAAI,GAAG,IAAI,kBAAkB,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;AAChI,CAAC;AAED,SAAS,gBAAgB,CAAC,QAAgB;IACxC,MAAM,GAAG,GAAG,QAAQ,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC;IACnC,OAAO,EAAE,KAAK,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,IAAI,GAAG,GAAG,CAAC,IAAI,GAAG,IAAI,MAAM,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC;AACxE,CAAC;AAED,SAAS,cAAc,CAAC,KAAU,EAAE,SAAiB,EAAE,QAAiB;IACtE,IAAI,CAAC,iBAAiB,CAAC,SAAS,CAAC,EAAE,CAAC;QAClC,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAChB,KAAK,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,2BAA2B,EAAE,CAAC,CAAC;QACnD,OAAO,IAAI,CAAC;IACd,CAAC;IACD,IAAI,QAAQ,KAAK,SAAS,EAAE,CAAC;QAC3B,MAAM,MAAM,GAAG,gBAAgB,CAAC,QAAQ,CAAC,CAAC;QAC1C,IAAI,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC;YAClB,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YAChB,KAAK,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,0BAA0B,EAAE,CAAC,CAAC;YAClD,OAAO,IAAI,CAAC;QACd,CAAC;QACD,OAAO,MAAM,CAAC,KAAK,CAAC;IACtB,CAAC;IACD,OAAO,CAAC,CAAC;AACX,CAAC;AAcD,MAAM,CAAC,KAAK,UAAU,YAAY,CAAC,OAAuB;IACxD,MAAM,IAAI,GAAG,OAAO,EAAE,IAAI,IAAI,IAAI,CAAC;IACnC,MAAM,QAAQ,GAAG,OAAO,EAAE,QAAQ,CAAC;IACnC,MAAM,SAAS,GAAG,QAAQ,CAAC,CAAC,CAAC,EAAE,QAAQ,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC;IAEtD,MAAM,GAAG,GAAG,OAAO,CAAC,EAAE,MAAM,EAAE,KAAK,EAAE,CAAC,CAAC;IAEvC,MAAM,GAAG,CAAC,QAAQ,CAAC,IAAI,EAAE;QACvB,MAAM,EAAE,OAAO,EAAE,UAAU,IAAI,CAAC,+BAA+B,CAAC;KACjE,CAAC,CAAC;IACH,MAAM,GAAG,CAAC,QAAQ,CAAC,SAAS,EAAE;QAC5B,GAAG,EAAE,IAAI;QACT,UAAU,EAAE,UAAU;QACtB,SAAS,EAAE,CAAC,WAAW,EAAE,KAAK,CAAC;KAChC,CAAC,CAAC;IACH,MAAM,GAAG,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC;IAE9B,MAAM,SAAS,GAAG,IAAI,CAAC,SAAS,EAAE,QAAQ,CAAC,CAAC;IAC5C,IAAI,UAAU,CAAC,SAAS,CAAC,EAAE,CAAC;QAC1B,MAAM,GAAG,CAAC,QAAQ,CAAC,aAAa,EAAE;YAChC,IAAI,EAAE,SAAS;YACf,MAAM,EAAE,GAAG;SACZ,CAAC,CAAC;IACL,CAAC;IAED,GAAG,CAAC,kBAAkB,CAAC,KAAK,EAAE,IAAI,EAAE,KAAK,EAAE,EAAE;QAC3C,MAAM,SAAS,GAAG,IAAI,CAAC,SAAS,EAAE,YAAY,CAAC,CAAC;QAChD,IAAI,UAAU,CAAC,SAAS,CAAC,EAAE,CAAC;YAC1B,KAAK,CAAC,IAAI,CAAC,0BAA0B,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC,CAAC;QAChF,CAAC;aAAM,CAAC;YACN,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,WAAW,EAAE,CAAC,CAAC;QAC/C,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,GAAG,CAAC,GAAG,CAAC,eAAe,EAAE,KAAK,EAAE,IAAI,EAAE,KAAK,EAAE,EAAE;QAC7C,MAAM,QAAQ,GAAG,KAAK,CAAC,YAAY,CAAC,SAAS,CAAC,CAAC;QAC/C,OAAO,QAAQ,CAAC;IAClB,CAAC,CAAC,CAAC;IAEH,GAAG,CAAC,GAAG,CAAC,oBAAoB,EAAE,KAAK,EAAE,IAAI,EAAE,KAAK,EAAE,EAAE;QAClD,MAAM,IAAI,GAAG,KAAK,CAAC,gBAAgB,CAAC,SAAS,CAAC,CAAC;QAC/C,OAAO,IAAI,CAAC;IACd,CAAC,CAAC,CAAC;IAEH,GAAG,CAAC,GAAG,CAAoC,mCAAmC,EAAE,KAAK,EAAE,GAAG,EAAE,KAAK,EAAE,EAAE;QACnG,MAAM,EAAE,SAAS,EAAE,GAAG,GAAG,CAAC,MAAM,CAAC;QACjC,IAAI,CAAC,iBAAiB,CAAC,SAAS,CAAC,EAAE,CAAC;YAClC,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YAChB,OAAO,EAAE,KAAK,EAAE,2BAA2B,EAAE,CAAC;QAChD,CAAC;QACD,MAAM,OAAO,GAAG,KAAK,CAAC,iBAAiB,CAAC,SAAS,EAAE,SAAS,CAAC,CAAC;QAC9D,MAAM,aAAa,GAAG,OAAO;aAC1B,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE;YACX,MAAM,MAAM,GAAG,KAAK,CAAC,cAAc,CAAC,GAAG,CAAC,CAAC;YACzC,MAAM,QAAQ,GAAG,KAAK,CAAC,cAAc,CAAC,GAAG,CAAC,OAAO,CAAC,GAAG,EAAE,GAAG,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;YACzE,IAAI,WAAW,GAAG,MAAM,CAAC,IAAI,CAAC;YAE9B,IAAI,QAAQ,KAAK,aAAa,EAAE,CAAC;gBAC/B,MAAM,SAAS,GAAG,KAAK,CAAC,gBAAgB,CAAC,YAAY,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;gBACxE,WAAW,GAAG,SAAS,CAAC,IAAI,CAAC;YAC/B,CAAC;iBAAM,IAAI,QAAQ,KAAK,kBAAkB,EAAE,CAAC;gBAC3C,MAAM,SAAS,GAAG,KAAK,CAAC,qBAAqB,CAAC,YAAY,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;gBAC7E,WAAW,GAAG,SAAS,CAAC,IAAI,CAAC;YAC/B,CAAC;iBAAM,IAAI,QAAQ,KAAK,WAAW,EAAE,CAAC;gBACpC,MAAM,SAAS,GAAG,KAAK,CAAC,eAAe,CAAC,YAAY,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;gBACvE,WAAW,GAAG,SAAS,CAAC,IAAI,CAAC;YAC/B,CAAC;YAED,OAAO;gBACL,EAAE,EAAE,GAAG,CAAC,EAAE;gBACV,SAAS,EAAE,GAAG,CAAC,SAAS;gBACxB,WAAW;gBACX,MAAM;aACP,CAAC;QACJ,CAAC,CAAC;aACD,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,QAAQ,KAAK,SAAS,IAAI,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;QAC9E,MAAM,QAAQ,GAAG,KAAK,CAAC,oBAAoB,CAAC,SAAS,EAAE,aAAa,CAAC,CAAC;QACtE,MAAM,UAAU,GAAG,aAAa,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;YAC3C,EAAE,EAAE,CAAC,CAAC,EAAE;YACR,KAAK,EAAE,CAAC,CAAC,MAAM,CAAC,KAAK;YACrB,QAAQ,EAAE,CAAC,CAAC,MAAM,CAAC,QAAQ;SAC5B,CAAC,CAAC,CAAC;QACJ,OAAO,EAAE,GAAG,QAAQ,EAAE,UAAU,EAAE,CAAC;IACrC,CAAC,CAAC,CAAC;IAEH,GAAG,CAAC,GAAG,CAAoC,mCAAmC,EAAE,KAAK,EAAE,GAAG,EAAE,KAAK,EAAE,EAAE;QACnG,MAAM,EAAE,SAAS,EAAE,GAAG,GAAG,CAAC,MAAM,CAAC;QACjC,IAAI,CAAC,iBAAiB,CAAC,SAAS,CAAC,EAAE,CAAC;YAClC,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YAChB,OAAO,EAAE,KAAK,EAAE,2BAA2B,EAAE,CAAC;QAChD,CAAC;QACD,MAAM,OAAO,GAAG,KAAK,CAAC,iBAAiB,CAAC,SAAS,EAAE,SAAS,CAAC,CAAC;QAC9D,MAAM,aAAa,GAAG,OAAO;aAC1B,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;YACb,EAAE,EAAE,GAAG,CAAC,EAAE;YACV,MAAM,EAAE,GAAG;YACX,MAAM,EAAE,KAAK,CAAC,cAAc,CAAC,GAAG,CAAC;SAClC,CAAC,CAAC;aACF,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,QAAQ,KAAK,SAAS,IAAI,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;QAE9E,MAAM,QAAQ,GAAG,KAAK,CAAC,YAAY,CAAC,SAAS,CAAC,CAAC;QAC/C,MAAM,WAAW,GAAG,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,SAAS,CAAC,CAAC;QAE7D,MAAM,IAAI,GAAG,KAAK,CAAC,gBAAgB,CAAC,SAAS,CAAC,CAAC;QAC/C,MAAM,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,SAAS,CAAC,CAAC;QAElD,MAAM,QAAQ,GAAG,KAAK,CAAC,oBAAoB,CACzC,SAAS,EACT,aAAa,EACb,WAAW,EAAE,UAAU,CACxB,CAAC;QAEF,IAAI,WAAW,EAAE,CAAC;YAChB,QAAQ,CAAC,SAAS,GAAG,WAAW,CAAC,SAAS,CAAC;YAC3C,QAAQ,CAAC,SAAS,GAAG,WAAW,CAAC,SAAS,CAAC;YAC3C,QAAQ,CAAC,WAAW,GAAG,IAAI,EAAE,QAAQ,EAAE,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,EAAE,CAAC;YAC5D,QAAQ,CAAC,aAAa,GAAG,WAAW,CAAC,QAAQ,IAAI,IAAI,CAAC;QACxD,CAAC;QAED,OAAO,QAAQ,CAAC;IAClB,CAAC,CAAC,CAAC;IAEH,GAAG,CAAC,GAAG,CACL,mDAAmD,EACnD,KAAK,EAAE,GAAG,EAAE,KAAK,EAAE,EAAE;QACnB,MAAM,EAAE,SAAS,EAAE,QAAQ,EAAE,GAAG,GAAG,CAAC,MAAM,CAAC;QAC3C,MAAM,GAAG,GAAG,cAAc,CAAC,KAAK,EAAE,SAAS,EAAE,QAAQ,CAAC,CAAC;QACvD,IAAI,GAAG,KAAK,IAAI;YAAE,OAAO;QACzB,MAAM,GAAG,GAAG,KAAK,CAAC,SAAS,CAAC,SAAS,EAAE,GAAG,EAAE,SAAS,CAAC,CAAC;QACvD,IAAI,CAAC,GAAG,EAAE,CAAC;YACT,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YAChB,OAAO,EAAE,KAAK,EAAE,kBAAkB,EAAE,CAAC;QACvC,CAAC;QACD,MAAM,MAAM,GAAG,KAAK,CAAC,cAAc,CAAC,GAAG,CAAC,CAAC;QACzC,OAAO,MAAM,CAAC;IAChB,CAAC,CACF,CAAC;IAEF,GAAG,CAAC,GAAG,CACL,kDAAkD,EAClD,KAAK,EAAE,GAAG,EAAE,KAAK,EAAE,EAAE;QACnB,MAAM,EAAE,SAAS,EAAE,QAAQ,EAAE,GAAG,GAAG,CAAC,MAAM,CAAC;QAC3C,MAAM,GAAG,GAAG,cAAc,CAAC,KAAK,EAAE,SAAS,EAAE,QAAQ,CAAC,CAAC;QACvD,IAAI,GAAG,KAAK,IAAI;YAAE,OAAO;QACzB,MAAM,GAAG,GAAG,KAAK,CAAC,SAAS,CAAC,SAAS,EAAE,GAAG,EAAE,SAAS,CAAC,CAAC;QACvD,IAAI,CAAC,GAAG,EAAE,CAAC;YACT,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YAChB,OAAO,EAAE,KAAK,EAAE,kBAAkB,EAAE,CAAC;QACvC,CAAC;QACD,MAAM,KAAK,GAAG,KAAK,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC;QACtC,OAAO,KAAK,CAAC;IACf,CAAC,CACF,CAAC;IAEF,GAAG,CAAC,GAAG,CACL,oDAAoD,EACpD,KAAK,EAAE,GAAG,EAAE,KAAK,EAAE,EAAE;QACnB,MAAM,EAAE,SAAS,EAAE,QAAQ,EAAE,GAAG,GAAG,CAAC,MAAM,CAAC;QAC3C,MAAM,GAAG,GAAG,cAAc,CAAC,KAAK,EAAE,SAAS,EAAE,QAAQ,CAAC,CAAC;QACvD,IAAI,GAAG,KAAK,IAAI;YAAE,OAAO;QACzB,MAAM,GAAG,GAAG,KAAK,CAAC,SAAS,CAAC,SAAS,EAAE,GAAG,EAAE,SAAS,CAAC,CAAC;QACvD,IAAI,CAAC,GAAG,EAAE,CAAC;YACT,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YAChB,OAAO,EAAE,KAAK,EAAE,kBAAkB,EAAE,CAAC;QACvC,CAAC;QACD,MAAM,OAAO,GAAG,KAAK,CAAC,cAAc,CAAC,GAAG,CAAC,CAAC;QAC1C,OAAO,OAAO,IAAI,EAAE,KAAK,EAAE,2BAA2B,EAAE,CAAC;IAC3D,CAAC,CACF,CAAC;IAEF,GAAG,CAAC,GAAG,CACL,gDAAgD,EAChD,KAAK,EAAE,GAAG,EAAE,KAAK,EAAE,EAAE;QACnB,MAAM,EAAE,SAAS,EAAE,QAAQ,EAAE,GAAG,GAAG,CAAC,MAAM,CAAC;QAC3C,MAAM,GAAG,GAAG,cAAc,CAAC,KAAK,EAAE,SAAS,EAAE,QAAQ,CAAC,CAAC;QACvD,IAAI,GAAG,KAAK,IAAI;YAAE,OAAO;QACzB,MAAM,OAAO,GAAG,KAAK,CAAC,YAAY,CAAC,SAAS,EAAE,GAAG,EAAE,SAAS,CAAC,CAAC;QAC9D,IAAI,CAAC,OAAO,EAAE,CAAC;YACb,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YAChB,OAAO,EAAE,KAAK,EAAE,mBAAmB,EAAE,CAAC;QACxC,CAAC;QACD,MAAM,GAAG,GAAG,KAAK,CAAC,SAAS,CAAC,SAAS,EAAE,GAAG,EAAE,SAAS,CAAC,CAAC;QACvD,MAAM,QAAQ,GAAG,GAAG,EAAE,OAAO;YAC3B,CAAC,CAAC,KAAK,CAAC,cAAc,CAAC,GAAG,CAAC,OAAO,CAAC,GAAG,EAAE,GAAG,CAAC,OAAO,CAAC,IAAI,CAAC;YACzD,CAAC,CAAC,IAAI,CAAC;QACT,IAAI,QAAQ,CAAC;QACb,IAAI,QAAQ,KAAK,WAAW,EAAE,CAAC;YAC7B,QAAQ,GAAG,SAAS,CAAC,sBAAsB,CAAC,OAAO,CAAC,CAAC;QACvD,CAAC;aAAM,IAAI,QAAQ,KAAK,kBAAkB,EAAE,CAAC;YAC3C,QAAQ,GAAG,SAAS,CAAC,4BAA4B,CAAC,OAAO,CAAC,CAAC;QAC7D,CAAC;aAAM,CAAC;YACN,QAAQ,GAAG,SAAS,CAAC,uBAAuB,CAAC,OAAO,CAAC,CAAC;QACxD,CAAC;QACD,OAAO,EAAE,GAAG,EAAE,OAAO,EAAE,QAAQ,EAAE,CAAC;IACpC,CAAC,CACF,CAAC;IAEF,GAAG,CAAC,GAAG,CACL,4CAA4C,EAC5C,KAAK,EAAE,GAAG,EAAE,KAAK,EAAE,EAAE;QACnB,MAAM,EAAE,SAAS,EAAE,QAAQ,EAAE,GAAG,GAAG,CAAC,MAAM,CAAC;QAC3C,MAAM,GAAG,GAAG,cAAc,CAAC,KAAK,EAAE,SAAS,EAAE,QAAQ,CAAC,CAAC;QACvD,IAAI,GAAG,KAAK,IAAI;YAAE,OAAO;QACzB,MAAM,GAAG,GAAG,KAAK,CAAC,SAAS,CAAC,SAAS,EAAE,GAAG,EAAE,SAAS,CAAC,CAAC;QACvD,IAAI,CAAC,GAAG,EAAE,CAAC;YACT,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YAChB,OAAO,EAAE,KAAK,EAAE,kBAAkB,EAAE,CAAC;QACvC,CAAC;QACD,OAAO,GAAG,CAAC;IACb,CAAC,CACF,CAAC;IAEF,GAAG,CAAC,GAAG,CAAoC,0BAA0B,EAAE,KAAK,EAAE,GAAG,EAAE,KAAK,EAAE,EAAE;QAC1F,MAAM,EAAE,SAAS,EAAE,GAAG,GAAG,CAAC,MAAM,CAAC;QACjC,IAAI,CAAC,iBAAiB,CAAC,SAAS,CAAC,EAAE,CAAC;YAClC,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YAChB,OAAO,EAAE,KAAK,EAAE,2BAA2B,EAAE,CAAC;QAChD,CAAC;QACD,MAAM,QAAQ,GAAG,KAAK,CAAC,YAAY,CAAC,SAAS,CAAC,CAAC;QAC/C,MAAM,WAAW,GAAG,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,SAAS,CAAC,CAAC;QAC7D,MAAM,OAAO,GAAG,KAAK,CAAC,iBAAiB,CAAC,SAAS,EAAE,SAAS,CAAC,CAAC;QAC9D,MAAM,QAAQ,GAAG,OAAO,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;YACrC,GAAG,GAAG;YACN,QAAQ,EAAE,GAAG,EAAE,OAAO;gBACpB,CAAC,CAAC,KAAK,CAAC,cAAc,CAAC,GAAG,CAAC,OAAO,CAAC,GAAG,EAAE,GAAG,CAAC,OAAO,CAAC,IAAI,CAAC;gBACzD,CAAC,CAAC,IAAI;SACT,CAAC,CAAC,CAAC;QACJ,OAAO;YACL,OAAO,EACL,WAAW;gBACV;oBACC,EAAE,EAAE,SAAS;oBACb,YAAY,EAAE,OAAO,CAAC,MAAM;oBAC5B,SAAS,EAAE,IAAI;oBACf,SAAS,EAAE,IAAI;iBACM;YACzB,OAAO,EAAE,QAAQ;SAClB,CAAC;IACJ,CAAC,CAAC,CAAC;IAEH,GAAG,CAAC,GAAG,CAAC,mBAAmB,EAAE,KAAK,EAAE,IAAI,EAAE,KAAK,EAAE,EAAE;QACjD,MAAM,OAAO,GAAG,MAAM,CAAC,qBAAqB,CAAC,QAAQ,CAAC,CAAC;QACvD,OAAO,EAAE,aAAa,EAAE,OAAO,EAAE,CAAC;IACpC,CAAC,CAAC,CAAC;IAEH,GAAG,CAAC,GAAG,CAAC,mBAAmB,EAAE,KAAK,EAAE,IAAI,EAAE,KAAK,EAAE,EAAE;QACjD,MAAM,CAAC,qBAAqB,CAAC,IAAI,EAAE,QAAQ,CAAC,CAAC;QAC7C,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,aAAa,EAAE,IAAI,EAAE,CAAC;IAChD,CAAC,CAAC,CAAC;IAEH,GAAG,CAAC,GAAG,CAAC,oBAAoB,EAAE,KAAK,EAAE,IAAI,EAAE,KAAK,EAAE,EAAE;QAClD,MAAM,CAAC,qBAAqB,CAAC,KAAK,EAAE,QAAQ,CAAC,CAAC;QAC9C,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,aAAa,EAAE,KAAK,EAAE,CAAC;IACjD,CAAC,CAAC,CAAC;IAEH,GAAG,CAAC,GAAG,CAAC,gBAAgB,EAAE,KAAK,EAAE,IAAI,EAAE,KAAK,EAAE,EAAE;QAC9C,OAAO,EAAE,QAAQ,EAAE,KAAK,CAAC,WAAW,CAAC,SAAS,CAAC,EAAE,CAAC;IACpD,CAAC,CAAC,CAAC;IAEH,GAAG,CAAC,IAAI,CAAoC,iCAAiC,EAAE,KAAK,EAAE,GAAG,EAAE,KAAK,EAAE,EAAE;QAClG,IAAI,CAAC;YACH,MAAM,EAAE,SAAS,EAAE,GAAG,GAAG,CAAC,MAAM,CAAC;YACjC,IAAI,CAAC,iBAAiB,CAAC,SAAS,CAAC,EAAE,CAAC;gBAClC,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;gBAChB,OAAO,EAAE,KAAK,EAAE,2BAA2B,EAAE,CAAC;YAChD,CAAC;YACD,MAAM,MAAM,GAAG,MAAM,KAAK,CAAC,gBAAgB,CAAC,SAAS,EAAE,SAAS,CAAC,CAAC;YAElE,KAAK;iBACF,IAAI,CAAC,iBAAiB,CAAC;iBACvB,MAAM,CAAC,qBAAqB,EAAE,iCAAiC,SAAS,OAAO,CAAC;iBAChF,IAAI,CAAC,MAAM,CAAC,CAAC;QAClB,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,MAAM,GAAG,GAAG,CAAU,CAAC;YACvB,IAAI,GAAG,CAAC,OAAO,KAAK,mBAAmB,EAAE,CAAC;gBACxC,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;gBAChB,OAAO,EAAE,KAAK,EAAE,mBAAmB,EAAE,CAAC;YACxC,CAAC;YACD,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YAChB,OAAO,EAAE,KAAK,EAAE,iBAAiB,GAAG,GAAG,CAAC,OAAO,EAAE,CAAC;QACpD,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,GAAG,CAAC,IAAI,CAAC,sBAAsB,EAAE,KAAK,EAAE,GAAG,EAAE,KAAK,EAAE,EAAE;QACpD,IAAI,CAAC;YACH,MAAM,IAAI,GAAG,MAAM,GAAG,CAAC,IAAI,EAAE,CAAC;YAC9B,IAAI,CAAC,IAAI,EAAE,CAAC;gBACV,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;gBAChB,OAAO,EAAE,KAAK,EAAE,2BAA2B,EAAE,CAAC;YAChD,CAAC;YAED,MAAM,UAAU,GAAG,MAAM,IAAI,CAAC,QAAQ,EAAE,CAAC;YACzC,MAAM,gBAAgB,GAAI,IAAI,CAAC,MAAM,CAAC,gBAAsC,EAAE,KAAK,IAAI,QAAQ,CAAC;YAEhG,MAAM,eAAe,GAAG,CAAC,QAAQ,EAAE,QAAQ,EAAE,MAAM,EAAE,WAAW,CAAC,CAAC;YAClE,IAAI,CAAC,eAAe,CAAC,QAAQ,CAAC,gBAAgB,CAAC,EAAE,CAAC;gBAChD,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;gBAChB,OAAO,EAAE,KAAK,EAAE,8BAA8B,gBAAgB,YAAY,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,EAAE,CAAC;YAC3G,CAAC;YAED,MAAM,MAAM,GAAG,MAAM,KAAK,CAAC,gBAAgB,CAAC,UAAU,EAAE;gBACtD,GAAG,SAAS;gBACZ,gBAAgB,EAAE,gBAA8D;aACjF,CAAC,CAAC;YAEH,OAAO,MAAM,CAAC;QAChB,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,MAAM,GAAG,GAAG,CAAU,CAAC;YACvB,IAAI,GAAG,CAAC,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAC,IAAI,GAAG,CAAC,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAC,EAAE,CAAC;gBACvE,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;gBAChB,OAAO,EAAE,KAAK,EAAE,GAAG,CAAC,OAAO,EAAE,CAAC;YAChC,CAAC;YACD,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YAChB,OAAO,EAAE,KAAK,EAAE,iBAAiB,GAAG,GAAG,CAAC,OAAO,EAAE,CAAC;QACpD,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,GAAG,CAAC,IAAI,CAAoC,iCAAiC,EAAE,KAAK,EAAE,GAAG,EAAE,KAAK,EAAE,EAAE;QAClG,IAAI,CAAC;YACH,MAAM,EAAE,SAAS,EAAE,GAAG,GAAG,CAAC,MAAM,CAAC;YACjC,IAAI,CAAC,iBAAiB,CAAC,SAAS,CAAC,EAAE,CAAC;gBAClC,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;gBAChB,OAAO,EAAE,KAAK,EAAE,2BAA2B,EAAE,CAAC;YAChD,CAAC;YACD,MAAM,KAAK,CAAC,aAAa,CAAC,SAAS,EAAE,SAAS,CAAC,CAAC;YAChD,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,SAAS,EAAE,CAAC;QACtC,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,MAAM,GAAG,GAAG,CAAU,CAAC;YACvB,IAAI,GAAG,CAAC,OAAO,KAAK,mBAAmB,EAAE,CAAC;gBACxC,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;gBAChB,OAAO,EAAE,KAAK,EAAE,mBAAmB,EAAE,CAAC;YACxC,CAAC;YACD,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YAChB,OAAO,EAAE,KAAK,EAAE,iBAAiB,GAAG,GAAG,CAAC,OAAO,EAAE,CAAC;QACpD,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,MAAM,GAAG,CAAC,MAAM,CAAC,EAAE,IAAI,EAAE,IAAI,EAAE,SAAS,EAAE,CAAC,CAAC;IAE5C,MAAM,IAAI,GAAG,oBAAoB,IAAI,EAAE,CAAC;IAExC,IAAI,OAAO,EAAE,IAAI,EAAE,CAAC;QAClB,MAAM,CAAC,oBAAoB,CAAC,CAAC,IAAI,CAAC,CAAC,EAAE,IAAI,EAAE,EAAE,EAAE;YAC7C,MAAM,GAAG,GACP,OAAO,CAAC,QAAQ,KAAK,QAAQ;gBAC3B,CAAC,CAAC,MAAM;gBACR,CAAC,CAAC,OAAO,CAAC,QAAQ,KAAK,OAAO;oBAC5B,CAAC,CAAC,OAAO;oBACT,CAAC,CAAC,UAAU,CAAC;YACnB,IAAI,CAAC,GAAG,GAAG,IAAI,IAAI,EAAE,CAAC,CAAC;QACzB,CAAC,CAAC,CAAC;IACL,CAAC;IAED,OAAO;QACL,GAAG,EAAE,IAAI;QACT,KAAK,EAAE,GAAG,EAAE,CAAC,GAAG,CAAC,KAAK,EAAE;KACzB,CAAC;AACJ,CAAC"}
|
|
@@ -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":"AAAA,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,EAAE,EAAE,UAAU,EAAE,SAAS,EAAE,MAAM,QAAQ,CAAC;AACzE,OAAO,OAAO,MAAM,SAAS,CAAC;AAC9B,OAAO,IAAI,MAAM,eAAe,CAAC;AACjC,OAAO,SAAS,MAAM,oBAAoB,CAAC;AAC3C,OAAO,EAAE,KAAK,EAAE,KAAK,EAAoB,MAAM,sBAAsB,CAAC;AAGtE,SAAS,YAAY;IACnB,MAAM,GAAG,GAAG,OAAO,CAAC,EAAE,MAAM,EAAE,KAAK,EAAE,CAAC,CAAC;IAEvC,GAAG,CAAC,QAAQ,CAAC,IAAI,EAAE,EAAE,MAAM,EAAE,GAAG,EAAE,CAAC,CAAC;IACpC,GAAG,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC;IAExB,GAAG,CAAC,GAAG,CAAC,GAAG,EAAE,KAAK,EAAE,IAAI,EAAE,KAAK,EAAE,EAAE;QACjC,KAAK,CAAC,IAAI,CAAC,0BAA0B,CAAC,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;IAC/D,CAAC,CAAC,CAAC;IAEH,GAAG,CAAC,GAAG,CAAC,eAAe,EAAE,KAAK,EAAE,IAAI,EAAE,KAAK,EAAE,EAAE;QAC7C,OAAO,KAAK,CAAC,YAAY,EAAE,CAAC;IAC9B,CAAC,CAAC,CAAC;IAEH,GAAG,CAAC,GAAG,CAAC,oBAAoB,EAAE,KAAK,EAAE,IAAI,EAAE,KAAK,EAAE,EAAE;QAClD,OAAO,KAAK,CAAC,gBAAgB,EAAE,CAAC;IAClC,CAAC,CAAC,CAAC;IAEH,GAAG,CAAC,GAAG,CAAoC,mCAAmC,EAAE,KAAK,EAAE,GAAG,EAAE,KAAK,EAAE,EAAE;QACnG,MAAM,EAAE,SAAS,EAAE,GAAG,GAAG,CAAC,MAAM,CAAC;QACjC,MAAM,OAAO,GAAG,KAAK,CAAC,iBAAiB,CAAC,SAAS,CAAC,CAAC;QACnD,MAAM,aAAa,GAAG,OAAO,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE;YACxC,MAAM,MAAM,GAAG,KAAK,CAAC,cAAc,CAAC,GAAG,CAAC,CAAC;YACzC,MAAM,QAAQ,GAAG,KAAK,CAAC,cAAc,CAAC,GAAG,CAAC,OAAO,CAAC,GAAG,EAAE,GAAG,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;YACzE,IAAI,WAAW,GAAG,MAAM,CAAC,IAAI,CAAC;YAC9B,IAAI,QAAQ,KAAK,aAAa,EAAE,CAAC;gBAC/B,WAAW,GAAG,KAAK,CAAC,gBAAgB,CAAC,YAAY,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC;YAC3E,CAAC;iBAAM,IAAI,QAAQ,KAAK,kBAAkB,EAAE,CAAC;gBAC3C,WAAW,GAAG,KAAK,CAAC,qBAAqB,CAAC,YAAY,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC;YAChF,CAAC;iBAAM,IAAI,QAAQ,KAAK,WAAW,EAAE,CAAC;gBACpC,WAAW,GAAG,KAAK,CAAC,eAAe,CAAC,YAAY,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC;YAC1E,CAAC;YACD,OAAO,EAAE,EAAE,EAAE,GAAG,CAAC,EAAE,EAAE,SAAS,EAAE,GAAG,CAAC,SAAS,EAAE,WAAW,EAAE,MAAM,EAAE,CAAC;QACvE,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,QAAQ,KAAK,SAAS,IAAI,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;QAC9E,MAAM,QAAQ,GAAG,EAAE,QAAQ,EAAE,EAAE,EAAE,UAAU,EAAE,EAAE,EAAE,CAAC;QAClD,OAAO,EAAE,GAAG,QAAQ,EAAE,UAAU,EAAE,aAAa,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC,EAAE,EAAE,KAAK,EAAE,CAAC,CAAC,MAAM,CAAC,KAAK,EAAE,QAAQ,EAAE,CAAC,CAAC,MAAM,CAAC,QAAQ,EAAE,CAAC,CAAC,EAAE,CAAC;IACnI,CAAC,CAAC,CAAC;IAEH,GAAG,CAAC,GAAG,CACL,oDAAoD,EACpD,KAAK,EAAE,GAAG,EAAE,KAAK,EAAE,EAAE;QACnB,MAAM,EAAE,SAAS,EAAE,QAAQ,EAAE,GAAG,GAAG,CAAC,MAAM,CAAC;QAC3C,MAAM,GAAG,GAAG,QAAQ,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC;QACnC,MAAM,GAAG,GAAG,KAAK,CAAC,SAAS,CAAC,SAAS,EAAE,GAAG,CAAC,CAAC;QAC5C,IAAI,CAAC,GAAG,EAAE,CAAC;YACT,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YAChB,OAAO,EAAE,KAAK,EAAE,kBAAkB,EAAE,CAAC;QACvC,CAAC;QACD,MAAM,OAAO,GAAG,KAAK,CAAC,cAAc,CAAC,GAAG,CAAC,CAAC;QAC1C,OAAO,OAAO,IAAI,EAAE,KAAK,EAAE,2BAA2B,EAAE,CAAC;IAC3D,CAAC,CACF,CAAC;IAEF,OAAO,GAAG,CAAC;AACb,CAAC;AAED,QAAQ,CAAC,YAAY,EAAE,GAAG,EAAE;IAC1B,IAAI,MAAM,GAA0B,IAAI,CAAC;IAEzC,UAAU,CAAC,GAAG,EAAE;QACd,EAAE,CAAC,aAAa,EAAE,CAAC;IACrB,CAAC,CAAC,CAAC;IAEH,SAAS,CAAC,KAAK,IAAI,EAAE;QACnB,IAAI,MAAM,EAAE,CAAC;YACX,MAAM,MAAM,CAAC,KAAK,EAAE,CAAC;YACrB,MAAM,GAAG,IAAI,CAAC;QAChB,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,kBAAkB,EAAE,GAAG,EAAE;QAChC,EAAE,CAAC,gDAAgD,EAAE,KAAK,IAAI,EAAE;YAC9D,MAAM,UAAU,GAAgB;gBAC9B,EAAE,EAAE,CAAC;gBACL,OAAO,EAAE,EAAE;gBACX,SAAS,EAAE,0BAA0B;gBACrC,UAAU,EAAE,0BAA0B;gBACtC,OAAO,EAAE,EAAE,MAAM,EAAE,MAAM,EAAE,GAAG,EAAE,qBAAqB,EAAE,OAAO,EAAE,EAAE,EAAE,IAAI,EAAE,IAAI,EAAE;gBAChF,QAAQ,EAAE,IAAI;gBACd,KAAK,EAAE,IAAI;gBACX,aAAa,EAAE,UAAU;gBACzB,YAAY,EAAE,UAAU;gBACxB,WAAW,EAAE,UAAU;aACxB,CAAC;YAEF,EAAE,CAAC,KAAK,CAAC,KAAK,EAAE,WAAW,CAAC,CAAC,eAAe,CAAC,UAAU,CAAC,CAAC;YAEzD,MAAM,GAAG,GAAG,YAAY,EAAE,CAAC;YAC3B,MAAM,GAAG,CAAC,KAAK,EAAE,CAAC;YAElB,MAAM,QAAQ,GAAG,MAAM,GAAG,CAAC,MAAM,CAAC;gBAChC,MAAM,EAAE,KAAK;gBACb,GAAG,EAAE,sCAAsC;aAC5C,CAAC,CAAC;YAEH,MAAM,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YACtC,MAAM,IAAI,GAAG,QAAQ,CAAC,IAAI,EAAE,CAAC;YAC7B,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,WAAW,EAAE,CAAC;YAChC,MAAM,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,WAAW,EAAE,CAAC;QAC3C,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,2CAA2C,EAAE,KAAK,IAAI,EAAE;YACzD,EAAE,CAAC,KAAK,CAAC,KAAK,EAAE,WAAW,CAAC,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC;YAEnD,MAAM,GAAG,GAAG,YAAY,EAAE,CAAC;YAC3B,MAAM,GAAG,CAAC,KAAK,EAAE,CAAC;YAElB,MAAM,QAAQ,GAAG,MAAM,GAAG,CAAC,MAAM,CAAC;gBAChC,MAAM,EAAE,KAAK;gBACb,GAAG,EAAE,wCAAwC;aAC9C,CAAC,CAAC;YAEH,MAAM,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QACxC,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
|