@greatlhd/ailo-endpoint-sdk 1.0.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 (89) hide show
  1. package/dist/blob/client.d.ts +15 -0
  2. package/dist/blob/client.d.ts.map +1 -0
  3. package/dist/blob/client.js +48 -0
  4. package/dist/blob/client.js.map +1 -0
  5. package/dist/blob/index.d.ts +3 -0
  6. package/dist/blob/index.d.ts.map +1 -0
  7. package/dist/blob/index.js +2 -0
  8. package/dist/blob/index.js.map +1 -0
  9. package/dist/bootstrap.d.ts +37 -0
  10. package/dist/bootstrap.d.ts.map +1 -0
  11. package/dist/bootstrap.js +145 -0
  12. package/dist/bootstrap.js.map +1 -0
  13. package/dist/config-io.d.ts +6 -0
  14. package/dist/config-io.d.ts.map +1 -0
  15. package/dist/config-io.js +43 -0
  16. package/dist/config-io.js.map +1 -0
  17. package/dist/connection-state.d.ts +24 -0
  18. package/dist/connection-state.d.ts.map +1 -0
  19. package/dist/connection-state.js +83 -0
  20. package/dist/connection-state.js.map +1 -0
  21. package/dist/connection-util.d.ts +12 -0
  22. package/dist/connection-util.d.ts.map +1 -0
  23. package/dist/connection-util.js +15 -0
  24. package/dist/connection-util.js.map +1 -0
  25. package/dist/endpoint-client.d.ts +92 -0
  26. package/dist/endpoint-client.d.ts.map +1 -0
  27. package/dist/endpoint-client.js +669 -0
  28. package/dist/endpoint-client.js.map +1 -0
  29. package/dist/endpoint-config.d.ts +28 -0
  30. package/dist/endpoint-config.d.ts.map +1 -0
  31. package/dist/endpoint-config.js +277 -0
  32. package/dist/endpoint-config.js.map +1 -0
  33. package/dist/errors.d.ts +17 -0
  34. package/dist/errors.d.ts.map +1 -0
  35. package/dist/errors.js +40 -0
  36. package/dist/errors.js.map +1 -0
  37. package/dist/fileref.d.ts +22 -0
  38. package/dist/fileref.d.ts.map +1 -0
  39. package/dist/fileref.js +41 -0
  40. package/dist/fileref.js.map +1 -0
  41. package/dist/index.d.ts +29 -0
  42. package/dist/index.d.ts.map +1 -0
  43. package/dist/index.js +18 -0
  44. package/dist/index.js.map +1 -0
  45. package/dist/local-endpoint-storage.d.ts +11 -0
  46. package/dist/local-endpoint-storage.d.ts.map +1 -0
  47. package/dist/local-endpoint-storage.js +143 -0
  48. package/dist/local-endpoint-storage.js.map +1 -0
  49. package/dist/logger.d.ts +25 -0
  50. package/dist/logger.d.ts.map +1 -0
  51. package/dist/logger.js +27 -0
  52. package/dist/logger.js.map +1 -0
  53. package/dist/media-util.d.ts +8 -0
  54. package/dist/media-util.d.ts.map +1 -0
  55. package/dist/media-util.js +78 -0
  56. package/dist/media-util.js.map +1 -0
  57. package/dist/middleware/index.d.ts +2 -0
  58. package/dist/middleware/index.d.ts.map +1 -0
  59. package/dist/middleware/index.js +2 -0
  60. package/dist/middleware/index.js.map +1 -0
  61. package/dist/middleware/media.d.ts +7 -0
  62. package/dist/middleware/media.d.ts.map +1 -0
  63. package/dist/middleware/media.js +112 -0
  64. package/dist/middleware/media.js.map +1 -0
  65. package/dist/skill-loader.d.ts +19 -0
  66. package/dist/skill-loader.d.ts.map +1 -0
  67. package/dist/skill-loader.js +131 -0
  68. package/dist/skill-loader.js.map +1 -0
  69. package/dist/tool-dispatch/index.d.ts +3 -0
  70. package/dist/tool-dispatch/index.d.ts.map +1 -0
  71. package/dist/tool-dispatch/index.js +2 -0
  72. package/dist/tool-dispatch/index.js.map +1 -0
  73. package/dist/tool-dispatch/normalize.d.ts +5 -0
  74. package/dist/tool-dispatch/normalize.d.ts.map +1 -0
  75. package/dist/tool-dispatch/normalize.js +100 -0
  76. package/dist/tool-dispatch/normalize.js.map +1 -0
  77. package/dist/types.d.ts +150 -0
  78. package/dist/types.d.ts.map +1 -0
  79. package/dist/types.js +13 -0
  80. package/dist/types.js.map +1 -0
  81. package/dist/utils/content.d.ts +7 -0
  82. package/dist/utils/content.d.ts.map +1 -0
  83. package/dist/utils/content.js +30 -0
  84. package/dist/utils/content.js.map +1 -0
  85. package/dist/utils/index.d.ts +2 -0
  86. package/dist/utils/index.d.ts.map +1 -0
  87. package/dist/utils/index.js +2 -0
  88. package/dist/utils/index.js.map +1 -0
  89. package/package.json +36 -0
