@renxqoo/renx-code 0.0.9 → 0.0.12
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 +31 -143
- package/bin/renx.cjs +79 -42
- package/bin/renx.exe +0 -0
- package/package.json +10 -28
- package/src/App.tsx +0 -297
- package/src/agent/runtime/event-format.ts +0 -258
- package/src/agent/runtime/model-types.ts +0 -13
- package/src/agent/runtime/runtime.context-usage.test.ts +0 -192
- package/src/agent/runtime/runtime.error-handling.test.ts +0 -235
- package/src/agent/runtime/runtime.simple.test.ts +0 -16
- package/src/agent/runtime/runtime.test.ts +0 -296
- package/src/agent/runtime/runtime.ts +0 -875
- package/src/agent/runtime/runtime.usage-forwarding.test.ts +0 -228
- package/src/agent/runtime/source-modules.test.ts +0 -38
- package/src/agent/runtime/source-modules.ts +0 -370
- package/src/agent/runtime/tool-call-buffer.test.ts +0 -65
- package/src/agent/runtime/tool-call-buffer.ts +0 -60
- package/src/agent/runtime/tool-confirmation.test.ts +0 -56
- package/src/agent/runtime/tool-confirmation.ts +0 -15
- package/src/agent/runtime/types.ts +0 -99
- package/src/commands/slash-commands.test.ts +0 -216
- package/src/commands/slash-commands.ts +0 -64
- package/src/components/chat/assistant-reply.test.tsx +0 -47
- package/src/components/chat/assistant-reply.tsx +0 -136
- package/src/components/chat/assistant-segment.test.ts +0 -99
- package/src/components/chat/assistant-segment.tsx +0 -125
- package/src/components/chat/assistant-tool-group.tsx +0 -900
- package/src/components/chat/code-block.test.tsx +0 -206
- package/src/components/chat/code-block.tsx +0 -313
- package/src/components/chat/prompt-card.tsx +0 -81
- package/src/components/chat/segment-groups.test.ts +0 -52
- package/src/components/chat/segment-groups.ts +0 -106
- package/src/components/chat/turn-item.tsx +0 -39
- package/src/components/conversation-panel.tsx +0 -43
- package/src/components/file-mention-menu.tsx +0 -77
- package/src/components/file-picker-dialog.tsx +0 -206
- package/src/components/footer-hints.tsx +0 -75
- package/src/components/model-picker-dialog.tsx +0 -248
- package/src/components/prompt.tsx +0 -233
- package/src/components/slash-command-menu.tsx +0 -65
- package/src/components/tool-confirm-dialog-content.test.ts +0 -103
- package/src/components/tool-confirm-dialog-content.ts +0 -186
- package/src/components/tool-confirm-dialog.tsx +0 -187
- package/src/components/tool-display-config.ts +0 -119
- package/src/context-usage-regressions.test.ts +0 -26
- package/src/files/attachment-capabilities.test.ts +0 -30
- package/src/files/attachment-capabilities.ts +0 -50
- package/src/files/attachment-content.ts +0 -153
- package/src/files/file-mention-query.test.ts +0 -34
- package/src/files/file-mention-query.ts +0 -32
- package/src/files/prompt-display.ts +0 -13
- package/src/files/types.ts +0 -5
- package/src/files/workspace-files.ts +0 -61
- package/src/hooks/agent-event-handlers.test.ts +0 -207
- package/src/hooks/agent-event-handlers.ts +0 -196
- package/src/hooks/chat-local-replies.fixed.test.ts +0 -119
- package/src/hooks/chat-local-replies.test.ts +0 -153
- package/src/hooks/chat-local-replies.ts +0 -63
- package/src/hooks/turn-updater.test.ts +0 -70
- package/src/hooks/turn-updater.ts +0 -166
- package/src/hooks/use-agent-chat.context.test.ts +0 -10
- package/src/hooks/use-agent-chat.status.test.ts +0 -14
- package/src/hooks/use-agent-chat.test.ts +0 -80
- package/src/hooks/use-agent-chat.ts +0 -621
- package/src/hooks/use-file-mention-menu.ts +0 -196
- package/src/hooks/use-file-picker.ts +0 -185
- package/src/hooks/use-model-picker.ts +0 -196
- package/src/hooks/use-slash-command-menu.ts +0 -154
- package/src/index.tsx +0 -55
- package/src/runtime/clipboard.test.ts +0 -43
- package/src/runtime/clipboard.ts +0 -89
- package/src/runtime/exit.test.ts +0 -177
- package/src/runtime/exit.ts +0 -98
- package/src/runtime/runtime-support.test.ts +0 -31
- package/src/runtime/terminal-theme.test.ts +0 -55
- package/src/runtime/terminal-theme.ts +0 -196
- package/src/types/chat.ts +0 -32
- package/src/types/message-content.ts +0 -48
- package/src/ui/open-code-theme.ts +0 -176
- package/src/ui/opencode-markdown.ts +0 -211
- package/src/ui/theme.simple.test.ts +0 -52
- package/src/ui/theme.test.ts +0 -151
- package/src/ui/theme.ts +0 -152
- package/src/utils/time.test.ts +0 -144
- package/src/utils/time.ts +0 -7
- package/tsconfig.json +0 -30
- package/vendor/agent-root/src/agent/ENTERPRISE_ACCEPTANCE_CHECKLIST.md +0 -95
- package/vendor/agent-root/src/agent/ENTERPRISE_REALTIME.html +0 -1345
- package/vendor/agent-root/src/agent/ENTERPRISE_REALTIME.md +0 -1353
- package/vendor/agent-root/src/agent/ERROR_CONTRACT.md +0 -60
- package/vendor/agent-root/src/agent/TEST_COVERAGE_ANALYSIS.md +0 -278
- package/vendor/agent-root/src/agent/__test__/error-contract.test.ts +0 -72
- package/vendor/agent-root/src/agent/__test__/types.test.ts +0 -137
- package/vendor/agent-root/src/agent/agent/__test__/abort-runtime.test.ts +0 -83
- package/vendor/agent-root/src/agent/agent/__test__/callback-safety.test.ts +0 -34
- package/vendor/agent-root/src/agent/agent/__test__/compaction.test.ts +0 -323
- package/vendor/agent-root/src/agent/agent/__test__/concurrency.test.ts +0 -290
- package/vendor/agent-root/src/agent/agent/__test__/error-normalizer.test.ts +0 -377
- package/vendor/agent-root/src/agent/agent/__test__/error.test.ts +0 -212
- package/vendor/agent-root/src/agent/agent/__test__/fault-injection.test.ts +0 -295
- package/vendor/agent-root/src/agent/agent/__test__/index.test.ts +0 -3607
- package/vendor/agent-root/src/agent/agent/__test__/logger.test.ts +0 -35
- package/vendor/agent-root/src/agent/agent/__test__/message-utils.test.ts +0 -517
- package/vendor/agent-root/src/agent/agent/__test__/telemetry.test.ts +0 -97
- package/vendor/agent-root/src/agent/agent/__test__/timeout-budget.test.ts +0 -479
- package/vendor/agent-root/src/agent/agent/__test__/tool-call-merge.test.ts +0 -80
- package/vendor/agent-root/src/agent/agent/__test__/tool-execution-ledger.test.ts +0 -76
- package/vendor/agent-root/src/agent/agent/__test__/write-buffer.test.ts +0 -173
- package/vendor/agent-root/src/agent/agent/__test__/write-file-session.test.ts +0 -109
- package/vendor/agent-root/src/agent/agent/abort-runtime.ts +0 -71
- package/vendor/agent-root/src/agent/agent/callback-safety.ts +0 -33
- package/vendor/agent-root/src/agent/agent/compaction.ts +0 -291
- package/vendor/agent-root/src/agent/agent/concurrency.ts +0 -103
- package/vendor/agent-root/src/agent/agent/error-normalizer.ts +0 -190
- package/vendor/agent-root/src/agent/agent/error.ts +0 -198
- package/vendor/agent-root/src/agent/agent/index.ts +0 -1772
- package/vendor/agent-root/src/agent/agent/logger.ts +0 -65
- package/vendor/agent-root/src/agent/agent/message-utils.ts +0 -101
- package/vendor/agent-root/src/agent/agent/stream-events.ts +0 -61
- package/vendor/agent-root/src/agent/agent/telemetry.ts +0 -123
- package/vendor/agent-root/src/agent/agent/timeout-budget.ts +0 -227
- package/vendor/agent-root/src/agent/agent/tool-call-merge.ts +0 -111
- package/vendor/agent-root/src/agent/agent/tool-execution-ledger.ts +0 -164
- package/vendor/agent-root/src/agent/agent/write-buffer.ts +0 -188
- package/vendor/agent-root/src/agent/agent/write-file-session.ts +0 -238
- package/vendor/agent-root/src/agent/app/__test__/agent-app-service.test.ts +0 -1053
- package/vendor/agent-root/src/agent/app/__test__/minimal-agent-application.test.ts +0 -158
- package/vendor/agent-root/src/agent/app/__test__/sqlite-agent-app-store.test.ts +0 -437
- package/vendor/agent-root/src/agent/app/agent-app-service.ts +0 -748
- package/vendor/agent-root/src/agent/app/contracts.ts +0 -109
- package/vendor/agent-root/src/agent/app/index.ts +0 -5
- package/vendor/agent-root/src/agent/app/minimal-agent-application.ts +0 -151
- package/vendor/agent-root/src/agent/app/ports.ts +0 -72
- package/vendor/agent-root/src/agent/app/sqlite-agent-app-store.ts +0 -1182
- package/vendor/agent-root/src/agent/app/sqlite-client.ts +0 -177
- package/vendor/agent-root/src/agent/docs/cli-app-layer/00-README.md +0 -36
- package/vendor/agent-root/src/agent/docs/cli-app-layer/01-scope-and-goals.md +0 -33
- package/vendor/agent-root/src/agent/docs/cli-app-layer/02-architecture-overview.md +0 -40
- package/vendor/agent-root/src/agent/docs/cli-app-layer/03-domain-model-and-contracts.md +0 -91
- package/vendor/agent-root/src/agent/docs/cli-app-layer/04-ports-and-interfaces.md +0 -116
- package/vendor/agent-root/src/agent/docs/cli-app-layer/05-run-orchestration-and-state-machine.md +0 -52
- package/vendor/agent-root/src/agent/docs/cli-app-layer/06-cli-commands-and-ux.md +0 -53
- package/vendor/agent-root/src/agent/docs/cli-app-layer/07-storage-design-local.md +0 -52
- package/vendor/agent-root/src/agent/docs/cli-app-layer/08-error-and-observability.md +0 -40
- package/vendor/agent-root/src/agent/docs/cli-app-layer/09-security-and-policy-boundary.md +0 -19
- package/vendor/agent-root/src/agent/docs/cli-app-layer/10-test-plan-and-acceptance.md +0 -28
- package/vendor/agent-root/src/agent/docs/cli-app-layer/11-implementation-phases.md +0 -26
- package/vendor/agent-root/src/agent/docs/cli-app-layer/12-open-questions-and-risks.md +0 -30
- package/vendor/agent-root/src/agent/docs/cli-app-layer/13-sqlite-schema-fields-and-rationale.md +0 -567
- package/vendor/agent-root/src/agent/docs/cli-app-layer/14-project-flow-mermaid.md +0 -583
- package/vendor/agent-root/src/agent/docs/cli-app-layer/15-openclaw-style-project-blueprint.md +0 -972
- package/vendor/agent-root/src/agent/error-contract.ts +0 -154
- package/vendor/agent-root/src/agent/prompts/system.ts +0 -246
- package/vendor/agent-root/src/agent/prompts/system1.ts +0 -208
- package/vendor/agent-root/src/agent/storage/__test__/file-history-store.test.ts +0 -98
- package/vendor/agent-root/src/agent/storage/file-history-store.ts +0 -313
- package/vendor/agent-root/src/agent/storage/file-storage-config.ts +0 -94
- package/vendor/agent-root/src/agent/storage/file-system.ts +0 -31
- package/vendor/agent-root/src/agent/storage/file-write-service.ts +0 -21
- package/vendor/agent-root/src/agent/tool/__test__/base-tool.test.ts +0 -413
- package/vendor/agent-root/src/agent/tool/__test__/bash-policy.test.ts +0 -356
- package/vendor/agent-root/src/agent/tool/__test__/bash.mocked-coverage.test.ts +0 -375
- package/vendor/agent-root/src/agent/tool/__test__/bash.test.ts +0 -372
- package/vendor/agent-root/src/agent/tool/__test__/error.test.ts +0 -108
- package/vendor/agent-root/src/agent/tool/__test__/file-edit-tool.test.ts +0 -258
- package/vendor/agent-root/src/agent/tool/__test__/file-history-tools.test.ts +0 -121
- package/vendor/agent-root/src/agent/tool/__test__/file-read-tool.test.ts +0 -210
- package/vendor/agent-root/src/agent/tool/__test__/glob.test.ts +0 -139
- package/vendor/agent-root/src/agent/tool/__test__/grep.mocked-coverage.test.ts +0 -456
- package/vendor/agent-root/src/agent/tool/__test__/grep.test.ts +0 -192
- package/vendor/agent-root/src/agent/tool/__test__/lsp.test.ts +0 -300
- package/vendor/agent-root/src/agent/tool/__test__/outside-workspace-confirmation.test.ts +0 -214
- package/vendor/agent-root/src/agent/tool/__test__/path-security.test.ts +0 -336
- package/vendor/agent-root/src/agent/tool/__test__/skill-loader.test.ts +0 -494
- package/vendor/agent-root/src/agent/tool/__test__/skill-parser.test.ts +0 -543
- package/vendor/agent-root/src/agent/tool/__test__/skill-tool.test.ts +0 -172
- package/vendor/agent-root/src/agent/tool/__test__/task-concurrency-and-version.test.ts +0 -116
- package/vendor/agent-root/src/agent/tool/__test__/task-create-get-list-update.test.ts +0 -267
- package/vendor/agent-root/src/agent/tool/__test__/task-create.test.ts +0 -519
- package/vendor/agent-root/src/agent/tool/__test__/task-errors.test.ts +0 -225
- package/vendor/agent-root/src/agent/tool/__test__/task-output-blocking.test.ts +0 -223
- package/vendor/agent-root/src/agent/tool/__test__/task-output.test.ts +0 -184
- package/vendor/agent-root/src/agent/tool/__test__/task-parent-abort.test.ts +0 -287
- package/vendor/agent-root/src/agent/tool/__test__/task-real-runner-adapter.test.ts +0 -190
- package/vendor/agent-root/src/agent/tool/__test__/task-run-lifecycle.test.ts +0 -352
- package/vendor/agent-root/src/agent/tool/__test__/task-store-runner-branches.test.ts +0 -395
- package/vendor/agent-root/src/agent/tool/__test__/task-store.test.ts +0 -391
- package/vendor/agent-root/src/agent/tool/__test__/task-subagent-config-integration.test.ts +0 -176
- package/vendor/agent-root/src/agent/tool/__test__/task-subagent-config.test.ts +0 -68
- package/vendor/agent-root/src/agent/tool/__test__/task-tools-core-edges.test.ts +0 -630
- package/vendor/agent-root/src/agent/tool/__test__/task-tools-runtime-edges.test.ts +0 -732
- package/vendor/agent-root/src/agent/tool/__test__/task-types.test.ts +0 -494
- package/vendor/agent-root/src/agent/tool/__test__/task-utils-branches.test.ts +0 -175
- package/vendor/agent-root/src/agent/tool/__test__/tool-manager.test.ts +0 -505
- package/vendor/agent-root/src/agent/tool/__test__/types.test.ts +0 -55
- package/vendor/agent-root/src/agent/tool/__test__/web-fetch.test.ts +0 -244
- package/vendor/agent-root/src/agent/tool/__test__/web-search.test.ts +0 -290
- package/vendor/agent-root/src/agent/tool/__test__/write-file.test.ts +0 -368
- package/vendor/agent-root/src/agent/tool/base-tool.ts +0 -345
- package/vendor/agent-root/src/agent/tool/bash-policy.ts +0 -636
- package/vendor/agent-root/src/agent/tool/bash.ts +0 -688
- package/vendor/agent-root/src/agent/tool/error.ts +0 -131
- package/vendor/agent-root/src/agent/tool/file-edit-tool.ts +0 -264
- package/vendor/agent-root/src/agent/tool/file-history-list.ts +0 -103
- package/vendor/agent-root/src/agent/tool/file-history-restore.ts +0 -149
- package/vendor/agent-root/src/agent/tool/file-read-tool.ts +0 -211
- package/vendor/agent-root/src/agent/tool/glob.ts +0 -171
- package/vendor/agent-root/src/agent/tool/grep.ts +0 -496
- package/vendor/agent-root/src/agent/tool/lsp.ts +0 -481
- package/vendor/agent-root/src/agent/tool/path-security.ts +0 -117
- package/vendor/agent-root/src/agent/tool/search/common.ts +0 -153
- package/vendor/agent-root/src/agent/tool/skill/index.ts +0 -13
- package/vendor/agent-root/src/agent/tool/skill/loader.ts +0 -229
- package/vendor/agent-root/src/agent/tool/skill/parser.ts +0 -124
- package/vendor/agent-root/src/agent/tool/skill/types.ts +0 -27
- package/vendor/agent-root/src/agent/tool/skill-tool.ts +0 -143
- package/vendor/agent-root/src/agent/tool/task-create.ts +0 -186
- package/vendor/agent-root/src/agent/tool/task-errors.ts +0 -42
- package/vendor/agent-root/src/agent/tool/task-get.ts +0 -116
- package/vendor/agent-root/src/agent/tool/task-graph.ts +0 -78
- package/vendor/agent-root/src/agent/tool/task-list.ts +0 -141
- package/vendor/agent-root/src/agent/tool/task-mock-runner-adapter.ts +0 -232
- package/vendor/agent-root/src/agent/tool/task-output.ts +0 -223
- package/vendor/agent-root/src/agent/tool/task-parent-abort.ts +0 -115
- package/vendor/agent-root/src/agent/tool/task-real-runner-adapter.ts +0 -336
- package/vendor/agent-root/src/agent/tool/task-runner-adapter.ts +0 -55
- package/vendor/agent-root/src/agent/tool/task-stop.ts +0 -187
- package/vendor/agent-root/src/agent/tool/task-store.ts +0 -217
- package/vendor/agent-root/src/agent/tool/task-subagent-config.ts +0 -149
- package/vendor/agent-root/src/agent/tool/task-types.ts +0 -264
- package/vendor/agent-root/src/agent/tool/task-update.ts +0 -315
- package/vendor/agent-root/src/agent/tool/task.ts +0 -209
- package/vendor/agent-root/src/agent/tool/tool-manager.ts +0 -361
- package/vendor/agent-root/src/agent/tool/tool-prompts.ts +0 -242
- package/vendor/agent-root/src/agent/tool/types.ts +0 -116
- package/vendor/agent-root/src/agent/tool/web-fetch.ts +0 -227
- package/vendor/agent-root/src/agent/tool/web-search.ts +0 -208
- package/vendor/agent-root/src/agent/tool/write-file.ts +0 -497
- package/vendor/agent-root/src/agent/types.ts +0 -232
- package/vendor/agent-root/src/agent/utils/__tests__/index.test.ts +0 -18
- package/vendor/agent-root/src/agent/utils/__tests__/message-utils.test.ts +0 -610
- package/vendor/agent-root/src/agent/utils/__tests__/message.test.ts +0 -223
- package/vendor/agent-root/src/agent/utils/__tests__/token.test.ts +0 -42
- package/vendor/agent-root/src/agent/utils/index.ts +0 -16
- package/vendor/agent-root/src/agent/utils/message.ts +0 -171
- package/vendor/agent-root/src/agent/utils/token.ts +0 -28
- package/vendor/agent-root/src/config/__tests__/load-config-to-env.test.ts +0 -238
- package/vendor/agent-root/src/config/__tests__/loader.test.ts +0 -361
- package/vendor/agent-root/src/config/__tests__/runtime.test.ts +0 -88
- package/vendor/agent-root/src/config/index.ts +0 -55
- package/vendor/agent-root/src/config/loader.ts +0 -494
- package/vendor/agent-root/src/config/paths.ts +0 -30
- package/vendor/agent-root/src/config/runtime.ts +0 -163
- package/vendor/agent-root/src/config/types.ts +0 -96
- package/vendor/agent-root/src/logger/index.ts +0 -57
- package/vendor/agent-root/src/logger/logger.ts +0 -819
- package/vendor/agent-root/src/logger/types.ts +0 -150
- package/vendor/agent-root/src/providers/__tests__/errors.test.ts +0 -441
- package/vendor/agent-root/src/providers/__tests__/index.test.ts +0 -16
- package/vendor/agent-root/src/providers/__tests__/openai-compatible.options.test.ts +0 -318
- package/vendor/agent-root/src/providers/__tests__/openai-compatible.test.ts +0 -600
- package/vendor/agent-root/src/providers/__tests__/registry.test.ts +0 -523
- package/vendor/agent-root/src/providers/__tests__/responses-adapter.test.ts +0 -298
- package/vendor/agent-root/src/providers/adapters/__tests__/anthropic.test.ts +0 -354
- package/vendor/agent-root/src/providers/adapters/__tests__/kimi.test.ts +0 -58
- package/vendor/agent-root/src/providers/adapters/__tests__/standard.test.ts +0 -261
- package/vendor/agent-root/src/providers/adapters/anthropic.ts +0 -572
- package/vendor/agent-root/src/providers/adapters/base.ts +0 -131
- package/vendor/agent-root/src/providers/adapters/kimi.ts +0 -48
- package/vendor/agent-root/src/providers/adapters/responses.ts +0 -732
- package/vendor/agent-root/src/providers/adapters/standard.ts +0 -120
- package/vendor/agent-root/src/providers/http/__tests__/client.timeout.test.ts +0 -313
- package/vendor/agent-root/src/providers/http/client.ts +0 -289
- package/vendor/agent-root/src/providers/http/stream-parser.ts +0 -109
- package/vendor/agent-root/src/providers/index.ts +0 -76
- package/vendor/agent-root/src/providers/kimi-headers.ts +0 -177
- package/vendor/agent-root/src/providers/openai-compatible.ts +0 -387
- package/vendor/agent-root/src/providers/registry/model-config.ts +0 -477
- package/vendor/agent-root/src/providers/registry/provider-factory.ts +0 -127
- package/vendor/agent-root/src/providers/registry.ts +0 -135
- package/vendor/agent-root/src/providers/types/api.ts +0 -284
- package/vendor/agent-root/src/providers/types/config.ts +0 -58
- package/vendor/agent-root/src/providers/types/errors.ts +0 -323
- package/vendor/agent-root/src/providers/types/index.ts +0 -72
- package/vendor/agent-root/src/providers/types/provider.ts +0 -45
- package/vendor/agent-root/src/providers/types/registry.ts +0 -68
|
@@ -1,494 +0,0 @@
|
|
|
1
|
-
import { describe, expect, it, beforeEach, afterEach } from 'vitest';
|
|
2
|
-
import * as fs from 'node:fs/promises';
|
|
3
|
-
import * as os from 'node:os';
|
|
4
|
-
import * as path from 'node:path';
|
|
5
|
-
import {
|
|
6
|
-
SkillLoader,
|
|
7
|
-
getSkillLoader,
|
|
8
|
-
initializeSkillLoader,
|
|
9
|
-
resetSkillLoader,
|
|
10
|
-
} from '../skill/loader';
|
|
11
|
-
|
|
12
|
-
describe('SkillLoader', () => {
|
|
13
|
-
let tempDir: string;
|
|
14
|
-
let skillDir: string;
|
|
15
|
-
let skillFile: string;
|
|
16
|
-
|
|
17
|
-
beforeEach(async () => {
|
|
18
|
-
tempDir = await fs.mkdtemp(path.join(os.tmpdir(), 'skill-loader-test-'));
|
|
19
|
-
skillDir = path.join(tempDir, 'test-skill');
|
|
20
|
-
await fs.mkdir(skillDir, { recursive: true });
|
|
21
|
-
skillFile = path.join(skillDir, 'SKILL.md');
|
|
22
|
-
|
|
23
|
-
await fs.writeFile(
|
|
24
|
-
skillFile,
|
|
25
|
-
`---
|
|
26
|
-
name: test-skill
|
|
27
|
-
description: A test skill
|
|
28
|
-
---
|
|
29
|
-
# Test Skill
|
|
30
|
-
This is a test skill.`
|
|
31
|
-
);
|
|
32
|
-
});
|
|
33
|
-
|
|
34
|
-
afterEach(async () => {
|
|
35
|
-
await fs.rm(tempDir, { recursive: true, force: true });
|
|
36
|
-
resetSkillLoader();
|
|
37
|
-
});
|
|
38
|
-
|
|
39
|
-
describe('constructor', () => {
|
|
40
|
-
it('creates loader with default options', () => {
|
|
41
|
-
const loader = new SkillLoader();
|
|
42
|
-
expect(loader).toBeDefined();
|
|
43
|
-
});
|
|
44
|
-
|
|
45
|
-
it('creates loader with custom skill roots', () => {
|
|
46
|
-
const loader = new SkillLoader({ skillRoots: [tempDir] });
|
|
47
|
-
expect(loader).toBeDefined();
|
|
48
|
-
});
|
|
49
|
-
|
|
50
|
-
it('creates loader with custom working directory', () => {
|
|
51
|
-
const loader = new SkillLoader({ workingDir: tempDir });
|
|
52
|
-
expect(loader).toBeDefined();
|
|
53
|
-
});
|
|
54
|
-
|
|
55
|
-
it('creates loader with both custom options', () => {
|
|
56
|
-
const loader = new SkillLoader({
|
|
57
|
-
skillRoots: [tempDir],
|
|
58
|
-
workingDir: tempDir,
|
|
59
|
-
});
|
|
60
|
-
expect(loader).toBeDefined();
|
|
61
|
-
});
|
|
62
|
-
});
|
|
63
|
-
|
|
64
|
-
describe('initialize', () => {
|
|
65
|
-
it('initializes loader and discovers skills', async () => {
|
|
66
|
-
const loader = new SkillLoader({ skillRoots: [tempDir] });
|
|
67
|
-
await loader.initialize();
|
|
68
|
-
|
|
69
|
-
const metadata = loader.getAllMetadata();
|
|
70
|
-
expect(metadata).toHaveLength(1);
|
|
71
|
-
expect(metadata[0].name).toBe('test-skill');
|
|
72
|
-
expect(metadata[0].description).toBe('A test skill');
|
|
73
|
-
});
|
|
74
|
-
|
|
75
|
-
it('does not reinitialize if already initialized', async () => {
|
|
76
|
-
const loader = new SkillLoader({ skillRoots: [tempDir] });
|
|
77
|
-
await loader.initialize();
|
|
78
|
-
|
|
79
|
-
// Add another skill after initialization
|
|
80
|
-
const skillDir2 = path.join(tempDir, 'test-skill-2');
|
|
81
|
-
await fs.mkdir(skillDir2, { recursive: true });
|
|
82
|
-
await fs.writeFile(
|
|
83
|
-
path.join(skillDir2, 'SKILL.md'),
|
|
84
|
-
`---
|
|
85
|
-
name: test-skill-2
|
|
86
|
-
description: Another test skill
|
|
87
|
-
---
|
|
88
|
-
# Test Skill 2`
|
|
89
|
-
);
|
|
90
|
-
|
|
91
|
-
// Should not discover the new skill
|
|
92
|
-
const metadata = loader.getAllMetadata();
|
|
93
|
-
expect(metadata).toHaveLength(1);
|
|
94
|
-
});
|
|
95
|
-
|
|
96
|
-
it('handles empty skill roots', async () => {
|
|
97
|
-
// When skillRoots is empty array, it should use default skill roots
|
|
98
|
-
const loader = new SkillLoader({ skillRoots: [] });
|
|
99
|
-
await loader.initialize();
|
|
100
|
-
|
|
101
|
-
const metadata = loader.getAllMetadata();
|
|
102
|
-
// Should discover skills from default roots (if any exist)
|
|
103
|
-
expect(Array.isArray(metadata)).toBe(true);
|
|
104
|
-
});
|
|
105
|
-
|
|
106
|
-
it('handles non-existent skill roots', async () => {
|
|
107
|
-
const loader = new SkillLoader({ skillRoots: ['/non/existent/path'] });
|
|
108
|
-
await loader.initialize();
|
|
109
|
-
|
|
110
|
-
const metadata = loader.getAllMetadata();
|
|
111
|
-
expect(metadata).toHaveLength(0);
|
|
112
|
-
});
|
|
113
|
-
|
|
114
|
-
it('handles skill roots that are files', async () => {
|
|
115
|
-
const filePath = path.join(tempDir, 'file.txt');
|
|
116
|
-
await fs.writeFile(filePath, 'test');
|
|
117
|
-
|
|
118
|
-
const loader = new SkillLoader({ skillRoots: [filePath] });
|
|
119
|
-
await loader.initialize();
|
|
120
|
-
|
|
121
|
-
const metadata = loader.getAllMetadata();
|
|
122
|
-
expect(metadata).toHaveLength(0);
|
|
123
|
-
});
|
|
124
|
-
|
|
125
|
-
it('handles skill roots with permission errors', async () => {
|
|
126
|
-
const restrictedDir = path.join(tempDir, 'restricted');
|
|
127
|
-
await fs.mkdir(restrictedDir, { recursive: true });
|
|
128
|
-
await fs.chmod(restrictedDir, 0o000);
|
|
129
|
-
|
|
130
|
-
try {
|
|
131
|
-
const loader = new SkillLoader({ skillRoots: [restrictedDir] });
|
|
132
|
-
await loader.initialize();
|
|
133
|
-
|
|
134
|
-
const metadata = loader.getAllMetadata();
|
|
135
|
-
expect(metadata).toHaveLength(0);
|
|
136
|
-
} finally {
|
|
137
|
-
await fs.chmod(restrictedDir, 0o755);
|
|
138
|
-
}
|
|
139
|
-
});
|
|
140
|
-
|
|
141
|
-
it('handles skill files with invalid frontmatter', async () => {
|
|
142
|
-
const invalidSkillDir = path.join(tempDir, 'invalid-skill');
|
|
143
|
-
await fs.mkdir(invalidSkillDir, { recursive: true });
|
|
144
|
-
await fs.writeFile(
|
|
145
|
-
path.join(invalidSkillDir, 'SKILL.md'),
|
|
146
|
-
`---
|
|
147
|
-
name: invalid-skill
|
|
148
|
-
---
|
|
149
|
-
# Invalid Skill`
|
|
150
|
-
);
|
|
151
|
-
|
|
152
|
-
const loader = new SkillLoader({ skillRoots: [tempDir] });
|
|
153
|
-
await loader.initialize();
|
|
154
|
-
|
|
155
|
-
const metadata = loader.getAllMetadata();
|
|
156
|
-
// When frontmatter is invalid, it uses fallback name (directory name) and derives description
|
|
157
|
-
expect(metadata).toHaveLength(2); // Both skills
|
|
158
|
-
expect(metadata.map((m) => m.name)).toContain('test-skill');
|
|
159
|
-
expect(metadata.map((m) => m.name)).toContain('invalid-skill');
|
|
160
|
-
});
|
|
161
|
-
|
|
162
|
-
it('handles skill files with invalid names', async () => {
|
|
163
|
-
const invalidNameDir = path.join(tempDir, 'Invalid-Name');
|
|
164
|
-
await fs.mkdir(invalidNameDir, { recursive: true });
|
|
165
|
-
await fs.writeFile(
|
|
166
|
-
path.join(invalidNameDir, 'SKILL.md'),
|
|
167
|
-
`---
|
|
168
|
-
name: Invalid-Name
|
|
169
|
-
description: Invalid name skill
|
|
170
|
-
---
|
|
171
|
-
# Invalid Name Skill`
|
|
172
|
-
);
|
|
173
|
-
|
|
174
|
-
const loader = new SkillLoader({ skillRoots: [tempDir] });
|
|
175
|
-
await loader.initialize();
|
|
176
|
-
|
|
177
|
-
const metadata = loader.getAllMetadata();
|
|
178
|
-
expect(metadata).toHaveLength(1); // Only the valid skill
|
|
179
|
-
expect(metadata[0].name).toBe('test-skill');
|
|
180
|
-
});
|
|
181
|
-
|
|
182
|
-
it('handles skill files with empty descriptions', async () => {
|
|
183
|
-
const emptyDescDir = path.join(tempDir, 'empty-desc');
|
|
184
|
-
await fs.mkdir(emptyDescDir, { recursive: true });
|
|
185
|
-
await fs.writeFile(
|
|
186
|
-
path.join(emptyDescDir, 'SKILL.md'),
|
|
187
|
-
`---
|
|
188
|
-
name: empty-desc
|
|
189
|
-
description:
|
|
190
|
-
---
|
|
191
|
-
# Empty Description Skill`
|
|
192
|
-
);
|
|
193
|
-
|
|
194
|
-
const loader = new SkillLoader({ skillRoots: [tempDir] });
|
|
195
|
-
await loader.initialize();
|
|
196
|
-
|
|
197
|
-
const metadata = loader.getAllMetadata();
|
|
198
|
-
// When description is empty, it derives description from markdown
|
|
199
|
-
expect(metadata).toHaveLength(2); // Both skills
|
|
200
|
-
expect(metadata.map((m) => m.name)).toContain('test-skill');
|
|
201
|
-
expect(metadata.map((m) => m.name)).toContain('empty-desc');
|
|
202
|
-
});
|
|
203
|
-
|
|
204
|
-
it('handles skill files that cannot be read', async () => {
|
|
205
|
-
// Skip on Windows as chmod doesn't work the same way
|
|
206
|
-
if (process.platform === 'win32') {
|
|
207
|
-
return;
|
|
208
|
-
}
|
|
209
|
-
const unreadableDir = path.join(tempDir, 'unreadable');
|
|
210
|
-
await fs.mkdir(unreadableDir, { recursive: true });
|
|
211
|
-
const unreadableFile = path.join(unreadableDir, 'SKILL.md');
|
|
212
|
-
await fs.writeFile(
|
|
213
|
-
unreadableFile,
|
|
214
|
-
`---
|
|
215
|
-
name: unreadable
|
|
216
|
-
description: Unreadable skill
|
|
217
|
-
---
|
|
218
|
-
# Unreadable Skill`
|
|
219
|
-
);
|
|
220
|
-
await fs.chmod(unreadableFile, 0o000);
|
|
221
|
-
|
|
222
|
-
try {
|
|
223
|
-
const loader = new SkillLoader({ skillRoots: [tempDir] });
|
|
224
|
-
await loader.initialize();
|
|
225
|
-
|
|
226
|
-
const metadata = loader.getAllMetadata();
|
|
227
|
-
expect(metadata).toHaveLength(1); // Only the valid skill
|
|
228
|
-
expect(metadata[0].name).toBe('test-skill');
|
|
229
|
-
} finally {
|
|
230
|
-
await fs.chmod(unreadableFile, 0o644);
|
|
231
|
-
}
|
|
232
|
-
});
|
|
233
|
-
|
|
234
|
-
it('handles duplicate skill names', async () => {
|
|
235
|
-
const duplicateDir = path.join(tempDir, 'duplicate');
|
|
236
|
-
await fs.mkdir(duplicateDir, { recursive: true });
|
|
237
|
-
await fs.writeFile(
|
|
238
|
-
path.join(duplicateDir, 'SKILL.md'),
|
|
239
|
-
`---
|
|
240
|
-
name: test-skill
|
|
241
|
-
description: Duplicate skill
|
|
242
|
-
---
|
|
243
|
-
# Duplicate Skill`
|
|
244
|
-
);
|
|
245
|
-
|
|
246
|
-
const loader = new SkillLoader({ skillRoots: [tempDir] });
|
|
247
|
-
await loader.initialize();
|
|
248
|
-
|
|
249
|
-
const metadata = loader.getAllMetadata();
|
|
250
|
-
expect(metadata).toHaveLength(1); // Only the first skill
|
|
251
|
-
expect(metadata[0].name).toBe('test-skill');
|
|
252
|
-
// The first discovered skill's description is kept
|
|
253
|
-
// Note: The order of discovery depends on directory traversal order
|
|
254
|
-
expect(['A test skill', 'Duplicate skill']).toContain(metadata[0].description);
|
|
255
|
-
});
|
|
256
|
-
|
|
257
|
-
it('handles nested skill directories', async () => {
|
|
258
|
-
const nestedDir = path.join(tempDir, 'nested', 'deep', 'skill');
|
|
259
|
-
await fs.mkdir(nestedDir, { recursive: true });
|
|
260
|
-
await fs.writeFile(
|
|
261
|
-
path.join(nestedDir, 'SKILL.md'),
|
|
262
|
-
`---
|
|
263
|
-
name: nested-skill
|
|
264
|
-
description: Nested skill
|
|
265
|
-
---
|
|
266
|
-
# Nested Skill`
|
|
267
|
-
);
|
|
268
|
-
|
|
269
|
-
const loader = new SkillLoader({ skillRoots: [tempDir] });
|
|
270
|
-
await loader.initialize();
|
|
271
|
-
|
|
272
|
-
const metadata = loader.getAllMetadata();
|
|
273
|
-
expect(metadata).toHaveLength(2); // Both skills
|
|
274
|
-
expect(metadata.map((m) => m.name)).toContain('test-skill');
|
|
275
|
-
expect(metadata.map((m) => m.name)).toContain('nested-skill');
|
|
276
|
-
});
|
|
277
|
-
|
|
278
|
-
it('handles circular symlinks', async () => {
|
|
279
|
-
const symlinkDir = path.join(tempDir, 'symlink');
|
|
280
|
-
await fs.symlink(tempDir, symlinkDir);
|
|
281
|
-
|
|
282
|
-
const loader = new SkillLoader({ skillRoots: [tempDir] });
|
|
283
|
-
await loader.initialize();
|
|
284
|
-
|
|
285
|
-
const metadata = loader.getAllMetadata();
|
|
286
|
-
expect(metadata).toHaveLength(1); // Should not hang or crash
|
|
287
|
-
});
|
|
288
|
-
});
|
|
289
|
-
|
|
290
|
-
describe('getAllMetadata', () => {
|
|
291
|
-
it('returns empty array when no skills', async () => {
|
|
292
|
-
// Create a new empty directory for this test
|
|
293
|
-
const emptyDir = await fs.mkdtemp(path.join(os.tmpdir(), 'empty-skill-'));
|
|
294
|
-
try {
|
|
295
|
-
const loader = new SkillLoader({ skillRoots: [emptyDir] });
|
|
296
|
-
await loader.initialize();
|
|
297
|
-
|
|
298
|
-
const metadata = loader.getAllMetadata();
|
|
299
|
-
expect(metadata).toHaveLength(0);
|
|
300
|
-
} finally {
|
|
301
|
-
await fs.rm(emptyDir, { recursive: true, force: true });
|
|
302
|
-
}
|
|
303
|
-
});
|
|
304
|
-
|
|
305
|
-
it('returns skills sorted by name', async () => {
|
|
306
|
-
// Create another skill
|
|
307
|
-
const skillDir2 = path.join(tempDir, 'another-skill');
|
|
308
|
-
await fs.mkdir(skillDir2, { recursive: true });
|
|
309
|
-
await fs.writeFile(
|
|
310
|
-
path.join(skillDir2, 'SKILL.md'),
|
|
311
|
-
`---
|
|
312
|
-
name: another-skill
|
|
313
|
-
description: Another test skill
|
|
314
|
-
---
|
|
315
|
-
# Another Test Skill`
|
|
316
|
-
);
|
|
317
|
-
|
|
318
|
-
const loader = new SkillLoader({ skillRoots: [tempDir] });
|
|
319
|
-
await loader.initialize();
|
|
320
|
-
|
|
321
|
-
const metadata = loader.getAllMetadata();
|
|
322
|
-
expect(metadata).toHaveLength(2);
|
|
323
|
-
expect(metadata[0].name).toBe('another-skill');
|
|
324
|
-
expect(metadata[1].name).toBe('test-skill');
|
|
325
|
-
});
|
|
326
|
-
});
|
|
327
|
-
|
|
328
|
-
describe('hasSkill', () => {
|
|
329
|
-
it('returns true for existing skill', async () => {
|
|
330
|
-
const loader = new SkillLoader({ skillRoots: [tempDir] });
|
|
331
|
-
await loader.initialize();
|
|
332
|
-
|
|
333
|
-
expect(loader.hasSkill('test-skill')).toBe(true);
|
|
334
|
-
});
|
|
335
|
-
|
|
336
|
-
it('returns false for non-existing skill', async () => {
|
|
337
|
-
const loader = new SkillLoader({ skillRoots: [tempDir] });
|
|
338
|
-
await loader.initialize();
|
|
339
|
-
|
|
340
|
-
expect(loader.hasSkill('non-existing')).toBe(false);
|
|
341
|
-
});
|
|
342
|
-
|
|
343
|
-
it('returns false before initialization', () => {
|
|
344
|
-
const loader = new SkillLoader({ skillRoots: [tempDir] });
|
|
345
|
-
|
|
346
|
-
expect(loader.hasSkill('test-skill')).toBe(false);
|
|
347
|
-
});
|
|
348
|
-
});
|
|
349
|
-
|
|
350
|
-
describe('loadSkill', () => {
|
|
351
|
-
it('loads skill content', async () => {
|
|
352
|
-
const loader = new SkillLoader({ skillRoots: [tempDir] });
|
|
353
|
-
await loader.initialize();
|
|
354
|
-
|
|
355
|
-
const skill = await loader.loadSkill('test-skill');
|
|
356
|
-
expect(skill).toBeDefined();
|
|
357
|
-
expect(skill!.metadata.name).toBe('test-skill');
|
|
358
|
-
expect(skill!.content).toContain('# Test Skill');
|
|
359
|
-
expect(skill!.content).toContain('This is a test skill.');
|
|
360
|
-
expect(skill!.loadedAt).toBeGreaterThan(0);
|
|
361
|
-
});
|
|
362
|
-
|
|
363
|
-
it('caches loaded skills', async () => {
|
|
364
|
-
const loader = new SkillLoader({ skillRoots: [tempDir] });
|
|
365
|
-
await loader.initialize();
|
|
366
|
-
|
|
367
|
-
const skill1 = await loader.loadSkill('test-skill');
|
|
368
|
-
const skill2 = await loader.loadSkill('test-skill');
|
|
369
|
-
|
|
370
|
-
expect(skill1).toBe(skill2); // Same object reference
|
|
371
|
-
});
|
|
372
|
-
|
|
373
|
-
it('returns null for non-existing skill', async () => {
|
|
374
|
-
const loader = new SkillLoader({ skillRoots: [tempDir] });
|
|
375
|
-
await loader.initialize();
|
|
376
|
-
|
|
377
|
-
const skill = await loader.loadSkill('non-existing');
|
|
378
|
-
expect(skill).toBeNull();
|
|
379
|
-
});
|
|
380
|
-
|
|
381
|
-
it('returns null for skill that cannot be read', async () => {
|
|
382
|
-
const loader = new SkillLoader({ skillRoots: [tempDir] });
|
|
383
|
-
await loader.initialize();
|
|
384
|
-
|
|
385
|
-
// Remove the skill file
|
|
386
|
-
await fs.rm(skillFile);
|
|
387
|
-
|
|
388
|
-
const skill = await loader.loadSkill('test-skill');
|
|
389
|
-
expect(skill).toBeNull();
|
|
390
|
-
});
|
|
391
|
-
|
|
392
|
-
it('returns null before initialization', async () => {
|
|
393
|
-
const loader = new SkillLoader({ skillRoots: [tempDir] });
|
|
394
|
-
|
|
395
|
-
const skill = await loader.loadSkill('test-skill');
|
|
396
|
-
expect(skill).toBeNull();
|
|
397
|
-
});
|
|
398
|
-
|
|
399
|
-
it('extracts file refs and shell commands', async () => {
|
|
400
|
-
// Create a skill with file refs and shell commands
|
|
401
|
-
const skillDir2 = path.join(tempDir, 'rich-skill');
|
|
402
|
-
await fs.mkdir(skillDir2, { recursive: true });
|
|
403
|
-
await fs.writeFile(
|
|
404
|
-
path.join(skillDir2, 'SKILL.md'),
|
|
405
|
-
`---
|
|
406
|
-
name: rich-skill
|
|
407
|
-
description: Rich skill
|
|
408
|
-
---
|
|
409
|
-
# Rich Skill
|
|
410
|
-
Check @file.txt and run !\`ls -la\`.`
|
|
411
|
-
);
|
|
412
|
-
|
|
413
|
-
const loader = new SkillLoader({ skillRoots: [tempDir] });
|
|
414
|
-
await loader.initialize();
|
|
415
|
-
|
|
416
|
-
const skill = await loader.loadSkill('rich-skill');
|
|
417
|
-
expect(skill).toBeDefined();
|
|
418
|
-
expect(skill!.fileRefs).toContain('file.txt');
|
|
419
|
-
expect(skill!.shellCommands).toContain('ls -la');
|
|
420
|
-
});
|
|
421
|
-
});
|
|
422
|
-
});
|
|
423
|
-
|
|
424
|
-
describe('getSkillLoader', () => {
|
|
425
|
-
afterEach(() => {
|
|
426
|
-
resetSkillLoader();
|
|
427
|
-
});
|
|
428
|
-
|
|
429
|
-
it('returns same loader for same options', () => {
|
|
430
|
-
const loader1 = getSkillLoader({ workingDir: '/tmp' });
|
|
431
|
-
const loader2 = getSkillLoader({ workingDir: '/tmp' });
|
|
432
|
-
|
|
433
|
-
expect(loader1).toBe(loader2);
|
|
434
|
-
});
|
|
435
|
-
|
|
436
|
-
it('returns different loader for different options', () => {
|
|
437
|
-
const loader1 = getSkillLoader({ workingDir: '/tmp' });
|
|
438
|
-
const loader2 = getSkillLoader({ workingDir: '/var' });
|
|
439
|
-
|
|
440
|
-
expect(loader1).not.toBe(loader2);
|
|
441
|
-
});
|
|
442
|
-
|
|
443
|
-
it('returns different loader for different skill roots', () => {
|
|
444
|
-
const loader1 = getSkillLoader({ skillRoots: ['/tmp'] });
|
|
445
|
-
const loader2 = getSkillLoader({ skillRoots: ['/var'] });
|
|
446
|
-
|
|
447
|
-
expect(loader1).not.toBe(loader2);
|
|
448
|
-
});
|
|
449
|
-
|
|
450
|
-
it('returns same loader for equivalent options', () => {
|
|
451
|
-
const loader1 = getSkillLoader({ skillRoots: ['/tmp', '/var'] });
|
|
452
|
-
const loader2 = getSkillLoader({ skillRoots: ['/var', '/tmp'] });
|
|
453
|
-
|
|
454
|
-
expect(loader1).toBe(loader2);
|
|
455
|
-
});
|
|
456
|
-
});
|
|
457
|
-
|
|
458
|
-
describe('initializeSkillLoader', () => {
|
|
459
|
-
afterEach(() => {
|
|
460
|
-
resetSkillLoader();
|
|
461
|
-
});
|
|
462
|
-
|
|
463
|
-
it('initializes and returns loader', async () => {
|
|
464
|
-
const tempDir = await fs.mkdtemp(path.join(os.tmpdir(), 'skill-init-test-'));
|
|
465
|
-
try {
|
|
466
|
-
const skillDir = path.join(tempDir, 'test-skill');
|
|
467
|
-
await fs.mkdir(skillDir, { recursive: true });
|
|
468
|
-
await fs.writeFile(
|
|
469
|
-
path.join(skillDir, 'SKILL.md'),
|
|
470
|
-
`---
|
|
471
|
-
name: test-skill
|
|
472
|
-
description: A test skill
|
|
473
|
-
---
|
|
474
|
-
# Test Skill`
|
|
475
|
-
);
|
|
476
|
-
|
|
477
|
-
const loader = await initializeSkillLoader({ skillRoots: [tempDir] });
|
|
478
|
-
expect(loader).toBeDefined();
|
|
479
|
-
expect(loader.hasSkill('test-skill')).toBe(true);
|
|
480
|
-
} finally {
|
|
481
|
-
await fs.rm(tempDir, { recursive: true, force: true });
|
|
482
|
-
}
|
|
483
|
-
});
|
|
484
|
-
});
|
|
485
|
-
|
|
486
|
-
describe('resetSkillLoader', () => {
|
|
487
|
-
it('resets global loader', () => {
|
|
488
|
-
const loader1 = getSkillLoader();
|
|
489
|
-
resetSkillLoader();
|
|
490
|
-
const loader2 = getSkillLoader();
|
|
491
|
-
|
|
492
|
-
expect(loader1).not.toBe(loader2);
|
|
493
|
-
});
|
|
494
|
-
});
|