@otto-assistant/otto 0.1.0 → 0.1.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.
Files changed (73) hide show
  1. package/README.md +142 -0
  2. package/dist/cli.js +406 -12
  3. package/dist/cli.js.map +1 -1
  4. package/dist/config.test.js +9 -9
  5. package/dist/config.test.js.map +1 -1
  6. package/dist/detect.test.js +4 -3
  7. package/dist/detect.test.js.map +1 -1
  8. package/dist/docker.d.ts +7 -0
  9. package/dist/docker.d.ts.map +1 -0
  10. package/dist/docker.js +17 -0
  11. package/dist/docker.js.map +1 -0
  12. package/dist/docker.test.d.ts +2 -0
  13. package/dist/docker.test.d.ts.map +1 -0
  14. package/dist/docker.test.js +12 -0
  15. package/dist/docker.test.js.map +1 -0
  16. package/dist/health.d.ts +4 -0
  17. package/dist/health.d.ts.map +1 -1
  18. package/dist/health.js +40 -1
  19. package/dist/health.js.map +1 -1
  20. package/dist/health.test.js +21 -2
  21. package/dist/health.test.js.map +1 -1
  22. package/dist/index.d.ts +11 -3
  23. package/dist/index.d.ts.map +1 -1
  24. package/dist/index.js +6 -2
  25. package/dist/index.js.map +1 -1
  26. package/dist/installer.test.js +2 -2
  27. package/dist/installer.test.js.map +1 -1
  28. package/dist/lifecycle.d.ts +6 -0
  29. package/dist/lifecycle.d.ts.map +1 -1
  30. package/dist/lifecycle.js +26 -11
  31. package/dist/lifecycle.js.map +1 -1
  32. package/dist/lifecycle.test.js +5 -4
  33. package/dist/lifecycle.test.js.map +1 -1
  34. package/dist/manifest.js +4 -4
  35. package/dist/manifest.js.map +1 -1
  36. package/dist/skills-baseline.d.ts +7 -0
  37. package/dist/skills-baseline.d.ts.map +1 -0
  38. package/dist/skills-baseline.js +9 -0
  39. package/dist/skills-baseline.js.map +1 -0
  40. package/dist/skills.d.ts +110 -0
  41. package/dist/skills.d.ts.map +1 -0
  42. package/dist/skills.js +429 -0
  43. package/dist/skills.js.map +1 -0
  44. package/dist/skills.test.d.ts +2 -0
  45. package/dist/skills.test.d.ts.map +1 -0
  46. package/dist/skills.test.js +416 -0
  47. package/dist/skills.test.js.map +1 -0
  48. package/dist/tenant.d.ts +13 -0
  49. package/dist/tenant.d.ts.map +1 -0
  50. package/dist/tenant.js +105 -0
  51. package/dist/tenant.js.map +1 -0
  52. package/dist/tenant.test.d.ts +2 -0
  53. package/dist/tenant.test.d.ts.map +1 -0
  54. package/dist/tenant.test.js +37 -0
  55. package/dist/tenant.test.js.map +1 -0
  56. package/package.json +15 -5
  57. package/src/cli.ts +457 -12
  58. package/src/config.test.ts +9 -9
  59. package/src/detect.test.ts +4 -3
  60. package/src/docker.test.ts +12 -0
  61. package/src/docker.ts +23 -0
  62. package/src/health.test.ts +23 -1
  63. package/src/health.ts +45 -1
  64. package/src/index.ts +37 -3
  65. package/src/installer.test.ts +2 -2
  66. package/src/lifecycle.test.ts +6 -5
  67. package/src/lifecycle.ts +29 -10
  68. package/src/manifest.ts +4 -4
  69. package/src/skills-baseline.ts +14 -0
  70. package/src/skills.test.ts +503 -0
  71. package/src/skills.ts +512 -0
  72. package/src/tenant.test.ts +49 -0
  73. package/src/tenant.ts +120 -0