@@ -0,0 +1,143 @@
1
+ import * as fs from "fs";
2
+ import * as os from "os";
3
+ import * as path from "path";
4
+ function safeEndpointDirSegment(id) {
5
+ const s = id.replace(/[^a-zA-Z0-9._-]+/g, "_").trim();
6
+ return s.slice(0, 200) || "default";
7
+ }
8
+ function loadSync(filePath) {
9
+ try {
10
+ const raw = fs.readFileSync(filePath, "utf-8");
11
+ const o = JSON.parse(raw);
12
+ if (typeof o !== "object" || o === null || Array.isArray(o))
13
+ return {};
14
+ const out = {};
15
+ for (const [k, v] of Object.entries(o)) {
16
+ if (typeof v === "string")
17
+ out[k] = v;
18
+ }
19
+ return out;
20
+ }
21
+ catch {
22
+ return {};
23
+ }
24
+ }
25
+ function saveSync(filePath, data) {
26
+ const tmp = `${filePath}.${process.pid}.tmp`;
27
+ fs.writeFileSync(tmp, JSON.stringify(data), "utf-8");
28
+ fs.renameSync(tmp, filePath);
29
+ }
30
+ export function createLocalEndpointStorage(endpointId) {
31
+ const dir = path.join(os.homedir(), ".ailo", "endpoint-data", safeEndpointDirSegment(endpointId));
32
+ fs.mkdirSync(dir, { recursive: true });
33
+ const filePath = path.join(dir, "kv.json");
34
+ let chain = Promise.resolve();
35
+ function runExclusive(fn) {
36
+ const p = chain.then(fn, fn);
37
+ chain = p.then(() => { }, () => { });
38
+ return p;
39
+ }
40
+ return {
41
+ getData(key) {
42
+ return runExclusive(async () => {
43
+ const o = loadSync(filePath);
44
+ return Object.prototype.hasOwnProperty.call(o, key) ? o[key] : null;
45
+ });
46
+ },
47
+ async setData(key, value) {
48
+ await runExclusive(async () => {
49
+ const o = loadSync(filePath);
50
+ o[key] = value;
51
+ saveSync(filePath, o);
52
+ });
53
+ },
54
+ async deleteData(key) {
55
+ await runExclusive(async () => {
56
+ const o = loadSync(filePath);
57
+ delete o[key];
58
+ saveSync(filePath, o);
59
+ });
60
+ },
61
+ };
62
+ }
63
+ export function createCachedEndpointStorage(endpointId, options) {
64
+ const flushDelayMs = options?.flushDelayMs ?? 100;
65
+ const maxCacheSize = options?.maxCacheSize ?? 1000;
66
+ const dir = path.join(os.homedir(), ".ailo", "endpoint-data", safeEndpointDirSegment(endpointId));
67
+ fs.mkdirSync(dir, { recursive: true });
68
+ const filePath = path.join(dir, "kv.json");
69
+ const cache = new Map();
70
+ const dirtyKeys = new Set();
71
+ let loaded = false;
72
+ let flushTimer = null;
73
+ let writeChain = Promise.resolve();
74
+ function ensureLoaded() {
75
+ if (loaded)
76
+ return;
77
+ const data = loadSync(filePath);
78
+ for (const [k, v] of Object.entries(data)) {
79
+ cache.set(k, v);
80
+ }
81
+ loaded = true;
82
+ }
83
+ function scheduleFlush() {
84
+ if (flushTimer)
85
+ return;
86
+ flushTimer = setTimeout(() => {
87
+ flushTimer = null;
88
+ void flush();
89
+ }, flushDelayMs);
90
+ }
91
+ async function flush() {
92
+ if (dirtyKeys.size === 0)
93
+ return;
94
+ const keysToFlush = [...dirtyKeys];
95
+ dirtyKeys.clear();
96
+ await writeChain;
97
+ writeChain = (async () => {
98
+ ensureLoaded();
99
+ const data = {};
100
+ for (const [k, v] of cache) {
101
+ data[k] = v;
102
+ }
103
+ saveSync(filePath, data);
104
+ })();
105
+ }
106
+ function clearCache() {
107
+ cache.clear();
108
+ dirtyKeys.clear();
109
+ loaded = false;
110
+ if (flushTimer) {
111
+ clearTimeout(flushTimer);
112
+ flushTimer = null;
113
+ }
114
+ }
115
+ return {
116
+ async getData(key) {
117
+ ensureLoaded();
118
+ return cache.has(key) ? cache.get(key) : null;
119
+ },
120
+ async setData(key, value) {
121
+ ensureLoaded();
122
+ if (cache.size >= maxCacheSize && !cache.has(key)) {
123
+ const oldestKey = cache.keys().next().value;
124
+ if (oldestKey) {
125
+ cache.delete(oldestKey);
126
+ dirtyKeys.add(oldestKey);
127
+ }
128
+ }
129
+ cache.set(key, value);
130
+ dirtyKeys.add(key);
131
+ scheduleFlush();
132
+ },
133
+ async deleteData(key) {
134
+ ensureLoaded();
135
+ cache.delete(key);
136
+ dirtyKeys.add(key);
137
+ scheduleFlush();
138
+ },
139
+ flush,
140
+ clearCache,
141
+ };
142
+ }
143
+ //# sourceMappingURL=local-endpoint-storage.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"local-endpoint-storage.js","sourceRoot":"","sources":["../src/local-endpoint-storage.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,25 @@
1
+ export type LogLevel = 'error' | 'warn' | 'info' | 'debug';
2
+ export interface LogData {
3
+ [key: string]: unknown;
4
+ }
5
+ export interface Logger {
6
+ error(event: string, data?: LogData): void;
7
+ warn(event: string, data?: LogData): void;
8
+ info(event: string, data?: LogData): void;
9
+ debug(event: string, data?: LogData): void;
10
+ }
11
+ export declare class ConsoleLogger implements Logger {
12
+ private prefix;
13
+ constructor(prefix?: string);
14
+ error(event: string, data?: LogData): void;
15
+ warn(event: string, data?: LogData): void;
16
+ info(event: string, data?: LogData): void;
17
+ debug(event: string, data?: LogData): void;
18
+ }
19
+ export declare class NoopLogger implements Logger {
20
+ error(): void;
21
+ warn(): void;
22
+ info(): void;
23
+ debug(): void;
24
+ }
25
+ //# sourceMappingURL=logger.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"logger.d.ts","sourceRoot":"","sources":["../src/logger.ts"],"names":[],"mappings":"AAAA,MAAM,MAAM,QAAQ,GAAG,OAAO,GAAG,MAAM,GAAG,MAAM,GAAG,OAAO,CAAC;AAE3D,MAAM,WAAW,OAAO;IACtB,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC;CACxB;AAED,MAAM,WAAW,MAAM;IACrB,KAAK,CAAC,KAAK,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,OAAO,GAAG,IAAI,CAAC;IAC3C,IAAI,CAAC,KAAK,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,OAAO,GAAG,IAAI,CAAC;IAC1C,IAAI,CAAC,KAAK,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,OAAO,GAAG,IAAI,CAAC;IAC1C,KAAK,CAAC,KAAK,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,OAAO,GAAG,IAAI,CAAC;CAC5C;AAED,qBAAa,aAAc,YAAW,MAAM;IAC9B,OAAO,CAAC,MAAM;gBAAN,MAAM,GAAE,MAAqB;IAEjD,KAAK,CAAC,KAAK,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,OAAO,GAAG,IAAI;IAI1C,IAAI,CAAC,KAAK,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,OAAO,GAAG,IAAI;IAIzC,IAAI,CAAC,KAAK,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,OAAO,GAAG,IAAI;IAIzC,KAAK,CAAC,KAAK,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,OAAO,GAAG,IAAI;CAK3C;AAED,qBAAa,UAAW,YAAW,MAAM;IACvC,KAAK,IAAI,IAAI;IACb,IAAI,IAAI,IAAI;IACZ,IAAI,IAAI,IAAI;IACZ,KAAK,IAAI,IAAI;CACd"}
package/dist/logger.js ADDED
@@ -0,0 +1,27 @@
1
+ export class ConsoleLogger {
2
+ prefix;
3
+ constructor(prefix = '[endpoint]') {
4
+ this.prefix = prefix;
5
+ }
6
+ error(event, data) {
7
+ console.error(`${this.prefix} [ERROR] ${event}`, data ?? '');
8
+ }
9
+ warn(event, data) {
10
+ console.warn(`${this.prefix} [WARN] ${event}`, data ?? '');
11
+ }
12
+ info(event, data) {
13
+ console.log(`${this.prefix} [INFO] ${event}`, data ?? '');
14
+ }
15
+ debug(event, data) {
16
+ if (process.env.DEBUG || process.env.NODE_ENV === 'development') {
17
+ console.log(`${this.prefix} [DEBUG] ${event}`, data ?? '');
18
+ }
19
+ }
20
+ }
21
+ export class NoopLogger {
22
+ error() { }
23
+ warn() { }
24
+ info() { }
25
+ debug() { }
26
+ }
27
+ //# sourceMappingURL=logger.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"logger.js","sourceRoot":"","sources":["../src/logger.ts"],"names":[],"mappings":"AAaA,MAAM,OAAO,aAAa;IACJ;IAApB,YAAoB,SAAiB,YAAY;QAA7B,WAAM,GAAN,MAAM,CAAuB;IAAG,CAAC;IAErD,KAAK,CAAC,KAAa,EAAE,IAAc;QACjC,OAAO,CAAC,KAAK,CAAC,GAAG,IAAI,CAAC,MAAM,YAAY,KAAK,EAAE,EAAE,IAAI,IAAI,EAAE,CAAC,CAAC;IAC/D,CAAC;IAED,IAAI,CAAC,KAAa,EAAE,IAAc;QAChC,OAAO,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,MAAM,WAAW,KAAK,EAAE,EAAE,IAAI,IAAI,EAAE,CAAC,CAAC;IAC7D,CAAC;IAED,IAAI,CAAC,KAAa,EAAE,IAAc;QAChC,OAAO,CAAC,GAAG,CAAC,GAAG,IAAI,CAAC,MAAM,WAAW,KAAK,EAAE,EAAE,IAAI,IAAI,EAAE,CAAC,CAAC;IAC5D,CAAC;IAED,KAAK,CAAC,KAAa,EAAE,IAAc;QACjC,IAAI,OAAO,CAAC,GAAG,CAAC,KAAK,IAAI,OAAO,CAAC,GAAG,CAAC,QAAQ,KAAK,aAAa,EAAE,CAAC;YAChE,OAAO,CAAC,GAAG,CAAC,GAAG,IAAI,CAAC,MAAM,YAAY,KAAK,EAAE,EAAE,IAAI,IAAI,EAAE,CAAC,CAAC;QAC7D,CAAC;IACH,CAAC;CACF;AAED,MAAM,OAAO,UAAU;IACrB,KAAK,KAAU,CAAC;IAChB,IAAI,KAAU,CAAC;IACf,IAAI,KAAU,CAAC;IACf,KAAK,KAAU,CAAC;CACjB"}
@@ -0,0 +1,8 @@
1
+ /**
2
+ * MIME type inference and media classification utilities.
3
+ */
4
+ /** Infer MIME type from a file path's extension. */
5
+ export declare function inferMime(filePath: string): string;
6
+ /** Classify a MIME type into a ContentPart.type value. */
7
+ export declare function classifyMedia(mime: string): "image" | "audio" | "video" | "pdf" | "file";
8
+ //# sourceMappingURL=media-util.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"media-util.d.ts","sourceRoot":"","sources":["../src/media-util.ts"],"names":[],"mappings":"AAAA;;GAEG;AA4DH,oDAAoD;AACpD,wBAAgB,SAAS,CAAC,QAAQ,EAAE,MAAM,GAAG,MAAM,CAGlD;AAED,0DAA0D;AAC1D,wBAAgB,aAAa,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,GAAG,OAAO,GAAG,OAAO,GAAG,KAAK,GAAG,MAAM,CAOxF"}
@@ -0,0 +1,78 @@
1
+ /**
2
+ * MIME type inference and media classification utilities.
3
+ */
4
+ import * as path from "path";
5
+ const MIME_MAP = {
6
+ // image
7
+ png: "image/png",
8
+ jpg: "image/jpeg",
9
+ jpeg: "image/jpeg",
10
+ gif: "image/gif",
11
+ webp: "image/webp",
12
+ svg: "image/svg+xml",
13
+ bmp: "image/bmp",
14
+ ico: "image/x-icon",
15
+ tiff: "image/tiff",
16
+ tif: "image/tiff",
17
+ avif: "image/avif",
18
+ // audio
19
+ mp3: "audio/mpeg",
20
+ wav: "audio/wav",
21
+ ogg: "audio/ogg",
22
+ opus: "audio/opus",
23
+ flac: "audio/flac",
24
+ aac: "audio/aac",
25
+ m4a: "audio/mp4",
26
+ wma: "audio/x-ms-wma",
27
+ // video
28
+ mp4: "video/mp4",
29
+ webm: "video/webm",
30
+ mkv: "video/x-matroska",
31
+ avi: "video/x-msvideo",
32
+ mov: "video/quicktime",
33
+ wmv: "video/x-ms-wmv",
34
+ flv: "video/x-flv",
35
+ m4v: "video/mp4",
36
+ // document
37
+ pdf: "application/pdf",
38
+ doc: "application/msword",
39
+ docx: "application/vnd.openxmlformats-officedocument.wordprocessingml.document",
40
+ xls: "application/vnd.ms-excel",
41
+ xlsx: "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet",
42
+ ppt: "application/vnd.ms-powerpoint",
43
+ pptx: "application/vnd.openxmlformats-officedocument.presentationml.presentation",
44
+ // text / data
45
+ txt: "text/plain",
46
+ csv: "text/csv",
47
+ json: "application/json",
48
+ xml: "application/xml",
49
+ html: "text/html",
50
+ md: "text/markdown",
51
+ yaml: "text/yaml",
52
+ yml: "text/yaml",
53
+ // archive
54
+ zip: "application/zip",
55
+ gz: "application/gzip",
56
+ tar: "application/x-tar",
57
+ "7z": "application/x-7z-compressed",
58
+ rar: "application/vnd.rar",
59
+ };
60
+ /** Infer MIME type from a file path's extension. */
61
+ export function inferMime(filePath) {
62
+ const ext = path.extname(filePath).toLowerCase().replace(".", "");
63
+ return MIME_MAP[ext] ?? "application/octet-stream";
64
+ }
65
+ /** Classify a MIME type into a ContentPart.type value. */
66
+ export function classifyMedia(mime) {
67
+ const m = mime.toLowerCase();
68
+ if (m.startsWith("image/"))
69
+ return "image";
70
+ if (m.startsWith("audio/"))
71
+ return "audio";
72
+ if (m.startsWith("video/"))
73
+ return "video";
74
+ if (m === "application/pdf")
75
+ return "pdf";
76
+ return "file";
77
+ }
78
+ //# sourceMappingURL=media-util.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"media-util.js","sourceRoot":"","sources":["../src/media-util.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,KAAK,IAAI,MAAM,MAAM,CAAC;AAE7B,MAAM,QAAQ,GAA2B;IACvC,QAAQ;IACR,GAAG,EAAE,WAAW;IAChB,GAAG,EAAE,YAAY;IACjB,IAAI,EAAE,YAAY;IAClB,GAAG,EAAE,WAAW;IAChB,IAAI,EAAE,YAAY;IAClB,GAAG,EAAE,eAAe;IACpB,GAAG,EAAE,WAAW;IAChB,GAAG,EAAE,cAAc;IACnB,IAAI,EAAE,YAAY;IAClB,GAAG,EAAE,YAAY;IACjB,IAAI,EAAE,YAAY;IAClB,QAAQ;IACR,GAAG,EAAE,YAAY;IACjB,GAAG,EAAE,WAAW;IAChB,GAAG,EAAE,WAAW;IAChB,IAAI,EAAE,YAAY;IAClB,IAAI,EAAE,YAAY;IAClB,GAAG,EAAE,WAAW;IAChB,GAAG,EAAE,WAAW;IAChB,GAAG,EAAE,gBAAgB;IACrB,QAAQ;IACR,GAAG,EAAE,WAAW;IAChB,IAAI,EAAE,YAAY;IAClB,GAAG,EAAE,kBAAkB;IACvB,GAAG,EAAE,iBAAiB;IACtB,GAAG,EAAE,iBAAiB;IACtB,GAAG,EAAE,gBAAgB;IACrB,GAAG,EAAE,aAAa;IAClB,GAAG,EAAE,WAAW;IAChB,WAAW;IACX,GAAG,EAAE,iBAAiB;IACtB,GAAG,EAAE,oBAAoB;IACzB,IAAI,EAAE,yEAAyE;IAC/E,GAAG,EAAE,0BAA0B;IAC/B,IAAI,EAAE,mEAAmE;IACzE,GAAG,EAAE,+BAA+B;IACpC,IAAI,EAAE,2EAA2E;IACjF,cAAc;IACd,GAAG,EAAE,YAAY;IACjB,GAAG,EAAE,UAAU;IACf,IAAI,EAAE,kBAAkB;IACxB,GAAG,EAAE,iBAAiB;IACtB,IAAI,EAAE,WAAW;IACjB,EAAE,EAAE,eAAe;IACnB,IAAI,EAAE,WAAW;IACjB,GAAG,EAAE,WAAW;IAChB,UAAU;IACV,GAAG,EAAE,iBAAiB;IACtB,EAAE,EAAE,kBAAkB;IACtB,GAAG,EAAE,mBAAmB;IACxB,IAAI,EAAE,6BAA6B;IACnC,GAAG,EAAE,qBAAqB;CAC3B,CAAC;AAEF,oDAAoD;AACpD,MAAM,UAAU,SAAS,CAAC,QAAgB;IACxC,MAAM,GAAG,GAAG,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC,WAAW,EAAE,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;IAClE,OAAO,QAAQ,CAAC,GAAG,CAAC,IAAI,0BAA0B,CAAC;AACrD,CAAC;AAED,0DAA0D;AAC1D,MAAM,UAAU,aAAa,CAAC,IAAY;IACxC,MAAM,CAAC,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;IAC7B,IAAI,CAAC,CAAC,UAAU,CAAC,QAAQ,CAAC;QAAE,OAAO,OAAO,CAAC;IAC3C,IAAI,CAAC,CAAC,UAAU,CAAC,QAAQ,CAAC;QAAE,OAAO,OAAO,CAAC;IAC3C,IAAI,CAAC,CAAC,UAAU,CAAC,QAAQ,CAAC;QAAE,OAAO,OAAO,CAAC;IAC3C,IAAI,CAAC,KAAK,iBAAiB;QAAE,OAAO,KAAK,CAAC;IAC1C,OAAO,MAAM,CAAC;AAChB,CAAC"}
@@ -0,0 +1,2 @@
1
+ export { autoUploadMedia, resolveFileArgsInPlace, resolveToLocal, materializeInlineMedia } from './media.js';
2
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/middleware/index.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,eAAe,EACf,sBAAsB,EACtB,cAAc,EACd,sBAAsB,EACvB,MAAM,YAAY,CAAC"}
@@ -0,0 +1,2 @@
1
+ export { autoUploadMedia, resolveFileArgsInPlace, resolveToLocal, materializeInlineMedia } from './media.js';
2
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/middleware/index.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,eAAe,EACf,sBAAsB,EACtB,cAAc,EACd,sBAAsB,EACvB,MAAM,YAAY,CAAC"}
@@ -0,0 +1,7 @@
1
+ import type { ContentPart } from "../types.js";
2
+ import type { BlobClient } from "../blob/client.js";
3
+ export declare function autoUploadMedia(content: ContentPart[], blobClient: BlobClient): Promise<void>;
4
+ export declare function resolveFileArgsInPlace(args: Record<string, unknown>, blobClient: BlobClient, blobUrlPrefix: string): Promise<void>;
5
+ export declare function resolveToLocal(pathOrUrl: string, blobClient: BlobClient, endpointId: string, blobUrlPrefix: string): Promise<string>;
6
+ export declare function materializeInlineMedia(data: string, mime: string | undefined, type: Exclude<ContentPart["type"], "text">): string;
7
+ //# sourceMappingURL=media.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"media.d.ts","sourceRoot":"","sources":["../../src/middleware/media.ts"],"names":[],"mappings":"AAIA,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,aAAa,CAAC;AAG/C,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,mBAAmB,CAAC;AAEpD,wBAAsB,eAAe,CACnC,OAAO,EAAE,WAAW,EAAE,EACtB,UAAU,EAAE,UAAU,GACrB,OAAO,CAAC,IAAI,CAAC,CAsBf;AAED,wBAAsB,sBAAsB,CAC1C,IAAI,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAC7B,UAAU,EAAE,UAAU,EACtB,aAAa,EAAE,MAAM,GACpB,OAAO,CAAC,IAAI,CAAC,CAwBf;AAED,wBAAsB,cAAc,CAClC,SAAS,EAAE,MAAM,EACjB,UAAU,EAAE,UAAU,EACtB,UAAU,EAAE,MAAM,EAClB,aAAa,EAAE,MAAM,GACpB,OAAO,CAAC,MAAM,CAAC,CAejB;AAoBD,wBAAgB,sBAAsB,CACpC,IAAI,EAAE,MAAM,EACZ,IAAI,EAAE,MAAM,GAAG,SAAS,EACxB,IAAI,EAAE,OAAO,CAAC,WAAW,CAAC,MAAM,CAAC,EAAE,MAAM,CAAC,GACzC,MAAM,CASR"}
@@ -0,0 +1,112 @@
1
+ import * as path from "path";
2
+ import * as fs from "fs";
3
+ import * as os from "os";
4
+ import * as crypto from "crypto";
5
+ import { inferMime } from "../media-util.js";
6
+ import { parseFileRef, isFileRef } from "../fileref.js";
7
+ export async function autoUploadMedia(content, blobClient) {
8
+ for (let i = 0; i < content.length; i++) {
9
+ const part = content[i];
10
+ if (!part.media?.path)
11
+ continue;
12
+ const absPath = path.resolve(part.media.path);
13
+ if (!fs.existsSync(absPath)) {
14
+ throw new Error(`auto-upload: file not found: ${absPath}`);
15
+ }
16
+ const { fileRef } = await blobClient.upload(absPath);
17
+ content[i] = {
18
+ ...part,
19
+ media: {
20
+ type: part.media.type,
21
+ fileRef,
22
+ mime: part.media.mime || inferMime(absPath),
23
+ name: part.media.name || path.basename(absPath),
24
+ sourcePath: absPath,
25
+ },
26
+ };
27
+ }
28
+ }
29
+ export async function resolveFileArgsInPlace(args, blobClient, blobUrlPrefix) {
30
+ for (const [key, value] of Object.entries(args)) {
31
+ if (typeof value === "string" && value.startsWith(blobUrlPrefix)) {
32
+ try {
33
+ args[key] = await blobClient.download(value);
34
+ }
35
+ catch {
36
+ // keep original
37
+ }
38
+ }
39
+ else if (Array.isArray(value)) {
40
+ for (let i = 0; i < value.length; i++) {
41
+ if (typeof value[i] === "string" && value[i].startsWith(blobUrlPrefix)) {
42
+ try {
43
+ value[i] = await blobClient.download(value[i]);
44
+ }
45
+ catch {
46
+ // keep
47
+ }
48
+ }
49
+ else if (typeof value[i] === "object" && value[i] !== null) {
50
+ await resolveFileArgsInPlace(value[i], blobClient, blobUrlPrefix);
51
+ }
52
+ }
53
+ }
54
+ else if (typeof value === "object" && value !== null) {
55
+ await resolveFileArgsInPlace(value, blobClient, blobUrlPrefix);
56
+ }
57
+ }
58
+ }
59
+ export async function resolveToLocal(pathOrUrl, blobClient, endpointId, blobUrlPrefix) {
60
+ if (isFileRef(pathOrUrl)) {
61
+ const parsed = parseFileRef(pathOrUrl);
62
+ if (parsed.type === "endpoint" && parsed.endpointId === endpointId) {
63
+ return parsed.path;
64
+ }
65
+ if (parsed.type === "blob") {
66
+ return blobClient.download(`${blobClient['opts'].httpBase}/api/blob/${parsed.blobId}`);
67
+ }
68
+ return pathOrUrl;
69
+ }
70
+ if (pathOrUrl.startsWith(blobUrlPrefix)) {
71
+ return blobClient.download(pathOrUrl);
72
+ }
73
+ return pathOrUrl;
74
+ }
75
+ function tmpExtensionForMedia(mime, type) {
76
+ const normalized = (mime || "").toLowerCase();
77
+ if (normalized === "image/png")
78
+ return ".png";
79
+ if (normalized === "image/jpeg")
80
+ return ".jpg";
81
+ if (normalized === "image/webp")
82
+ return ".webp";
83
+ if (normalized === "image/gif")
84
+ return ".gif";
85
+ if (normalized === "audio/mpeg")
86
+ return ".mp3";
87
+ if (normalized === "audio/wav" || normalized === "audio/x-wav")
88
+ return ".wav";
89
+ if (normalized === "audio/ogg")
90
+ return ".ogg";
91
+ if (normalized === "video/mp4")
92
+ return ".mp4";
93
+ if (normalized === "application/pdf")
94
+ return ".pdf";
95
+ if (type === "image")
96
+ return ".png";
97
+ if (type === "audio")
98
+ return ".mp3";
99
+ if (type === "video")
100
+ return ".mp4";
101
+ if (type === "pdf")
102
+ return ".pdf";
103
+ return ".bin";
104
+ }
105
+ export function materializeInlineMedia(data, mime, type) {
106
+ const dir = path.join(os.tmpdir(), "ailo_mcp_media");
107
+ fs.mkdirSync(dir, { recursive: true });
108
+ const filePath = path.join(dir, `${Date.now()}_${crypto.randomUUID()}${tmpExtensionForMedia(mime, type)}`);
109
+ fs.writeFileSync(filePath, Buffer.from(data, "base64"));
110
+ return filePath;
111
+ }
112
+ //# sourceMappingURL=media.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"media.js","sourceRoot":"","sources":["../../src/middleware/media.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,19 @@
1
+ /**
2
+ * skill-loader.ts — Scans local SKILL.md files and parses them into SkillMeta[].
3
+ *
4
+ * Compatible with the ~/.agents/skills/ convention used by Cursor, CoPaw and others.
5
+ * Each skill lives in its own subdirectory with a SKILL.md file:
6
+ *
7
+ * ~/.agents/skills/
8
+ * pdf-processing/
9
+ * SKILL.md ← YAML frontmatter (name, description) + markdown body
10
+ * ui-analysis/
11
+ * SKILL.md
12
+ */
13
+ import type { SkillMeta } from "./types.js";
14
+ /**
15
+ * Scan one or more directories for `*​/SKILL.md` files, parse frontmatter, return SkillMeta[].
16
+ * Directories that don't exist are silently skipped.
17
+ */
18
+ export declare function loadSkills(dirs?: string[]): SkillMeta[];
19
+ //# sourceMappingURL=skill-loader.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"skill-loader.d.ts","sourceRoot":"","sources":["../src/skill-loader.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;GAWG;AAMH,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,YAAY,CAAC;AAE5C;;;GAGG;AACH,wBAAgB,UAAU,CAAC,IAAI,CAAC,EAAE,MAAM,EAAE,GAAG,SAAS,EAAE,CA8CvD"}
@@ -0,0 +1,131 @@
1
+ /**
2
+ * skill-loader.ts — Scans local SKILL.md files and parses them into SkillMeta[].
3
+ *
4
+ * Compatible with the ~/.agents/skills/ convention used by Cursor, CoPaw and others.
5
+ * Each skill lives in its own subdirectory with a SKILL.md file:
6
+ *
7
+ * ~/.agents/skills/
8
+ * pdf-processing/
9
+ * SKILL.md ← YAML frontmatter (name, description) + markdown body
10
+ * ui-analysis/
11
+ * SKILL.md
12
+ */
13
+ import fs from "fs";
14
+ import path from "path";
15
+ import os from "os";
16
+ import yaml from "js-yaml";
17
+ /**
18
+ * Scan one or more directories for `*​/SKILL.md` files, parse frontmatter, return SkillMeta[].
19
+ * Directories that don't exist are silently skipped.
20
+ */
21
+ export function loadSkills(dirs) {
22
+ const resolved = (dirs && dirs.length > 0)
23
+ ? dirs
24
+ : [path.join(os.homedir(), ".agents", "skills")];
25
+ const skills = [];
26
+ const seen = new Set();
27
+ for (const dir of resolved) {
28
+ const expanded = dir.startsWith("~")
29
+ ? path.join(os.homedir(), dir.slice(1))
30
+ : dir;
31
+ if (!fs.existsSync(expanded))
32
+ continue;
33
+ let entries;
34
+ try {
35
+ entries = fs.readdirSync(expanded, { withFileTypes: true });
36
+ }
37
+ catch {
38
+ continue;
39
+ }
40
+ for (const entry of entries) {
41
+ if (!entry.isDirectory())
42
+ continue;
43
+ const skillFile = path.join(expanded, entry.name, "SKILL.md");
44
+ if (!fs.existsSync(skillFile))
45
+ continue;
46
+ try {
47
+ const raw = fs.readFileSync(skillFile, "utf-8");
48
+ const meta = parseSkillMd(raw, entry.name);
49
+ if (meta) {
50
+ if (seen.has(meta.name)) {
51
+ const idx = skills.findIndex((s) => s.name === meta.name);
52
+ if (idx >= 0)
53
+ skills[idx] = meta;
54
+ }
55
+ else {
56
+ seen.add(meta.name);
57
+ skills.push(meta);
58
+ }
59
+ }
60
+ }
61
+ catch {
62
+ // skip unreadable files
63
+ }
64
+ }
65
+ }
66
+ return skills;
67
+ }
68
+ /**
69
+ * Parse a SKILL.md file with optional YAML frontmatter.
70
+ *
71
+ * Frontmatter format:
72
+ * ```
73
+ * ---
74
+ * name: pdf-processing
75
+ * description: Extract text and tables from PDF files
76
+ * ---
77
+ * (markdown body)
78
+ * ```
79
+ *
80
+ * If no frontmatter, `dirName` is used as the name and the first non-empty line as description.
81
+ */
82
+ function parseSkillMd(raw, dirName) {
83
+ const trimmed = raw.trim();
84
+ if (!trimmed)
85
+ return null;
86
+ let name = dirName;
87
+ let description = "";
88
+ let content;
89
+ if (trimmed.startsWith("---")) {
90
+ const endIdx = trimmed.indexOf("---", 3);
91
+ if (endIdx > 0) {
92
+ const frontmatter = trimmed.slice(3, endIdx);
93
+ content = trimmed.slice(endIdx + 3).trim();
94
+ try {
95
+ const parsed = yaml.load(frontmatter);
96
+ if (parsed && typeof parsed === "object") {
97
+ if (typeof parsed.name === "string" && parsed.name)
98
+ name = parsed.name;
99
+ if (typeof parsed.description === "string" && parsed.description)
100
+ description = parsed.description;
101
+ }
102
+ }
103
+ catch {
104
+ // Fall back to simple string parsing if yaml fails
105
+ for (const line of frontmatter.split("\n")) {
106
+ const colonIdx = line.indexOf(":");
107
+ if (colonIdx < 0)
108
+ continue;
109
+ const key = line.slice(0, colonIdx).trim().toLowerCase();
110
+ const val = line.slice(colonIdx + 1).trim().replace(/^["']|["']$/g, "");
111
+ if (key === "name" && val)
112
+ name = val;
113
+ if (key === "description" && val)
114
+ description = val;
115
+ }
116
+ }
117
+ }
118
+ else {
119
+ content = trimmed;
120
+ }
121
+ }
122
+ else {
123
+ content = trimmed;
124
+ }
125
+ if (!description) {
126
+ const firstLine = content.split("\n").find((l) => l.trim() && !l.startsWith("#"));
127
+ description = firstLine?.trim().slice(0, 200) ?? name;
128
+ }
129
+ return { name, description, content };
130
+ }
131
+ //# sourceMappingURL=skill-loader.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"skill-loader.js","sourceRoot":"","sources":["../src/skill-loader.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ export { normalizeToolResult } from './normalize.js';
2
+ export type { ToolHandler } from './normalize.js';
3
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/tool-dispatch/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,mBAAmB,EAAE,MAAM,gBAAgB,CAAC;AACrD,YAAY,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC"}
@@ -0,0 +1,2 @@
1
+ export { normalizeToolResult } from './normalize.js';
2
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/tool-dispatch/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,mBAAmB,EAAE,MAAM,gBAAgB,CAAC"}
@@ -0,0 +1,5 @@
1
+ import type { ContentPart, ToolHandler } from "../types.js";
2
+ import type { BlobClient } from "../blob/client.js";
3
+ export declare function normalizeToolResult(result: unknown, blobClient: BlobClient): Promise<ContentPart[] | unknown>;
4
+ export type { ToolHandler };
5
+ //# sourceMappingURL=normalize.d.ts.map