jinn-cli 0.1.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.
Files changed (227) hide show
  1. package/dist/bin/jimmy.d.ts +3 -0
  2. package/dist/bin/jimmy.d.ts.map +1 -0
  3. package/dist/bin/jimmy.js +148 -0
  4. package/dist/bin/jimmy.js.map +1 -0
  5. package/dist/src/cli/chrome-allow.d.ts +5 -0
  6. package/dist/src/cli/chrome-allow.d.ts.map +1 -0
  7. package/dist/src/cli/chrome-allow.js +241 -0
  8. package/dist/src/cli/chrome-allow.js.map +1 -0
  9. package/dist/src/cli/create.d.ts +2 -0
  10. package/dist/src/cli/create.d.ts.map +1 -0
  11. package/dist/src/cli/create.js +72 -0
  12. package/dist/src/cli/create.js.map +1 -0
  13. package/dist/src/cli/instances.d.ts +14 -0
  14. package/dist/src/cli/instances.d.ts.map +1 -0
  15. package/dist/src/cli/instances.js +43 -0
  16. package/dist/src/cli/instances.js.map +1 -0
  17. package/dist/src/cli/list.d.ts +2 -0
  18. package/dist/src/cli/list.d.ts.map +1 -0
  19. package/dist/src/cli/list.js +38 -0
  20. package/dist/src/cli/list.js.map +1 -0
  21. package/dist/src/cli/migrate.d.ts +5 -0
  22. package/dist/src/cli/migrate.d.ts.map +1 -0
  23. package/dist/src/cli/migrate.js +203 -0
  24. package/dist/src/cli/migrate.js.map +1 -0
  25. package/dist/src/cli/nuke.d.ts +2 -0
  26. package/dist/src/cli/nuke.d.ts.map +1 -0
  27. package/dist/src/cli/nuke.js +91 -0
  28. package/dist/src/cli/nuke.js.map +1 -0
  29. package/dist/src/cli/remove.d.ts +4 -0
  30. package/dist/src/cli/remove.d.ts.map +1 -0
  31. package/dist/src/cli/remove.js +47 -0
  32. package/dist/src/cli/remove.js.map +1 -0
  33. package/dist/src/cli/setup.d.ts +4 -0
  34. package/dist/src/cli/setup.d.ts.map +1 -0
  35. package/dist/src/cli/setup.js +483 -0
  36. package/dist/src/cli/setup.js.map +1 -0
  37. package/dist/src/cli/skills.d.ts +28 -0
  38. package/dist/src/cli/skills.d.ts.map +1 -0
  39. package/dist/src/cli/skills.js +284 -0
  40. package/dist/src/cli/skills.js.map +1 -0
  41. package/dist/src/cli/start.d.ts +5 -0
  42. package/dist/src/cli/start.d.ts.map +1 -0
  43. package/dist/src/cli/start.js +34 -0
  44. package/dist/src/cli/start.js.map +1 -0
  45. package/dist/src/cli/status.d.ts +2 -0
  46. package/dist/src/cli/status.d.ts.map +1 -0
  47. package/dist/src/cli/status.js +60 -0
  48. package/dist/src/cli/status.js.map +1 -0
  49. package/dist/src/cli/stop.d.ts +2 -0
  50. package/dist/src/cli/stop.d.ts.map +1 -0
  51. package/dist/src/cli/stop.js +11 -0
  52. package/dist/src/cli/stop.js.map +1 -0
  53. package/dist/src/connectors/slack/format.d.ts +10 -0
  54. package/dist/src/connectors/slack/format.d.ts.map +1 -0
  55. package/dist/src/connectors/slack/format.js +55 -0
  56. package/dist/src/connectors/slack/format.js.map +1 -0
  57. package/dist/src/connectors/slack/index.d.ts +18 -0
  58. package/dist/src/connectors/slack/index.d.ts.map +1 -0
  59. package/dist/src/connectors/slack/index.js +122 -0
  60. package/dist/src/connectors/slack/index.js.map +1 -0
  61. package/dist/src/connectors/slack/threads.d.ts +2 -0
  62. package/dist/src/connectors/slack/threads.d.ts.map +1 -0
  63. package/dist/src/connectors/slack/threads.js +10 -0
  64. package/dist/src/connectors/slack/threads.js.map +1 -0
  65. package/dist/src/cron/jobs.d.ts +5 -0
  66. package/dist/src/cron/jobs.d.ts.map +1 -0
  67. package/dist/src/cron/jobs.js +23 -0
  68. package/dist/src/cron/jobs.js.map +1 -0
  69. package/dist/src/cron/runner.d.ts +3 -0
  70. package/dist/src/cron/runner.d.ts.map +1 -0
  71. package/dist/src/cron/runner.js +118 -0
  72. package/dist/src/cron/runner.js.map +1 -0
  73. package/dist/src/cron/scheduler.d.ts +5 -0
  74. package/dist/src/cron/scheduler.d.ts.map +1 -0
  75. package/dist/src/cron/scheduler.js +39 -0
  76. package/dist/src/cron/scheduler.js.map +1 -0
  77. package/dist/src/engines/claude.d.ts +14 -0
  78. package/dist/src/engines/claude.d.ts.map +1 -0
  79. package/dist/src/engines/claude.js +264 -0
  80. package/dist/src/engines/claude.js.map +1 -0
  81. package/dist/src/engines/codex.d.ts +15 -0
  82. package/dist/src/engines/codex.d.ts.map +1 -0
  83. package/dist/src/engines/codex.js +346 -0
  84. package/dist/src/engines/codex.js.map +1 -0
  85. package/dist/src/gateway/api.d.ts +13 -0
  86. package/dist/src/gateway/api.d.ts.map +1 -0
  87. package/dist/src/gateway/api.js +819 -0
  88. package/dist/src/gateway/api.js.map +1 -0
  89. package/dist/src/gateway/daemon-entry.d.ts +2 -0
  90. package/dist/src/gateway/daemon-entry.d.ts.map +1 -0
  91. package/dist/src/gateway/daemon-entry.js +12 -0
  92. package/dist/src/gateway/daemon-entry.js.map +1 -0
  93. package/dist/src/gateway/lifecycle.d.ts +10 -0
  94. package/dist/src/gateway/lifecycle.d.ts.map +1 -0
  95. package/dist/src/gateway/lifecycle.js +124 -0
  96. package/dist/src/gateway/lifecycle.js.map +1 -0
  97. package/dist/src/gateway/org.d.ts +10 -0
  98. package/dist/src/gateway/org.d.ts.map +1 -0
  99. package/dist/src/gateway/org.js +71 -0
  100. package/dist/src/gateway/org.js.map +1 -0
  101. package/dist/src/gateway/server.d.ts +4 -0
  102. package/dist/src/gateway/server.d.ts.map +1 -0
  103. package/dist/src/gateway/server.js +301 -0
  104. package/dist/src/gateway/server.js.map +1 -0
  105. package/dist/src/gateway/watcher.d.ts +14 -0
  106. package/dist/src/gateway/watcher.d.ts.map +1 -0
  107. package/dist/src/gateway/watcher.js +104 -0
  108. package/dist/src/gateway/watcher.js.map +1 -0
  109. package/dist/src/index.d.ts +5 -0
  110. package/dist/src/index.d.ts.map +1 -0
  111. package/dist/src/index.js +4 -0
  112. package/dist/src/index.js.map +1 -0
  113. package/dist/src/sessions/context.d.ts +20 -0
  114. package/dist/src/sessions/context.d.ts.map +1 -0
  115. package/dist/src/sessions/context.js +532 -0
  116. package/dist/src/sessions/context.js.map +1 -0
  117. package/dist/src/sessions/manager.d.ts +38 -0
  118. package/dist/src/sessions/manager.d.ts.map +1 -0
  119. package/dist/src/sessions/manager.js +208 -0
  120. package/dist/src/sessions/manager.js.map +1 -0
  121. package/dist/src/sessions/queue.d.ts +14 -0
  122. package/dist/src/sessions/queue.d.ts.map +1 -0
  123. package/dist/src/sessions/queue.js +42 -0
  124. package/dist/src/sessions/queue.js.map +1 -0
  125. package/dist/src/sessions/registry.d.ts +46 -0
  126. package/dist/src/sessions/registry.d.ts.map +1 -0
  127. package/dist/src/sessions/registry.js +193 -0
  128. package/dist/src/sessions/registry.js.map +1 -0
  129. package/dist/src/shared/config.d.ts +3 -0
  130. package/dist/src/shared/config.d.ts.map +1 -0
  131. package/dist/src/shared/config.js +11 -0
  132. package/dist/src/shared/config.js.map +1 -0
  133. package/dist/src/shared/logger.d.ts +12 -0
  134. package/dist/src/shared/logger.d.ts.map +1 -0
  135. package/dist/src/shared/logger.js +35 -0
  136. package/dist/src/shared/logger.js.map +1 -0
  137. package/dist/src/shared/paths.d.ts +19 -0
  138. package/dist/src/shared/paths.d.ts.map +1 -0
  139. package/dist/src/shared/paths.js +31 -0
  140. package/dist/src/shared/paths.js.map +1 -0
  141. package/dist/src/shared/types.d.ts +166 -0
  142. package/dist/src/shared/types.d.ts.map +1 -0
  143. package/dist/src/shared/types.js +4 -0
  144. package/dist/src/shared/types.js.map +1 -0
  145. package/dist/src/shared/version.d.ts +15 -0
  146. package/dist/src/shared/version.d.ts.map +1 -0
  147. package/dist/src/shared/version.js +56 -0
  148. package/dist/src/shared/version.js.map +1 -0
  149. package/dist/web/404.html +1 -0
  150. package/dist/web/_next/static/chunks/198-fd91406a158c5c25.js +1 -0
  151. package/dist/web/_next/static/chunks/517.62389e8d3c929c43.js +1 -0
  152. package/dist/web/_next/static/chunks/534-17c49c944e0d0fe1.js +1 -0
  153. package/dist/web/_next/static/chunks/573-070537ec2452d03e.js +1 -0
  154. package/dist/web/_next/static/chunks/590-2c34156c7417317e.js +1 -0
  155. package/dist/web/_next/static/chunks/704-af2893821e1d18dc.js +1 -0
  156. package/dist/web/_next/static/chunks/7273c211.06e3b6021d90b73f.js +1 -0
  157. package/dist/web/_next/static/chunks/73-c226535579393e21.js +1 -0
  158. package/dist/web/_next/static/chunks/743-5bb03adbb0e4ddec.js +1 -0
  159. package/dist/web/_next/static/chunks/874.97d5a27895061057.js +1 -0
  160. package/dist/web/_next/static/chunks/8e6518bb-c26e82767f1faf66.js +1 -0
  161. package/dist/web/_next/static/chunks/app/_not-found/page-bb075b0779827928.js +1 -0
  162. package/dist/web/_next/static/chunks/app/chat/page-6d5bc707a45c92c6.js +1 -0
  163. package/dist/web/_next/static/chunks/app/costs/page-d6c03718defdb599.js +1 -0
  164. package/dist/web/_next/static/chunks/app/cron/page-4c563eef2b6231fe.js +1 -0
  165. package/dist/web/_next/static/chunks/app/kanban/page-55a73165a36f4077.js +1 -0
  166. package/dist/web/_next/static/chunks/app/layout-5129b67d5f126cf0.js +1 -0
  167. package/dist/web/_next/static/chunks/app/logs/page-e18889d67e48c9c9.js +1 -0
  168. package/dist/web/_next/static/chunks/app/org/page-d5cd8d9b7864737b.js +1 -0
  169. package/dist/web/_next/static/chunks/app/page-b81992940fd1dbc6.js +1 -0
  170. package/dist/web/_next/static/chunks/app/sessions/page-2eef6ac7882a28ba.js +1 -0
  171. package/dist/web/_next/static/chunks/app/settings/page-4fb01b9b09500170.js +1 -0
  172. package/dist/web/_next/static/chunks/app/skills/page-df9465e314561bb5.js +1 -0
  173. package/dist/web/_next/static/chunks/framework-077b27ad7787463c.js +1 -0
  174. package/dist/web/_next/static/chunks/main-app-437f51faf74fbb3b.js +1 -0
  175. package/dist/web/_next/static/chunks/main-f1c74cefd4965abf.js +1 -0
  176. package/dist/web/_next/static/chunks/pages/_app-77a85fe7d6bca671.js +1 -0
  177. package/dist/web/_next/static/chunks/pages/_error-68febf4b34900064.js +1 -0
  178. package/dist/web/_next/static/chunks/polyfills-42372ed130431b0a.js +1 -0
  179. package/dist/web/_next/static/chunks/webpack-0f39b7e91dce9791.js +1 -0
  180. package/dist/web/_next/static/css/4a6a5bca9238c104.css +1 -0
  181. package/dist/web/_next/static/vLvOwhC8JocJzSHTHKKOv/_buildManifest.js +1 -0
  182. package/dist/web/_next/static/vLvOwhC8JocJzSHTHKKOv/_ssgManifest.js +1 -0
  183. package/dist/web/chat.html +1 -0
  184. package/dist/web/chat.txt +20 -0
  185. package/dist/web/costs.html +16 -0
  186. package/dist/web/costs.txt +20 -0
  187. package/dist/web/cron.html +1 -0
  188. package/dist/web/cron.txt +20 -0
  189. package/dist/web/index.html +1 -0
  190. package/dist/web/index.txt +20 -0
  191. package/dist/web/kanban.html +1 -0
  192. package/dist/web/kanban.txt +20 -0
  193. package/dist/web/logs.html +7 -0
  194. package/dist/web/logs.txt +20 -0
  195. package/dist/web/org.html +1 -0
  196. package/dist/web/org.txt +20 -0
  197. package/dist/web/sessions.html +1 -0
  198. package/dist/web/sessions.txt +20 -0
  199. package/dist/web/settings.html +1 -0
  200. package/dist/web/settings.txt +20 -0
  201. package/dist/web/skills.html +1 -0
  202. package/dist/web/skills.txt +20 -0
  203. package/package.json +43 -0
  204. package/template/AGENTS.md +167 -0
  205. package/template/CLAUDE.md +106 -0
  206. package/template/config.default.yaml +27 -0
  207. package/template/docs/architecture.md +74 -0
  208. package/template/docs/connectors.md +72 -0
  209. package/template/docs/cron.md +137 -0
  210. package/template/docs/org.md +105 -0
  211. package/template/docs/overview.md +39 -0
  212. package/template/docs/self-modification.md +65 -0
  213. package/template/docs/skills.md +58 -0
  214. package/template/knowledge/.gitkeep +0 -0
  215. package/template/migrations/.gitkeep +0 -0
  216. package/template/migrations/0.1.0/MIGRATION.md +25 -0
  217. package/template/skills/cron-manager/SKILL.md +127 -0
  218. package/template/skills/find-and-install/SKILL.md +92 -0
  219. package/template/skills/management/SKILL.md +203 -0
  220. package/template/skills/migrate/SKILL.md +154 -0
  221. package/template/skills/new/SKILL.md +19 -0
  222. package/template/skills/onboarding/SKILL.md +106 -0
  223. package/template/skills/self-heal/SKILL.md +114 -0
  224. package/template/skills/skill-creator/SKILL.md +112 -0
  225. package/template/skills/status/SKILL.md +19 -0
  226. package/template/skills/sync/SKILL.md +67 -0
  227. package/template/skills.json +3 -0
