@tachu/extensions 1.0.0-alpha.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.
Files changed (210) hide show
  1. package/CHANGELOG.md +119 -0
  2. package/LICENSE +201 -0
  3. package/README.md +1104 -0
  4. package/README_ZH.md +1082 -0
  5. package/dist/backends/file.d.ts +18 -0
  6. package/dist/backends/file.d.ts.map +1 -0
  7. package/dist/backends/file.js +85 -0
  8. package/dist/backends/file.js.map +1 -0
  9. package/dist/backends/index.d.ts +4 -0
  10. package/dist/backends/index.d.ts.map +1 -0
  11. package/dist/backends/index.js +4 -0
  12. package/dist/backends/index.js.map +1 -0
  13. package/dist/backends/terminal.d.ts +18 -0
  14. package/dist/backends/terminal.d.ts.map +1 -0
  15. package/dist/backends/terminal.js +81 -0
  16. package/dist/backends/terminal.js.map +1 -0
  17. package/dist/backends/web.d.ts +18 -0
  18. package/dist/backends/web.d.ts.map +1 -0
  19. package/dist/backends/web.js +55 -0
  20. package/dist/backends/web.js.map +1 -0
  21. package/dist/common/net.d.ts +39 -0
  22. package/dist/common/net.d.ts.map +1 -0
  23. package/dist/common/net.js +177 -0
  24. package/dist/common/net.js.map +1 -0
  25. package/dist/common/path.d.ts +51 -0
  26. package/dist/common/path.d.ts.map +1 -0
  27. package/dist/common/path.js +76 -0
  28. package/dist/common/path.js.map +1 -0
  29. package/dist/common/process.d.ts +19 -0
  30. package/dist/common/process.d.ts.map +1 -0
  31. package/dist/common/process.js +67 -0
  32. package/dist/common/process.js.map +1 -0
  33. package/dist/index.d.ts +13 -0
  34. package/dist/index.d.ts.map +1 -0
  35. package/dist/index.js +13 -0
  36. package/dist/index.js.map +1 -0
  37. package/dist/mcp/index.d.ts +3 -0
  38. package/dist/mcp/index.d.ts.map +1 -0
  39. package/dist/mcp/index.js +3 -0
  40. package/dist/mcp/index.js.map +1 -0
  41. package/dist/mcp/sse-adapter.d.ts +82 -0
  42. package/dist/mcp/sse-adapter.d.ts.map +1 -0
  43. package/dist/mcp/sse-adapter.js +201 -0
  44. package/dist/mcp/sse-adapter.js.map +1 -0
  45. package/dist/mcp/stdio-adapter.d.ts +85 -0
  46. package/dist/mcp/stdio-adapter.d.ts.map +1 -0
  47. package/dist/mcp/stdio-adapter.js +203 -0
  48. package/dist/mcp/stdio-adapter.js.map +1 -0
  49. package/dist/memory/fs-memory-system.d.ts +147 -0
  50. package/dist/memory/fs-memory-system.d.ts.map +1 -0
  51. package/dist/memory/fs-memory-system.js +266 -0
  52. package/dist/memory/fs-memory-system.js.map +1 -0
  53. package/dist/memory/index.d.ts +2 -0
  54. package/dist/memory/index.d.ts.map +1 -0
  55. package/dist/memory/index.js +2 -0
  56. package/dist/memory/index.js.map +1 -0
  57. package/dist/observability/index.d.ts +3 -0
  58. package/dist/observability/index.d.ts.map +1 -0
  59. package/dist/observability/index.js +3 -0
  60. package/dist/observability/index.js.map +1 -0
  61. package/dist/observability/jsonl-emitter.d.ts +58 -0
  62. package/dist/observability/jsonl-emitter.d.ts.map +1 -0
  63. package/dist/observability/jsonl-emitter.js +96 -0
  64. package/dist/observability/jsonl-emitter.js.map +1 -0
  65. package/dist/observability/otel-emitter.d.ts +52 -0
  66. package/dist/observability/otel-emitter.d.ts.map +1 -0
  67. package/dist/observability/otel-emitter.js +143 -0
  68. package/dist/observability/otel-emitter.js.map +1 -0
  69. package/dist/providers/anthropic.d.ts +73 -0
  70. package/dist/providers/anthropic.d.ts.map +1 -0
  71. package/dist/providers/anthropic.js +521 -0
  72. package/dist/providers/anthropic.js.map +1 -0
  73. package/dist/providers/index.d.ts +5 -0
  74. package/dist/providers/index.d.ts.map +1 -0
  75. package/dist/providers/index.js +5 -0
  76. package/dist/providers/index.js.map +1 -0
  77. package/dist/providers/mock.d.ts +81 -0
  78. package/dist/providers/mock.d.ts.map +1 -0
  79. package/dist/providers/mock.js +160 -0
  80. package/dist/providers/mock.js.map +1 -0
  81. package/dist/providers/openai.d.ts +95 -0
  82. package/dist/providers/openai.d.ts.map +1 -0
  83. package/dist/providers/openai.js +529 -0
  84. package/dist/providers/openai.js.map +1 -0
  85. package/dist/providers/qwen.d.ts +145 -0
  86. package/dist/providers/qwen.d.ts.map +1 -0
  87. package/dist/providers/qwen.js +669 -0
  88. package/dist/providers/qwen.js.map +1 -0
  89. package/dist/rules/index.d.ts +9 -0
  90. package/dist/rules/index.d.ts.map +1 -0
  91. package/dist/rules/index.js +15 -0
  92. package/dist/rules/index.js.map +1 -0
  93. package/dist/rules/no-hallucination.md +11 -0
  94. package/dist/rules/no-sensitive-output.md +11 -0
  95. package/dist/rules/prefer-concise-response.md +11 -0
  96. package/dist/rules/require-tool-verification.md +11 -0
  97. package/dist/safety/default-gate.d.ts +112 -0
  98. package/dist/safety/default-gate.d.ts.map +1 -0
  99. package/dist/safety/default-gate.js +188 -0
  100. package/dist/safety/default-gate.js.map +1 -0
  101. package/dist/safety/index.d.ts +2 -0
  102. package/dist/safety/index.d.ts.map +1 -0
  103. package/dist/safety/index.js +2 -0
  104. package/dist/safety/index.js.map +1 -0
  105. package/dist/tools/_shared/web-client.d.ts +18 -0
  106. package/dist/tools/_shared/web-client.d.ts.map +1 -0
  107. package/dist/tools/_shared/web-client.js +46 -0
  108. package/dist/tools/_shared/web-client.js.map +1 -0
  109. package/dist/tools/apply-patch/descriptor.md +27 -0
  110. package/dist/tools/apply-patch/executor.d.ts +19 -0
  111. package/dist/tools/apply-patch/executor.d.ts.map +1 -0
  112. package/dist/tools/apply-patch/executor.js +193 -0
  113. package/dist/tools/apply-patch/executor.js.map +1 -0
  114. package/dist/tools/fetch-url/descriptor.md +44 -0
  115. package/dist/tools/fetch-url/executor.d.ts +28 -0
  116. package/dist/tools/fetch-url/executor.d.ts.map +1 -0
  117. package/dist/tools/fetch-url/executor.js +115 -0
  118. package/dist/tools/fetch-url/executor.js.map +1 -0
  119. package/dist/tools/index.d.ts +12 -0
  120. package/dist/tools/index.d.ts.map +1 -0
  121. package/dist/tools/index.js +286 -0
  122. package/dist/tools/index.js.map +1 -0
  123. package/dist/tools/list-dir/descriptor.md +29 -0
  124. package/dist/tools/list-dir/executor.d.ts +22 -0
  125. package/dist/tools/list-dir/executor.d.ts.map +1 -0
  126. package/dist/tools/list-dir/executor.js +48 -0
  127. package/dist/tools/list-dir/executor.js.map +1 -0
  128. package/dist/tools/read-file/descriptor.md +28 -0
  129. package/dist/tools/read-file/executor.d.ts +15 -0
  130. package/dist/tools/read-file/executor.d.ts.map +1 -0
  131. package/dist/tools/read-file/executor.js +22 -0
  132. package/dist/tools/read-file/executor.js.map +1 -0
  133. package/dist/tools/run-shell/descriptor.md +39 -0
  134. package/dist/tools/run-shell/executor.d.ts +20 -0
  135. package/dist/tools/run-shell/executor.d.ts.map +1 -0
  136. package/dist/tools/run-shell/executor.js +76 -0
  137. package/dist/tools/run-shell/executor.js.map +1 -0
  138. package/dist/tools/search-code/descriptor.md +31 -0
  139. package/dist/tools/search-code/executor.d.ts +23 -0
  140. package/dist/tools/search-code/executor.d.ts.map +1 -0
  141. package/dist/tools/search-code/executor.js +122 -0
  142. package/dist/tools/search-code/executor.js.map +1 -0
  143. package/dist/tools/shared.d.ts +47 -0
  144. package/dist/tools/shared.d.ts.map +1 -0
  145. package/dist/tools/shared.js +27 -0
  146. package/dist/tools/shared.js.map +1 -0
  147. package/dist/tools/web-fetch/descriptor.md +198 -0
  148. package/dist/tools/web-fetch/errors.d.ts +32 -0
  149. package/dist/tools/web-fetch/errors.d.ts.map +1 -0
  150. package/dist/tools/web-fetch/errors.js +91 -0
  151. package/dist/tools/web-fetch/errors.js.map +1 -0
  152. package/dist/tools/web-fetch/executor.d.ts +10 -0
  153. package/dist/tools/web-fetch/executor.d.ts.map +1 -0
  154. package/dist/tools/web-fetch/executor.js +191 -0
  155. package/dist/tools/web-fetch/executor.js.map +1 -0
  156. package/dist/tools/web-fetch/index.d.ts +4 -0
  157. package/dist/tools/web-fetch/index.d.ts.map +1 -0
  158. package/dist/tools/web-fetch/index.js +3 -0
  159. package/dist/tools/web-fetch/index.js.map +1 -0
  160. package/dist/tools/web-fetch/types.d.ts +157 -0
  161. package/dist/tools/web-fetch/types.d.ts.map +1 -0
  162. package/dist/tools/web-fetch/types.js +7 -0
  163. package/dist/tools/web-fetch/types.js.map +1 -0
  164. package/dist/tools/web-search/descriptor.md +89 -0
  165. package/dist/tools/web-search/errors.d.ts +33 -0
  166. package/dist/tools/web-search/errors.d.ts.map +1 -0
  167. package/dist/tools/web-search/errors.js +45 -0
  168. package/dist/tools/web-search/errors.js.map +1 -0
  169. package/dist/tools/web-search/executor.d.ts +10 -0
  170. package/dist/tools/web-search/executor.d.ts.map +1 -0
  171. package/dist/tools/web-search/executor.js +185 -0
  172. package/dist/tools/web-search/executor.js.map +1 -0
  173. package/dist/tools/web-search/index.d.ts +4 -0
  174. package/dist/tools/web-search/index.d.ts.map +1 -0
  175. package/dist/tools/web-search/index.js +3 -0
  176. package/dist/tools/web-search/index.js.map +1 -0
  177. package/dist/tools/web-search/types.d.ts +86 -0
  178. package/dist/tools/web-search/types.d.ts.map +1 -0
  179. package/dist/tools/web-search/types.js +7 -0
  180. package/dist/tools/web-search/types.js.map +1 -0
  181. package/dist/tools/write-file/descriptor.md +30 -0
  182. package/dist/tools/write-file/executor.d.ts +16 -0
  183. package/dist/tools/write-file/executor.d.ts.map +1 -0
  184. package/dist/tools/write-file/executor.js +18 -0
  185. package/dist/tools/write-file/executor.js.map +1 -0
  186. package/dist/transformers/document-to-text.d.ts +23 -0
  187. package/dist/transformers/document-to-text.d.ts.map +1 -0
  188. package/dist/transformers/document-to-text.js +69 -0
  189. package/dist/transformers/document-to-text.js.map +1 -0
  190. package/dist/transformers/image-to-text.d.ts +38 -0
  191. package/dist/transformers/image-to-text.d.ts.map +1 -0
  192. package/dist/transformers/image-to-text.js +82 -0
  193. package/dist/transformers/image-to-text.js.map +1 -0
  194. package/dist/transformers/index.d.ts +3 -0
  195. package/dist/transformers/index.d.ts.map +1 -0
  196. package/dist/transformers/index.js +3 -0
  197. package/dist/transformers/index.js.map +1 -0
  198. package/dist/vector/index.d.ts +3 -0
  199. package/dist/vector/index.d.ts.map +1 -0
  200. package/dist/vector/index.js +3 -0
  201. package/dist/vector/index.js.map +1 -0
  202. package/dist/vector/local-fs.d.ts +76 -0
  203. package/dist/vector/local-fs.d.ts.map +1 -0
  204. package/dist/vector/local-fs.js +153 -0
  205. package/dist/vector/local-fs.js.map +1 -0
  206. package/dist/vector/qdrant.d.ts +85 -0
  207. package/dist/vector/qdrant.d.ts.map +1 -0
  208. package/dist/vector/qdrant.js +208 -0
  209. package/dist/vector/qdrant.js.map +1 -0
  210. package/package.json +74 -0
