open-design/apps/daemon/tests/skills.test.ts
marco 5dd70b5016
Some checks failed
ci / Validate workspace (push) Successful in 12m32s
landing-page-ci / Validate landing page (push) Successful in 9m41s
landing-page-deploy / Deploy landing page (push) Failing after 5m23s
github-metrics / Generate repository metrics SVG (push) Failing after 2m3s
refresh-contributors-wall / Refresh contributors wall cache bust (push) Failing after 11s
Initial import: open-design source for helix-mind.ai distribution
This repository contains the open-design daemon CLI source code, built
and packaged at https://helix-mind.ai/cli/open-design/latest.tgz for use
by the HelixMind /design slash command.

Licenses: Apache-2.0 (root) + MIT (skills/*)
2026-05-06 20:50:24 +02:00

144 lines
5.2 KiB
TypeScript

import { mkdirSync, mkdtempSync, writeFileSync } from 'node:fs';
import { tmpdir } from 'node:os';
import { fileURLToPath } from 'node:url';
import path from 'node:path';
import { describe, expect, it } from 'vitest';
import { SKILLS_CWD_ALIAS } from '../src/cwd-aliases.js';
import { listSkills } from '../src/skills.js';
const __filename = fileURLToPath(import.meta.url);
const __dirname = path.dirname(__filename);
const repoRoot = path.resolve(__dirname, '../../..');
const skillsRoot = path.join(repoRoot, 'skills');
const liveArtifactRoot = path.join(skillsRoot, 'live-artifact');
function fresh(): string {
return mkdtempSync(path.join(tmpdir(), 'od-skills-'));
}
function writeSkill(
root: string,
folder: string,
options: {
name?: string;
description?: string;
body?: string;
withAttachments?: boolean;
} = {},
) {
const dir = path.join(root, folder);
mkdirSync(dir, { recursive: true });
const fm = [
'---',
`name: ${options.name ?? folder}`,
`description: ${options.description ?? 'A test skill.'}`,
'---',
'',
options.body ?? '# Test skill body',
'',
].join('\n');
writeFileSync(path.join(dir, 'SKILL.md'), fm);
if (options.withAttachments) {
mkdirSync(path.join(dir, 'assets'), { recursive: true });
writeFileSync(
path.join(dir, 'assets', 'template.html'),
'<html><body>seed</body></html>',
);
}
}
describe('listSkills', () => {
it('includes the built-in live-artifact skill catalog entry', async () => {
const skills = await listSkills(skillsRoot);
const skill = skills.find((entry: { id: string }) => entry.id === 'live-artifact');
expect(skill).toBeTruthy();
expect(skill).toMatchObject({
id: 'live-artifact',
name: 'live-artifact',
mode: 'prototype',
previewType: 'html',
});
expect(skill.triggers.length).toBeGreaterThan(0);
expect(skill.body).toContain(`> **Skill root (absolute fallback):** \`${liveArtifactRoot}\``);
expect(skill.body).toContain(`${SKILLS_CWD_ALIAS}/live-artifact/`);
expect(skill.body).toContain('references/artifact-schema.md');
expect(skill.body).toContain('references/connector-policy.md');
expect(skill.body).toContain('references/refresh-contract.md');
expect(skill.body).toContain('"$OD_NODE_BIN" "$OD_BIN" tools live-artifacts create --input artifact.json');
expect(skill.body).toContain('do not ask “where should the data come from?” before checking daemon connector tools');
expect(skill.body).toContain('notion.notion_search');
expect(skill.body).toContain('`OD_DAEMON_URL`');
expect(skill.body).toContain('`OD_TOOL_TOKEN`');
});
});
describe('listSkills preamble', () => {
it('emits both a cwd-relative skill root and an absolute fallback', async () => {
const root = fresh();
writeSkill(root, 'demo-skill', {
withAttachments: true,
body: 'Use `assets/template.html` to bootstrap.',
});
const skills = await listSkills(root);
expect(skills).toHaveLength(1);
const [skill] = skills;
// The cwd-relative alias path is the primary one — that's what makes
// the agent stay inside its working directory when reading skill
// side files (issue #430).
expect(skill.body).toContain(`${SKILLS_CWD_ALIAS}/demo-skill/`);
expect(skill.body).toContain(
`${SKILLS_CWD_ALIAS}/demo-skill/assets/template.html`,
);
// The absolute fallback is required for two cases the relative path
// cannot serve:
// - calls without a project (cwd defaults to PROJECT_ROOT, where
// the absolute path is in fact an in-cwd path);
// - environments where `stageActiveSkill()` failed.
// Claude/Copilot are additionally given `--add-dir` for that path.
expect(skill.body).toContain(skill.dir);
expect(skill.body).toMatch(/Skill root \(absolute fallback\)/);
expect(skill.body).toMatch(/Skill root \(relative to project\)/);
});
it('uses the on-disk folder name in the alias path even when `name` differs', async () => {
const root = fresh();
writeSkill(root, 'guizang-ppt', {
name: 'magazine-web-ppt',
withAttachments: true,
});
const skills = await listSkills(root);
expect(skills).toHaveLength(1);
const [skill] = skills;
// `id`/`name` reflect the frontmatter value (used elsewhere as a stable
// public id), but the on-disk alias path must use the actual folder
// name — that is what the daemon-staged junction maps to.
expect(skill.id).toBe('magazine-web-ppt');
expect(skill.body).toContain(`${SKILLS_CWD_ALIAS}/guizang-ppt/`);
expect(skill.body).not.toContain(`${SKILLS_CWD_ALIAS}/magazine-web-ppt/`);
});
it('does not emit a preamble for skills without side files', async () => {
const root = fresh();
writeSkill(root, 'lone-skill', {
withAttachments: false,
body: 'Body without external files.',
});
const skills = await listSkills(root);
expect(skills).toHaveLength(1);
const [skill] = skills;
expect(skill.body).not.toContain(SKILLS_CWD_ALIAS);
expect(skill.body).not.toContain('Skill root');
expect(skill.body).toContain('Body without external files.');
});
});