@@ -0,0 +1,301 @@
1
+ import http from "node:http";
2
+ import fs from "node:fs";
3
+ import path from "node:path";
4
+ import { fileURLToPath } from "node:url";
5
+ import { randomUUID } from "node:crypto";
6
+ import { WebSocketServer } from "ws";
7
+ import { loadConfig } from "../shared/config.js";
8
+ import { configureLogger, logger } from "../shared/logger.js";
9
+ import { initDb, recoverStaleSessions } from "../sessions/registry.js";
10
+ import { SessionManager } from "../sessions/manager.js";
11
+ import { ClaudeEngine } from "../engines/claude.js";
12
+ import { CodexEngine } from "../engines/codex.js";
13
+ import { handleApiRequest } from "./api.js";
14
+ import { startWatchers, stopWatchers, syncSkillSymlinks } from "./watcher.js";
15
+ import { SlackConnector } from "../connectors/slack/index.js";
16
+ import { loadJobs } from "../cron/jobs.js";
17
+ import { startScheduler, reloadScheduler, stopScheduler } from "../cron/scheduler.js";
18
+ import { scanOrg } from "./org.js";
19
+ const __filename = fileURLToPath(import.meta.url);
20
+ const __dirname = path.dirname(__filename);
21
+ const MIME_TYPES = {
22
+ ".html": "text/html",
23
+ ".js": "application/javascript",
24
+ ".css": "text/css",
25
+ ".json": "application/json",
26
+ ".png": "image/png",
27
+ ".svg": "image/svg+xml",
28
+ ".ico": "image/x-icon",
29
+ ".woff": "font/woff",
30
+ ".woff2": "font/woff2",
31
+ };
32
+ function serveStatic(req, res, webDir) {
33
+ if (!fs.existsSync(webDir))
34
+ return false;
35
+ // Strip query string before resolving file path
36
+ const urlPath = (req.url || "/").split("?")[0];
37
+ let filePath = path.join(webDir, urlPath);
38
+ if (filePath.endsWith("/"))
39
+ filePath = path.join(filePath, "index.html");
40
+ // Prevent directory traversal
41
+ const resolved = path.resolve(filePath);
42
+ if (!resolved.startsWith(path.resolve(webDir))) {
43
+ res.writeHead(403);
44
+ res.end("Forbidden");
45
+ return true;
46
+ }
47
+ if (!fs.existsSync(resolved) || fs.statSync(resolved).isDirectory()) {
48
+ // Next.js static export produces /chat.html, /sessions.html, etc.
49
+ // Try appending .html before falling back to index.html
50
+ const htmlPath = resolved.endsWith("/")
51
+ ? path.join(resolved, "index.html")
52
+ : resolved + ".html";
53
+ if (fs.existsSync(htmlPath) && !fs.statSync(htmlPath).isDirectory()) {
54
+ res.writeHead(200, { "Content-Type": "text/html" });
55
+ fs.createReadStream(htmlPath).pipe(res);
56
+ return true;
57
+ }
58
+ // SPA fallback: serve index.html for non-API, non-WS routes
59
+ const indexPath = path.join(webDir, "index.html");
60
+ if (fs.existsSync(indexPath)) {
61
+ res.writeHead(200, { "Content-Type": "text/html" });
62
+ fs.createReadStream(indexPath).pipe(res);
63
+ return true;
64
+ }
65
+ return false;
66
+ }
67
+ const ext = path.extname(resolved);
68
+ const contentType = MIME_TYPES[ext] || "application/octet-stream";
69
+ res.writeHead(200, { "Content-Type": contentType });
70
+ fs.createReadStream(resolved).pipe(res);
71
+ return true;
72
+ }
73
+ export async function startGateway(config) {
74
+ const bootId = randomUUID().slice(0, 8);
75
+ // Configure logging
76
+ configureLogger({
77
+ level: config.logging.level,
78
+ stdout: config.logging.stdout,
79
+ file: config.logging.file,
80
+ });
81
+ const gatewayName = config.portal?.portalName || "Jinn";
82
+ logger.info(`Starting ${gatewayName} gateway (boot ${bootId}, pid ${process.pid})...`);
83
+ // Initialize database and recover any sessions stuck from a previous run
84
+ initDb();
85
+ const recovered = recoverStaleSessions();
86
+ if (recovered > 0) {
87
+ logger.info(`Recovered ${recovered} stale session(s) stuck in "running" state`);
88
+ }
89
+ // Set up engines
90
+ const claudeEngine = new ClaudeEngine();
91
+ const codexEngine = new CodexEngine();
92
+ const engines = new Map();
93
+ engines.set("claude", claudeEngine);
94
+ engines.set("codex", codexEngine);
95
+ // Derive connector names from config
96
+ const connectorNames = [];
97
+ if (config.connectors?.slack?.appToken && config.connectors?.slack?.botToken) {
98
+ connectorNames.push("slack");
99
+ }
100
+ // Session manager
101
+ const sessionManager = new SessionManager(config, engines, connectorNames);
102
+ // Build employee registry
103
+ let employeeRegistry = scanOrg();
104
+ logger.info(`Loaded ${employeeRegistry.size} employee(s) from org directory`);
105
+ // Start connectors
106
+ const connectors = [];
107
+ const connectorMap = new Map();
108
+ if (config.connectors?.slack?.appToken && config.connectors?.slack?.botToken) {
109
+ try {
110
+ const slack = new SlackConnector({
111
+ appToken: config.connectors.slack.appToken,
112
+ botToken: config.connectors.slack.botToken,
113
+ });
114
+ slack.onMessage((msg) => {
115
+ // Always route to COO. Employee mentions are detected by the COO
116
+ // in the message text and delegated via child sessions.
117
+ sessionManager.route(msg, slack).catch((err) => {
118
+ logger.error(`Slack route error: ${err instanceof Error ? err.message : err}`);
119
+ });
120
+ });
121
+ await slack.start();
122
+ connectors.push(slack);
123
+ connectorMap.set("slack", slack);
124
+ }
125
+ catch (err) {
126
+ logger.error(`Failed to start Slack connector: ${err instanceof Error ? err.message : err}`);
127
+ }
128
+ }
129
+ // Start cron scheduler
130
+ const cronJobs = loadJobs();
131
+ startScheduler(cronJobs, engines, config, connectorMap);
132
+ logger.info(`Loaded ${cronJobs.length} cron job(s)`);
133
+ // Mutable config reference for hot-reload
134
+ let currentConfig = config;
135
+ const startTime = Date.now();
136
+ // Broadcast function (defined early so apiContext can reference it)
137
+ const wsClients = new Set();
138
+ const emit = (event, payload) => {
139
+ const message = JSON.stringify({ event, payload, ts: Date.now() });
140
+ for (const client of wsClients) {
141
+ if (client.readyState === 1) {
142
+ try {
143
+ client.send(message);
144
+ }
145
+ catch (err) {
146
+ logger.warn(`WebSocket send failed, removing dead client: ${err instanceof Error ? err.message : err}`);
147
+ wsClients.delete(client);
148
+ }
149
+ }
150
+ }
151
+ };
152
+ // API context
153
+ const apiContext = {
154
+ config: currentConfig,
155
+ sessionManager,
156
+ startTime,
157
+ getConfig: () => currentConfig,
158
+ emit,
159
+ connectors: connectorMap,
160
+ };
161
+ // Resolve web UI directory — bundled into dist/web/ by postbuild script
162
+ // At runtime __dirname is dist/src/gateway/, so ../../web resolves to dist/web/
163
+ const webDir = path.resolve(__dirname, "..", "..", "web");
164
+ // Create HTTP server
165
+ const server = http.createServer((req, res) => {
166
+ const url = req.url || "/";
167
+ // CORS headers for development
168
+ res.setHeader("Access-Control-Allow-Origin", "*");
169
+ res.setHeader("Access-Control-Allow-Methods", "GET, POST, PUT, DELETE, OPTIONS");
170
+ res.setHeader("Access-Control-Allow-Headers", "Content-Type");
171
+ if (req.method === "OPTIONS") {
172
+ res.writeHead(204);
173
+ res.end();
174
+ return;
175
+ }
176
+ // API routes
177
+ if (url.startsWith("/api/")) {
178
+ handleApiRequest(req, res, apiContext);
179
+ return;
180
+ }
181
+ // Static files for web UI
182
+ if (!serveStatic(req, res, webDir)) {
183
+ if (url === "/" || url === "/index.html") {
184
+ res.writeHead(503, { "Content-Type": "text/html" });
185
+ res.end("<html><body><h1>Web UI not built</h1><p>Run <code>pnpm build</code> from the project root to build the web UI.</p></body></html>");
186
+ }
187
+ else {
188
+ res.writeHead(404, { "Content-Type": "application/json" });
189
+ res.end(JSON.stringify({ error: "Not found" }));
190
+ }
191
+ }
192
+ });
193
+ // WebSocket server
194
+ const wss = new WebSocketServer({ noServer: true });
195
+ wss.on("connection", (ws) => {
196
+ wsClients.add(ws);
197
+ logger.info(`WebSocket client connected (${wsClients.size} total)`);
198
+ ws.on("close", () => {
199
+ wsClients.delete(ws);
200
+ logger.info(`WebSocket client disconnected (${wsClients.size} total)`);
201
+ });
202
+ ws.on("error", (err) => {
203
+ logger.error(`WebSocket error: ${err.message}`);
204
+ wsClients.delete(ws);
205
+ });
206
+ });
207
+ server.on("upgrade", (req, socket, head) => {
208
+ if (req.url === "/ws") {
209
+ wss.handleUpgrade(req, socket, head, (ws) => {
210
+ wss.emit("connection", ws, req);
211
+ });
212
+ }
213
+ else {
214
+ socket.destroy();
215
+ }
216
+ });
217
+ // Sync skill symlinks to .claude/skills/ and .agents/skills/
218
+ syncSkillSymlinks();
219
+ // Start file watchers
220
+ startWatchers({
221
+ onConfigReload: () => {
222
+ try {
223
+ currentConfig = loadConfig();
224
+ apiContext.config = currentConfig;
225
+ logger.info("Config reloaded successfully");
226
+ emit("config:reloaded", {});
227
+ }
228
+ catch (err) {
229
+ logger.error(`Failed to reload config: ${err instanceof Error ? err.message : err}`);
230
+ }
231
+ },
232
+ onCronReload: () => {
233
+ const updatedJobs = loadJobs();
234
+ reloadScheduler(updatedJobs);
235
+ logger.info(`Cron jobs reloaded (${updatedJobs.length} job(s))`);
236
+ emit("cron:reloaded", {});
237
+ },
238
+ onOrgChange: () => {
239
+ employeeRegistry = scanOrg();
240
+ logger.info(`Org directory changed, reloaded ${employeeRegistry.size} employee(s)`);
241
+ emit("org:changed", {});
242
+ },
243
+ onSkillsChange: () => {
244
+ logger.info("Skills changed, notifying clients");
245
+ emit("skills:changed", {});
246
+ },
247
+ });
248
+ // Start listening
249
+ const port = config.gateway.port || 7777;
250
+ const host = config.gateway.host || "127.0.0.1";
251
+ await new Promise((resolve, reject) => {
252
+ server.on("error", (err) => {
253
+ if (err.code === "EADDRINUSE") {
254
+ const msg = `Port ${port} is already in use.`;
255
+ logger.error(msg);
256
+ console.error(`\nError: ${msg}`);
257
+ console.error(`\nTry: jinn start -p ${port + 1}`);
258
+ console.error(`Or update the port in config.yaml\n`);
259
+ process.exit(1);
260
+ }
261
+ reject(err);
262
+ });
263
+ server.listen(port, host, () => {
264
+ logger.info(`${gatewayName} gateway listening on http://${host}:${port} (boot ${bootId})`);
265
+ resolve();
266
+ });
267
+ });
268
+ // Return cleanup function
269
+ return async () => {
270
+ logger.info("Gateway cleanup starting...");
271
+ // Terminate live engine subprocesses before tearing down the gateway.
272
+ claudeEngine.killAll();
273
+ codexEngine.killAll();
274
+ // Stop cron scheduler
275
+ stopScheduler();
276
+ // Stop connectors
277
+ for (const connector of connectors) {
278
+ try {
279
+ await connector.stop();
280
+ }
281
+ catch (err) {
282
+ logger.error(`Failed to stop ${connector.name} connector: ${err instanceof Error ? err.message : err}`);
283
+ }
284
+ }
285
+ // Stop watchers
286
+ await stopWatchers();
287
+ // Close WebSocket connections
288
+ for (const client of wsClients) {
289
+ client.close(1001, "Server shutting down");
290
+ }
291
+ wsClients.clear();
292
+ // Close WebSocket server
293
+ await new Promise((resolve) => wss.close(() => resolve()));
294
+ // Close HTTP server
295
+ await new Promise((resolve, reject) => {
296
+ server.close((err) => (err ? reject(err) : resolve()));
297
+ });
298
+ logger.info("Gateway shutdown complete");
299
+ };
300
+ }
301
+ //# sourceMappingURL=server.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server.js","sourceRoot":"","sources":["../../../src/gateway/server.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,14 @@
1
+ export interface WatcherCallbacks {
2
+ onConfigReload: () => void;
3
+ onCronReload: () => void;
4
+ onOrgChange: () => void;
5
+ onSkillsChange: () => void;
6
+ }
7
+ /**
8
+ * Sync symlinks in .claude/skills/ and .agents/skills/ to match skills/.
9
+ * Each skill directory gets a relative symlink: ../../skills/<name>
10
+ */
11
+ export declare function syncSkillSymlinks(): void;
12
+ export declare function startWatchers(callbacks: WatcherCallbacks): void;
13
+ export declare function stopWatchers(): Promise<void>;
14
+ //# sourceMappingURL=watcher.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"watcher.d.ts","sourceRoot":"","sources":["../../../src/gateway/watcher.ts"],"names":[],"mappings":"AAMA,MAAM,WAAW,gBAAgB;IAC/B,cAAc,EAAE,MAAM,IAAI,CAAC;IAC3B,YAAY,EAAE,MAAM,IAAI,CAAC;IACzB,WAAW,EAAE,MAAM,IAAI,CAAC;IACxB,cAAc,EAAE,MAAM,IAAI,CAAC;CAC5B;AAYD;;;GAGG;AACH,wBAAgB,iBAAiB,IAAI,IAAI,CA0CxC;AAED,wBAAgB,aAAa,CAAC,SAAS,EAAE,gBAAgB,GAAG,IAAI,CAuD/D;AAED,wBAAsB,YAAY,IAAI,OAAO,CAAC,IAAI,CAAC,CAIlD"}
@@ -0,0 +1,104 @@
1
+ import fs from "node:fs";
2
+ import path from "node:path";
3
+ import { watch } from "chokidar";
4
+ import { CONFIG_PATH, CRON_JOBS, ORG_DIR, SKILLS_DIR, CLAUDE_SKILLS_DIR, AGENTS_SKILLS_DIR } from "../shared/paths.js";
5
+ import { logger } from "../shared/logger.js";
6
+ let watchers = [];
7
+ function debounce(fn, ms) {
8
+ let timer = null;
9
+ return () => {
10
+ if (timer)
11
+ clearTimeout(timer);
12
+ timer = setTimeout(fn, ms);
13
+ };
14
+ }
15
+ /**
16
+ * Sync symlinks in .claude/skills/ and .agents/skills/ to match skills/.
17
+ * Each skill directory gets a relative symlink: ../../skills/<name>
18
+ */
19
+ export function syncSkillSymlinks() {
20
+ const targetDirs = [CLAUDE_SKILLS_DIR, AGENTS_SKILLS_DIR];
21
+ // Get current skill directories
22
+ let skillNames = [];
23
+ if (fs.existsSync(SKILLS_DIR)) {
24
+ skillNames = fs.readdirSync(SKILLS_DIR, { withFileTypes: true })
25
+ .filter((e) => e.isDirectory())
26
+ .map((e) => e.name);
27
+ }
28
+ for (const targetDir of targetDirs) {
29
+ fs.mkdirSync(targetDir, { recursive: true });
30
+ // Remove stale symlinks
31
+ const existing = fs.readdirSync(targetDir, { withFileTypes: true });
32
+ for (const entry of existing) {
33
+ if (!skillNames.includes(entry.name)) {
34
+ const linkPath = path.join(targetDir, entry.name);
35
+ try {
36
+ fs.unlinkSync(linkPath);
37
+ logger.debug(`Removed stale skill symlink: ${linkPath}`);
38
+ }
39
+ catch {
40
+ // ignore
41
+ }
42
+ }
43
+ }
44
+ // Create missing symlinks
45
+ for (const name of skillNames) {
46
+ const linkPath = path.join(targetDir, name);
47
+ const relTarget = path.join("..", "..", "skills", name);
48
+ if (!fs.existsSync(linkPath)) {
49
+ try {
50
+ fs.symlinkSync(relTarget, linkPath);
51
+ logger.debug(`Created skill symlink: ${linkPath} -> ${relTarget}`);
52
+ }
53
+ catch {
54
+ // ignore — may fail on some platforms
55
+ }
56
+ }
57
+ }
58
+ }
59
+ }
60
+ export function startWatchers(callbacks) {
61
+ const DEBOUNCE_MS = 500;
62
+ const configWatcher = watch(CONFIG_PATH, {
63
+ ignoreInitial: true,
64
+ awaitWriteFinish: { stabilityThreshold: 300 },
65
+ });
66
+ configWatcher.on("change", debounce(() => {
67
+ logger.info("config.yaml changed, reloading...");
68
+ callbacks.onConfigReload();
69
+ }, DEBOUNCE_MS));
70
+ const cronWatcher = watch(CRON_JOBS, {
71
+ ignoreInitial: true,
72
+ awaitWriteFinish: { stabilityThreshold: 300 },
73
+ });
74
+ cronWatcher.on("change", debounce(() => {
75
+ logger.info("cron/jobs.json changed, reloading...");
76
+ callbacks.onCronReload();
77
+ }, DEBOUNCE_MS));
78
+ const orgWatcher = watch(ORG_DIR, {
79
+ ignoreInitial: true,
80
+ awaitWriteFinish: { stabilityThreshold: 300 },
81
+ });
82
+ orgWatcher.on("all", debounce(() => {
83
+ logger.info("org/ directory changed, reloading...");
84
+ callbacks.onOrgChange();
85
+ }, DEBOUNCE_MS));
86
+ // Watch skills/ directory for added/removed skill folders → sync symlinks
87
+ const skillsWatcher = watch(SKILLS_DIR, {
88
+ ignoreInitial: true,
89
+ depth: 0,
90
+ });
91
+ skillsWatcher.on("all", debounce(() => {
92
+ logger.info("skills/ directory changed, syncing symlinks...");
93
+ syncSkillSymlinks();
94
+ callbacks.onSkillsChange();
95
+ }, DEBOUNCE_MS));
96
+ watchers = [configWatcher, cronWatcher, orgWatcher, skillsWatcher];
97
+ logger.info("File watchers started");
98
+ }
99
+ export async function stopWatchers() {
100
+ await Promise.all(watchers.map((w) => w.close()));
101
+ watchers = [];
102
+ logger.info("File watchers stopped");
103
+ }
104
+ //# sourceMappingURL=watcher.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"watcher.js","sourceRoot":"","sources":["../../../src/gateway/watcher.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ export type { Engine, EngineRunOpts, EngineResult, Connector, IncomingMessage, Attachment, Target, Session, CronJob, CronDelivery, Employee, Department, JinnConfig, } from "./shared/types.js";
2
+ export { loadConfig } from "./shared/config.js";
3
+ export { configureLogger, logger } from "./shared/logger.js";
4
+ export { JINN_HOME, CONFIG_PATH, SESSIONS_DB, CRON_JOBS, CRON_RUNS, ORG_DIR, SKILLS_DIR, DOCS_DIR, LOGS_DIR, TMP_DIR, PID_FILE, TEMPLATE_DIR, } from "./shared/paths.js";
5
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/index.ts"],"names":[],"mappings":"AAAA,YAAY,EACV,MAAM,EACN,aAAa,EACb,YAAY,EACZ,SAAS,EACT,eAAe,EACf,UAAU,EACV,MAAM,EACN,OAAO,EACP,OAAO,EACP,YAAY,EACZ,QAAQ,EACR,UAAU,EACV,UAAU,GACX,MAAM,mBAAmB,CAAC;AAE3B,OAAO,EAAE,UAAU,EAAE,MAAM,oBAAoB,CAAC;AAChD,OAAO,EAAE,eAAe,EAAE,MAAM,EAAE,MAAM,oBAAoB,CAAC;AAC7D,OAAO,EACL,SAAS,EACT,WAAW,EACX,WAAW,EACX,SAAS,EACT,SAAS,EACT,OAAO,EACP,UAAU,EACV,QAAQ,EACR,QAAQ,EACR,OAAO,EACP,QAAQ,EACR,YAAY,GACb,MAAM,mBAAmB,CAAC"}
@@ -0,0 +1,4 @@
1
+ export { loadConfig } from "./shared/config.js";
2
+ export { configureLogger, logger } from "./shared/logger.js";
3
+ export { JINN_HOME, CONFIG_PATH, SESSIONS_DB, CRON_JOBS, CRON_RUNS, ORG_DIR, SKILLS_DIR, DOCS_DIR, LOGS_DIR, TMP_DIR, PID_FILE, TEMPLATE_DIR, } from "./shared/paths.js";
4
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/index.ts"],"names":[],"mappings":"AAgBA,OAAO,EAAE,UAAU,EAAE,MAAM,oBAAoB,CAAC;AAChD,OAAO,EAAE,eAAe,EAAE,MAAM,EAAE,MAAM,oBAAoB,CAAC;AAC7D,OAAO,EACL,SAAS,EACT,WAAW,EACX,WAAW,EACX,SAAS,EACT,SAAS,EACT,OAAO,EACP,UAAU,EACV,QAAQ,EACR,QAAQ,EACR,OAAO,EACP,QAAQ,EACR,YAAY,GACb,MAAM,mBAAmB,CAAC"}
@@ -0,0 +1,20 @@
1
+ import type { Employee, JinnConfig } from "../shared/types.js";
2
+ /**
3
+ * Build a rich system prompt for engine sessions.
4
+ * This is what makes Jinn "smart" — the engine sees all of this context
5
+ * before responding to the user.
6
+ */
7
+ export declare function buildContext(opts: {
8
+ source: string;
9
+ channel: string;
10
+ thread?: string;
11
+ user: string;
12
+ employee?: Employee;
13
+ connectors?: string[];
14
+ config?: JinnConfig;
15
+ sessionId?: string;
16
+ portalName?: string;
17
+ operatorName?: string;
18
+ language?: string;
19
+ }): string;
20
+ //# sourceMappingURL=context.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"context.d.ts","sourceRoot":"","sources":["../../../src/sessions/context.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,QAAQ,EAAE,UAAU,EAAE,MAAM,oBAAoB,CAAC;AAK/D;;;;GAIG;AACH,wBAAgB,YAAY,CAAC,IAAI,EAAE;IACjC,MAAM,EAAE,MAAM,CAAC;IACf,OAAO,EAAE,MAAM,CAAC;IAChB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,IAAI,EAAE,MAAM,CAAC;IACb,QAAQ,CAAC,EAAE,QAAQ,CAAC;IACpB,UAAU,CAAC,EAAE,MAAM,EAAE,CAAC;IACtB,MAAM,CAAC,EAAE,UAAU,CAAC;IACpB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,QAAQ,CAAC,EAAE,MAAM,CAAC;CACnB,GAAG,MAAM,CAqET"}