@@ -0,0 +1,416 @@
1
+ import { describe, it, expect, beforeEach, afterEach } from "vitest";
2
+ import fs from "node:fs";
3
+ import path from "node:path";
4
+ import os from "node:os";
5
+ import { execFileSync } from "node:child_process";
6
+ import { SKILLS_INDEX_PATH, OPENCODE_SKILLS_DIR, DEFAULT_SKILL_REPOS, parseSkillMd, listInstalledSkills, removeSkill, loadSkillsIndex, saveSkillsIndex, isIndexStale, searchSkills, getAllIndexedSkills, installSkillFromIndex, installSkillsBaseline, fetchRepoSkillsIndex, } from "./skills.js";
7
+ // ---------------------------------------------------------------------------
8
+ // Helpers
9
+ // ---------------------------------------------------------------------------
10
+ function makeTmp(prefix = "otto-skills-test-") {
11
+ return fs.mkdtempSync(path.join(os.tmpdir(), prefix));
12
+ }
13
+ function writeSkillMd(dir, content) {
14
+ fs.mkdirSync(dir, { recursive: true });
15
+ fs.writeFileSync(path.join(dir, "SKILL.md"), content, "utf-8");
16
+ }
17
+ const VALID_SKILL_MD = `---
18
+ name: my-skill
19
+ description: A test skill
20
+ ---
21
+
22
+ # My Skill
23
+
24
+ Body text here.
25
+ `;
26
+ const VALID_SKILL_MD_WITH_META = `---
27
+ name: my-skill
28
+ description: A test skill with metadata
29
+ metadata:
30
+ author: otto
31
+ version: "1.0"
32
+ ---
33
+
34
+ # My Skill
35
+
36
+ Body text here.
37
+ `;
38
+ const NO_FRONTMATTER_MD = `# No frontmatter
39
+
40
+ Just a regular markdown file.
41
+ `;
42
+ const MISSING_NAME_MD = `---
43
+ description: Has description but no name
44
+ ---
45
+
46
+ # No name
47
+ `;
48
+ const MISSING_DESC_MD = `---
49
+ name: has-name
50
+ ---
51
+
52
+ # No description
53
+ `;
54
+ // ---------------------------------------------------------------------------
55
+ // Constants
56
+ // ---------------------------------------------------------------------------
57
+ describe("skills constants", () => {
58
+ it("DEFAULT_SKILL_REPOS includes key repos", () => {
59
+ expect(DEFAULT_SKILL_REPOS).toContain("otto-assistant/skills");
60
+ expect(DEFAULT_SKILL_REPOS).toContain("anthropics/skills");
61
+ expect(DEFAULT_SKILL_REPOS).toContain("vercel-labs/agent-skills");
62
+ expect(DEFAULT_SKILL_REPOS.length).toBeGreaterThanOrEqual(3);
63
+ });
64
+ it("SKILLS_INDEX_PATH returns ~/.cache/otto/skills-index.json", () => {
65
+ const home = process.env.HOME || process.env.USERPROFILE || "/root";
66
+ expect(SKILLS_INDEX_PATH()).toContain(".cache/otto/skills-index.json");
67
+ });
68
+ it("OPENCODE_SKILLS_DIR returns ~/.config/opencode/skills", () => {
69
+ expect(OPENCODE_SKILLS_DIR()).toContain(".config/opencode/skills");
70
+ });
71
+ });
72
+ // ---------------------------------------------------------------------------
73
+ // Parser
74
+ // ---------------------------------------------------------------------------
75
+ describe("parseSkillMd", () => {
76
+ it("parses valid SKILL.md with required fields only", () => {
77
+ const result = parseSkillMd(VALID_SKILL_MD);
78
+ expect(result).not.toBeNull();
79
+ expect(result.name).toBe("my-skill");
80
+ expect(result.description).toBe("A test skill");
81
+ expect(result.metadata).toBeUndefined();
82
+ });
83
+ it("parses valid SKILL.md with optional metadata", () => {
84
+ const result = parseSkillMd(VALID_SKILL_MD_WITH_META);
85
+ expect(result).not.toBeNull();
86
+ expect(result.name).toBe("my-skill");
87
+ expect(result.description).toBe("A test skill with metadata");
88
+ expect(result.metadata).toEqual({ author: "otto", version: "1.0" });
89
+ });
90
+ it("returns null for missing frontmatter", () => {
91
+ expect(parseSkillMd(NO_FRONTMATTER_MD)).toBeNull();
92
+ });
93
+ it("returns null for missing name", () => {
94
+ expect(parseSkillMd(MISSING_NAME_MD)).toBeNull();
95
+ });
96
+ it("returns null for missing description", () => {
97
+ expect(parseSkillMd(MISSING_DESC_MD)).toBeNull();
98
+ });
99
+ it("returns null for empty string", () => {
100
+ expect(parseSkillMd("")).toBeNull();
101
+ });
102
+ });
103
+ // ---------------------------------------------------------------------------
104
+ // Local skill discovery
105
+ // ---------------------------------------------------------------------------
106
+ describe("listInstalledSkills", () => {
107
+ let tmpBase;
108
+ beforeEach(() => {
109
+ tmpBase = makeTmp();
110
+ });
111
+ afterEach(() => {
112
+ fs.rmSync(tmpBase, { recursive: true, force: true });
113
+ });
114
+ it("lists skills from given base dir", () => {
115
+ writeSkillMd(path.join(tmpBase, "installed-skill"), VALID_SKILL_MD);
116
+ const skills = listInstalledSkills(tmpBase);
117
+ expect(skills).toHaveLength(1);
118
+ expect(skills[0].name).toBe("my-skill");
119
+ });
120
+ it("returns empty array for non-existent dir", () => {
121
+ const skills = listInstalledSkills(path.join(tmpBase, "nonexistent"));
122
+ expect(skills).toEqual([]);
123
+ });
124
+ });
125
+ // ---------------------------------------------------------------------------
126
+ // Remove skill
127
+ // ---------------------------------------------------------------------------
128
+ describe("removeSkill", () => {
129
+ let tmpTarget;
130
+ beforeEach(() => {
131
+ tmpTarget = makeTmp();
132
+ });
133
+ afterEach(() => {
134
+ fs.rmSync(tmpTarget, { recursive: true, force: true });
135
+ });
136
+ it("deletes installed skill", () => {
137
+ writeSkillMd(path.join(tmpTarget, "remove-me"), VALID_SKILL_MD);
138
+ expect(fs.existsSync(path.join(tmpTarget, "remove-me", "SKILL.md"))).toBe(true);
139
+ const result = removeSkill("remove-me", tmpTarget);
140
+ expect(result).toBe(true);
141
+ expect(fs.existsSync(path.join(tmpTarget, "remove-me"))).toBe(false);
142
+ });
143
+ it("returns false if not installed", () => {
144
+ const result = removeSkill("not-there", tmpTarget);
145
+ expect(result).toBe(false);
146
+ });
147
+ });
148
+ // ---------------------------------------------------------------------------
149
+ // Skills Index — load, save, stale check
150
+ // ---------------------------------------------------------------------------
151
+ describe("Skills Index", () => {
152
+ let tmpDir;
153
+ beforeEach(() => {
154
+ tmpDir = makeTmp();
155
+ });
156
+ afterEach(() => {
157
+ fs.rmSync(tmpDir, { recursive: true, force: true });
158
+ });
159
+ it("loadSkillsIndex returns empty index when file missing", () => {
160
+ const indexPath = path.join(tmpDir, "skills-index.json");
161
+ const idx = loadSkillsIndex(indexPath);
162
+ expect(idx.version).toBe(1);
163
+ expect(idx.repos).toEqual({});
164
+ });
165
+ it("saveSkillsIndex + loadSkillsIndex roundtrip", () => {
166
+ const indexPath = path.join(tmpDir, "skills-index.json");
167
+ const idx = {
168
+ version: 1,
169
+ updated: new Date().toISOString(),
170
+ repos: {
171
+ "anthropics/skills": {
172
+ fetched: new Date().toISOString(),
173
+ skills: [
174
+ { name: "pdf", description: "Create PDF files", source: "anthropics/skills", path: "skills/pdf" },
175
+ ],
176
+ },
177
+ },
178
+ };
179
+ saveSkillsIndex(idx, indexPath);
180
+ const loaded = loadSkillsIndex(indexPath);
181
+ expect(loaded.repos["anthropics/skills"].skills).toHaveLength(1);
182
+ expect(loaded.repos["anthropics/skills"].skills[0].name).toBe("pdf");
183
+ });
184
+ it("isIndexStale returns true for old timestamps", () => {
185
+ const oldDate = new Date(Date.now() - 25 * 60 * 60 * 1000).toISOString();
186
+ expect(isIndexStale(oldDate, 24)).toBe(true);
187
+ expect(isIndexStale(new Date().toISOString(), 24)).toBe(false);
188
+ expect(isIndexStale("", 24)).toBe(true);
189
+ });
190
+ });
191
+ // ---------------------------------------------------------------------------
192
+ // Search
193
+ // ---------------------------------------------------------------------------
194
+ describe("searchSkills", () => {
195
+ let tmpIndex;
196
+ beforeEach(() => {
197
+ tmpIndex = path.join(makeTmp(), "skills-index.json");
198
+ });
199
+ afterEach(() => {
200
+ fs.rmSync(path.dirname(tmpIndex), { recursive: true, force: true });
201
+ });
202
+ function makeTestIndex() {
203
+ return {
204
+ version: 1,
205
+ updated: new Date().toISOString(),
206
+ repos: {
207
+ "anthropics/skills": {
208
+ fetched: new Date().toISOString(),
209
+ skills: [
210
+ { name: "frontend-design", description: "Design web frontends", source: "anthropics/skills", path: "skills/frontend-design" },
211
+ { name: "pdf", description: "Create PDF documents", source: "anthropics/skills", path: "skills/pdf" },
212
+ ],
213
+ },
214
+ "vercel-labs/agent-skills": {
215
+ fetched: new Date().toISOString(),
216
+ skills: [
217
+ { name: "react-best-practices", description: "React and Next.js performance patterns", source: "vercel-labs/agent-skills", path: "skills/react-best-practices" },
218
+ ],
219
+ },
220
+ },
221
+ };
222
+ }
223
+ it("finds skills by name substring", () => {
224
+ saveSkillsIndex(makeTestIndex(), tmpIndex);
225
+ const results = searchSkills("react", tmpIndex);
226
+ expect(results).toHaveLength(1);
227
+ expect(results[0].name).toBe("react-best-practices");
228
+ });
229
+ it("finds skills by description substring", () => {
230
+ saveSkillsIndex(makeTestIndex(), tmpIndex);
231
+ const results = searchSkills("PDF", tmpIndex);
232
+ expect(results).toHaveLength(1);
233
+ expect(results[0].name).toBe("pdf");
234
+ });
235
+ it("returns empty array for no matches", () => {
236
+ saveSkillsIndex(makeTestIndex(), tmpIndex);
237
+ const results = searchSkills("nonexistent-xyz", tmpIndex);
238
+ expect(results).toHaveLength(0);
239
+ });
240
+ it("finds skills across multiple repos", () => {
241
+ saveSkillsIndex(makeTestIndex(), tmpIndex);
242
+ const results = searchSkills("design", tmpIndex);
243
+ expect(results.length).toBeGreaterThanOrEqual(1);
244
+ });
245
+ });
246
+ // ---------------------------------------------------------------------------
247
+ // getAllIndexedSkills
248
+ // ---------------------------------------------------------------------------
249
+ describe("getAllIndexedSkills", () => {
250
+ let tmpIndex;
251
+ beforeEach(() => {
252
+ tmpIndex = path.join(makeTmp(), "skills-index.json");
253
+ });
254
+ afterEach(() => {
255
+ fs.rmSync(path.dirname(tmpIndex), { recursive: true, force: true });
256
+ });
257
+ it("returns all skills from all repos", () => {
258
+ const idx = {
259
+ version: 1,
260
+ updated: new Date().toISOString(),
261
+ repos: {
262
+ "anthropics/skills": {
263
+ fetched: new Date().toISOString(),
264
+ skills: [
265
+ { name: "pdf", description: "Create PDFs", source: "anthropics/skills", path: "skills/pdf" },
266
+ ],
267
+ },
268
+ "vercel-labs/agent-skills": {
269
+ fetched: new Date().toISOString(),
270
+ skills: [
271
+ { name: "react-best", description: "React patterns", source: "vercel-labs/agent-skills", path: "skills/react-best" },
272
+ ],
273
+ },
274
+ },
275
+ };
276
+ saveSkillsIndex(idx, tmpIndex);
277
+ const all = getAllIndexedSkills(tmpIndex);
278
+ expect(all).toHaveLength(2);
279
+ });
280
+ });
281
+ // ---------------------------------------------------------------------------
282
+ // installSkillFromIndex (unit tests — no network)
283
+ // ---------------------------------------------------------------------------
284
+ describe("installSkillFromIndex", () => {
285
+ let tmpTarget;
286
+ let tmpIndex;
287
+ beforeEach(() => {
288
+ tmpTarget = makeTmp();
289
+ tmpIndex = path.join(makeTmp(), "skills-index.json");
290
+ });
291
+ afterEach(() => {
292
+ fs.rmSync(tmpTarget, { recursive: true, force: true });
293
+ fs.rmSync(path.dirname(tmpIndex), { recursive: true, force: true });
294
+ });
295
+ it("returns false if skill not found in index", () => {
296
+ const idx = { version: 1, updated: "", repos: {} };
297
+ saveSkillsIndex(idx, tmpIndex);
298
+ const result = installSkillFromIndex("nonexistent", tmpTarget, tmpIndex);
299
+ expect(result).toBe(false);
300
+ });
301
+ });
302
+ // ---------------------------------------------------------------------------
303
+ // installSkillsBaseline
304
+ // ---------------------------------------------------------------------------
305
+ describe("installSkillsBaseline", () => {
306
+ let tmpTarget;
307
+ let tmpIndex;
308
+ beforeEach(() => {
309
+ tmpTarget = makeTmp();
310
+ tmpIndex = path.join(makeTmp(), "skills-index.json");
311
+ });
312
+ afterEach(() => {
313
+ fs.rmSync(tmpTarget, { recursive: true, force: true });
314
+ fs.rmSync(path.dirname(tmpIndex), { recursive: true, force: true });
315
+ });
316
+ it("reports installed, already present, and failed skills", () => {
317
+ // preinstall one skill
318
+ writeSkillMd(path.join(tmpTarget, "existing-skill"), `---\nname: existing-skill\ndescription: Existing\n---\n`);
319
+ const idx = {
320
+ version: 1,
321
+ updated: new Date().toISOString(),
322
+ repos: {
323
+ "otto-assistant/skills": {
324
+ fetched: new Date().toISOString(),
325
+ skills: [
326
+ {
327
+ name: "otto-subagent-threads",
328
+ description: "Enforce Discord threads",
329
+ source: "otto-assistant/skills",
330
+ path: "skills/otto-subagent-threads",
331
+ },
332
+ ],
333
+ },
334
+ },
335
+ };
336
+ saveSkillsIndex(idx, tmpIndex);
337
+ const result = installSkillsBaseline([
338
+ "existing-skill",
339
+ "otto-subagent-threads",
340
+ "missing-skill",
341
+ ], tmpTarget, tmpIndex);
342
+ expect(result.alreadyPresent).toContain("existing-skill");
343
+ if (hasGhAuth()) {
344
+ expect(result.installed).toContain("otto-subagent-threads");
345
+ }
346
+ else {
347
+ expect(result.failed).toContain("otto-subagent-threads");
348
+ }
349
+ expect(result.failed).toContain("missing-skill");
350
+ }, 20_000);
351
+ });
352
+ // ---------------------------------------------------------------------------
353
+ // Integration tests (require gh auth)
354
+ // ---------------------------------------------------------------------------
355
+ function hasGhAuth() {
356
+ try {
357
+ execFileSync("gh", ["auth", "status"], { encoding: "utf-8", stdio: "pipe" });
358
+ return true;
359
+ }
360
+ catch {
361
+ return false;
362
+ }
363
+ }
364
+ describe("GitHub API integration", () => {
365
+ it("fetchRepoSkillsIndex indexes skills from otto-assistant/skills", () => {
366
+ if (!hasGhAuth())
367
+ return;
368
+ const entries = fetchRepoSkillsIndex("otto-assistant/skills");
369
+ expect(entries.length).toBeGreaterThanOrEqual(3);
370
+ for (const entry of entries) {
371
+ expect(entry.name).toBeTruthy();
372
+ expect(entry.description).toBeTruthy();
373
+ expect(entry.source).toBe("otto-assistant/skills");
374
+ }
375
+ }, 15_000);
376
+ it("fetchRepoSkillsIndex indexes skills from anthropics/skills", () => {
377
+ if (!hasGhAuth())
378
+ return;
379
+ const entries = fetchRepoSkillsIndex("anthropics/skills");
380
+ expect(entries.length).toBeGreaterThanOrEqual(5);
381
+ for (const entry of entries) {
382
+ expect(entry.name).toBeTruthy();
383
+ expect(entry.description).toBeTruthy();
384
+ }
385
+ }, 30_000);
386
+ it("installSkillFromIndex installs a real skill", () => {
387
+ if (!hasGhAuth())
388
+ return;
389
+ const tmpTarget = makeTmp();
390
+ const tmpIdx = path.join(makeTmp(), "skills-index.json");
391
+ try {
392
+ const idx = {
393
+ version: 1,
394
+ updated: new Date().toISOString(),
395
+ repos: {
396
+ "otto-assistant/skills": {
397
+ fetched: new Date().toISOString(),
398
+ skills: [
399
+ { name: "otto-subagent-threads", description: "Enforce Discord threads", source: "otto-assistant/skills", path: "skills/otto-subagent-threads" },
400
+ ],
401
+ },
402
+ },
403
+ };
404
+ saveSkillsIndex(idx, tmpIdx);
405
+ const result = installSkillFromIndex("otto-subagent-threads", tmpTarget, tmpIdx);
406
+ expect(result).toBe(true);
407
+ const skillMd = fs.readFileSync(path.join(tmpTarget, "otto-subagent-threads", "SKILL.md"), "utf-8");
408
+ expect(skillMd).toContain("name: otto-subagent-threads");
409
+ }
410
+ finally {
411
+ fs.rmSync(tmpTarget, { recursive: true, force: true });
412
+ fs.rmSync(path.dirname(tmpIdx), { recursive: true, force: true });
413
+ }
414
+ }, 15_000);
415
+ });
416
+ //# sourceMappingURL=skills.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"skills.test.js","sourceRoot":"","sources":["../src/skills.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,13 @@
1
+ export type TenantMode = "safe" | "admin";
2
+ export interface TenantInitResult {
3
+ created: string[];
4
+ }
5
+ export declare function deriveComposeProjectName(tenantPath: string): string;
6
+ export declare function resolveTenantImage(input: {
7
+ composeImage: string;
8
+ envImage?: string;
9
+ }): string;
10
+ export declare function resolveTenantMode(mode?: string): TenantMode;
11
+ export declare function ensureTenantMemoryLayout(memoryPath: string): TenantInitResult;
12
+ export declare function ensureTenantScaffold(tenantPath: string): TenantInitResult;
13
+ //# sourceMappingURL=tenant.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tenant.d.ts","sourceRoot":"","sources":["../src/tenant.ts"],"names":[],"mappings":"AAGA,MAAM,MAAM,UAAU,GAAG,MAAM,GAAG,OAAO,CAAA;AAEzC,MAAM,WAAW,gBAAgB;IAC/B,OAAO,EAAE,MAAM,EAAE,CAAA;CAClB;AAQD,wBAAgB,wBAAwB,CAAC,UAAU,EAAE,MAAM,GAAG,MAAM,CAEnE;AAED,wBAAgB,kBAAkB,CAAC,KAAK,EAAE;IAAE,YAAY,EAAE,MAAM,CAAC;IAAC,QAAQ,CAAC,EAAE,MAAM,CAAA;CAAE,GAAG,MAAM,CAE7F;AAED,wBAAgB,iBAAiB,CAAC,IAAI,CAAC,EAAE,MAAM,GAAG,UAAU,CAE3D;AAcD,wBAAgB,wBAAwB,CAAC,UAAU,EAAE,MAAM,GAAG,gBAAgB,CAmB7E;AAED,wBAAgB,oBAAoB,CAAC,UAAU,EAAE,MAAM,GAAG,gBAAgB,CA2DzE"}
package/dist/tenant.js ADDED
@@ -0,0 +1,105 @@
1
+ import fs from "node:fs";
2
+ import path from "node:path";
3
+ function getLeafName(inputPath) {
4
+ const normalized = path.resolve(inputPath);
5
+ const base = path.basename(normalized);
6
+ return base || "tenant";
7
+ }
8
+ export function deriveComposeProjectName(tenantPath) {
9
+ return `otto-${getLeafName(tenantPath)}`;
10
+ }
11
+ export function resolveTenantImage(input) {
12
+ return input.envImage?.trim() || input.composeImage;
13
+ }
14
+ export function resolveTenantMode(mode) {
15
+ return mode?.trim().toLowerCase() === "admin" ? "admin" : "safe";
16
+ }
17
+ function ensureDir(dirPath) {
18
+ if (fs.existsSync(dirPath))
19
+ return false;
20
+ fs.mkdirSync(dirPath, { recursive: true });
21
+ return true;
22
+ }
23
+ function ensureFile(filePath) {
24
+ if (fs.existsSync(filePath))
25
+ return false;
26
+ fs.writeFileSync(filePath, "", "utf-8");
27
+ return true;
28
+ }
29
+ export function ensureTenantMemoryLayout(memoryPath) {
30
+ const created = [];
31
+ ensureDir(memoryPath);
32
+ const mempalacePath = path.join(memoryPath, "mempalace");
33
+ if (ensureDir(mempalacePath))
34
+ created.push("mempalace/");
35
+ const kimakiPath = path.join(memoryPath, "kimaki");
36
+ if (ensureDir(kimakiPath))
37
+ created.push("kimaki/");
38
+ const opencodePath = path.join(memoryPath, "opencode");
39
+ if (ensureDir(opencodePath))
40
+ created.push("opencode/");
41
+ for (const fileName of ["AGENTS.md", "soul.md", "persona.md"]) {
42
+ const filePath = path.join(memoryPath, fileName);
43
+ if (ensureFile(filePath))
44
+ created.push(fileName);
45
+ }
46
+ return { created };
47
+ }
48
+ export function ensureTenantScaffold(tenantPath) {
49
+ const created = [];
50
+ ensureDir(tenantPath);
51
+ const projectsPath = path.join(tenantPath, "projects");
52
+ if (ensureDir(projectsPath))
53
+ created.push("projects/");
54
+ const memoryPath = path.join(tenantPath, "memory");
55
+ if (ensureDir(memoryPath))
56
+ created.push("memory/");
57
+ const memoryResult = ensureTenantMemoryLayout(memoryPath);
58
+ created.push(...memoryResult.created);
59
+ const composePath = path.join(tenantPath, "compose.yml");
60
+ if (ensureFile(composePath)) {
61
+ const projectName = deriveComposeProjectName(tenantPath);
62
+ const compose = [
63
+ "services:",
64
+ " otto:",
65
+ " image: ghcr.io/otto-assistant/otto:stable",
66
+ " env_file:",
67
+ " - .env",
68
+ " volumes:",
69
+ " - ./projects:/workspace/projects",
70
+ " - ./memory:/workspace/memory",
71
+ " - ./memory/kimaki:/root/.kimaki",
72
+ " - ./memory/opencode:/root/.config/opencode",
73
+ " working_dir: /workspace",
74
+ " restart: unless-stopped",
75
+ "",
76
+ ].join("\n");
77
+ fs.writeFileSync(composePath, compose, "utf-8");
78
+ created.push("compose.yml");
79
+ const envPath = path.join(tenantPath, ".env");
80
+ const envContent = [
81
+ `COMPOSE_PROJECT_NAME=${projectName}`,
82
+ "",
83
+ "# ===== Required =====",
84
+ "# Discord bot token — https://discord.com/developers/applications",
85
+ "# Kimaki reads this env var in non-interactive/headless mode",
86
+ "KIMAKI_BOT_TOKEN=",
87
+ "",
88
+ "# AI provider (choose one)",
89
+ "# OPENAI_API_KEY=sk-...",
90
+ "# ANTHROPIC_API_KEY=sk-ant-...",
91
+ "# GOOGLE_API_KEY=...",
92
+ "",
93
+ "# ===== Optional =====",
94
+ "# Runtime mode: safe (default) or admin (elevated host access)",
95
+ "OTTO_MODE=safe",
96
+ "",
97
+ ].join("\n");
98
+ if (ensureFile(envPath)) {
99
+ fs.writeFileSync(envPath, envContent, "utf-8");
100
+ created.push(".env");
101
+ }
102
+ }
103
+ return { created };
104
+ }
105
+ //# sourceMappingURL=tenant.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tenant.js","sourceRoot":"","sources":["../src/tenant.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=tenant.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tenant.test.d.ts","sourceRoot":"","sources":["../src/tenant.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,37 @@
1
+ import fs from "node:fs";
2
+ import os from "node:os";
3
+ import path from "node:path";
4
+ import { describe, expect, it } from "vitest";
5
+ import { deriveComposeProjectName, ensureTenantMemoryLayout, ensureTenantScaffold, resolveTenantImage, resolveTenantMode, } from "./tenant.js";
6
+ describe("tenant", () => {
7
+ it("uses otto-<folder_name> default", () => {
8
+ expect(deriveComposeProjectName("/tmp/my-tenant")).toBe("otto-my-tenant");
9
+ });
10
+ it("uses image from compose.yml when env override missing", () => {
11
+ const resolved = resolveTenantImage({
12
+ composeImage: "otto-assistant/otto:stable",
13
+ envImage: undefined,
14
+ });
15
+ expect(resolved).toBe("otto-assistant/otto:stable");
16
+ });
17
+ it("defaults to safe mode when OTTO_MODE is unset", () => {
18
+ expect(resolveTenantMode(undefined)).toBe("safe");
19
+ });
20
+ it("creates required memory files and mempalace dir", () => {
21
+ const root = fs.mkdtempSync(path.join(os.tmpdir(), "otto-memory-"));
22
+ const report = ensureTenantMemoryLayout(root);
23
+ expect(report.created).toContain("AGENTS.md");
24
+ expect(report.created).toContain("soul.md");
25
+ expect(report.created).toContain("persona.md");
26
+ expect(report.created).toContain("mempalace/");
27
+ });
28
+ it("creates compose scaffold for tenant", () => {
29
+ const tenantPath = fs.mkdtempSync(path.join(os.tmpdir(), "otto-tenant-"));
30
+ const report = ensureTenantScaffold(tenantPath);
31
+ expect(report.created).toContain("compose.yml");
32
+ expect(fs.existsSync(path.join(tenantPath, "compose.yml"))).toBe(true);
33
+ expect(fs.existsSync(path.join(tenantPath, "memory", "AGENTS.md"))).toBe(true);
34
+ expect(fs.existsSync(path.join(tenantPath, "projects"))).toBe(true);
35
+ });
36
+ });
37
+ //# sourceMappingURL=tenant.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tenant.test.js","sourceRoot":"","sources":["../src/tenant.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,SAAS,CAAA;AACxB,OAAO,EAAE,MAAM,SAAS,CAAA;AACxB,OAAO,IAAI,MAAM,WAAW,CAAA;AAC5B,OAAO,EAAE,QAAQ,EAAE,MAAM,EAAE,EAAE,EAAE,MAAM,QAAQ,CAAA;AAC7C,OAAO,EACL,wBAAwB,EACxB,wBAAwB,EACxB,oBAAoB,EACpB,kBAAkB,EAClB,iBAAiB,GAClB,MAAM,aAAa,CAAA;AAEpB,QAAQ,CAAC,QAAQ,EAAE,GAAG,EAAE;IACtB,EAAE,CAAC,iCAAiC,EAAE,GAAG,EAAE;QACzC,MAAM,CAAC,wBAAwB,CAAC,gBAAgB,CAAC,CAAC,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAA;IAC3E,CAAC,CAAC,CAAA;IAEF,EAAE,CAAC,uDAAuD,EAAE,GAAG,EAAE;QAC/D,MAAM,QAAQ,GAAG,kBAAkB,CAAC;YAClC,YAAY,EAAE,4BAA4B;YAC1C,QAAQ,EAAE,SAAS;SACpB,CAAC,CAAA;QACF,MAAM,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,4BAA4B,CAAC,CAAA;IACrD,CAAC,CAAC,CAAA;IAEF,EAAE,CAAC,+CAA+C,EAAE,GAAG,EAAE;QACvD,MAAM,CAAC,iBAAiB,CAAC,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,CAAA;IACnD,CAAC,CAAC,CAAA;IAEF,EAAE,CAAC,iDAAiD,EAAE,GAAG,EAAE;QACzD,MAAM,IAAI,GAAG,EAAE,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,MAAM,EAAE,EAAE,cAAc,CAAC,CAAC,CAAA;QACnE,MAAM,MAAM,GAAG,wBAAwB,CAAC,IAAI,CAAC,CAAA;QAE7C,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,SAAS,CAAC,WAAW,CAAC,CAAA;QAC7C,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,SAAS,CAAC,SAAS,CAAC,CAAA;QAC3C,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,SAAS,CAAC,YAAY,CAAC,CAAA;QAC9C,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,SAAS,CAAC,YAAY,CAAC,CAAA;IAChD,CAAC,CAAC,CAAA;IAEF,EAAE,CAAC,qCAAqC,EAAE,GAAG,EAAE;QAC7C,MAAM,UAAU,GAAG,EAAE,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,MAAM,EAAE,EAAE,cAAc,CAAC,CAAC,CAAA;QACzE,MAAM,MAAM,GAAG,oBAAoB,CAAC,UAAU,CAAC,CAAA;QAE/C,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,SAAS,CAAC,aAAa,CAAC,CAAA;QAC/C,MAAM,CAAC,EAAE,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,aAAa,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;QACtE,MAAM,CAAC,EAAE,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,QAAQ,EAAE,WAAW,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;QAC9E,MAAM,CAAC,EAAE,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,UAAU,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;IACrE,CAAC,CAAC,CAAA;AACJ,CAAC,CAAC,CAAA"}