@pancake-apps/server 0.0.1
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/dist/index.cjs +985 -0
- package/dist/index.cjs.map +1 -0
- package/dist/index.d.cts +558 -0
- package/dist/index.d.ts +558 -0
- package/dist/index.js +380 -0
- package/dist/index.js.map +1 -0
- package/dist/server-3KHS3VCE.js +426 -0
- package/dist/server-3KHS3VCE.js.map +1 -0
- package/dist/stdio-TAQIWN3E.js +120 -0
- package/dist/stdio-TAQIWN3E.js.map +1 -0
- package/package.json +66 -0
package/dist/index.js
ADDED
|
@@ -0,0 +1,380 @@
|
|
|
1
|
+
import { zodToJsonSchema } from 'zod-to-json-schema';
|
|
2
|
+
import * as fs from 'fs/promises';
|
|
3
|
+
import * as fsSync from 'fs';
|
|
4
|
+
import * as path from 'path';
|
|
5
|
+
|
|
6
|
+
// src/normalizers.ts
|
|
7
|
+
|
|
8
|
+
// src/view.ts
|
|
9
|
+
function isViewConfig(value) {
|
|
10
|
+
return typeof value === "object" && value !== null && "description" in value && "ui" in value;
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
// src/action.ts
|
|
14
|
+
function isActionConfig(value) {
|
|
15
|
+
return typeof value === "object" && value !== null && "description" in value && "input" in value && "output" in value && "handler" in value;
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
// src/data.ts
|
|
19
|
+
function isDataConfig(value) {
|
|
20
|
+
return typeof value === "object" && value !== null && "description" in value && "input" in value && "output" in value && "handler" in value && !("ui" in value);
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
// src/tool.ts
|
|
24
|
+
function isToolConfig(value) {
|
|
25
|
+
return typeof value === "object" && value !== null && "name" in value && "description" in value && "input" in value && "output" in value && "handler" in value;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
// src/normalizers.ts
|
|
29
|
+
function toJsonSchema(schema) {
|
|
30
|
+
if (!schema) {
|
|
31
|
+
return { type: "object", properties: {} };
|
|
32
|
+
}
|
|
33
|
+
return zodToJsonSchema(schema);
|
|
34
|
+
}
|
|
35
|
+
function normalizeView(name, config) {
|
|
36
|
+
const toolName = `view:${name}`;
|
|
37
|
+
return {
|
|
38
|
+
name: toolName,
|
|
39
|
+
description: config.description,
|
|
40
|
+
inputSchema: toJsonSchema(config.input),
|
|
41
|
+
outputSchema: toJsonSchema(config.data),
|
|
42
|
+
handler: async (input, ctx) => {
|
|
43
|
+
if (config.handler) {
|
|
44
|
+
const validated = config.input ? config.input.parse(input) : input;
|
|
45
|
+
return config.handler(validated, {
|
|
46
|
+
...ctx,
|
|
47
|
+
viewName: name
|
|
48
|
+
});
|
|
49
|
+
}
|
|
50
|
+
return {};
|
|
51
|
+
},
|
|
52
|
+
ui: config.ui,
|
|
53
|
+
visibility: config.visibility ?? "both",
|
|
54
|
+
category: "view"
|
|
55
|
+
};
|
|
56
|
+
}
|
|
57
|
+
function normalizeAction(name, config) {
|
|
58
|
+
const toolName = `action:${name}`;
|
|
59
|
+
return {
|
|
60
|
+
name: toolName,
|
|
61
|
+
description: config.description,
|
|
62
|
+
inputSchema: toJsonSchema(config.input),
|
|
63
|
+
outputSchema: toJsonSchema(config.output),
|
|
64
|
+
handler: async (input, ctx) => {
|
|
65
|
+
const validated = config.input.parse(input);
|
|
66
|
+
return config.handler(validated, {
|
|
67
|
+
...ctx,
|
|
68
|
+
actionName: name
|
|
69
|
+
});
|
|
70
|
+
},
|
|
71
|
+
ui: config.ui,
|
|
72
|
+
visibility: config.visibility ?? "both",
|
|
73
|
+
category: "action"
|
|
74
|
+
};
|
|
75
|
+
}
|
|
76
|
+
function normalizeData(name, config) {
|
|
77
|
+
const toolName = `data:${name}`;
|
|
78
|
+
return {
|
|
79
|
+
name: toolName,
|
|
80
|
+
description: config.description,
|
|
81
|
+
inputSchema: toJsonSchema(config.input),
|
|
82
|
+
outputSchema: toJsonSchema(config.output),
|
|
83
|
+
handler: async (input, ctx) => {
|
|
84
|
+
const validated = config.input.parse(input);
|
|
85
|
+
return config.handler(validated, {
|
|
86
|
+
...ctx,
|
|
87
|
+
dataName: name
|
|
88
|
+
});
|
|
89
|
+
},
|
|
90
|
+
ui: void 0,
|
|
91
|
+
// Data fetchers don't have UI
|
|
92
|
+
visibility: config.visibility ?? "both",
|
|
93
|
+
category: "data"
|
|
94
|
+
};
|
|
95
|
+
}
|
|
96
|
+
function normalizeTool(name, config) {
|
|
97
|
+
return {
|
|
98
|
+
name: config.name || name,
|
|
99
|
+
description: config.description,
|
|
100
|
+
inputSchema: toJsonSchema(config.input),
|
|
101
|
+
outputSchema: toJsonSchema(config.output),
|
|
102
|
+
handler: async (input, ctx) => {
|
|
103
|
+
const validated = config.input.parse(input);
|
|
104
|
+
return config.handler(validated, ctx);
|
|
105
|
+
},
|
|
106
|
+
ui: config.ui,
|
|
107
|
+
visibility: config.visibility ?? "both",
|
|
108
|
+
category: "tool"
|
|
109
|
+
};
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
// src/app.ts
|
|
113
|
+
var TypedEventEmitter = class {
|
|
114
|
+
listeners = /* @__PURE__ */ new Map();
|
|
115
|
+
on(event, handler) {
|
|
116
|
+
const handlers = this.listeners.get(event) ?? /* @__PURE__ */ new Set();
|
|
117
|
+
handlers.add(handler);
|
|
118
|
+
this.listeners.set(event, handlers);
|
|
119
|
+
return () => handlers.delete(handler);
|
|
120
|
+
}
|
|
121
|
+
emit(event, payload) {
|
|
122
|
+
const handlers = this.listeners.get(event);
|
|
123
|
+
if (handlers) {
|
|
124
|
+
for (const handler of handlers) {
|
|
125
|
+
handler(payload);
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
};
|
|
130
|
+
var MiddlewareChain = class {
|
|
131
|
+
middlewares = [];
|
|
132
|
+
constructor(initial) {
|
|
133
|
+
if (initial) {
|
|
134
|
+
this.middlewares = [...initial];
|
|
135
|
+
}
|
|
136
|
+
}
|
|
137
|
+
add(middleware) {
|
|
138
|
+
this.middlewares.push(middleware);
|
|
139
|
+
}
|
|
140
|
+
async execute(ctx, handler) {
|
|
141
|
+
const state = /* @__PURE__ */ new Map();
|
|
142
|
+
const middlewareCtx = { ...ctx, state };
|
|
143
|
+
let index = 0;
|
|
144
|
+
let result;
|
|
145
|
+
const next = async () => {
|
|
146
|
+
if (index < this.middlewares.length) {
|
|
147
|
+
const mw = this.middlewares[index++];
|
|
148
|
+
await mw(middlewareCtx, next);
|
|
149
|
+
} else {
|
|
150
|
+
result = await handler();
|
|
151
|
+
}
|
|
152
|
+
};
|
|
153
|
+
await next();
|
|
154
|
+
return result;
|
|
155
|
+
}
|
|
156
|
+
};
|
|
157
|
+
var PluginManager = class {
|
|
158
|
+
constructor(plugins = []) {
|
|
159
|
+
this.plugins = plugins;
|
|
160
|
+
}
|
|
161
|
+
async runHook(hook, context) {
|
|
162
|
+
for (const plugin of this.plugins) {
|
|
163
|
+
const fn = plugin[hook];
|
|
164
|
+
if (typeof fn === "function") {
|
|
165
|
+
await fn.call(plugin, context);
|
|
166
|
+
}
|
|
167
|
+
}
|
|
168
|
+
}
|
|
169
|
+
};
|
|
170
|
+
function createApp(config) {
|
|
171
|
+
if (!config.name) {
|
|
172
|
+
throw new Error("App name is required");
|
|
173
|
+
}
|
|
174
|
+
if (!config.version) {
|
|
175
|
+
throw new Error("App version is required");
|
|
176
|
+
}
|
|
177
|
+
const events = new TypedEventEmitter();
|
|
178
|
+
const middlewareChain = new MiddlewareChain(config.middleware);
|
|
179
|
+
const pluginManager = new PluginManager(config.plugins);
|
|
180
|
+
const tools = /* @__PURE__ */ new Map();
|
|
181
|
+
const uiResources = /* @__PURE__ */ new Map();
|
|
182
|
+
for (const [name, viewConfig] of Object.entries(config.views ?? {})) {
|
|
183
|
+
const tool = normalizeView(name, viewConfig);
|
|
184
|
+
tools.set(tool.name, tool);
|
|
185
|
+
if (tool.ui) {
|
|
186
|
+
const uri = `pancake://ui/view/${name}`;
|
|
187
|
+
uiResources.set(uri, {
|
|
188
|
+
name,
|
|
189
|
+
uri,
|
|
190
|
+
definition: tool.ui
|
|
191
|
+
});
|
|
192
|
+
}
|
|
193
|
+
}
|
|
194
|
+
for (const [name, actionConfig] of Object.entries(config.actions ?? {})) {
|
|
195
|
+
const tool = normalizeAction(name, actionConfig);
|
|
196
|
+
tools.set(tool.name, tool);
|
|
197
|
+
if (tool.ui) {
|
|
198
|
+
const uri = `pancake://ui/action/${name}`;
|
|
199
|
+
uiResources.set(uri, {
|
|
200
|
+
name,
|
|
201
|
+
uri,
|
|
202
|
+
definition: tool.ui
|
|
203
|
+
});
|
|
204
|
+
}
|
|
205
|
+
}
|
|
206
|
+
for (const [name, dataConfig] of Object.entries(config.data ?? {})) {
|
|
207
|
+
const tool = normalizeData(name, dataConfig);
|
|
208
|
+
tools.set(tool.name, tool);
|
|
209
|
+
}
|
|
210
|
+
for (const [name, toolConfig] of Object.entries(config.tools ?? {})) {
|
|
211
|
+
const tool = normalizeTool(name, toolConfig);
|
|
212
|
+
tools.set(tool.name, tool);
|
|
213
|
+
if (tool.ui) {
|
|
214
|
+
const uri = `pancake://ui/tool/${name}`;
|
|
215
|
+
uiResources.set(uri, {
|
|
216
|
+
name,
|
|
217
|
+
uri,
|
|
218
|
+
definition: tool.ui
|
|
219
|
+
});
|
|
220
|
+
}
|
|
221
|
+
}
|
|
222
|
+
events.emit("app:init", { config });
|
|
223
|
+
let server = null;
|
|
224
|
+
const app = {
|
|
225
|
+
async start(options = {}) {
|
|
226
|
+
const port = options.port ?? 3e3;
|
|
227
|
+
const host = options.host ?? "0.0.0.0";
|
|
228
|
+
const transport = options.transport ?? "http";
|
|
229
|
+
for (const plugin of config.plugins ?? []) {
|
|
230
|
+
await pluginManager.runHook("onInit", { app: config, plugin });
|
|
231
|
+
}
|
|
232
|
+
const { createServer } = await import('./server-3KHS3VCE.js');
|
|
233
|
+
const serverConfig = {
|
|
234
|
+
name: config.name,
|
|
235
|
+
version: config.version,
|
|
236
|
+
tools,
|
|
237
|
+
uiResources,
|
|
238
|
+
middlewareChain,
|
|
239
|
+
events,
|
|
240
|
+
port,
|
|
241
|
+
host,
|
|
242
|
+
transport
|
|
243
|
+
};
|
|
244
|
+
if (config.config) {
|
|
245
|
+
const cfg = {};
|
|
246
|
+
if (config.config.cors !== void 0) cfg.cors = config.config.cors;
|
|
247
|
+
if (config.config.debug !== void 0) cfg.debug = config.config.debug;
|
|
248
|
+
if (config.config.serverRoute !== void 0) cfg.serverRoute = config.config.serverRoute;
|
|
249
|
+
if (config.config.devServer !== void 0) cfg.devServer = config.config.devServer;
|
|
250
|
+
serverConfig.config = cfg;
|
|
251
|
+
}
|
|
252
|
+
server = await createServer(serverConfig);
|
|
253
|
+
for (const plugin of config.plugins ?? []) {
|
|
254
|
+
await pluginManager.runHook("onStart", { app: config, plugin });
|
|
255
|
+
}
|
|
256
|
+
events.emit("app:start", { port, transport });
|
|
257
|
+
},
|
|
258
|
+
async stop() {
|
|
259
|
+
if (server) {
|
|
260
|
+
for (const plugin of config.plugins ?? []) {
|
|
261
|
+
await pluginManager.runHook("onShutdown", { app: config, plugin });
|
|
262
|
+
}
|
|
263
|
+
await server.close();
|
|
264
|
+
server = null;
|
|
265
|
+
events.emit("app:shutdown", { graceful: true });
|
|
266
|
+
}
|
|
267
|
+
},
|
|
268
|
+
handler() {
|
|
269
|
+
return async (_req, _res, next) => {
|
|
270
|
+
next();
|
|
271
|
+
};
|
|
272
|
+
},
|
|
273
|
+
async handleRequest(_req) {
|
|
274
|
+
return new Response("Not implemented", { status: 501 });
|
|
275
|
+
},
|
|
276
|
+
use(middleware) {
|
|
277
|
+
middlewareChain.add(middleware);
|
|
278
|
+
},
|
|
279
|
+
on(event, handler) {
|
|
280
|
+
return events.on(event, handler);
|
|
281
|
+
}
|
|
282
|
+
};
|
|
283
|
+
return app;
|
|
284
|
+
}
|
|
285
|
+
function discoverViews(dir) {
|
|
286
|
+
const views = {};
|
|
287
|
+
const absoluteDir = path.resolve(process.cwd(), dir);
|
|
288
|
+
try {
|
|
289
|
+
const entries = fsSync.readdirSync(absoluteDir, { withFileTypes: true });
|
|
290
|
+
for (const entry of entries) {
|
|
291
|
+
if (entry.isDirectory()) {
|
|
292
|
+
const indexHtml = path.join(absoluteDir, entry.name, "index.html");
|
|
293
|
+
const metadataJson = path.join(absoluteDir, entry.name, "metadata.json");
|
|
294
|
+
if (fileExistsSync(indexHtml) && fileExistsSync(metadataJson)) {
|
|
295
|
+
const metadata = parseMetadataSync(metadataJson);
|
|
296
|
+
views[entry.name] = createViewFromDiscovery(entry.name, indexHtml, metadata);
|
|
297
|
+
}
|
|
298
|
+
} else if (entry.name.endsWith(".html")) {
|
|
299
|
+
const baseName = entry.name.replace(".html", "");
|
|
300
|
+
const metadataPath = path.join(absoluteDir, `${baseName}.json`);
|
|
301
|
+
const htmlPath = path.join(absoluteDir, entry.name);
|
|
302
|
+
if (fileExistsSync(metadataPath)) {
|
|
303
|
+
const metadata = parseMetadataSync(metadataPath);
|
|
304
|
+
views[baseName] = createViewFromDiscovery(baseName, htmlPath, metadata);
|
|
305
|
+
}
|
|
306
|
+
}
|
|
307
|
+
}
|
|
308
|
+
} catch (error) {
|
|
309
|
+
console.warn(`[Pancake] Could not discover views from ${dir}:`, error);
|
|
310
|
+
}
|
|
311
|
+
return views;
|
|
312
|
+
}
|
|
313
|
+
async function discoverViewsAsync(dir) {
|
|
314
|
+
const views = {};
|
|
315
|
+
const absoluteDir = path.resolve(process.cwd(), dir);
|
|
316
|
+
try {
|
|
317
|
+
const entries = await fs.readdir(absoluteDir, { withFileTypes: true });
|
|
318
|
+
for (const entry of entries) {
|
|
319
|
+
if (entry.isDirectory()) {
|
|
320
|
+
const indexHtml = path.join(absoluteDir, entry.name, "index.html");
|
|
321
|
+
const metadataJson = path.join(absoluteDir, entry.name, "metadata.json");
|
|
322
|
+
if (await fileExists(indexHtml) && await fileExists(metadataJson)) {
|
|
323
|
+
const metadata = await parseMetadata(metadataJson);
|
|
324
|
+
views[entry.name] = createViewFromDiscovery(entry.name, indexHtml, metadata);
|
|
325
|
+
}
|
|
326
|
+
} else if (entry.name.endsWith(".html")) {
|
|
327
|
+
const baseName = entry.name.replace(".html", "");
|
|
328
|
+
const metadataPath = path.join(absoluteDir, `${baseName}.json`);
|
|
329
|
+
const htmlPath = path.join(absoluteDir, entry.name);
|
|
330
|
+
if (await fileExists(metadataPath)) {
|
|
331
|
+
const metadata = await parseMetadata(metadataPath);
|
|
332
|
+
views[baseName] = createViewFromDiscovery(baseName, htmlPath, metadata);
|
|
333
|
+
}
|
|
334
|
+
}
|
|
335
|
+
}
|
|
336
|
+
} catch (error) {
|
|
337
|
+
console.warn(`[Pancake] Could not discover views from ${dir}:`, error);
|
|
338
|
+
}
|
|
339
|
+
return views;
|
|
340
|
+
}
|
|
341
|
+
async function fileExists(filePath) {
|
|
342
|
+
try {
|
|
343
|
+
await fs.access(filePath);
|
|
344
|
+
return true;
|
|
345
|
+
} catch {
|
|
346
|
+
return false;
|
|
347
|
+
}
|
|
348
|
+
}
|
|
349
|
+
function fileExistsSync(filePath) {
|
|
350
|
+
try {
|
|
351
|
+
fsSync.accessSync(filePath);
|
|
352
|
+
return true;
|
|
353
|
+
} catch {
|
|
354
|
+
return false;
|
|
355
|
+
}
|
|
356
|
+
}
|
|
357
|
+
async function parseMetadata(filePath) {
|
|
358
|
+
const content = await fs.readFile(filePath, "utf-8");
|
|
359
|
+
return JSON.parse(content);
|
|
360
|
+
}
|
|
361
|
+
function parseMetadataSync(filePath) {
|
|
362
|
+
const content = fsSync.readFileSync(filePath, "utf-8");
|
|
363
|
+
return JSON.parse(content);
|
|
364
|
+
}
|
|
365
|
+
function createViewFromDiscovery(name, htmlPath, metadata) {
|
|
366
|
+
return {
|
|
367
|
+
description: metadata.description,
|
|
368
|
+
// Note: input/data from JSON are JSON Schema, not Zod
|
|
369
|
+
// We'll need to handle this specially in the normalizer
|
|
370
|
+
visibility: metadata.visibility ?? "both",
|
|
371
|
+
ui: {
|
|
372
|
+
html: htmlPath,
|
|
373
|
+
name
|
|
374
|
+
}
|
|
375
|
+
};
|
|
376
|
+
}
|
|
377
|
+
|
|
378
|
+
export { createApp, discoverViews, discoverViewsAsync, isActionConfig, isDataConfig, isToolConfig, isViewConfig };
|
|
379
|
+
//# sourceMappingURL=index.js.map
|
|
380
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/view.ts","../src/action.ts","../src/data.ts","../src/tool.ts","../src/normalizers.ts","../src/app.ts","../src/discovery/index.ts"],"names":[],"mappings":"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","file":"index.js","sourcesContent":["import type { ViewConfig } from './types/view.js';\n\n/**\n * Check if a value is a ViewConfig\n */\nexport function isViewConfig(value: unknown): value is ViewConfig {\n return (\n typeof value === 'object' &&\n value !== null &&\n 'description' in value &&\n 'ui' in value\n );\n}\n","import type { ActionConfig } from './types/action.js';\n\n/**\n * Check if a value is an ActionConfig\n */\nexport function isActionConfig(value: unknown): value is ActionConfig {\n return (\n typeof value === 'object' &&\n value !== null &&\n 'description' in value &&\n 'input' in value &&\n 'output' in value &&\n 'handler' in value\n );\n}\n","import type { DataConfig } from './types/data.js';\n\n/**\n * Check if a value is a DataConfig\n */\nexport function isDataConfig(value: unknown): value is DataConfig {\n return (\n typeof value === 'object' &&\n value !== null &&\n 'description' in value &&\n 'input' in value &&\n 'output' in value &&\n 'handler' in value &&\n !('ui' in value) // No UI means it's data, not view\n );\n}\n","import type { ToolConfig } from './types/tool.js';\n\n/**\n * Check if a value is a ToolConfig\n */\nexport function isToolConfig(value: unknown): value is ToolConfig {\n return (\n typeof value === 'object' &&\n value !== null &&\n 'name' in value &&\n 'description' in value &&\n 'input' in value &&\n 'output' in value &&\n 'handler' in value\n );\n}\n","import { zodToJsonSchema } from 'zod-to-json-schema';\nimport type { z } from 'zod';\nimport type { ViewConfig } from './types/view.js';\nimport type { ActionConfig } from './types/action.js';\nimport type { DataConfig } from './types/data.js';\nimport type { ToolConfig, InternalTool } from './types/tool.js';\nimport type { ToolContext } from './types/context.js';\nimport { isViewConfig } from './view.js';\nimport { isActionConfig } from './action.js';\nimport { isDataConfig } from './data.js';\nimport { isToolConfig } from './tool.js';\n\n/**\n * Convert a Zod schema to JSON Schema, handling undefined schemas\n */\nfunction toJsonSchema(schema: z.ZodType | undefined): Record<string, unknown> {\n if (!schema) {\n return { type: 'object', properties: {} };\n }\n return zodToJsonSchema(schema) as Record<string, unknown>;\n}\n\n/**\n * Normalize a view config to an internal tool\n */\nexport function normalizeView(\n name: string,\n config: ViewConfig\n): InternalTool {\n const toolName = `view:${name}`;\n\n return {\n name: toolName,\n description: config.description,\n inputSchema: toJsonSchema(config.input),\n outputSchema: toJsonSchema(config.data),\n handler: async (input: unknown, ctx: ToolContext) => {\n if (config.handler) {\n // Validate input with Zod schema if defined\n const validated = config.input ? config.input.parse(input) : input;\n return config.handler(validated, {\n ...ctx,\n viewName: name,\n });\n }\n // Views without handlers just return empty data\n return {};\n },\n ui: config.ui,\n visibility: config.visibility ?? 'both',\n category: 'view',\n };\n}\n\n/**\n * Normalize an action config to an internal tool\n */\nexport function normalizeAction(\n name: string,\n config: ActionConfig\n): InternalTool {\n const toolName = `action:${name}`;\n\n return {\n name: toolName,\n description: config.description,\n inputSchema: toJsonSchema(config.input),\n outputSchema: toJsonSchema(config.output),\n handler: async (input: unknown, ctx: ToolContext) => {\n // Validate input with Zod schema\n const validated = config.input.parse(input);\n return config.handler(validated, {\n ...ctx,\n actionName: name,\n });\n },\n ui: config.ui,\n visibility: config.visibility ?? 'both',\n category: 'action',\n };\n}\n\n/**\n * Normalize a data config to an internal tool\n */\nexport function normalizeData(\n name: string,\n config: DataConfig\n): InternalTool {\n const toolName = `data:${name}`;\n\n return {\n name: toolName,\n description: config.description,\n inputSchema: toJsonSchema(config.input),\n outputSchema: toJsonSchema(config.output),\n handler: async (input: unknown, ctx: ToolContext) => {\n // Validate input with Zod schema\n const validated = config.input.parse(input);\n return config.handler(validated, {\n ...ctx,\n dataName: name,\n });\n },\n ui: undefined, // Data fetchers don't have UI\n visibility: config.visibility ?? 'both',\n category: 'data',\n };\n}\n\n/**\n * Normalize a tool config to an internal tool\n */\nexport function normalizeTool(\n name: string,\n config: ToolConfig\n): InternalTool {\n return {\n name: config.name || name,\n description: config.description,\n inputSchema: toJsonSchema(config.input),\n outputSchema: toJsonSchema(config.output),\n handler: async (input: unknown, ctx: ToolContext) => {\n // Validate input with Zod schema\n const validated = config.input.parse(input);\n return config.handler(validated, ctx);\n },\n ui: config.ui,\n visibility: config.visibility ?? 'both',\n category: 'tool',\n };\n}\n\n/**\n * Normalize any config to an internal tool based on its type\n */\nexport function normalizeAny(\n name: string,\n config: unknown,\n category?: 'view' | 'action' | 'data' | 'tool'\n): InternalTool {\n if (isViewConfig(config) || category === 'view') {\n return normalizeView(name, config as ViewConfig);\n }\n\n if (isActionConfig(config) || category === 'action') {\n return normalizeAction(name, config as ActionConfig);\n }\n\n if (isDataConfig(config) || category === 'data') {\n return normalizeData(name, config as DataConfig);\n }\n\n if (isToolConfig(config) || category === 'tool') {\n return normalizeTool(name, config as ToolConfig);\n }\n\n throw new Error(\n `Unable to normalize config for \"${name}\". ` +\n `Ensure it has the required properties (description, input, output, handler, etc.).`\n );\n}\n","import type { RequestHandler } from 'express';\nimport type {\n AppConfig,\n App,\n StartOptions,\n Middleware,\n EventMap,\n Plugin,\n PluginContext,\n} from './types/app.js';\nimport type { InternalTool } from './types/tool.js';\nimport type { UIDefinition } from './types/protocol.js';\nimport { normalizeView, normalizeAction, normalizeData, normalizeTool } from './normalizers.js';\n\n/**\n * Event emitter for typed events\n */\nclass TypedEventEmitter<TEvents extends { [K in keyof TEvents]: unknown }> {\n private listeners = new Map<keyof TEvents, Set<(payload: unknown) => void>>();\n\n on<K extends keyof TEvents>(\n event: K,\n handler: (payload: TEvents[K]) => void\n ): () => void {\n const handlers = this.listeners.get(event) ?? new Set();\n handlers.add(handler as (payload: unknown) => void);\n this.listeners.set(event, handlers);\n\n return () => handlers.delete(handler as (payload: unknown) => void);\n }\n\n emit<K extends keyof TEvents>(event: K, payload: TEvents[K]): void {\n const handlers = this.listeners.get(event);\n if (handlers) {\n for (const handler of handlers) {\n handler(payload);\n }\n }\n }\n}\n\n/**\n * Middleware chain implementation\n */\nclass MiddlewareChain {\n private middlewares: Middleware[] = [];\n\n constructor(initial?: Middleware[]) {\n if (initial) {\n this.middlewares = [...initial];\n }\n }\n\n add(middleware: Middleware): void {\n this.middlewares.push(middleware);\n }\n\n async execute(\n ctx: { toolName: string; input: unknown; metadata: Record<string, unknown> },\n handler: () => Promise<unknown>\n ): Promise<unknown> {\n const state = new Map<string, unknown>();\n const middlewareCtx = { ...ctx, state };\n let index = 0;\n let result: unknown;\n\n const next = async (): Promise<void> => {\n if (index < this.middlewares.length) {\n const mw = this.middlewares[index++];\n await mw(middlewareCtx, next);\n } else {\n result = await handler();\n }\n };\n\n await next();\n return result;\n }\n}\n\n/**\n * Plugin manager for running lifecycle hooks\n */\nclass PluginManager {\n constructor(private plugins: Plugin[] = []) {}\n\n async runHook<K extends 'onInit' | 'onStart' | 'onShutdown'>(\n hook: K,\n context: PluginContext\n ): Promise<void> {\n for (const plugin of this.plugins) {\n const fn = plugin[hook];\n if (typeof fn === 'function') {\n await fn.call(plugin, context);\n }\n }\n }\n}\n\n/**\n * UI resource manager\n */\ninterface UIResource {\n name: string;\n uri: string;\n definition: UIDefinition;\n}\n\n/**\n * Create a Pancake application.\n *\n * @example\n * ```ts\n * import { createApp, defineView, defineAction, defineData } from '@pancake-apps/server';\n * import { z } from 'zod';\n *\n * const app = createApp({\n * name: 'my-app',\n * version: '1.0.0',\n * views: {\n * hello: defineView({\n * description: 'Greeting view',\n * input: z.object({ name: z.string() }),\n * ui: { html: './views/hello.html' },\n * }),\n * },\n * actions: {\n * save: defineAction({\n * description: 'Save data',\n * input: z.object({ data: z.string() }),\n * output: z.object({ success: z.boolean() }),\n * handler: async ({ data }) => ({ success: true }),\n * }),\n * },\n * });\n *\n * app.start({ port: 3000 });\n * ```\n */\nexport function createApp(config: AppConfig): App {\n // Validate config\n if (!config.name) {\n throw new Error('App name is required');\n }\n if (!config.version) {\n throw new Error('App version is required');\n }\n\n // Initialize event emitter\n const events = new TypedEventEmitter<EventMap>();\n\n // Initialize middleware chain\n const middlewareChain = new MiddlewareChain(config.middleware);\n\n // Initialize plugin manager\n const pluginManager = new PluginManager(config.plugins);\n\n // Normalize all definitions to internal tools\n const tools = new Map<string, InternalTool>();\n const uiResources = new Map<string, UIResource>();\n\n // Register views\n for (const [name, viewConfig] of Object.entries(config.views ?? {})) {\n const tool = normalizeView(name, viewConfig);\n tools.set(tool.name, tool);\n\n // Register UI resource\n if (tool.ui) {\n const uri = `pancake://ui/view/${name}`;\n uiResources.set(uri, {\n name,\n uri,\n definition: tool.ui,\n });\n }\n }\n\n // Register actions\n for (const [name, actionConfig] of Object.entries(config.actions ?? {})) {\n const tool = normalizeAction(name, actionConfig);\n tools.set(tool.name, tool);\n\n // Register UI resource if present\n if (tool.ui) {\n const uri = `pancake://ui/action/${name}`;\n uiResources.set(uri, {\n name,\n uri,\n definition: tool.ui,\n });\n }\n }\n\n // Register data fetchers\n for (const [name, dataConfig] of Object.entries(config.data ?? {})) {\n const tool = normalizeData(name, dataConfig);\n tools.set(tool.name, tool);\n }\n\n // Register low-level tools\n for (const [name, toolConfig] of Object.entries(config.tools ?? {})) {\n const tool = normalizeTool(name, toolConfig);\n tools.set(tool.name, tool);\n\n // Register UI resource if present\n if (tool.ui) {\n const uri = `pancake://ui/tool/${name}`;\n uiResources.set(uri, {\n name,\n uri,\n definition: tool.ui,\n });\n }\n }\n\n // Emit init event\n events.emit('app:init', { config });\n\n // Server state\n let server: { close: () => Promise<void> } | null = null;\n\n // Create app instance\n const app: App = {\n async start(options: StartOptions = {}) {\n const port = options.port ?? 3000;\n const host = options.host ?? '0.0.0.0';\n const transport = options.transport ?? 'http';\n\n // Run plugin onInit hooks\n for (const plugin of config.plugins ?? []) {\n await pluginManager.runHook('onInit', { app: config, plugin });\n }\n\n // Dynamically import server implementation to avoid bundling express in builds\n // that don't need it\n const { createServer } = await import('./server/index.js');\n\n const serverConfig: Parameters<typeof createServer>[0] = {\n name: config.name,\n version: config.version,\n tools,\n uiResources,\n middlewareChain,\n events,\n port,\n host,\n transport,\n };\n if (config.config) {\n const cfg: NonNullable<typeof serverConfig.config> = {};\n if (config.config.cors !== undefined) cfg.cors = config.config.cors;\n if (config.config.debug !== undefined) cfg.debug = config.config.debug;\n if (config.config.serverRoute !== undefined) cfg.serverRoute = config.config.serverRoute;\n if (config.config.devServer !== undefined) cfg.devServer = config.config.devServer;\n serverConfig.config = cfg;\n }\n server = await createServer(serverConfig);\n\n // Run plugin onStart hooks\n for (const plugin of config.plugins ?? []) {\n await pluginManager.runHook('onStart', { app: config, plugin });\n }\n\n events.emit('app:start', { port, transport });\n },\n\n async stop() {\n if (server) {\n // Run plugin onShutdown hooks\n for (const plugin of config.plugins ?? []) {\n await pluginManager.runHook('onShutdown', { app: config, plugin });\n }\n\n await server.close();\n server = null;\n events.emit('app:shutdown', { graceful: true });\n }\n },\n\n handler(): RequestHandler {\n // Return a placeholder that will be replaced when server starts\n // For integrating with existing Express apps\n return async (_req, _res, next) => {\n next();\n };\n },\n\n async handleRequest(_req: Request): Promise<Response> {\n // For serverless/edge environments\n return new Response('Not implemented', { status: 501 });\n },\n\n use(middleware: Middleware) {\n middlewareChain.add(middleware);\n },\n\n on<K extends keyof EventMap>(\n event: K,\n handler: (payload: EventMap[K]) => void\n ): () => void {\n return events.on(event, handler);\n },\n };\n\n return app;\n}\n","import * as fs from 'node:fs/promises';\nimport * as fsSync from 'node:fs';\nimport * as path from 'node:path';\nimport type { ViewConfig, ViewMetadata, DiscoveredView } from '../types/view.js';\n\n/**\n * Discover views from a directory.\n *\n * Supports two patterns:\n * 1. Flat files: `views/hello.html` + `views/hello.json`\n * 2. Folder-based: `views/hello/index.html` + `views/hello/metadata.json`\n *\n * @example\n * ```ts\n * const app = createApp({\n * name: 'my-app',\n * version: '1.0.0',\n * views: {\n * ...discoverViews('./src/views'),\n * },\n * });\n * ```\n */\nexport function discoverViews(dir: string): Record<string, ViewConfig> {\n // Return a synchronous-looking object that will be populated\n // We need to use a synchronous approach for the API\n const views: Record<string, ViewConfig> = {};\n\n // Synchronous discovery using Node.js fs sync methods\n const absoluteDir = path.resolve(process.cwd(), dir);\n\n try {\n const entries = fsSync.readdirSync(absoluteDir, { withFileTypes: true });\n\n for (const entry of entries) {\n if (entry.isDirectory()) {\n // Pattern 2: views/xxx/index.html + metadata.json\n const indexHtml = path.join(absoluteDir, entry.name, 'index.html');\n const metadataJson = path.join(absoluteDir, entry.name, 'metadata.json');\n\n if (fileExistsSync(indexHtml) && fileExistsSync(metadataJson)) {\n const metadata = parseMetadataSync(metadataJson);\n views[entry.name] = createViewFromDiscovery(entry.name, indexHtml, metadata);\n }\n } else if (entry.name.endsWith('.html')) {\n // Pattern 1: views/xxx.html + xxx.json\n const baseName = entry.name.replace('.html', '');\n const metadataPath = path.join(absoluteDir, `${baseName}.json`);\n const htmlPath = path.join(absoluteDir, entry.name);\n\n if (fileExistsSync(metadataPath)) {\n const metadata = parseMetadataSync(metadataPath);\n views[baseName] = createViewFromDiscovery(baseName, htmlPath, metadata);\n }\n }\n }\n } catch (error) {\n // Directory might not exist, return empty\n console.warn(`[Pancake] Could not discover views from ${dir}:`, error);\n }\n\n return views;\n}\n\n/**\n * Async version of discoverViews for dynamic usage\n */\nexport async function discoverViewsAsync(dir: string): Promise<Record<string, ViewConfig>> {\n const views: Record<string, ViewConfig> = {};\n const absoluteDir = path.resolve(process.cwd(), dir);\n\n try {\n const entries = await fs.readdir(absoluteDir, { withFileTypes: true });\n\n for (const entry of entries) {\n if (entry.isDirectory()) {\n // Pattern 2: views/xxx/index.html + metadata.json\n const indexHtml = path.join(absoluteDir, entry.name, 'index.html');\n const metadataJson = path.join(absoluteDir, entry.name, 'metadata.json');\n\n if ((await fileExists(indexHtml)) && (await fileExists(metadataJson))) {\n const metadata = await parseMetadata(metadataJson);\n views[entry.name] = createViewFromDiscovery(entry.name, indexHtml, metadata);\n }\n } else if (entry.name.endsWith('.html')) {\n // Pattern 1: views/xxx.html + xxx.json\n const baseName = entry.name.replace('.html', '');\n const metadataPath = path.join(absoluteDir, `${baseName}.json`);\n const htmlPath = path.join(absoluteDir, entry.name);\n\n if (await fileExists(metadataPath)) {\n const metadata = await parseMetadata(metadataPath);\n views[baseName] = createViewFromDiscovery(baseName, htmlPath, metadata);\n }\n }\n }\n } catch (error) {\n console.warn(`[Pancake] Could not discover views from ${dir}:`, error);\n }\n\n return views;\n}\n\n/**\n * Check if a file exists (async)\n */\nasync function fileExists(filePath: string): Promise<boolean> {\n try {\n await fs.access(filePath);\n return true;\n } catch {\n return false;\n }\n}\n\n/**\n * Check if a file exists (sync)\n */\nfunction fileExistsSync(filePath: string): boolean {\n try {\n fsSync.accessSync(filePath);\n return true;\n } catch {\n return false;\n }\n}\n\n/**\n * Parse metadata JSON file (async)\n */\nasync function parseMetadata(filePath: string): Promise<ViewMetadata> {\n const content = await fs.readFile(filePath, 'utf-8');\n return JSON.parse(content) as ViewMetadata;\n}\n\n/**\n * Parse metadata JSON file (sync)\n */\nfunction parseMetadataSync(filePath: string): ViewMetadata {\n const content = fsSync.readFileSync(filePath, 'utf-8');\n return JSON.parse(content) as ViewMetadata;\n}\n\n/**\n * Create a ViewConfig from discovered files\n */\nfunction createViewFromDiscovery(\n name: string,\n htmlPath: string,\n metadata: ViewMetadata\n): ViewConfig {\n return {\n description: metadata.description,\n // Note: input/data from JSON are JSON Schema, not Zod\n // We'll need to handle this specially in the normalizer\n visibility: metadata.visibility ?? 'both',\n ui: {\n html: htmlPath,\n name,\n },\n };\n}\n\nexport type { DiscoveredView, ViewMetadata };\n"]}
|