@@ -0,0 +1,185 @@
1
+ import { buildWebFetchJsonHeaders, getWebFetchEndpointBase, readWebFetchClientTimeoutMs, } from "../_shared/web-client";
2
+ import { assertNotAborted } from "../shared";
3
+ import { getMalformedResponseError, getNetworkError } from "../web-fetch/errors";
4
+ import { asWebSearchClientError, getSearchTimeoutError, mapSearchServerErrorToClient, } from "./errors";
5
+ function toSearchRequest(input) {
6
+ const { timeoutMs: _timeoutMs, ...rest } = input;
7
+ return {
8
+ ...rest,
9
+ traceId: null,
10
+ };
11
+ }
12
+ function isRecord(value) {
13
+ return typeof value === "object" && value !== null && !Array.isArray(value);
14
+ }
15
+ function isSearchResultExtract(value) {
16
+ if (!isRecord(value))
17
+ return false;
18
+ if (typeof value.status !== "number")
19
+ return false;
20
+ if (value.renderedWith !== "static" && value.renderedWith !== "browser")
21
+ return false;
22
+ if (typeof value.body !== "string")
23
+ return false;
24
+ if (typeof value.wordCount !== "number")
25
+ return false;
26
+ if (typeof value.truncated !== "boolean")
27
+ return false;
28
+ if (!Array.isArray(value.warnings))
29
+ return false;
30
+ if (value.title !== undefined && typeof value.title !== "string")
31
+ return false;
32
+ return true;
33
+ }
34
+ function isSearchResultItem(value) {
35
+ if (!isRecord(value))
36
+ return false;
37
+ if (typeof value.title !== "string")
38
+ return false;
39
+ if (typeof value.url !== "string")
40
+ return false;
41
+ if (typeof value.snippet !== "string")
42
+ return false;
43
+ if (value.publishedAt !== undefined && value.publishedAt !== null && typeof value.publishedAt !== "string") {
44
+ return false;
45
+ }
46
+ if (value.score !== undefined && value.score !== null && typeof value.score !== "number") {
47
+ return false;
48
+ }
49
+ if (value.extract !== undefined && !isSearchResultExtract(value.extract))
50
+ return false;
51
+ return true;
52
+ }
53
+ function isSearchResponse(value) {
54
+ if (!isRecord(value))
55
+ return false;
56
+ if (typeof value.query !== "string")
57
+ return false;
58
+ if (typeof value.provider !== "string")
59
+ return false;
60
+ if (!Array.isArray(value.results))
61
+ return false;
62
+ for (const item of value.results) {
63
+ if (!isSearchResultItem(item))
64
+ return false;
65
+ }
66
+ if (typeof value.totalResults !== "number")
67
+ return false;
68
+ if (typeof value.searchedAtMs !== "number")
69
+ return false;
70
+ if (!Array.isArray(value.warnings))
71
+ return false;
72
+ if (typeof value.traceId !== "string")
73
+ return false;
74
+ return true;
75
+ }
76
+ function toToolOutput(res) {
77
+ return {
78
+ query: res.query,
79
+ provider: res.provider,
80
+ results: res.results.map((r) => {
81
+ const row = {
82
+ title: r.title,
83
+ url: r.url,
84
+ snippet: r.snippet,
85
+ };
86
+ if (r.publishedAt !== undefined)
87
+ row.publishedAt = r.publishedAt;
88
+ if (r.score !== undefined)
89
+ row.score = r.score;
90
+ if (r.extract !== undefined) {
91
+ row.extract = {
92
+ status: r.extract.status,
93
+ renderedWith: r.extract.renderedWith,
94
+ body: r.extract.body,
95
+ wordCount: r.extract.wordCount,
96
+ truncated: r.extract.truncated,
97
+ warnings: r.extract.warnings,
98
+ };
99
+ if (r.extract.title !== undefined)
100
+ row.extract.title = r.extract.title;
101
+ }
102
+ return row;
103
+ }),
104
+ totalResults: res.totalResults,
105
+ warnings: res.warnings,
106
+ };
107
+ }
108
+ function tryParseErrorBody(text) {
109
+ try {
110
+ const parsed = JSON.parse(text);
111
+ if (!isRecord(parsed))
112
+ return null;
113
+ const err = parsed.error;
114
+ if (!isRecord(err))
115
+ return null;
116
+ if (typeof err.code !== "string")
117
+ return null;
118
+ if (typeof err.message !== "string")
119
+ return null;
120
+ if (typeof err.requestId !== "string")
121
+ return null;
122
+ const body = {
123
+ error: {
124
+ code: err.code,
125
+ message: err.message,
126
+ requestId: err.requestId,
127
+ ...(isRecord(err.detail) ? { detail: err.detail } : {}),
128
+ },
129
+ };
130
+ return body;
131
+ }
132
+ catch {
133
+ return null;
134
+ }
135
+ }
136
+ /**
137
+ * 调用 Web Fetch Server `POST /v1/search`。
138
+ *
139
+ * @see docs/adr/decisions/0003a-web-fetch-api-contract.md §Endpoint 3
140
+ * @see docs/adr/decisions/0003d-web-fetch-errors.md §2.2
141
+ */
142
+ export async function executeWebSearch(input, ctx) {
143
+ assertNotAborted(ctx.abortSignal);
144
+ const endpoint = getWebFetchEndpointBase();
145
+ const url = `${endpoint}/v1/search`;
146
+ const timeoutMs = readWebFetchClientTimeoutMs(input.timeoutMs);
147
+ const signal = AbortSignal.any([AbortSignal.timeout(timeoutMs), ctx.abortSignal]);
148
+ const headers = buildWebFetchJsonHeaders();
149
+ let response;
150
+ try {
151
+ response = await fetch(url, {
152
+ method: "POST",
153
+ headers,
154
+ body: JSON.stringify(toSearchRequest(input)),
155
+ signal,
156
+ });
157
+ }
158
+ catch (e) {
159
+ if (e instanceof Error && e.name === "AbortError") {
160
+ if (ctx.abortSignal.aborted) {
161
+ throw ctx.abortSignal.reason ?? e;
162
+ }
163
+ throw getSearchTimeoutError(timeoutMs);
164
+ }
165
+ throw asWebSearchClientError(getNetworkError(endpoint, e));
166
+ }
167
+ if (!response.ok) {
168
+ const text = await response.text();
169
+ const parsed = tryParseErrorBody(text);
170
+ throw mapSearchServerErrorToClient(response.status, parsed, { endpoint });
171
+ }
172
+ const text = await response.text();
173
+ let parsedBody;
174
+ try {
175
+ parsedBody = JSON.parse(text);
176
+ }
177
+ catch {
178
+ throw asWebSearchClientError(getMalformedResponseError());
179
+ }
180
+ if (!isSearchResponse(parsedBody)) {
181
+ throw asWebSearchClientError(getMalformedResponseError());
182
+ }
183
+ return toToolOutput(parsedBody);
184
+ }
185
+ //# sourceMappingURL=executor.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"executor.js","sourceRoot":"","sources":["../../../src/tools/web-search/executor.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,4 @@
1
+ export { asWebSearchClientError, getSearchTimeoutError, mapSearchServerErrorToClient, WebSearchClientError, } from "./errors";
2
+ export { executeWebSearch } from "./executor";
3
+ export type { SearchRequest, SearchResponse, SearchResultItem, WebSearchToolInput, WebSearchToolOutput, } from "./types";
4
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/tools/web-search/index.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,sBAAsB,EACtB,qBAAqB,EACrB,4BAA4B,EAC5B,oBAAoB,GACrB,MAAM,UAAU,CAAC;AAClB,OAAO,EAAE,gBAAgB,EAAE,MAAM,YAAY,CAAC;AAC9C,YAAY,EACV,aAAa,EACb,cAAc,EACd,gBAAgB,EAChB,kBAAkB,EAClB,mBAAmB,GACpB,MAAM,SAAS,CAAC"}
@@ -0,0 +1,3 @@
1
+ export { asWebSearchClientError, getSearchTimeoutError, mapSearchServerErrorToClient, WebSearchClientError, } from "./errors";
2
+ export { executeWebSearch } from "./executor";
3
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/tools/web-search/index.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,sBAAsB,EACtB,qBAAqB,EACrB,4BAA4B,EAC5B,oBAAoB,GACrB,MAAM,UAAU,CAAC;AAClB,OAAO,EAAE,gBAAgB,EAAE,MAAM,YAAY,CAAC"}
@@ -0,0 +1,86 @@
1
+ /**
2
+ * Web Search 工具类型(客户端本地复刻,不依赖 `@tachu/web-fetch-server`)。
3
+ *
4
+ * @see docs/adr/decisions/0003b-web-fetch-types.md §2 / §6.2
5
+ */
6
+ import type { ExtractRequest, ExtractResponse } from "../web-fetch/types";
7
+ /**
8
+ * POST /v1/search 请求体。
9
+ * @see docs/adr/decisions/0003a-web-fetch-api-contract.md §Endpoint 3
10
+ */
11
+ export interface SearchRequest {
12
+ query: string;
13
+ maxResults?: number;
14
+ language?: string | null;
15
+ region?: string | null;
16
+ timeRange?: "day" | "week" | "month" | "year" | null;
17
+ safeSearch?: "off" | "moderate" | "strict";
18
+ includeDomains?: string[];
19
+ excludeDomains?: string[];
20
+ fetchTopN?: number;
21
+ fetchOptions?: Omit<ExtractRequest, "url" | "traceId">;
22
+ traceId?: string | null;
23
+ }
24
+ export interface SearchResponse {
25
+ query: string;
26
+ provider: string;
27
+ results: SearchResultItem[];
28
+ totalResults: number;
29
+ searchedAtMs: number;
30
+ warnings: string[];
31
+ traceId: string;
32
+ }
33
+ export interface SearchResultItem {
34
+ title: string;
35
+ url: string;
36
+ snippet: string;
37
+ publishedAt?: string | null;
38
+ score?: number | null;
39
+ extract?: Pick<ExtractResponse, "status" | "renderedWith" | "title" | "body" | "wordCount" | "truncated" | "warnings">;
40
+ }
41
+ /**
42
+ * `web-search` 工具入参(`timeoutMs` 仅客户端使用)。
43
+ * @see docs/adr/decisions/0003b-web-fetch-types.md §6.2
44
+ */
45
+ export interface WebSearchToolInput {
46
+ query: string;
47
+ maxResults?: number;
48
+ language?: string;
49
+ region?: string;
50
+ timeRange?: "day" | "week" | "month" | "year";
51
+ safeSearch?: "off" | "moderate" | "strict";
52
+ includeDomains?: string[];
53
+ excludeDomains?: string[];
54
+ /** 渲染前 N 条结果为 Markdown;上限 5 */
55
+ fetchTopN?: number;
56
+ /** `fetchTopN > 0` 时透传至服务端 extract 管线 */
57
+ fetchOptions?: Omit<ExtractRequest, "url" | "traceId">;
58
+ timeoutMs?: number;
59
+ }
60
+ /**
61
+ * `web-search` 工具成功出参(不含 `searchedAtMs` / `traceId` 等追踪字段)。
62
+ * @see docs/adr/decisions/0003b-web-fetch-types.md §6.2
63
+ */
64
+ export interface WebSearchToolOutput {
65
+ query: string;
66
+ provider: string;
67
+ results: Array<{
68
+ title: string;
69
+ url: string;
70
+ snippet: string;
71
+ publishedAt?: string | null;
72
+ score?: number | null;
73
+ extract?: {
74
+ status: number;
75
+ renderedWith: "static" | "browser";
76
+ title?: string;
77
+ body: string;
78
+ wordCount: number;
79
+ truncated: boolean;
80
+ warnings?: string[];
81
+ };
82
+ }>;
83
+ totalResults: number;
84
+ warnings: string[];
85
+ }
86
+ //# sourceMappingURL=types.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../../src/tools/web-search/types.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,7 @@
1
+ /**
2
+ * Web Search 工具类型(客户端本地复刻,不依赖 `@tachu/web-fetch-server`)。
3
+ *
4
+ * @see docs/adr/decisions/0003b-web-fetch-types.md §2 / §6.2
5
+ */
6
+ export {};
7
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../../../src/tools/web-search/types.ts"],"names":[],"mappings":"AAAA;;;;GAIG"}
@@ -0,0 +1,30 @@
1
+ ---
2
+ kind: tool
3
+ name: write-file
4
+ description: 写入工作区内文件内容
5
+ sideEffect: write
6
+ idempotent: false
7
+ requiresApproval: true
8
+ timeout: 5000
9
+ inputSchema:
10
+ type: object
11
+ properties:
12
+ path:
13
+ type: string
14
+ content:
15
+ type: string
16
+ encoding:
17
+ type: string
18
+ enum: [utf-8, base64]
19
+ createDirs:
20
+ type: boolean
21
+ required: [path, content]
22
+ outputSchema:
23
+ type: object
24
+ properties:
25
+ bytesWritten:
26
+ type: number
27
+ execute: write-file
28
+ ---
29
+
30
+ 向工作区写入文件,支持 UTF-8 与 base64 输入。
@@ -0,0 +1,16 @@
1
+ import type { ToolExecutor } from "../shared";
2
+ interface WriteFileInput {
3
+ path: string;
4
+ content: string;
5
+ encoding?: "utf-8" | "base64";
6
+ createDirs?: boolean;
7
+ }
8
+ interface WriteFileOutput {
9
+ bytesWritten: number;
10
+ }
11
+ /**
12
+ * 写入文件 Tool 执行器。
13
+ */
14
+ export declare const writeFileExecutor: ToolExecutor<WriteFileInput, WriteFileOutput>;
15
+ export {};
16
+ //# sourceMappingURL=executor.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"executor.d.ts","sourceRoot":"","sources":["../../../src/tools/write-file/executor.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,WAAW,CAAC;AAG9C,UAAU,cAAc;IACtB,IAAI,EAAE,MAAM,CAAC;IACb,OAAO,EAAE,MAAM,CAAC;IAChB,QAAQ,CAAC,EAAE,OAAO,GAAG,QAAQ,CAAC;IAC9B,UAAU,CAAC,EAAE,OAAO,CAAC;CACtB;AAED,UAAU,eAAe;IACvB,YAAY,EAAE,MAAM,CAAC;CACtB;AAED;;GAEG;AACH,eAAO,MAAM,iBAAiB,EAAE,YAAY,CAAC,cAAc,EAAE,eAAe,CAa3E,CAAC"}
@@ -0,0 +1,18 @@
1
+ import { mkdir, writeFile } from "node:fs/promises";
2
+ import { dirname } from "node:path";
3
+ import { resolveAllowedPath } from "../../common/path";
4
+ import { assertNotAborted, resolveSandboxPolicy } from "../shared";
5
+ /**
6
+ * 写入文件 Tool 执行器。
7
+ */
8
+ export const writeFileExecutor = async (input, context) => {
9
+ assertNotAborted(context.abortSignal);
10
+ const target = resolveAllowedPath(input.path, resolveSandboxPolicy(context));
11
+ if (input.createDirs) {
12
+ await mkdir(dirname(target), { recursive: true });
13
+ }
14
+ const payload = input.encoding === "base64" ? Buffer.from(input.content, "base64") : Buffer.from(input.content);
15
+ await writeFile(target, payload);
16
+ return { bytesWritten: payload.byteLength };
17
+ };
18
+ //# sourceMappingURL=executor.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"executor.js","sourceRoot":"","sources":["../../../src/tools/write-file/executor.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,EAAE,SAAS,EAAE,MAAM,kBAAkB,CAAC;AACpD,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpC,OAAO,EAAE,kBAAkB,EAAE,MAAM,mBAAmB,CAAC;AAEvD,OAAO,EAAE,gBAAgB,EAAE,oBAAoB,EAAE,MAAM,WAAW,CAAC;AAanE;;GAEG;AACH,MAAM,CAAC,MAAM,iBAAiB,GAAkD,KAAK,EACnF,KAAK,EACL,OAAO,EACP,EAAE;IACF,gBAAgB,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC;IACtC,MAAM,MAAM,GAAG,kBAAkB,CAAC,KAAK,CAAC,IAAI,EAAE,oBAAoB,CAAC,OAAO,CAAC,CAAC,CAAC;IAC7E,IAAI,KAAK,CAAC,UAAU,EAAE,CAAC;QACrB,MAAM,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;IACpD,CAAC;IACD,MAAM,OAAO,GACX,KAAK,CAAC,QAAQ,KAAK,QAAQ,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,OAAO,EAAE,QAAQ,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;IAClG,MAAM,SAAS,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IACjC,OAAO,EAAE,YAAY,EAAE,OAAO,CAAC,UAAU,EAAE,CAAC;AAC9C,CAAC,CAAC"}
@@ -0,0 +1,23 @@
1
+ import type { InputEnvelope, InputMetadata, InputTransformer, ModelCapabilities } from "@tachu/core";
2
+ /**
3
+ * 文档转文本转换器(PDF / DOCX)。
4
+ */
5
+ export declare class DocumentToTextTransformer implements InputTransformer {
6
+ readonly modality = "document";
7
+ /**
8
+ * 判断是否需要文档降级。
9
+ *
10
+ * @param metadata 输入元信息
11
+ * @param modelCapabilities 模型能力
12
+ * @returns true 表示需要转换
13
+ */
14
+ canHandle(metadata: InputMetadata, modelCapabilities: ModelCapabilities): boolean;
15
+ /**
16
+ * 将文档转换为文本信封。
17
+ *
18
+ * @param envelope 输入信封
19
+ * @returns 文本信封
20
+ */
21
+ transform(envelope: InputEnvelope): Promise<InputEnvelope>;
22
+ }
23
+ //# sourceMappingURL=document-to-text.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"document-to-text.d.ts","sourceRoot":"","sources":["../../src/transformers/document-to-text.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,aAAa,EAAE,aAAa,EAAE,gBAAgB,EAAE,iBAAiB,EAAE,MAAM,aAAa,CAAC;AAkBrG;;GAEG;AACH,qBAAa,yBAA0B,YAAW,gBAAgB;IAChE,QAAQ,CAAC,QAAQ,cAAc;IAE/B;;;;;;OAMG;IACH,SAAS,CAAC,QAAQ,EAAE,aAAa,EAAE,iBAAiB,EAAE,iBAAiB,GAAG,OAAO;IAMjF;;;;;OAKG;IACG,SAAS,CAAC,QAAQ,EAAE,aAAa,GAAG,OAAO,CAAC,aAAa,CAAC;CAkCjE"}
@@ -0,0 +1,69 @@
1
+ import { readFile } from "node:fs/promises";
2
+ import { PDFParse } from "pdf-parse";
3
+ import mammoth from "mammoth";
4
+ import { ValidationError } from "@tachu/core";
5
+ const MAX_TEXT_CHARS = 500_000;
6
+ const truncateIfNeeded = (text) => {
7
+ if (text.length <= MAX_TEXT_CHARS) {
8
+ return text;
9
+ }
10
+ const trimmed = text.slice(0, MAX_TEXT_CHARS);
11
+ return `${trimmed}\n\n[TRUNCATED:${text.length - MAX_TEXT_CHARS}]`;
12
+ };
13
+ /**
14
+ * 文档转文本转换器(PDF / DOCX)。
15
+ */
16
+ export class DocumentToTextTransformer {
17
+ modality = "document";
18
+ /**
19
+ * 判断是否需要文档降级。
20
+ *
21
+ * @param metadata 输入元信息
22
+ * @param modelCapabilities 模型能力
23
+ * @returns true 表示需要转换
24
+ */
25
+ canHandle(metadata, modelCapabilities) {
26
+ const modality = metadata.modality ?? "";
27
+ const isDocument = modality === "document" || modality === "file";
28
+ return isDocument && !modelCapabilities.supportedModalities.includes("file");
29
+ }
30
+ /**
31
+ * 将文档转换为文本信封。
32
+ *
33
+ * @param envelope 输入信封
34
+ * @returns 文本信封
35
+ */
36
+ async transform(envelope) {
37
+ const content = envelope.content;
38
+ if (!content.path) {
39
+ throw new ValidationError("VALIDATION_DOCUMENT_PATH", "文档输入缺少 path");
40
+ }
41
+ const mimeType = content.mimeType ?? envelope.metadata.mimeType ?? "";
42
+ let text = "";
43
+ if (mimeType.includes("pdf") || content.path.toLowerCase().endsWith(".pdf")) {
44
+ const buffer = await readFile(content.path);
45
+ const parser = new PDFParse({ data: buffer });
46
+ const parsed = await parser.getText();
47
+ text = parsed.text;
48
+ await parser.destroy();
49
+ }
50
+ else if (mimeType.includes("word") ||
51
+ mimeType.includes("officedocument") ||
52
+ content.path.toLowerCase().endsWith(".docx")) {
53
+ const result = await mammoth.extractRawText({ path: content.path });
54
+ text = result.value;
55
+ }
56
+ else {
57
+ text = await readFile(content.path, "utf8");
58
+ }
59
+ return {
60
+ content: truncateIfNeeded(text),
61
+ metadata: {
62
+ ...envelope.metadata,
63
+ modality: "text",
64
+ mimeType: "text/plain",
65
+ },
66
+ };
67
+ }
68
+ }
69
+ //# sourceMappingURL=document-to-text.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"document-to-text.js","sourceRoot":"","sources":["../../src/transformers/document-to-text.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,kBAAkB,CAAC;AAC5C,OAAO,EAAE,QAAQ,EAAE,MAAM,WAAW,CAAC;AACrC,OAAO,OAAO,MAAM,SAAS,CAAC;AAE9B,OAAO,EAAE,eAAe,EAAE,MAAM,aAAa,CAAC;AAO9C,MAAM,cAAc,GAAG,OAAO,CAAC;AAE/B,MAAM,gBAAgB,GAAG,CAAC,IAAY,EAAU,EAAE;IAChD,IAAI,IAAI,CAAC,MAAM,IAAI,cAAc,EAAE,CAAC;QAClC,OAAO,IAAI,CAAC;IACd,CAAC;IACD,MAAM,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,EAAE,cAAc,CAAC,CAAC;IAC9C,OAAO,GAAG,OAAO,kBAAkB,IAAI,CAAC,MAAM,GAAG,cAAc,GAAG,CAAC;AACrE,CAAC,CAAC;AAEF;;GAEG;AACH,MAAM,OAAO,yBAAyB;IAC3B,QAAQ,GAAG,UAAU,CAAC;IAE/B;;;;;;OAMG;IACH,SAAS,CAAC,QAAuB,EAAE,iBAAoC;QACrE,MAAM,QAAQ,GAAG,QAAQ,CAAC,QAAQ,IAAI,EAAE,CAAC;QACzC,MAAM,UAAU,GAAG,QAAQ,KAAK,UAAU,IAAI,QAAQ,KAAK,MAAM,CAAC;QAClE,OAAO,UAAU,IAAI,CAAC,iBAAiB,CAAC,mBAAmB,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;IAC/E,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,SAAS,CAAC,QAAuB;QACrC,MAAM,OAAO,GAAG,QAAQ,CAAC,OAA0B,CAAC;QACpD,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC;YAClB,MAAM,IAAI,eAAe,CAAC,0BAA0B,EAAE,aAAa,CAAC,CAAC;QACvE,CAAC;QACD,MAAM,QAAQ,GAAG,OAAO,CAAC,QAAQ,IAAI,QAAQ,CAAC,QAAQ,CAAC,QAAQ,IAAI,EAAE,CAAC;QAEtE,IAAI,IAAI,GAAG,EAAE,CAAC;QACd,IAAI,QAAQ,CAAC,QAAQ,CAAC,KAAK,CAAC,IAAI,OAAO,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC,QAAQ,CAAC,MAAM,CAAC,EAAE,CAAC;YAC5E,MAAM,MAAM,GAAG,MAAM,QAAQ,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;YAC5C,MAAM,MAAM,GAAG,IAAI,QAAQ,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,CAAC,CAAC;YAC9C,MAAM,MAAM,GAAG,MAAM,MAAM,CAAC,OAAO,EAAE,CAAC;YACtC,IAAI,GAAG,MAAM,CAAC,IAAI,CAAC;YACnB,MAAM,MAAM,CAAC,OAAO,EAAE,CAAC;QACzB,CAAC;aAAM,IACL,QAAQ,CAAC,QAAQ,CAAC,MAAM,CAAC;YACzB,QAAQ,CAAC,QAAQ,CAAC,gBAAgB,CAAC;YACnC,OAAO,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC,QAAQ,CAAC,OAAO,CAAC,EAC5C,CAAC;YACD,MAAM,MAAM,GAAG,MAAM,OAAO,CAAC,cAAc,CAAC,EAAE,IAAI,EAAE,OAAO,CAAC,IAAI,EAAE,CAAC,CAAC;YACpE,IAAI,GAAG,MAAM,CAAC,KAAK,CAAC;QACtB,CAAC;aAAM,CAAC;YACN,IAAI,GAAG,MAAM,QAAQ,CAAC,OAAO,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC;QAC9C,CAAC;QAED,OAAO;YACL,OAAO,EAAE,gBAAgB,CAAC,IAAI,CAAC;YAC/B,QAAQ,EAAE;gBACR,GAAG,QAAQ,CAAC,QAAQ;gBACpB,QAAQ,EAAE,MAAM;gBAChB,QAAQ,EAAE,YAAY;aACvB;SACF,CAAC;IACJ,CAAC;CACF"}
@@ -0,0 +1,38 @@
1
+ import type { InputEnvelope, InputMetadata, InputTransformer, ModelCapabilities, ProviderAdapter } from "@tachu/core";
2
+ interface ImageToTextTransformerOptions {
3
+ provider: ProviderAdapter;
4
+ model?: string;
5
+ promptTemplate?: string;
6
+ }
7
+ /**
8
+ * 图像转文本转换器(基于 Provider Vision 能力)。
9
+ */
10
+ export declare class ImageToTextTransformer implements InputTransformer {
11
+ readonly modality = "image";
12
+ private readonly provider;
13
+ private readonly model;
14
+ private readonly promptTemplate;
15
+ /**
16
+ * 创建图像转换器。
17
+ *
18
+ * @param options 配置项
19
+ */
20
+ constructor(options: ImageToTextTransformerOptions);
21
+ /**
22
+ * 判断是否需要执行降级转换。
23
+ *
24
+ * @param metadata 输入元信息
25
+ * @param modelCapabilities 模型能力
26
+ * @returns true 表示需要转换
27
+ */
28
+ canHandle(metadata: InputMetadata, modelCapabilities: ModelCapabilities): boolean;
29
+ /**
30
+ * 将图像输入转换为文本信封。
31
+ *
32
+ * @param envelope 输入信封
33
+ * @returns 转换后的文本信封
34
+ */
35
+ transform(envelope: InputEnvelope): Promise<InputEnvelope>;
36
+ }
37
+ export {};
38
+ //# sourceMappingURL=image-to-text.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"image-to-text.d.ts","sourceRoot":"","sources":["../../src/transformers/image-to-text.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAEV,aAAa,EACb,aAAa,EACb,gBAAgB,EAEhB,iBAAiB,EACjB,eAAe,EAChB,MAAM,aAAa,CAAC;AAGrB,UAAU,6BAA6B;IACrC,QAAQ,EAAE,eAAe,CAAC;IAC1B,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,cAAc,CAAC,EAAE,MAAM,CAAC;CACzB;AAUD;;GAEG;AACH,qBAAa,sBAAuB,YAAW,gBAAgB;IAC7D,QAAQ,CAAC,QAAQ,WAAW;IAE5B,OAAO,CAAC,QAAQ,CAAC,QAAQ,CAAkB;IAC3C,OAAO,CAAC,QAAQ,CAAC,KAAK,CAAS;IAC/B,OAAO,CAAC,QAAQ,CAAC,cAAc,CAAS;IAExC;;;;OAIG;gBACS,OAAO,EAAE,6BAA6B;IAMlD;;;;;;OAMG;IACH,SAAS,CAAC,QAAQ,EAAE,aAAa,EAAE,iBAAiB,EAAE,iBAAiB,GAAG,OAAO;IAOjF;;;;;OAKG;IACG,SAAS,CAAC,QAAQ,EAAE,aAAa,GAAG,OAAO,CAAC,aAAa,CAAC;CA4CjE"}
@@ -0,0 +1,82 @@
1
+ import { ProviderError } from "@tachu/core";
2
+ const DEFAULT_PROMPT = "请描述图片的主要内容、关键信息和可见文本。";
3
+ /**
4
+ * 图像转文本转换器(基于 Provider Vision 能力)。
5
+ */
6
+ export class ImageToTextTransformer {
7
+ modality = "image";
8
+ provider;
9
+ model;
10
+ promptTemplate;
11
+ /**
12
+ * 创建图像转换器。
13
+ *
14
+ * @param options 配置项
15
+ */
16
+ constructor(options) {
17
+ this.provider = options.provider;
18
+ this.model = options.model ?? "gpt-4o";
19
+ this.promptTemplate = options.promptTemplate ?? DEFAULT_PROMPT;
20
+ }
21
+ /**
22
+ * 判断是否需要执行降级转换。
23
+ *
24
+ * @param metadata 输入元信息
25
+ * @param modelCapabilities 模型能力
26
+ * @returns true 表示需要转换
27
+ */
28
+ canHandle(metadata, modelCapabilities) {
29
+ return (metadata.modality === "image" &&
30
+ !modelCapabilities.supportedModalities.includes("image"));
31
+ }
32
+ /**
33
+ * 将图像输入转换为文本信封。
34
+ *
35
+ * @param envelope 输入信封
36
+ * @returns 转换后的文本信封
37
+ */
38
+ async transform(envelope) {
39
+ const content = envelope.content;
40
+ const imageUrl = content.imageUrl ??
41
+ (content.imageBase64
42
+ ? `data:${content.mimeType ?? "image/png"};base64,${content.imageBase64}`
43
+ : undefined);
44
+ if (!imageUrl) {
45
+ throw new ProviderError("PROVIDER_INVALID_INPUT", "缺少 imageUrl 或 imageBase64");
46
+ }
47
+ try {
48
+ const parts = [
49
+ { type: "text", text: this.promptTemplate },
50
+ { type: "image_url", image_url: { url: imageUrl } },
51
+ ];
52
+ const request = {
53
+ model: this.model,
54
+ messages: [
55
+ {
56
+ role: "user",
57
+ content: parts,
58
+ },
59
+ ],
60
+ };
61
+ const response = await this.provider.chat(request);
62
+ return {
63
+ content: response.content,
64
+ metadata: {
65
+ ...envelope.metadata,
66
+ modality: "text",
67
+ mimeType: "text/plain",
68
+ },
69
+ };
70
+ }
71
+ catch (error) {
72
+ if (error instanceof ProviderError) {
73
+ throw error;
74
+ }
75
+ throw new ProviderError("PROVIDER_UPSTREAM_ERROR", "图像转文本失败", {
76
+ cause: error,
77
+ retryable: true,
78
+ });
79
+ }
80
+ }
81
+ }
82
+ //# sourceMappingURL=image-to-text.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"image-to-text.js","sourceRoot":"","sources":["../../src/transformers/image-to-text.ts"],"names":[],"mappings":"AASA,OAAO,EAAE,aAAa,EAAE,MAAM,aAAa,CAAC;AAc5C,MAAM,cAAc,GAAG,uBAAuB,CAAC;AAE/C;;GAEG;AACH,MAAM,OAAO,sBAAsB;IACxB,QAAQ,GAAG,OAAO,CAAC;IAEX,QAAQ,CAAkB;IAC1B,KAAK,CAAS;IACd,cAAc,CAAS;IAExC;;;;OAIG;IACH,YAAY,OAAsC;QAChD,IAAI,CAAC,QAAQ,GAAG,OAAO,CAAC,QAAQ,CAAC;QACjC,IAAI,CAAC,KAAK,GAAG,OAAO,CAAC,KAAK,IAAI,QAAQ,CAAC;QACvC,IAAI,CAAC,cAAc,GAAG,OAAO,CAAC,cAAc,IAAI,cAAc,CAAC;IACjE,CAAC;IAED;;;;;;OAMG;IACH,SAAS,CAAC,QAAuB,EAAE,iBAAoC;QACrE,OAAO,CACL,QAAQ,CAAC,QAAQ,KAAK,OAAO;YAC7B,CAAC,iBAAiB,CAAC,mBAAmB,CAAC,QAAQ,CAAC,OAAO,CAAC,CACzD,CAAC;IACJ,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,SAAS,CAAC,QAAuB;QACrC,MAAM,OAAO,GAAG,QAAQ,CAAC,OAAuB,CAAC;QACjD,MAAM,QAAQ,GACZ,OAAO,CAAC,QAAQ;YAChB,CAAC,OAAO,CAAC,WAAW;gBAClB,CAAC,CAAC,QAAQ,OAAO,CAAC,QAAQ,IAAI,WAAW,WAAW,OAAO,CAAC,WAAW,EAAE;gBACzE,CAAC,CAAC,SAAS,CAAC,CAAC;QACjB,IAAI,CAAC,QAAQ,EAAE,CAAC;YACd,MAAM,IAAI,aAAa,CAAC,wBAAwB,EAAE,2BAA2B,CAAC,CAAC;QACjF,CAAC;QAED,IAAI,CAAC;YACH,MAAM,KAAK,GAAyB;gBAClC,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,IAAI,CAAC,cAAc,EAAE;gBAC3C,EAAE,IAAI,EAAE,WAAW,EAAE,SAAS,EAAE,EAAE,GAAG,EAAE,QAAQ,EAAE,EAAE;aACpD,CAAC;YACF,MAAM,OAAO,GAAgB;gBAC3B,KAAK,EAAE,IAAI,CAAC,KAAK;gBACjB,QAAQ,EAAE;oBACR;wBACE,IAAI,EAAE,MAAM;wBACZ,OAAO,EAAE,KAAK;qBACf;iBACF;aACF,CAAC;YACF,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;YACnD,OAAO;gBACL,OAAO,EAAE,QAAQ,CAAC,OAAO;gBACzB,QAAQ,EAAE;oBACR,GAAG,QAAQ,CAAC,QAAQ;oBACpB,QAAQ,EAAE,MAAM;oBAChB,QAAQ,EAAE,YAAY;iBACvB;aACF,CAAC;QACJ,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,IAAI,KAAK,YAAY,aAAa,EAAE,CAAC;gBACnC,MAAM,KAAK,CAAC;YACd,CAAC;YACD,MAAM,IAAI,aAAa,CAAC,yBAAyB,EAAE,SAAS,EAAE;gBAC5D,KAAK,EAAE,KAAK;gBACZ,SAAS,EAAE,IAAI;aAChB,CAAC,CAAC;QACL,CAAC;IACH,CAAC;CACF"}
@@ -0,0 +1,3 @@
1
+ export { ImageToTextTransformer } from "./image-to-text";
2
+ export { DocumentToTextTransformer } from "./document-to-text";
3
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/transformers/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,sBAAsB,EAAE,MAAM,iBAAiB,CAAC;AACzD,OAAO,EAAE,yBAAyB,EAAE,MAAM,oBAAoB,CAAC"}
@@ -0,0 +1,3 @@
1
+ export { ImageToTextTransformer } from "./image-to-text";
2
+ export { DocumentToTextTransformer } from "./document-to-text";
3
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/transformers/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,sBAAsB,EAAE,MAAM,iBAAiB,CAAC;AACzD,OAAO,EAAE,yBAAyB,EAAE,MAAM,oBAAoB,CAAC"}
@@ -0,0 +1,3 @@
1
+ export { LocalFsVectorStore } from "./local-fs";
2
+ export { QdrantVectorStore } from "./qdrant";
3
+ //# sourceMappingURL=index.d.ts.map