@archznn/crewloop-skills 0.5.0 → 0.7.0
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 +4 -16
- package/package.json +3 -3
- package/packages/cli/dist/agents.js +1 -1
- package/packages/cli/dist/agents.js.map +1 -1
- package/packages/cli/dist/cli.d.ts.map +1 -1
- package/packages/cli/dist/cli.js +31 -37
- package/packages/cli/dist/cli.js.map +1 -1
- package/packages/cli/dist/hooks.d.ts +6 -4
- package/packages/cli/dist/hooks.d.ts.map +1 -1
- package/packages/cli/dist/hooks.js +258 -98
- package/packages/cli/dist/hooks.js.map +1 -1
- package/packages/cli/dist/tests/cli.test.js +21 -0
- package/packages/cli/dist/tests/cli.test.js.map +1 -1
- package/packages/cli/dist/tests/hooks.test.js +253 -27
- package/packages/cli/dist/tests/hooks.test.js.map +1 -1
- package/references/conventions.md +1 -10
- package/references/workflow.md +1 -1
- package/servers/dashboard/README.md +55 -1
- package/servers/dashboard/bin/crewloop-shim.js +4 -0
- package/servers/dashboard/dist/adapters/agy.d.ts +19 -0
- package/servers/dashboard/dist/adapters/agy.d.ts.map +1 -0
- package/servers/dashboard/dist/adapters/agy.js +108 -0
- package/servers/dashboard/dist/adapters/agy.js.map +1 -0
- package/servers/dashboard/dist/adapters/codex.d.ts.map +1 -1
- package/servers/dashboard/dist/adapters/codex.js +2 -0
- package/servers/dashboard/dist/adapters/codex.js.map +1 -1
- package/servers/dashboard/dist/adapters/kimi.d.ts +1 -1
- package/servers/dashboard/dist/adapters/kimi.d.ts.map +1 -1
- package/servers/dashboard/dist/adapters/kimi.js +9 -0
- package/servers/dashboard/dist/adapters/kimi.js.map +1 -1
- package/servers/dashboard/dist/adapters/shim.d.ts +1 -1
- package/servers/dashboard/dist/adapters/shim.d.ts.map +1 -1
- package/servers/dashboard/dist/adapters/shim.js +32 -11
- package/servers/dashboard/dist/adapters/shim.js.map +1 -1
- package/servers/dashboard/dist/adapters/shim.test.js +46 -4
- package/servers/dashboard/dist/adapters/shim.test.js.map +1 -1
- package/servers/dashboard/dist/lib/constants.d.ts +5 -0
- package/servers/dashboard/dist/lib/constants.d.ts.map +1 -0
- package/servers/dashboard/dist/lib/constants.js +46 -0
- package/servers/dashboard/dist/lib/constants.js.map +1 -0
- package/servers/dashboard/dist/lib/format.d.ts +6 -0
- package/servers/dashboard/dist/lib/format.d.ts.map +1 -0
- package/servers/dashboard/dist/lib/format.js +52 -0
- package/servers/dashboard/dist/lib/format.js.map +1 -0
- package/servers/dashboard/dist/lib/graph.d.ts +22 -0
- package/servers/dashboard/dist/lib/graph.d.ts.map +1 -0
- package/servers/dashboard/dist/lib/graph.js +45 -0
- package/servers/dashboard/dist/lib/graph.js.map +1 -0
- package/servers/dashboard/dist/lib/invocations.d.ts +32 -0
- package/servers/dashboard/dist/lib/invocations.d.ts.map +1 -0
- package/servers/dashboard/dist/lib/invocations.js +135 -0
- package/servers/dashboard/dist/lib/invocations.js.map +1 -0
- package/servers/dashboard/dist/lib/invocations.test.d.ts +2 -0
- package/servers/dashboard/dist/lib/invocations.test.d.ts.map +1 -0
- package/servers/dashboard/dist/lib/invocations.test.js +68 -0
- package/servers/dashboard/dist/lib/invocations.test.js.map +1 -0
- package/servers/dashboard/dist/lib/paths.d.ts +2 -0
- package/servers/dashboard/dist/lib/paths.d.ts.map +1 -0
- package/servers/dashboard/dist/lib/paths.js +40 -0
- package/servers/dashboard/dist/lib/paths.js.map +1 -0
- package/servers/dashboard/dist/presenter.d.ts.map +1 -1
- package/servers/dashboard/dist/presenter.js +2 -0
- package/servers/dashboard/dist/presenter.js.map +1 -1
- package/servers/dashboard/dist/public/assets/index-DjmMKbPN.css +1 -0
- package/servers/dashboard/dist/public/assets/index-DzOqMleZ.js +5323 -0
- package/servers/dashboard/dist/public/assets/index-DzOqMleZ.js.map +1 -0
- package/servers/dashboard/dist/public/index.html +16 -0
- package/servers/dashboard/dist/server.d.ts.map +1 -1
- package/servers/dashboard/dist/server.js +5 -1
- package/servers/dashboard/dist/server.js.map +1 -1
- package/servers/dashboard/dist/skills/infer.d.ts.map +1 -1
- package/servers/dashboard/dist/skills/infer.js +0 -6
- package/servers/dashboard/dist/skills/infer.js.map +1 -1
- package/servers/dashboard/dist/skills/infer.test.js +10 -3
- package/servers/dashboard/dist/skills/infer.test.js.map +1 -1
- package/servers/dashboard/dist/skills/mapping.d.ts +0 -3
- package/servers/dashboard/dist/skills/mapping.d.ts.map +1 -1
- package/servers/dashboard/dist/skills/mapping.js +0 -18
- package/servers/dashboard/dist/skills/mapping.js.map +1 -1
- package/servers/dashboard/dist/skills/registry.d.ts.map +1 -1
- package/servers/dashboard/dist/skills/registry.js +0 -1
- package/servers/dashboard/dist/skills/registry.js.map +1 -1
- package/servers/dashboard/dist/tests/adapters.test.d.ts +2 -0
- package/servers/dashboard/dist/tests/adapters.test.d.ts.map +1 -0
- package/servers/dashboard/dist/tests/adapters.test.js +180 -0
- package/servers/dashboard/dist/tests/adapters.test.js.map +1 -0
- package/servers/dashboard/dist/tests/lib-helpers.test.d.ts +2 -0
- package/servers/dashboard/dist/tests/lib-helpers.test.d.ts.map +1 -0
- package/servers/dashboard/dist/tests/lib-helpers.test.js +123 -0
- package/servers/dashboard/dist/tests/lib-helpers.test.js.map +1 -0
- package/servers/dashboard/dist/tests/shim.test.d.ts +2 -0
- package/servers/dashboard/dist/tests/shim.test.d.ts.map +1 -0
- package/servers/dashboard/dist/tests/shim.test.js +133 -0
- package/servers/dashboard/dist/tests/shim.test.js.map +1 -0
- package/servers/dashboard/dist/types.d.ts +5 -2
- package/servers/dashboard/dist/types.d.ts.map +1 -1
- package/servers/dashboard/package.json +24 -6
- package/servers/dashboard/src/adapters/agy.ts +136 -0
- package/servers/dashboard/src/adapters/codex.ts +2 -0
- package/servers/dashboard/src/adapters/kimi.ts +11 -1
- package/servers/dashboard/src/adapters/shim.test.ts +57 -4
- package/servers/dashboard/src/adapters/shim.ts +31 -11
- package/servers/dashboard/src/lib/constants.ts +44 -0
- package/servers/dashboard/src/lib/format.ts +44 -0
- package/servers/dashboard/src/lib/graph.ts +69 -0
- package/servers/dashboard/src/lib/invocations.test.ts +70 -0
- package/servers/dashboard/src/lib/invocations.ts +172 -0
- package/servers/dashboard/src/lib/paths.ts +35 -0
- package/servers/dashboard/src/presenter.ts +2 -0
- package/servers/dashboard/src/server.ts +5 -1
- package/servers/dashboard/src/skills/infer.test.ts +11 -3
- package/servers/dashboard/src/skills/infer.ts +1 -8
- package/servers/dashboard/src/skills/mapping.ts +0 -20
- package/servers/dashboard/src/skills/registry.ts +0 -1
- package/servers/dashboard/src/tests/adapters.test.ts +198 -0
- package/servers/dashboard/src/tests/lib-helpers.test.ts +133 -0
- package/servers/dashboard/src/tests/shim.test.ts +153 -0
- package/servers/dashboard/src/types.ts +5 -3
- package/servers/dashboard/ui/index.html +15 -0
- package/servers/dashboard/ui/postcss.config.js +6 -0
- package/servers/dashboard/ui/src/App.tsx +360 -0
- package/servers/dashboard/ui/src/components/ActiveSkillPanel.tsx +69 -0
- package/servers/dashboard/ui/src/components/ActivityGraph.tsx +74 -0
- package/servers/dashboard/ui/src/components/CommandPalette.tsx +200 -0
- package/servers/dashboard/ui/src/components/FileActivity.tsx +20 -0
- package/servers/dashboard/ui/src/components/FileDiff.tsx +68 -0
- package/servers/dashboard/ui/src/components/FileList.tsx +64 -0
- package/servers/dashboard/ui/src/components/FilterBar.tsx +208 -0
- package/servers/dashboard/ui/src/components/Network3D.tsx +178 -0
- package/servers/dashboard/ui/src/components/SessionSelector.tsx +95 -0
- package/servers/dashboard/ui/src/components/Sidebar.tsx +110 -0
- package/servers/dashboard/ui/src/components/TelemetryPanel.tsx +57 -0
- package/servers/dashboard/ui/src/components/Timeline.tsx +57 -0
- package/servers/dashboard/ui/src/components/TimelineRow.tsx +112 -0
- package/servers/dashboard/ui/src/components/TopBar.tsx +116 -0
- package/servers/dashboard/ui/src/components/ViewHeader.tsx +19 -0
- package/servers/dashboard/ui/src/components/ui/Icon.tsx +105 -0
- package/servers/dashboard/ui/src/components/ui/StatusBadge.tsx +19 -0
- package/servers/dashboard/ui/src/components/views/FilesView.tsx +23 -0
- package/servers/dashboard/ui/src/components/views/NetworkView.tsx +20 -0
- package/servers/dashboard/ui/src/components/views/Overview.tsx +135 -0
- package/servers/dashboard/ui/src/components/views/SessionsView.tsx +84 -0
- package/servers/dashboard/ui/src/components/views/SettingsView.tsx +138 -0
- package/servers/dashboard/ui/src/components/views/SkillsView.tsx +92 -0
- package/servers/dashboard/ui/src/components/views/TimelineView.tsx +46 -0
- package/servers/dashboard/ui/src/contexts/FilterContext.tsx +41 -0
- package/servers/dashboard/ui/src/contexts/PinnedSessionsContext.tsx +80 -0
- package/servers/dashboard/ui/src/contexts/SettingsContext.tsx +60 -0
- package/servers/dashboard/ui/src/hooks/useCommandPalette.ts +36 -0
- package/servers/dashboard/ui/src/hooks/useKeyboardShortcut.ts +38 -0
- package/servers/dashboard/ui/src/hooks/useNow.ts +12 -0
- package/servers/dashboard/ui/src/hooks/useReducedMotion.ts +15 -0
- package/servers/dashboard/ui/src/hooks/useSessions.ts +64 -0
- package/servers/dashboard/ui/src/hooks/useTheme.ts +30 -0
- package/servers/dashboard/ui/src/hooks/useViewport.ts +19 -0
- package/servers/dashboard/ui/src/hooks/useWebSocket.ts +118 -0
- package/servers/dashboard/ui/src/lib/export.test.ts +33 -0
- package/servers/dashboard/ui/src/lib/export.ts +39 -0
- package/servers/dashboard/ui/src/lib/filter.test.ts +95 -0
- package/servers/dashboard/ui/src/lib/filter.ts +178 -0
- package/servers/dashboard/ui/src/lib/format.test.ts +25 -0
- package/servers/dashboard/ui/src/lib/search.test.ts +52 -0
- package/servers/dashboard/ui/src/lib/search.ts +60 -0
- package/servers/dashboard/ui/src/lib/settings.test.ts +50 -0
- package/servers/dashboard/ui/src/lib/settings.ts +56 -0
- package/servers/dashboard/ui/src/lib/types.ts +124 -0
- package/servers/dashboard/ui/src/main.tsx +19 -0
- package/servers/dashboard/ui/src/styles/index.css +155 -0
- package/servers/dashboard/ui/tailwind.config.js +45 -0
- package/servers/dashboard/ui/tsconfig.json +33 -0
- package/servers/dashboard/ui/tsconfig.node.json +10 -0
- package/servers/dashboard/ui/vite.config.ts +37 -0
- package/servers/dashboard/ui/vitest.config.ts +8 -0
- package/skills/accessibility-auditor/SKILL.md +0 -20
- package/skills/architect/SKILL.md +0 -45
- package/skills/designer/SKILL.md +0 -30
- package/skills/docs-writer/SKILL.md +0 -13
- package/skills/engineer/SKILL.md +0 -30
- package/skills/maintainer/SKILL.md +0 -20
- package/skills/orchestrator/SKILL.md +0 -13
- package/skills/product-manager/SKILL.md +0 -20
- package/skills/researcher/SKILL.md +0 -20
- package/skills/reviewer/SKILL.md +0 -30
- package/skills/security-guard/SKILL.md +0 -20
- package/skills/shipper/SKILL.md +0 -33
- package/skills/tester/SKILL.md +0 -20
- package/packages/cli/dist/mcp.d.ts +0 -28
- package/packages/cli/dist/mcp.d.ts.map +0 -1
- package/packages/cli/dist/mcp.js +0 -148
- package/packages/cli/dist/mcp.js.map +0 -1
- package/packages/cli/dist/tests/mcp.test.d.ts +0 -2
- package/packages/cli/dist/tests/mcp.test.d.ts.map +0 -1
- package/packages/cli/dist/tests/mcp.test.js +0 -232
- package/packages/cli/dist/tests/mcp.test.js.map +0 -1
- package/references/obsidian-mcp-usage.md +0 -190
- package/servers/dashboard/public/app.js +0 -516
- package/servers/dashboard/public/index.html +0 -96
- package/servers/dashboard/public/styles.css +0 -819
- package/servers/obsidian-mcp/README.md +0 -82
- package/servers/obsidian-mcp/pyproject.toml +0 -32
- package/servers/obsidian-mcp/src/obsidian_mcp/__init__.py +0 -0
- package/servers/obsidian-mcp/src/obsidian_mcp/config.py +0 -47
- package/servers/obsidian-mcp/src/obsidian_mcp/indexer/__init__.py +0 -0
- package/servers/obsidian-mcp/src/obsidian_mcp/indexer/embeddings.py +0 -105
- package/servers/obsidian-mcp/src/obsidian_mcp/indexer/indexer.py +0 -79
- package/servers/obsidian-mcp/src/obsidian_mcp/indexer/store.py +0 -141
- package/servers/obsidian-mcp/src/obsidian_mcp/indexer/sync.py +0 -37
- package/servers/obsidian-mcp/src/obsidian_mcp/learning/__init__.py +0 -0
- package/servers/obsidian-mcp/src/obsidian_mcp/learning/detector.py +0 -66
- package/servers/obsidian-mcp/src/obsidian_mcp/learning/note_generator.py +0 -40
- package/servers/obsidian-mcp/src/obsidian_mcp/main.py +0 -4
- package/servers/obsidian-mcp/src/obsidian_mcp/models.py +0 -42
- package/servers/obsidian-mcp/src/obsidian_mcp/privacy/__init__.py +0 -0
- package/servers/obsidian-mcp/src/obsidian_mcp/privacy/filter.py +0 -68
- package/servers/obsidian-mcp/src/obsidian_mcp/rag/__init__.py +0 -0
- package/servers/obsidian-mcp/src/obsidian_mcp/rag/engine.py +0 -50
- package/servers/obsidian-mcp/src/obsidian_mcp/rag/graph_search.py +0 -55
- package/servers/obsidian-mcp/src/obsidian_mcp/rag/text_search.py +0 -37
- package/servers/obsidian-mcp/src/obsidian_mcp/rag/vector_search.py +0 -118
- package/servers/obsidian-mcp/src/obsidian_mcp/server.py +0 -61
- package/servers/obsidian-mcp/src/obsidian_mcp/tools/__init__.py +0 -0
- package/servers/obsidian-mcp/src/obsidian_mcp/tools/create.py +0 -43
- package/servers/obsidian-mcp/src/obsidian_mcp/tools/delete.py +0 -16
- package/servers/obsidian-mcp/src/obsidian_mcp/tools/learn.py +0 -42
- package/servers/obsidian-mcp/src/obsidian_mcp/tools/list.py +0 -16
- package/servers/obsidian-mcp/src/obsidian_mcp/tools/read.py +0 -15
- package/servers/obsidian-mcp/src/obsidian_mcp/tools/registry.py +0 -130
- package/servers/obsidian-mcp/src/obsidian_mcp/tools/related.py +0 -20
- package/servers/obsidian-mcp/src/obsidian_mcp/tools/search.py +0 -26
- package/servers/obsidian-mcp/src/obsidian_mcp/tools/sync.py +0 -22
- package/servers/obsidian-mcp/src/obsidian_mcp/tools/update.py +0 -34
- package/servers/obsidian-mcp/src/obsidian_mcp/vault/__init__.py +0 -0
- package/servers/obsidian-mcp/src/obsidian_mcp/vault/parser.py +0 -82
- package/servers/obsidian-mcp/src/obsidian_mcp/vault/repository.py +0 -68
- package/servers/obsidian-mcp/src/obsidian_mcp/vault/writer.py +0 -61
- package/servers/obsidian-mcp/tests/conftest.py +0 -39
- package/servers/obsidian-mcp/tests/test_async_tools.py +0 -87
- package/servers/obsidian-mcp/tests/test_edge_cases.py +0 -59
- package/servers/obsidian-mcp/tests/test_indexer.py +0 -27
- package/servers/obsidian-mcp/tests/test_integration.py +0 -90
- package/servers/obsidian-mcp/tests/test_learning.py +0 -34
- package/servers/obsidian-mcp/tests/test_privacy.py +0 -31
- package/servers/obsidian-mcp/tests/test_privacy_config.py +0 -44
- package/servers/obsidian-mcp/tests/test_rag.py +0 -64
- package/servers/obsidian-mcp/tests/test_read_raw.py +0 -37
- package/servers/obsidian-mcp/tests/test_tfidf_fallback.py +0 -54
- package/servers/obsidian-mcp/tests/test_tools.py +0 -108
- package/servers/obsidian-mcp/tests/test_vault.py +0 -103
- package/servers/obsidian-mcp/tests/test_writer.py +0 -139
- package/skills/obsidian-second-brain/SKILL.md +0 -298
|
@@ -0,0 +1,60 @@
|
|
|
1
|
+
import { createContext, useCallback, useContext, useEffect, useMemo, useState, type ReactNode } from 'react';
|
|
2
|
+
import type { DashboardSettings, Theme } from '../lib/types';
|
|
3
|
+
import { loadSettings, saveSettings } from '../lib/settings';
|
|
4
|
+
import { useReducedMotion } from '../hooks/useReducedMotion';
|
|
5
|
+
|
|
6
|
+
interface SettingsContextValue {
|
|
7
|
+
settings: DashboardSettings;
|
|
8
|
+
resolvedTheme: 'dark' | 'light';
|
|
9
|
+
reducedMotion: boolean;
|
|
10
|
+
setSettings: (updater: DashboardSettings | ((prev: DashboardSettings) => DashboardSettings)) => void;
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
const SettingsContext = createContext<SettingsContextValue | null>(null);
|
|
14
|
+
|
|
15
|
+
function resolveTheme(theme: Theme): 'dark' | 'light' {
|
|
16
|
+
if (theme !== 'system') return theme;
|
|
17
|
+
return window.matchMedia('(prefers-color-scheme: light)').matches ? 'light' : 'dark';
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
export function SettingsProvider({ children }: { children: ReactNode }) {
|
|
21
|
+
const [settings, setSettingsState] = useState<DashboardSettings>(() => loadSettings());
|
|
22
|
+
const systemReducedMotion = useReducedMotion();
|
|
23
|
+
const resolvedTheme = useMemo(() => resolveTheme(settings.theme), [settings.theme]);
|
|
24
|
+
const reducedMotion = settings.reducedMotion || systemReducedMotion;
|
|
25
|
+
|
|
26
|
+
useEffect(() => {
|
|
27
|
+
const root = document.documentElement;
|
|
28
|
+
root.classList.remove('dark', 'light');
|
|
29
|
+
root.classList.add(resolvedTheme);
|
|
30
|
+
}, [resolvedTheme]);
|
|
31
|
+
|
|
32
|
+
useEffect(() => {
|
|
33
|
+
const root = document.documentElement;
|
|
34
|
+
root.classList.remove('density-compact', 'density-comfortable');
|
|
35
|
+
root.classList.add(`density-${settings.density}`);
|
|
36
|
+
}, [settings.density]);
|
|
37
|
+
|
|
38
|
+
const setSettings = useCallback(
|
|
39
|
+
(updater: DashboardSettings | ((prev: DashboardSettings) => DashboardSettings)) => {
|
|
40
|
+
setSettingsState((prev) => {
|
|
41
|
+
const next = typeof updater === 'function' ? (updater as (p: DashboardSettings) => DashboardSettings)(prev) : updater;
|
|
42
|
+
saveSettings(next);
|
|
43
|
+
return next;
|
|
44
|
+
});
|
|
45
|
+
},
|
|
46
|
+
[]
|
|
47
|
+
);
|
|
48
|
+
|
|
49
|
+
return (
|
|
50
|
+
<SettingsContext.Provider value={{ settings, resolvedTheme, reducedMotion, setSettings }}>
|
|
51
|
+
{children}
|
|
52
|
+
</SettingsContext.Provider>
|
|
53
|
+
);
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
export function useSettings(): SettingsContextValue {
|
|
57
|
+
const ctx = useContext(SettingsContext);
|
|
58
|
+
if (!ctx) throw new Error('useSettings must be used within SettingsProvider');
|
|
59
|
+
return ctx;
|
|
60
|
+
}
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
import { useCallback, useEffect, useState } from 'react';
|
|
2
|
+
|
|
3
|
+
export interface CommandPaletteState {
|
|
4
|
+
open: boolean;
|
|
5
|
+
query: string;
|
|
6
|
+
selectedIndex: number;
|
|
7
|
+
}
|
|
8
|
+
|
|
9
|
+
export interface CommandPaletteActions {
|
|
10
|
+
setOpen: (open: boolean) => void;
|
|
11
|
+
setQuery: (query: string) => void;
|
|
12
|
+
setSelectedIndex: (index: number | ((prev: number) => number)) => void;
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
export function useCommandPalette(): CommandPaletteState & CommandPaletteActions {
|
|
16
|
+
const [open, setOpen] = useState(false);
|
|
17
|
+
const [query, setQuery] = useState('');
|
|
18
|
+
const [selectedIndex, setSelectedIndexState] = useState(0);
|
|
19
|
+
|
|
20
|
+
const setSelectedIndex = useCallback(
|
|
21
|
+
(index: number | ((prev: number) => number)) => {
|
|
22
|
+
setSelectedIndexState((prev) => (typeof index === 'function' ? index(prev) : index));
|
|
23
|
+
},
|
|
24
|
+
[]
|
|
25
|
+
);
|
|
26
|
+
|
|
27
|
+
useEffect(() => {
|
|
28
|
+
setSelectedIndexState(0);
|
|
29
|
+
}, [query]);
|
|
30
|
+
|
|
31
|
+
useEffect(() => {
|
|
32
|
+
if (open) setQuery('');
|
|
33
|
+
}, [open]);
|
|
34
|
+
|
|
35
|
+
return { open, setOpen, query, setQuery, selectedIndex, setSelectedIndex };
|
|
36
|
+
}
|
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
import { useEffect } from 'react';
|
|
2
|
+
|
|
3
|
+
interface ShortcutOptions {
|
|
4
|
+
meta?: boolean;
|
|
5
|
+
ctrl?: boolean;
|
|
6
|
+
preventDefault?: boolean;
|
|
7
|
+
disabled?: boolean;
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
export function useKeyboardShortcut(
|
|
11
|
+
key: string,
|
|
12
|
+
callback: () => void,
|
|
13
|
+
options: ShortcutOptions = {}
|
|
14
|
+
): void {
|
|
15
|
+
const { meta = false, ctrl = false, preventDefault = true, disabled = false } = options;
|
|
16
|
+
|
|
17
|
+
useEffect(() => {
|
|
18
|
+
if (disabled) return;
|
|
19
|
+
|
|
20
|
+
function handler(e: KeyboardEvent) {
|
|
21
|
+
const target = e.target as HTMLElement | null;
|
|
22
|
+
const tag = target?.tagName ?? '';
|
|
23
|
+
if (tag === 'INPUT' || tag === 'TEXTAREA' || tag === 'SELECT' || target?.isContentEditable) {
|
|
24
|
+
return;
|
|
25
|
+
}
|
|
26
|
+
if (e.key.toLowerCase() !== key.toLowerCase()) return;
|
|
27
|
+
if (meta && !(e.metaKey || e.ctrlKey)) return;
|
|
28
|
+
if (ctrl && !e.ctrlKey) return;
|
|
29
|
+
if (!meta && !ctrl && (e.metaKey || e.ctrlKey)) return;
|
|
30
|
+
|
|
31
|
+
if (preventDefault) e.preventDefault();
|
|
32
|
+
callback();
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
document.addEventListener('keydown', handler);
|
|
36
|
+
return () => document.removeEventListener('keydown', handler);
|
|
37
|
+
}, [key, callback, meta, ctrl, preventDefault, disabled]);
|
|
38
|
+
}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import { useEffect, useState } from 'react';
|
|
2
|
+
|
|
3
|
+
export function useNow(interval = 10_000): number {
|
|
4
|
+
const [now, setNow] = useState(Date.now());
|
|
5
|
+
|
|
6
|
+
useEffect(() => {
|
|
7
|
+
const id = window.setInterval(() => setNow(Date.now()), interval);
|
|
8
|
+
return () => window.clearInterval(id);
|
|
9
|
+
}, [interval]);
|
|
10
|
+
|
|
11
|
+
return now;
|
|
12
|
+
}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
import { useEffect, useState } from 'react';
|
|
2
|
+
|
|
3
|
+
export function useReducedMotion(): boolean {
|
|
4
|
+
const [reduced, setReduced] = useState(false);
|
|
5
|
+
|
|
6
|
+
useEffect(() => {
|
|
7
|
+
const mq = window.matchMedia('(prefers-reduced-motion: reduce)');
|
|
8
|
+
setReduced(mq.matches);
|
|
9
|
+
const handler = (e: MediaQueryListEvent) => setReduced(e.matches);
|
|
10
|
+
mq.addEventListener('change', handler);
|
|
11
|
+
return () => mq.removeEventListener('change', handler);
|
|
12
|
+
}, []);
|
|
13
|
+
|
|
14
|
+
return reduced;
|
|
15
|
+
}
|
|
@@ -0,0 +1,64 @@
|
|
|
1
|
+
import { useCallback, useState } from 'react';
|
|
2
|
+
import type { ClientSession, ClientWebSocketMessage } from '../../../src/types';
|
|
3
|
+
|
|
4
|
+
export interface SessionsState {
|
|
5
|
+
sessions: Map<string, ClientSession>;
|
|
6
|
+
selectedSessionId: string | null;
|
|
7
|
+
}
|
|
8
|
+
|
|
9
|
+
function defaultSelected(
|
|
10
|
+
sessions: Map<string, ClientSession>,
|
|
11
|
+
activeSessionId: string | undefined,
|
|
12
|
+
current: string | null
|
|
13
|
+
): string | null {
|
|
14
|
+
if (current && sessions.has(current)) return current;
|
|
15
|
+
if (activeSessionId && sessions.has(activeSessionId)) return activeSessionId;
|
|
16
|
+
const first = sessions.keys().next().value;
|
|
17
|
+
return first || null;
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
export function useSessions() {
|
|
21
|
+
const [state, setState] = useState<SessionsState>({
|
|
22
|
+
sessions: new Map(),
|
|
23
|
+
selectedSessionId: null,
|
|
24
|
+
});
|
|
25
|
+
|
|
26
|
+
const selectSession = useCallback((id: string | null) => {
|
|
27
|
+
setState((prev) => ({ ...prev, selectedSessionId: id }));
|
|
28
|
+
}, []);
|
|
29
|
+
|
|
30
|
+
const handleMessage = useCallback((msg: ClientWebSocketMessage) => {
|
|
31
|
+
setState((prev) => {
|
|
32
|
+
const sessions = new Map(prev.sessions);
|
|
33
|
+
let activeSessionId: string | undefined;
|
|
34
|
+
|
|
35
|
+
if (msg.type === 'snapshot') {
|
|
36
|
+
sessions.clear();
|
|
37
|
+
for (const s of msg.sessions) {
|
|
38
|
+
sessions.set(s.id, s);
|
|
39
|
+
}
|
|
40
|
+
} else if (msg.type === 'update') {
|
|
41
|
+
const s = msg.session;
|
|
42
|
+
sessions.set(s.id, s);
|
|
43
|
+
if (msg.isActive) activeSessionId = s.id;
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
return {
|
|
47
|
+
sessions,
|
|
48
|
+
selectedSessionId: defaultSelected(sessions, activeSessionId, prev.selectedSessionId),
|
|
49
|
+
};
|
|
50
|
+
});
|
|
51
|
+
}, []);
|
|
52
|
+
|
|
53
|
+
const sortedSessions = Array.from(state.sessions.values()).sort(
|
|
54
|
+
(a, b) => (b.lastActivity || 0) - (a.lastActivity || 0)
|
|
55
|
+
);
|
|
56
|
+
|
|
57
|
+
return {
|
|
58
|
+
sessions: state.sessions,
|
|
59
|
+
selectedSessionId: state.selectedSessionId,
|
|
60
|
+
selectSession,
|
|
61
|
+
handleMessage,
|
|
62
|
+
sortedSessions,
|
|
63
|
+
};
|
|
64
|
+
}
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
import { useEffect, useState } from 'react';
|
|
2
|
+
|
|
3
|
+
type Theme = 'dark' | 'light' | 'system';
|
|
4
|
+
|
|
5
|
+
function resolveTheme(theme: Theme): 'dark' | 'light' {
|
|
6
|
+
if (theme !== 'system') return theme;
|
|
7
|
+
if (window.matchMedia('(prefers-color-scheme: light)').matches) return 'light';
|
|
8
|
+
return 'dark';
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
export function useTheme() {
|
|
12
|
+
const [theme, setThemeState] = useState<Theme>(() => {
|
|
13
|
+
const stored = localStorage.getItem('crewloop-theme') as Theme | null;
|
|
14
|
+
return stored || 'system';
|
|
15
|
+
});
|
|
16
|
+
const resolved = resolveTheme(theme);
|
|
17
|
+
|
|
18
|
+
useEffect(() => {
|
|
19
|
+
const root = document.documentElement;
|
|
20
|
+
root.classList.remove('dark', 'light');
|
|
21
|
+
root.classList.add(resolved);
|
|
22
|
+
}, [resolved]);
|
|
23
|
+
|
|
24
|
+
const setTheme = (value: Theme) => {
|
|
25
|
+
localStorage.setItem('crewloop-theme', value);
|
|
26
|
+
setThemeState(value);
|
|
27
|
+
};
|
|
28
|
+
|
|
29
|
+
return { theme, resolved, setTheme };
|
|
30
|
+
}
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import { useEffect, useState } from 'react';
|
|
2
|
+
|
|
3
|
+
export type Breakpoint = 'mobile' | 'tablet' | 'desktop';
|
|
4
|
+
|
|
5
|
+
export function useViewport(): { width: number; breakpoint: Breakpoint } {
|
|
6
|
+
const [width, setWidth] = useState(window.innerWidth);
|
|
7
|
+
|
|
8
|
+
useEffect(() => {
|
|
9
|
+
const handler = () => setWidth(window.innerWidth);
|
|
10
|
+
window.addEventListener('resize', handler);
|
|
11
|
+
return () => window.removeEventListener('resize', handler);
|
|
12
|
+
}, []);
|
|
13
|
+
|
|
14
|
+
let breakpoint: Breakpoint = 'desktop';
|
|
15
|
+
if (width < 768) breakpoint = 'mobile';
|
|
16
|
+
else if (width < 1024) breakpoint = 'tablet';
|
|
17
|
+
|
|
18
|
+
return { width, breakpoint };
|
|
19
|
+
}
|
|
@@ -0,0 +1,118 @@
|
|
|
1
|
+
import { useEffect, useRef, useState } from 'react';
|
|
2
|
+
import type { ClientWebSocketMessage } from '../../../src/types';
|
|
3
|
+
|
|
4
|
+
type ConnectionStatus = 'connecting' | 'connected' | 'disconnected';
|
|
5
|
+
|
|
6
|
+
export function useWebSocket(
|
|
7
|
+
url: string,
|
|
8
|
+
onMessage: (msg: ClientWebSocketMessage) => void
|
|
9
|
+
): { status: ConnectionStatus; send: (data: unknown) => void } {
|
|
10
|
+
const [status, setStatus] = useState<ConnectionStatus>('connecting');
|
|
11
|
+
const wsRef = useRef<WebSocket | null>(null);
|
|
12
|
+
const reconnectTimerRef = useRef<number | null>(null);
|
|
13
|
+
const pingTimerRef = useRef<number | null>(null);
|
|
14
|
+
const lastPongRef = useRef<number>(0);
|
|
15
|
+
const onMessageRef = useRef(onMessage);
|
|
16
|
+
|
|
17
|
+
useEffect(() => {
|
|
18
|
+
onMessageRef.current = onMessage;
|
|
19
|
+
}, [onMessage]);
|
|
20
|
+
|
|
21
|
+
useEffect(() => {
|
|
22
|
+
let active = true;
|
|
23
|
+
|
|
24
|
+
function scheduleReconnect() {
|
|
25
|
+
if (reconnectTimerRef.current) return;
|
|
26
|
+
reconnectTimerRef.current = window.setTimeout(() => {
|
|
27
|
+
reconnectTimerRef.current = null;
|
|
28
|
+
if (active) connect();
|
|
29
|
+
}, 3000);
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
function startPing(ws: WebSocket) {
|
|
33
|
+
stopPing();
|
|
34
|
+
lastPongRef.current = Date.now();
|
|
35
|
+
pingTimerRef.current = window.setInterval(() => {
|
|
36
|
+
if (ws.readyState === WebSocket.OPEN) {
|
|
37
|
+
ws.send(JSON.stringify({ type: 'ping' }));
|
|
38
|
+
}
|
|
39
|
+
if (Date.now() - lastPongRef.current > 35000) {
|
|
40
|
+
ws.close();
|
|
41
|
+
}
|
|
42
|
+
}, 15000);
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
function stopPing() {
|
|
46
|
+
if (pingTimerRef.current) {
|
|
47
|
+
clearInterval(pingTimerRef.current);
|
|
48
|
+
pingTimerRef.current = null;
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
function connect() {
|
|
53
|
+
setStatus('connecting');
|
|
54
|
+
try {
|
|
55
|
+
const ws = new WebSocket(url);
|
|
56
|
+
wsRef.current = ws;
|
|
57
|
+
|
|
58
|
+
ws.addEventListener('open', () => {
|
|
59
|
+
if (!active) {
|
|
60
|
+
ws.close();
|
|
61
|
+
return;
|
|
62
|
+
}
|
|
63
|
+
setStatus('connected');
|
|
64
|
+
startPing(ws);
|
|
65
|
+
});
|
|
66
|
+
|
|
67
|
+
ws.addEventListener('message', (event) => {
|
|
68
|
+
let msg: ClientWebSocketMessage;
|
|
69
|
+
try {
|
|
70
|
+
msg = JSON.parse(event.data);
|
|
71
|
+
} catch {
|
|
72
|
+
return;
|
|
73
|
+
}
|
|
74
|
+
if ((msg as { type: string }).type === 'pong') {
|
|
75
|
+
lastPongRef.current = Date.now();
|
|
76
|
+
return;
|
|
77
|
+
}
|
|
78
|
+
onMessageRef.current(msg);
|
|
79
|
+
});
|
|
80
|
+
|
|
81
|
+
ws.addEventListener('close', () => {
|
|
82
|
+
setStatus('disconnected');
|
|
83
|
+
stopPing();
|
|
84
|
+
if (active) scheduleReconnect();
|
|
85
|
+
});
|
|
86
|
+
|
|
87
|
+
ws.addEventListener('error', () => {
|
|
88
|
+
ws.close();
|
|
89
|
+
});
|
|
90
|
+
} catch {
|
|
91
|
+
setStatus('disconnected');
|
|
92
|
+
scheduleReconnect();
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
connect();
|
|
97
|
+
|
|
98
|
+
return () => {
|
|
99
|
+
active = false;
|
|
100
|
+
if (reconnectTimerRef.current) {
|
|
101
|
+
clearTimeout(reconnectTimerRef.current);
|
|
102
|
+
reconnectTimerRef.current = null;
|
|
103
|
+
}
|
|
104
|
+
stopPing();
|
|
105
|
+
wsRef.current?.close();
|
|
106
|
+
wsRef.current = null;
|
|
107
|
+
};
|
|
108
|
+
}, [url]);
|
|
109
|
+
|
|
110
|
+
const send = (data: unknown) => {
|
|
111
|
+
const ws = wsRef.current;
|
|
112
|
+
if (ws && ws.readyState === WebSocket.OPEN) {
|
|
113
|
+
ws.send(JSON.stringify(data));
|
|
114
|
+
}
|
|
115
|
+
};
|
|
116
|
+
|
|
117
|
+
return { status, send };
|
|
118
|
+
}
|
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
import { describe, it, expect } from 'vitest';
|
|
2
|
+
import { toExportableEvent, toJson, filename } from './export';
|
|
3
|
+
import type { ToolInvocation } from '../../../src/lib/invocations';
|
|
4
|
+
|
|
5
|
+
describe('export', () => {
|
|
6
|
+
it('maps an invocation to an exportable event', () => {
|
|
7
|
+
const inv: ToolInvocation = {
|
|
8
|
+
id: 'e1',
|
|
9
|
+
tool: 'Read',
|
|
10
|
+
eventType: 'tool_end',
|
|
11
|
+
status: 'success',
|
|
12
|
+
startTime: 1000,
|
|
13
|
+
skill: 'engineer',
|
|
14
|
+
detail: 'opened file',
|
|
15
|
+
input: { path: 'src/index.ts' },
|
|
16
|
+
durationMs: 50,
|
|
17
|
+
};
|
|
18
|
+
const ev = toExportableEvent(inv);
|
|
19
|
+
expect(ev.id).toBe('e1');
|
|
20
|
+
expect(ev.tool).toBe('Read');
|
|
21
|
+
expect(ev.path).toBe('src/index.ts');
|
|
22
|
+
expect(ev.durationMs).toBe(50);
|
|
23
|
+
});
|
|
24
|
+
|
|
25
|
+
it('produces valid JSON', () => {
|
|
26
|
+
const blob = toJson([{ id: 'e1', timestamp: 1, eventType: 'tool_end', status: 'success' }]);
|
|
27
|
+
expect(blob.type).toBe('application/json');
|
|
28
|
+
});
|
|
29
|
+
|
|
30
|
+
it('generates a filename with timestamp', () => {
|
|
31
|
+
expect(filename('json')).toMatch(/^crewloop-events-\d{4}-\d{2}-\d{2}-\d{6}\.json$/);
|
|
32
|
+
});
|
|
33
|
+
});
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
import type { ToolInvocation } from '../../../src/lib/invocations';
|
|
2
|
+
import { resolvePath } from '../../../src/lib/paths';
|
|
3
|
+
import type { ExportableEvent } from './types';
|
|
4
|
+
|
|
5
|
+
export function toExportableEvent(inv: ToolInvocation): ExportableEvent {
|
|
6
|
+
return {
|
|
7
|
+
id: inv.id,
|
|
8
|
+
timestamp: inv.startTime,
|
|
9
|
+
tool: inv.tool,
|
|
10
|
+
eventType: inv.eventType,
|
|
11
|
+
status: inv.status,
|
|
12
|
+
skill: inv.skill,
|
|
13
|
+
detail: inv.detail,
|
|
14
|
+
path: resolvePath(inv.input, inv.output),
|
|
15
|
+
durationMs: inv.durationMs,
|
|
16
|
+
};
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
export function toJson(events: ExportableEvent[]): Blob {
|
|
20
|
+
return new Blob([JSON.stringify(events, null, 2)], { type: 'application/json' });
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
export function download(blob: Blob, filename: string): void {
|
|
24
|
+
const url = URL.createObjectURL(blob);
|
|
25
|
+
const a = document.createElement('a');
|
|
26
|
+
a.href = url;
|
|
27
|
+
a.download = filename;
|
|
28
|
+
document.body.appendChild(a);
|
|
29
|
+
a.click();
|
|
30
|
+
document.body.removeChild(a);
|
|
31
|
+
URL.revokeObjectURL(url);
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
export function filename(extension: 'json'): string {
|
|
35
|
+
const now = new Date();
|
|
36
|
+
const pad = (n: number) => n.toString().padStart(2, '0');
|
|
37
|
+
const stamp = `${now.getFullYear()}-${pad(now.getMonth() + 1)}-${pad(now.getDate())}-${pad(now.getHours())}${pad(now.getMinutes())}${pad(now.getSeconds())}`;
|
|
38
|
+
return `crewloop-events-${stamp}.${extension}`;
|
|
39
|
+
}
|
|
@@ -0,0 +1,95 @@
|
|
|
1
|
+
import { describe, it, expect } from 'vitest';
|
|
2
|
+
import { buildOptions, filterInvocations, filterSessions, filterGraph } from './filter';
|
|
3
|
+
import type { AgentSource, ClientSession, EventStatus } from '../../../src/types';
|
|
4
|
+
import type { ToolInvocation } from '../../../src/lib/invocations';
|
|
5
|
+
import type { Graph3D } from '../../../src/lib/graph';
|
|
6
|
+
import { DEFAULT_FILTER_STATE } from './types';
|
|
7
|
+
|
|
8
|
+
function makeSession(id: string, source: ClientSession['source'], overrides?: Partial<ClientSession>): ClientSession {
|
|
9
|
+
return {
|
|
10
|
+
id,
|
|
11
|
+
source,
|
|
12
|
+
lifecycle: 'running',
|
|
13
|
+
events: [],
|
|
14
|
+
startTime: 0,
|
|
15
|
+
lastActivity: 1000,
|
|
16
|
+
toolCounts: {},
|
|
17
|
+
...overrides,
|
|
18
|
+
} as ClientSession;
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
function makeInv(tool: string, overrides?: Partial<ToolInvocation>): ToolInvocation {
|
|
22
|
+
return {
|
|
23
|
+
id: Math.random().toString(),
|
|
24
|
+
tool,
|
|
25
|
+
eventType: 'tool_end',
|
|
26
|
+
status: 'success',
|
|
27
|
+
startTime: 500,
|
|
28
|
+
...overrides,
|
|
29
|
+
} as ToolInvocation;
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
describe('filter', () => {
|
|
33
|
+
it('builds options from selected session', () => {
|
|
34
|
+
const session = makeSession('s1', 'kimi', {
|
|
35
|
+
activeSkill: { name: 'engineer', confidence: 'explicit' },
|
|
36
|
+
events: [{ id: 'e1', timestamp: 0, event_type: 'tool_end', tool: 'Read' }],
|
|
37
|
+
});
|
|
38
|
+
const sessions = new Map<string, ClientSession>([['s1', session]]);
|
|
39
|
+
expect(buildOptions(sessions, 's1')).toEqual({
|
|
40
|
+
sources: ['kimi'],
|
|
41
|
+
skills: ['engineer'],
|
|
42
|
+
statuses: [],
|
|
43
|
+
tools: ['Read'],
|
|
44
|
+
opTypes: ['read'],
|
|
45
|
+
});
|
|
46
|
+
});
|
|
47
|
+
|
|
48
|
+
it('filters invocations by tool and status', () => {
|
|
49
|
+
const invs = [
|
|
50
|
+
makeInv('Read', { status: 'success' }),
|
|
51
|
+
makeInv('Edit', { status: 'error' }),
|
|
52
|
+
makeInv('Bash', { status: 'running' }),
|
|
53
|
+
];
|
|
54
|
+
const filters = { ...DEFAULT_FILTER_STATE, tools: ['Read', 'Edit'], statuses: ['success' as EventStatus] };
|
|
55
|
+
expect(filterInvocations(invs, undefined, filters, 1000)).toHaveLength(1);
|
|
56
|
+
});
|
|
57
|
+
|
|
58
|
+
it('filters invocations by time range', () => {
|
|
59
|
+
const invs = [
|
|
60
|
+
makeInv('Read', { startTime: 64000 }),
|
|
61
|
+
makeInv('Edit', { startTime: 100 }),
|
|
62
|
+
];
|
|
63
|
+
const filters = { ...DEFAULT_FILTER_STATE, timeRange: '1m' as const };
|
|
64
|
+
expect(filterInvocations(invs, undefined, filters, 65000)).toHaveLength(1);
|
|
65
|
+
});
|
|
66
|
+
|
|
67
|
+
it('filters sessions by source and pin order', () => {
|
|
68
|
+
const a = makeSession('a', 'kimi', { lastActivity: 2000 });
|
|
69
|
+
const b = makeSession('b', 'codex', { lastActivity: 3000 });
|
|
70
|
+
const filters = { ...DEFAULT_FILTER_STATE, sources: ['kimi', 'codex'] as AgentSource[] };
|
|
71
|
+
const result = filterSessions([a, b], filters, [{ id: 'a', pinnedAt: 0 }], 4000);
|
|
72
|
+
expect(result.map((s) => s.id)).toEqual(['a', 'b']);
|
|
73
|
+
});
|
|
74
|
+
|
|
75
|
+
it('filters graph while keeping skill root', () => {
|
|
76
|
+
const graph: Graph3D = {
|
|
77
|
+
nodes: [
|
|
78
|
+
{ id: 'skill:eng', type: 'skill', label: 'engineer', weight: 1 },
|
|
79
|
+
{ id: 'tool:Read', type: 'tool', label: 'Read', weight: 1 },
|
|
80
|
+
{ id: 'tool:Edit', type: 'tool', label: 'Edit', weight: 1 },
|
|
81
|
+
{ id: 'file:a', type: 'file', label: 'a', weight: 1 },
|
|
82
|
+
],
|
|
83
|
+
links: [
|
|
84
|
+
{ source: 'skill:eng', target: 'tool:Read', weight: 1 },
|
|
85
|
+
{ source: 'skill:eng', target: 'tool:Edit', weight: 1 },
|
|
86
|
+
{ source: 'tool:Read', target: 'file:a', weight: 1 },
|
|
87
|
+
],
|
|
88
|
+
};
|
|
89
|
+
const invs = [makeInv('Read'), makeInv('Edit')];
|
|
90
|
+
const filters = { ...DEFAULT_FILTER_STATE, tools: ['Read'] };
|
|
91
|
+
const result = filterGraph(graph, invs, filters);
|
|
92
|
+
expect(result.nodes.map((n) => n.id).sort()).toEqual(['file:a', 'skill:eng', 'tool:Read']);
|
|
93
|
+
expect(result.links).toHaveLength(2);
|
|
94
|
+
});
|
|
95
|
+
});
|