strapi-content-embeddings 0.1.3 → 0.1.5

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 (36) hide show
  1. package/README.md +187 -0
  2. package/dist/_chunks/{App-Swmo_WMf.js → App-Rq72tIgS.js} +37 -55
  3. package/dist/_chunks/App-Rq72tIgS.js.map +1 -0
  4. package/dist/_chunks/{App-BlCKKuQN.mjs → App-j180lztd.mjs} +37 -55
  5. package/dist/_chunks/App-j180lztd.mjs.map +1 -0
  6. package/dist/_chunks/en-B4KWt_jN.js +1 -0
  7. package/dist/_chunks/en-B4KWt_jN.js.map +1 -0
  8. package/dist/_chunks/en-Byx4XI2L.mjs +1 -0
  9. package/dist/_chunks/en-Byx4XI2L.mjs.map +1 -0
  10. package/dist/_chunks/{index-CXVoFiJp.mjs → index-B3j0IFUi.mjs} +70 -27
  11. package/dist/_chunks/index-B3j0IFUi.mjs.map +1 -0
  12. package/dist/_chunks/{index-BpKkUIJY.js → index-jf6vikTZ.js} +70 -27
  13. package/dist/_chunks/index-jf6vikTZ.js.map +1 -0
  14. package/dist/admin/index.js +2 -1
  15. package/dist/admin/index.js.map +1 -0
  16. package/dist/admin/index.mjs +2 -1
  17. package/dist/admin/index.mjs.map +1 -0
  18. package/dist/admin/src/components/custom/MarkdownEditor.d.ts +1 -1
  19. package/dist/server/index.js +850 -57
  20. package/dist/server/index.js.map +1 -0
  21. package/dist/server/index.mjs +850 -57
  22. package/dist/server/index.mjs.map +1 -0
  23. package/dist/server/src/config/index.d.ts +9 -0
  24. package/dist/server/src/controllers/controller.d.ts +14 -0
  25. package/dist/server/src/controllers/index.d.ts +2 -0
  26. package/dist/server/src/index.d.ts +38 -2
  27. package/dist/server/src/mcp/tools/create-embedding.d.ts +6 -0
  28. package/dist/server/src/mcp/tools/index.d.ts +4 -0
  29. package/dist/server/src/plugin-manager.d.ts +16 -0
  30. package/dist/server/src/routes/content-api.d.ts +10 -0
  31. package/dist/server/src/routes/index.d.ts +10 -0
  32. package/dist/server/src/services/embeddings.d.ts +43 -2
  33. package/dist/server/src/services/index.d.ts +23 -2
  34. package/dist/server/src/services/sync.d.ts +48 -0
  35. package/dist/server/src/utils/chunking.d.ts +44 -0
  36. package/package.json +1 -1
package/README.md CHANGED
@@ -11,6 +11,8 @@ A Strapi v5 plugin that creates vector embeddings from your content using OpenAI
11
11
  - **Content Manager Integration**: Create embeddings directly from any content type's edit view
12
12
  - **Standalone Embeddings**: Create embeddings independent of content types
13
13
  - **Multiple Embedding Models**: Support for OpenAI's text-embedding-3-small, text-embedding-3-large, and text-embedding-ada-002
14
+ - **Database Sync**: Sync embeddings from Neon DB to Strapi with cron-compatible endpoints
15
+ - **Automatic Chunking**: Split large content into multiple embeddings with overlap for context preservation
14
16
 
15
17
  ## Requirements
16
18
 
@@ -206,6 +208,191 @@ All endpoints require admin authentication.
206
208
  | `GET` | `/strapi-content-embeddings/embeddings/find/:id` | Get a single embedding |
207
209
  | `GET` | `/strapi-content-embeddings/embeddings/embeddings-query?query=...` | RAG query |
208
210
 
211
+ ## Database Sync (Neon to Strapi)
212
+
213
+ The plugin provides endpoints to sync embeddings from Neon DB (source of truth) to Strapi. These endpoints are designed to be triggered manually or via cron jobs.
214
+
215
+ ### Sync Endpoints
216
+
217
+ | Method | Endpoint | Description |
218
+ |--------|----------|-------------|
219
+ | `GET/POST` | `/api/strapi-content-embeddings/sync` | Sync embeddings from Neon to Strapi |
220
+ | `GET` | `/api/strapi-content-embeddings/sync/status` | Check sync status without making changes |
221
+
222
+ ### Query Parameters
223
+
224
+ | Parameter | Type | Default | Description |
225
+ |-----------|------|---------|-------------|
226
+ | `dryRun` | boolean | `false` | Preview changes without applying them |
227
+ | `removeOrphans` | boolean | `false` | Remove Strapi entries that don't exist in Neon |
228
+
229
+ ### Usage Examples
230
+
231
+ **Check sync status:**
232
+ ```bash
233
+ curl "http://localhost:1337/api/strapi-content-embeddings/sync/status" \
234
+ -H "Authorization: Bearer YOUR_API_TOKEN"
235
+ ```
236
+
237
+ Response:
238
+ ```json
239
+ {
240
+ "neonCount": 150,
241
+ "strapiCount": 145,
242
+ "inSync": false,
243
+ "missingInStrapi": 5,
244
+ "missingInNeon": 0,
245
+ "contentDifferences": 2
246
+ }
247
+ ```
248
+
249
+ **Dry run (preview changes):**
250
+ ```bash
251
+ curl "http://localhost:1337/api/strapi-content-embeddings/sync?dryRun=true" \
252
+ -H "Authorization: Bearer YOUR_API_TOKEN"
253
+ ```
254
+
255
+ **Run sync:**
256
+ ```bash
257
+ curl "http://localhost:1337/api/strapi-content-embeddings/sync" \
258
+ -H "Authorization: Bearer YOUR_API_TOKEN"
259
+ ```
260
+
261
+ **Sync and remove orphans:**
262
+ ```bash
263
+ curl "http://localhost:1337/api/strapi-content-embeddings/sync?removeOrphans=true" \
264
+ -H "Authorization: Bearer YOUR_API_TOKEN"
265
+ ```
266
+
267
+ ### Sync Response
268
+
269
+ ```json
270
+ {
271
+ "success": true,
272
+ "timestamp": "2024-01-07T12:00:00.000Z",
273
+ "neonCount": 150,
274
+ "strapiCount": 150,
275
+ "actions": {
276
+ "created": 5,
277
+ "updated": 2,
278
+ "orphansRemoved": 0
279
+ },
280
+ "details": {
281
+ "created": ["doc1 (Title 1)", "doc2 (Title 2)"],
282
+ "updated": ["doc3 (Title 3)"],
283
+ "orphansRemoved": []
284
+ },
285
+ "errors": []
286
+ }
287
+ ```
288
+
289
+ ### Cron Job Example
290
+
291
+ ```bash
292
+ # Sync every hour
293
+ 0 * * * * curl -s "https://your-strapi.com/api/strapi-content-embeddings/sync" \
294
+ -H "Authorization: Bearer YOUR_API_TOKEN" >> /var/log/embeddings-sync.log
295
+ ```
296
+
297
+ ## Content Chunking
298
+
299
+ For large content that exceeds the recommended size for embeddings (~4000 characters / ~1000 tokens), the plugin supports automatic chunking.
300
+
301
+ ### How Chunking Works
302
+
303
+ 1. **Smart Splitting**: Content is split at natural boundaries (paragraphs, sentences, words) to preserve meaning
304
+ 2. **Overlap**: Chunks include overlapping content (default: 200 chars) to maintain context between chunks
305
+ 3. **Metadata**: Each chunk stores metadata linking it to the original content and other chunks
306
+ 4. **Titles**: Chunk titles include part numbers (e.g., "My Document [Part 1/3]")
307
+
308
+ ### Configuration
309
+
310
+ Add chunking options to your plugin config:
311
+
312
+ ```typescript
313
+ // config/plugins.ts
314
+ export default ({ env }) => ({
315
+ "strapi-content-embeddings": {
316
+ enabled: true,
317
+ config: {
318
+ openAIApiKey: env("OPENAI_API_KEY"),
319
+ neonConnectionString: env("NEON_CONNECTION_STRING"),
320
+ // Chunking options
321
+ chunkSize: 4000, // Max characters per chunk (default: 4000)
322
+ chunkOverlap: 200, // Overlap between chunks (default: 200)
323
+ autoChunk: false, // Auto-chunk large content globally (default: false)
324
+ },
325
+ },
326
+ });
327
+ ```
328
+
329
+ ### Using Chunking
330
+
331
+ #### Via MCP Tool
332
+
333
+ ```json
334
+ {
335
+ "tool": "create_embedding",
336
+ "arguments": {
337
+ "title": "My Long Document",
338
+ "content": "... very long content ...",
339
+ "autoChunk": true
340
+ }
341
+ }
342
+ ```
343
+
344
+ #### Programmatic Usage
345
+
346
+ ```typescript
347
+ // Create with automatic chunking
348
+ const result = await strapi
349
+ .plugin("strapi-content-embeddings")
350
+ .service("embeddings")
351
+ .createChunkedEmbedding({
352
+ data: {
353
+ title: "My Long Document",
354
+ content: "... very long content ...",
355
+ },
356
+ });
357
+
358
+ console.log(result);
359
+ // {
360
+ // entity: { ... first chunk ... },
361
+ // chunks: [ ... all chunks ... ],
362
+ // totalChunks: 5,
363
+ // wasChunked: true
364
+ // }
365
+
366
+ // Or use createEmbedding with autoChunk flag
367
+ const embedding = await strapi
368
+ .plugin("strapi-content-embeddings")
369
+ .service("embeddings")
370
+ .createEmbedding({
371
+ data: {
372
+ title: "My Document",
373
+ content: "... long content ...",
374
+ autoChunk: true, // Enable chunking
375
+ },
376
+ });
377
+ ```
378
+
379
+ ### Chunk Metadata
380
+
381
+ Each chunk embedding includes metadata:
382
+
383
+ ```json
384
+ {
385
+ "isChunk": true,
386
+ "chunkIndex": 0,
387
+ "totalChunks": 5,
388
+ "startOffset": 0,
389
+ "endOffset": 4200,
390
+ "originalTitle": "My Long Document",
391
+ "parentDocumentId": "abc123",
392
+ "estimatedTokens": 1050
393
+ }
394
+ ```
395
+
209
396
  ## How It Works
210
397
 
211
398
  1. **Embedding Creation**: When you create an embedding, the content is sent to OpenAI's embedding API to generate a vector representation (1536 or 3072 dimensions depending on the model).
@@ -6,11 +6,11 @@ const reactRouterDom = require("react-router-dom");
6
6
  const react = require("react");
7
7
  const designSystem = require("@strapi/design-system");
8
8
  const icons = require("@strapi/icons");
9
- const reactIntl = require("react-intl");
10
9
  const qs = require("qs");
11
- const index = require("./index-BpKkUIJY.js");
10
+ const index = require("./index-jf6vikTZ.js");
12
11
  const styled = require("styled-components");
13
12
  const ReactMarkdown = require("react-markdown");
13
+ const reactIntl = require("react-intl");
14
14
  const _interopDefault = (e) => e && e.__esModule ? e : { default: e };
15
15
  const qs__default = /* @__PURE__ */ _interopDefault(qs);
16
16
  const styled__default = /* @__PURE__ */ _interopDefault(styled);
@@ -476,7 +476,6 @@ function debounce(func, wait) {
476
476
  };
477
477
  }
478
478
  function HomePage() {
479
- const { formatMessage } = reactIntl.useIntl();
480
479
  const { get } = admin.useFetchClient();
481
480
  const navigate = reactRouterDom.useNavigate();
482
481
  const [embeddings, setEmbeddings] = react.useState(null);
@@ -484,19 +483,14 @@ function HomePage() {
484
483
  const [isLoading, setIsLoading] = react.useState(true);
485
484
  const buildQuery = (searchTerm) => qs__default.default.stringify({
486
485
  filters: searchTerm ? {
487
- $or: [
488
- { title: { $containsi: searchTerm } },
489
- { content: { $containsi: searchTerm } }
490
- ]
486
+ $or: [{ title: { $containsi: searchTerm } }, { content: { $containsi: searchTerm } }]
491
487
  } : void 0
492
488
  });
493
489
  const fetchData = react.useCallback(
494
490
  async (searchTerm) => {
495
491
  setIsLoading(true);
496
492
  try {
497
- const response = await get(
498
- `/${index.PLUGIN_ID}/embeddings/find?${buildQuery(searchTerm)}`
499
- );
493
+ const response = await get(`/${index.PLUGIN_ID}/embeddings/find?${buildQuery(searchTerm)}`);
500
494
  setEmbeddings(response.data);
501
495
  } catch (error) {
502
496
  console.error("Failed to fetch embeddings:", error);
@@ -507,10 +501,7 @@ function HomePage() {
507
501
  },
508
502
  [get]
509
503
  );
510
- const debouncedFetch = react.useMemo(
511
- () => debounce(fetchData, 500),
512
- [fetchData]
513
- );
504
+ const debouncedFetch = react.useMemo(() => debounce(fetchData, 500), [fetchData]);
514
505
  react.useEffect(() => {
515
506
  debouncedFetch(search);
516
507
  }, [search, debouncedFetch]);
@@ -522,50 +513,36 @@ function HomePage() {
522
513
  };
523
514
  if (isLoading && !embeddings) {
524
515
  return /* @__PURE__ */ jsxRuntime.jsxs(designSystem.Main, { children: [
525
- /* @__PURE__ */ jsxRuntime.jsx(
526
- admin.Layouts.Header,
527
- {
528
- title: formatMessage({
529
- id: "HomePage.header.title",
530
- defaultMessage: "Content Embeddings"
531
- }),
532
- subtitle: formatMessage({
533
- id: "HomePage.header.subtitle",
534
- defaultMessage: "Manage your content embeddings"
535
- })
536
- }
537
- ),
516
+ /* @__PURE__ */ jsxRuntime.jsx(admin.Layouts.Header, { title: "Content Embeddings", subtitle: "Manage your content embeddings" }),
538
517
  /* @__PURE__ */ jsxRuntime.jsx(admin.Layouts.Content, { children: /* @__PURE__ */ jsxRuntime.jsx(designSystem.Flex, { justifyContent: "center", padding: 8, children: /* @__PURE__ */ jsxRuntime.jsx(designSystem.Loader, { children: "Loading..." }) }) }),
539
518
  /* @__PURE__ */ jsxRuntime.jsx(ChatModal, {})
540
519
  ] });
541
520
  }
542
521
  if (embeddings?.totalCount === 0 && !search) {
543
522
  return /* @__PURE__ */ jsxRuntime.jsxs(designSystem.Main, { children: [
544
- /* @__PURE__ */ jsxRuntime.jsx(
545
- admin.Layouts.Header,
546
- {
547
- title: formatMessage({
548
- id: "HomePage.header.title",
549
- defaultMessage: "Content Embeddings"
550
- }),
551
- subtitle: formatMessage({
552
- id: "HomePage.header.subtitle",
553
- defaultMessage: "Manage your content embeddings"
554
- })
555
- }
556
- ),
523
+ /* @__PURE__ */ jsxRuntime.jsx(admin.Layouts.Header, { title: "Content Embeddings", subtitle: "Manage your content embeddings" }),
557
524
  /* @__PURE__ */ jsxRuntime.jsx(admin.Layouts.Content, { children: /* @__PURE__ */ jsxRuntime.jsx(EmptyState, {}) }),
558
525
  /* @__PURE__ */ jsxRuntime.jsx(ChatModal, {})
559
526
  ] });
560
527
  }
528
+ const renderEmbeddingsContent = () => {
529
+ if (isLoading) {
530
+ return /* @__PURE__ */ jsxRuntime.jsx(designSystem.Flex, { justifyContent: "center", padding: 8, children: /* @__PURE__ */ jsxRuntime.jsx(designSystem.Loader, { children: "Loading..." }) });
531
+ }
532
+ if (embeddings?.data && embeddings.data.length > 0) {
533
+ return /* @__PURE__ */ jsxRuntime.jsx(EmbeddingsTable, { data: embeddings.data });
534
+ }
535
+ return /* @__PURE__ */ jsxRuntime.jsxs(designSystem.Box, { padding: 8, textAlign: "center", children: [
536
+ 'No embeddings found matching "',
537
+ search,
538
+ '"'
539
+ ] });
540
+ };
561
541
  return /* @__PURE__ */ jsxRuntime.jsxs(designSystem.Main, { children: [
562
542
  /* @__PURE__ */ jsxRuntime.jsx(
563
543
  admin.Layouts.Header,
564
544
  {
565
- title: formatMessage({
566
- id: "HomePage.header.title",
567
- defaultMessage: "Content Embeddings"
568
- }),
545
+ title: "Content Embeddings",
569
546
  subtitle: `${embeddings?.count || 0} results found`,
570
547
  primaryAction: /* @__PURE__ */ jsxRuntime.jsx(designSystem.Button, { startIcon: /* @__PURE__ */ jsxRuntime.jsx(icons.Plus, {}), onClick: handleCreateNew, children: "Create new embedding" })
571
548
  }
@@ -581,11 +558,7 @@ function HomePage() {
581
558
  startAction: /* @__PURE__ */ jsxRuntime.jsx(icons.Search, {})
582
559
  }
583
560
  ) }),
584
- isLoading ? /* @__PURE__ */ jsxRuntime.jsx(designSystem.Flex, { justifyContent: "center", padding: 8, children: /* @__PURE__ */ jsxRuntime.jsx(designSystem.Loader, { children: "Loading..." }) }) : embeddings?.data && embeddings.data.length > 0 ? /* @__PURE__ */ jsxRuntime.jsx(EmbeddingsTable, { data: embeddings.data }) : /* @__PURE__ */ jsxRuntime.jsxs(designSystem.Box, { padding: 8, textAlign: "center", children: [
585
- 'No embeddings found matching "',
586
- search,
587
- '"'
588
- ] })
561
+ renderEmbeddingsContent()
589
562
  ] }),
590
563
  /* @__PURE__ */ jsxRuntime.jsx(ChatModal, {})
591
564
  ] });
@@ -648,7 +621,7 @@ function BackLink({ to }) {
648
621
  }
649
622
  return /* @__PURE__ */ jsxRuntime.jsx(designSystem.Link, { tag: reactRouterDom.NavLink, to: "..", relative: "path", startIcon: /* @__PURE__ */ jsxRuntime.jsx(icons.ArrowLeft, {}), children: "Go back" });
650
623
  }
651
- const MAX_CONTENT_LENGTH = 4e3;
624
+ const CHUNK_SIZE = 4e3;
652
625
  function CreateEmbeddings() {
653
626
  const { formatMessage } = reactIntl.useIntl();
654
627
  const navigate = reactRouterDom.useNavigate();
@@ -658,8 +631,10 @@ function CreateEmbeddings() {
658
631
  const [content, setContent] = react.useState("");
659
632
  const [metadata, setMetadata] = react.useState("");
660
633
  const [error, setError] = react.useState(null);
661
- const isValid = title.trim() && content.trim() && content.length <= MAX_CONTENT_LENGTH;
634
+ const isValid = title.trim() && content.trim();
662
635
  const contentLength = content.length;
636
+ const willChunk = contentLength > CHUNK_SIZE;
637
+ const estimatedChunks = willChunk ? Math.ceil(contentLength / (CHUNK_SIZE - 200)) : 1;
663
638
  function parseMetadata() {
664
639
  if (!metadata.trim()) return null;
665
640
  try {
@@ -672,7 +647,7 @@ function CreateEmbeddings() {
672
647
  e.preventDefault();
673
648
  e.stopPropagation();
674
649
  if (!isValid) {
675
- setError("Please provide a title and content (max 4000 characters)");
650
+ setError("Please provide a title and content");
676
651
  return;
677
652
  }
678
653
  if (metadata.trim()) {
@@ -707,12 +682,18 @@ function CreateEmbeddings() {
707
682
  id: "CreateEmbeddings.header.title",
708
683
  defaultMessage: "Create Embedding"
709
684
  }),
710
- subtitle: formatMessage(
685
+ subtitle: willChunk ? formatMessage(
686
+ {
687
+ id: "CreateEmbeddings.header.subtitle.chunked",
688
+ defaultMessage: "Content: {length} characters (will create ~{chunks} embeddings)"
689
+ },
690
+ { length: contentLength, chunks: estimatedChunks }
691
+ ) : formatMessage(
711
692
  {
712
693
  id: "CreateEmbeddings.header.subtitle",
713
- defaultMessage: "Content: {length}/{max} characters"
694
+ defaultMessage: "Content: {length} characters"
714
695
  },
715
- { length: contentLength, max: MAX_CONTENT_LENGTH }
696
+ { length: contentLength }
716
697
  ),
717
698
  primaryAction: /* @__PURE__ */ jsxRuntime.jsx(
718
699
  designSystem.Button,
@@ -1006,3 +987,4 @@ const App = () => {
1006
987
  ] });
1007
988
  };
1008
989
  exports.App = App;
990
+ //# sourceMappingURL=App-Rq72tIgS.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"App-Rq72tIgS.js","sources":["../../admin/src/components/custom/Illo.tsx","../../admin/src/components/custom/EmptyState.tsx","../../admin/src/components/custom/EmbeddingsTable.tsx","../../admin/src/components/custom/Markdown.tsx","../../admin/src/components/custom/ChatModal.tsx","../../admin/src/pages/HomePage.tsx","../../admin/src/components/forms/CreateEmbeddingForm.tsx","../../admin/src/components/custom/BackLink.tsx","../../admin/src/pages/CreateEmbeddings.tsx","../../admin/src/pages/EmbeddingDetails.tsx","../../admin/src/pages/App.tsx"],"sourcesContent":["export const Illo = () => (\n <svg width=\"159\" height=\"88\" viewBox=\"0 0 159 88\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path\n fillRule=\"evenodd\"\n clipRule=\"evenodd\"\n d=\"M134.933 17.417C137.768 17.417 140.067 19.7153 140.067 22.5503C140.067 25.3854 137.768 27.6837 134.933 27.6837H105.6C108.435 27.6837 110.733 29.9819 110.733 32.817C110.733 35.6521 108.435 37.9503 105.6 37.9503H121.733C124.568 37.9503 126.867 40.2486 126.867 43.0837C126.867 45.9187 124.568 48.217 121.733 48.217H114.272C110.698 48.217 107.8 50.5153 107.8 53.3503C107.8 55.2404 109.267 56.9515 112.2 58.4837C115.035 58.4837 117.333 60.7819 117.333 63.617C117.333 66.4521 115.035 68.7503 112.2 68.7503H51.3333C48.4982 68.7503 46.2 66.4521 46.2 63.617C46.2 60.7819 48.4982 58.4837 51.3333 58.4837H22.7333C19.8982 58.4837 17.6 56.1854 17.6 53.3503C17.6 50.5153 19.8982 48.217 22.7333 48.217H52.0666C54.9017 48.217 57.2 45.9187 57.2 43.0837C57.2 40.2486 54.9017 37.9503 52.0666 37.9503H33.7333C30.8982 37.9503 28.6 35.6521 28.6 32.817C28.6 29.9819 30.8982 27.6837 33.7333 27.6837H63.0666C60.2316 27.6837 57.9333 25.3854 57.9333 22.5503C57.9333 19.7153 60.2316 17.417 63.0666 17.417H134.933ZM134.933 37.9503C137.768 37.9503 140.067 40.2486 140.067 43.0837C140.067 45.9187 137.768 48.217 134.933 48.217C132.098 48.217 129.8 45.9187 129.8 43.0837C129.8 40.2486 132.098 37.9503 134.933 37.9503Z\"\n fill=\"#DBDBFA\"\n />\n\n <path\n fillRule=\"evenodd\"\n clipRule=\"evenodd\"\n d=\"M95.826 16.6834L102.647 66.4348L103.26 71.4261C103.458 73.034 102.314 74.4976 100.706 74.695L57.7621 79.9679C56.1542 80.1653 54.6906 79.0219 54.4932 77.4139L47.8816 23.5671C47.7829 22.7631 48.3546 22.0313 49.1586 21.9326C49.1637 21.932 49.1688 21.9313 49.1739 21.9307L52.7367 21.5311L95.826 16.6834ZM55.6176 21.208L58.9814 20.8306Z\"\n fill=\"white\"\n />\n\n <path\n d=\"M55.6176 21.208L58.9814 20.8306M95.826 16.6834L102.647 66.4348L103.26 71.4261C103.458 73.034 102.314 74.4976 100.706 74.695L57.7621 79.9679C56.1542 80.1653 54.6906 79.0219 54.4932 77.4139L47.8816 23.5671C47.7829 22.7631 48.3546 22.0313 49.1586 21.9326C49.1637 21.932 49.1688 21.9313 49.1739 21.9307L52.7367 21.5311L95.826 16.6834Z\"\n stroke=\"#7E7BF6\"\n strokeWidth=\"2.5\"\n />\n\n <path\n fillRule=\"evenodd\"\n clipRule=\"evenodd\"\n d=\"M93.9695 19.8144L100.144 64.9025L100.699 69.4258C100.878 70.8831 99.8559 72.2077 98.416 72.3845L59.9585 77.1065C58.5185 77.2833 57.2062 76.2453 57.0272 74.7881L51.0506 26.112C50.9519 25.308 51.5236 24.5762 52.3276 24.4775L57.0851 23.8934\"\n fill=\"#F0F0FF\"\n />\n\n <path\n fillRule=\"evenodd\"\n clipRule=\"evenodd\"\n d=\"M97.701 7.33301H64.2927C63.7358 7.33301 63.2316 7.55873 62.8667 7.92368C62.5017 8.28862 62.276 8.79279 62.276 9.34967V65.083C62.276 65.6399 62.5017 66.1441 62.8667 66.509C63.2316 66.874 63.7358 67.0997 64.2927 67.0997H107.559C108.116 67.0997 108.62 66.874 108.985 66.509C109.35 66.1441 109.576 65.6399 109.576 65.083V19.202C109.576 18.6669 109.363 18.1537 108.985 17.7755L99.1265 7.92324C98.7484 7.54531 98.2356 7.33301 97.701 7.33301Z\"\n fill=\"white\"\n stroke=\"#7F7CFA\"\n strokeWidth=\"2.5\"\n />\n\n <path\n d=\"M98.026 8.17871V16.6833C98.026 17.8983 99.011 18.8833 100.226 18.8833H106.044\"\n stroke=\"#807EFA\"\n strokeWidth=\"2.5\"\n strokeLinecap=\"round\"\n strokeLinejoin=\"round\"\n />\n\n <path\n d=\"M70.1594 56.2838H89.2261M70.1594 18.8838H89.2261H70.1594ZM70.1594 27.6838H101.693H70.1594ZM70.1594 37.2171H101.693H70.1594ZM70.1594 46.7505H101.693H70.1594Z\"\n stroke=\"#817FFA\"\n strokeWidth=\"2.5\"\n strokeLinecap=\"round\"\n strokeLinejoin=\"round\"\n />\n </svg>\n);\n\n","import { EmptyStateLayout, Button } from '@strapi/design-system';\nimport { Link } from 'react-router-dom';\nimport { PLUGIN_ID } from \"../../pluginId\"\nimport { Plus } from '@strapi/icons';\nimport { Illo } from './Illo';\n\nexport function EmptyState() {\n return (\n <EmptyStateLayout\n icon={<Illo />}\n content=\"Let's create our first embedding...\"\n action={\n <Link to={`/plugins/${PLUGIN_ID}/embeddings`}>\n <Button startIcon={<Plus />}>Create new embedding</Button>\n </Link>\n }\n />\n );\n}","import React from \"react\";\nimport { useNavigate } from \"react-router-dom\";\nimport styled from \"styled-components\";\nimport {\n Box,\n Table,\n Thead,\n Tbody,\n Tr,\n Td,\n Th,\n Typography,\n VisuallyHidden,\n Flex,\n IconButton,\n} from \"@strapi/design-system\";\nimport { ArrowRight } from \"@strapi/icons\";\nimport { PLUGIN_ID } from \"../../pluginId\";\n\nconst StyledTr = styled(Tr)`\n cursor: pointer;\n &:hover {\n background-color: #f0f0ff;\n }\n`;\n\ninterface Embedding {\n id: number;\n documentId: string;\n title: string;\n content?: string;\n embeddingId?: string;\n}\n\ninterface EmbeddingsTableProps {\n data: Embedding[];\n}\n\nexport function EmbeddingsTable({ data }: EmbeddingsTableProps) {\n const navigate = useNavigate();\n\n const handleRowClick = (documentId: string) => {\n navigate(`/plugins/${PLUGIN_ID}/embeddings/${documentId}`);\n };\n\n return (\n <Box padding={8} background=\"neutral100\">\n <Table colCount={5} rowCount={data.length + 1}>\n <Thead>\n <Tr>\n <Th>\n <Typography variant=\"sigma\">ID</Typography>\n </Th>\n <Th>\n <Typography variant=\"sigma\">Title</Typography>\n </Th>\n <Th>\n <Typography variant=\"sigma\">Content</Typography>\n </Th>\n <Th>\n <Typography variant=\"sigma\">Embed ID</Typography>\n </Th>\n <Th>\n <VisuallyHidden>Actions</VisuallyHidden>\n </Th>\n </Tr>\n </Thead>\n <Tbody>\n {data?.map((entry) => (\n <StyledTr\n key={entry.documentId}\n onClick={() => handleRowClick(entry.documentId)}\n >\n <Td>\n <Typography textColor=\"neutral800\">\n {entry.documentId.slice(0, 8)}...\n </Typography>\n </Td>\n <Td>\n <Typography textColor=\"neutral800\">\n {entry.title?.slice(0, 30)}\n {entry.title?.length > 30 ? \"...\" : \"\"}\n </Typography>\n </Td>\n <Td>\n <Typography textColor=\"neutral800\">\n {entry.content?.slice(0, 30)}\n {entry.content && entry.content.length > 30 ? \"...\" : \"\"}\n </Typography>\n </Td>\n <Td>\n <Typography textColor=\"neutral800\">\n {entry.embeddingId?.slice(0, 8)}\n {entry.embeddingId && entry.embeddingId.length > 8\n ? \"...\"\n : \"\"}\n </Typography>\n </Td>\n <Td>\n <Flex>\n <IconButton\n withTooltip={false}\n label=\"View details\"\n onClick={(e: React.MouseEvent) => {\n e.stopPropagation();\n handleRowClick(entry.documentId);\n }}\n >\n <ArrowRight />\n </IconButton>\n </Flex>\n </Td>\n </StyledTr>\n ))}\n </Tbody>\n </Table>\n </Box>\n );\n}\n","import React from \"react\";\nimport ReactMarkdown from \"react-markdown\";\nimport styled from \"styled-components\";\n\nconst MarkdownWrapper = styled.div`\n /* Headers */\n h1 {\n font-size: 2.25rem;\n font-weight: 700;\n margin-bottom: 1rem;\n color: #272728;\n }\n\n h2 {\n font-size: 1.75rem;\n font-weight: 700;\n margin-bottom: 1rem;\n color: #272728;\n }\n\n h3 {\n font-size: 1.5rem;\n font-weight: 700;\n margin-bottom: 1rem;\n color: #272728;\n }\n\n h4 {\n font-size: 1.25rem;\n font-weight: 700;\n margin-bottom: 1rem;\n color: #272728;\n }\n\n h5 {\n font-size: 1.125rem;\n font-weight: 700;\n margin-bottom: 1rem;\n color: #272728;\n }\n\n h6 {\n font-size: 1rem;\n font-weight: 700;\n margin-bottom: 1rem;\n color: #9d4edd;\n }\n\n /* Horizontal rules */\n hr {\n border-color: #d1d5db;\n margin-top: 2rem;\n margin-bottom: 2rem;\n }\n\n a {\n color: #4945ff;\n text-decoration: underline;\n }\n\n /* Paragraphs */\n p {\n margin-bottom: 1rem;\n line-height: 1.5rem;\n color: #39393a;\n }\n\n /* Emphasis */\n strong {\n font-weight: 700;\n }\n\n em {\n font-style: italic;\n }\n\n del {\n text-decoration: line-through;\n }\n\n /* Blockquotes */\n blockquote {\n border-left-width: 1px;\n border-color: #9ca3af;\n padding-left: 1rem;\n padding-top: 0.5rem;\n padding-bottom: 0.5rem;\n margin-bottom: 1rem;\n }\n\n /* Lists */\n ul {\n list-style-type: disc;\n padding-left: 1rem;\n margin-bottom: 1rem;\n }\n\n ol {\n list-style-type: decimal;\n padding-left: 1rem;\n margin-bottom: 1rem;\n }\n\n li {\n margin-bottom: 0.5rem;\n }\n\n li > ul {\n list-style-type: disc;\n padding-left: 1rem;\n margin-bottom: 0.5rem;\n }\n\n li > ol {\n list-style-type: decimal;\n padding-left: 1rem;\n margin-bottom: 0.5rem;\n }\n\n /* Code blocks */\n pre {\n font-family: monospace;\n background-color: #1f2937;\n color: #f9fafb;\n border-radius: 0.375rem;\n padding: 1rem;\n margin-top: 1.5rem;\n margin-bottom: 1.5rem;\n line-height: 1.5rem;\n overflow: auto;\n }\n\n code {\n font-family: monospace;\n background-color: #1f2937;\n color: #f9fafb;\n border-radius: 0.375rem;\n padding-left: 0.5rem;\n padding-right: 0.5rem;\n padding-top: 0.25rem;\n padding-bottom: 0.25rem;\n }\n\n /* Tables */\n table {\n width: 100%;\n border-collapse: collapse;\n border-color: #d1d5db;\n margin-top: 1.5rem;\n margin-bottom: 1.5rem;\n }\n\n th {\n background-color: #1f2937;\n text-align: left;\n padding-top: 0.5rem;\n padding-bottom: 0.5rem;\n padding-left: 1rem;\n padding-right: 1rem;\n font-weight: 600;\n border-bottom-width: 1px;\n border-color: #d1d5db;\n }\n\n td {\n padding-top: 0.5rem;\n padding-bottom: 0.5rem;\n padding-left: 1rem;\n padding-right: 1rem;\n border-bottom-width: 1px;\n border-color: #d1d5db;\n }\n\n /* Images */\n img {\n width: 100%;\n object-fit: cover;\n border-radius: 0.75rem;\n margin-top: 1.5rem;\n margin-bottom: 1.5rem;\n }\n`;\n\ninterface MarkdownProps {\n children: string;\n}\n\nexport function Markdown({ children }: MarkdownProps) {\n return (\n <MarkdownWrapper>\n <ReactMarkdown>{children}</ReactMarkdown>\n </MarkdownWrapper>\n );\n}\n","import React, { useState, useRef, useEffect } from \"react\";\nimport { useNavigate } from \"react-router-dom\";\nimport styled from \"styled-components\";\nimport qs from \"qs\";\nimport { useFetchClient } from \"@strapi/strapi/admin\";\nimport {\n Button,\n Typography,\n Box,\n TextInput,\n Modal,\n Accordion,\n Link,\n} from \"@strapi/design-system\";\n\nimport { PLUGIN_ID } from \"../../pluginId\";\nimport { RobotIcon } from \"./RobotIcon\";\nimport { Markdown } from \"./Markdown\";\n\nconst StyledButton = styled(Button)`\n position: fixed;\n bottom: 1.5rem;\n right: 1.5rem;\n height: 3.5rem;\n width: 3.5rem;\n border-radius: 50%;\n display: flex;\n align-items: center;\n justify-content: center;\n box-shadow: 0 4px 6px -1px rgba(0, 0, 0, 0.1),\n 0 2px 4px -1px rgba(0, 0, 0, 0.06);\n z-index: 100;\n\n svg {\n height: 1.75rem;\n width: 1.75rem;\n }\n`;\n\nconst ResponseContainer = styled.div`\n border: solid 1px #e3e9f3;\n border-radius: 4px;\n padding: 1rem;\n color: #32324d;\n font-weight: 400;\n font-size: 0.875rem;\n display: block;\n width: 100%;\n max-height: 400px;\n background: inherit;\n overflow-y: auto;\n scroll-behavior: smooth;\n`;\n\ninterface SourceDocument {\n pageContent: string;\n metadata: {\n id: string;\n title: string;\n collectionType?: string;\n fieldName?: string;\n };\n}\n\ninterface QueryResponse {\n text: string;\n sourceDocuments: SourceDocument[];\n}\n\ninterface AccordionDetailsProps {\n title: string;\n content: React.ReactNode;\n children?: React.ReactNode;\n}\n\nfunction AccordionDetails({ title, content, children }: AccordionDetailsProps) {\n return (\n <Box padding={1} background=\"primary100\">\n <Accordion.Root size=\"S\">\n <Accordion.Item value=\"acc-1\">\n <Accordion.Header>\n <Accordion.Trigger>{title}</Accordion.Trigger>\n </Accordion.Header>\n <Accordion.Content>\n <Box padding={3}>\n <Typography>{content}</Typography>\n {children && <Box padding={1}>{children}</Box>}\n </Box>\n </Accordion.Content>\n </Accordion.Item>\n </Accordion.Root>\n </Box>\n );\n}\n\ninterface ShowResponseProps {\n data: QueryResponse[];\n onNavigate: (id: string) => void;\n}\n\nfunction ShowResponse({ data, onNavigate }: ShowResponseProps) {\n return (\n <>\n {data.map((item, index) => (\n <Box key={index} marginBottom={4}>\n <Box padding={1}>\n <Markdown>{item.text}</Markdown>\n </Box>\n\n {item.sourceDocuments?.length > 0 &&\n item.sourceDocuments.map((doc, docIndex) => (\n <AccordionDetails\n key={docIndex}\n title=\"Original Source Document\"\n content={<Markdown>{doc.pageContent}</Markdown>}\n >\n <Link\n onClick={() => onNavigate(doc.metadata.id)}\n style={{ cursor: \"pointer\" }}\n >\n View Source for {doc.metadata.title}\n </Link>\n </AccordionDetails>\n ))}\n </Box>\n ))}\n </>\n );\n}\n\nexport function ChatModal() {\n const { get } = useFetchClient();\n const containerRef = useRef<HTMLDivElement>(null);\n const navigate = useNavigate();\n\n const [isVisible, setIsVisible] = useState(false);\n const [isLoading, setIsLoading] = useState(false);\n const [inputValue, setInputValue] = useState(\"\");\n const [data, setData] = useState<QueryResponse[]>([]);\n\n useEffect(() => {\n if (containerRef.current) {\n containerRef.current.scrollTop = containerRef.current.scrollHeight;\n }\n }, [data]);\n\n function handleNavigate(id: string) {\n setIsVisible(false);\n navigate(`/plugins/${PLUGIN_ID}/embeddings/${id}`);\n }\n\n async function handleQueryEmbeddings(e: React.FormEvent) {\n e.preventDefault();\n if (!inputValue.trim() || isLoading) return;\n\n setIsLoading(true);\n try {\n const response = await get(\n `/${PLUGIN_ID}/embeddings/embeddings-query?${qs.stringify({\n query: inputValue,\n })}`\n );\n if (response.data && !response.data.error) {\n setData((prev) => [...prev, response.data as QueryResponse]);\n }\n setInputValue(\"\");\n } catch (error) {\n console.error(\"Query failed:\", error);\n } finally {\n setIsLoading(false);\n }\n }\n\n return (\n <>\n <StyledButton onClick={() => setIsVisible(true)} aria-label=\"Open chat\">\n <RobotIcon height={28} width={28} />\n </StyledButton>\n\n <Modal.Root open={isVisible} onOpenChange={setIsVisible}>\n <Modal.Content>\n <Modal.Header>\n <Modal.Title>Chat With Your Data</Modal.Title>\n </Modal.Header>\n <Modal.Body>\n {data.length > 0 && (\n <Box padding={1} marginBottom={4}>\n <ResponseContainer ref={containerRef}>\n <ShowResponse data={data} onNavigate={handleNavigate} />\n </ResponseContainer>\n </Box>\n )}\n <Box padding={1}>\n <form onSubmit={handleQueryEmbeddings}>\n <TextInput\n placeholder=\"Enter your question\"\n type=\"text\"\n aria-label=\"Question\"\n name=\"question\"\n onChange={(e: React.ChangeEvent<HTMLInputElement>) =>\n setInputValue(e.target.value)\n }\n value={inputValue}\n />\n </form>\n </Box>\n </Modal.Body>\n <Modal.Footer>\n <Modal.Close>\n <Button variant=\"tertiary\">Cancel</Button>\n </Modal.Close>\n <Button\n onClick={handleQueryEmbeddings}\n disabled={!inputValue.trim() || isLoading}\n loading={isLoading}\n >\n {isLoading ? \"Sending...\" : \"Send\"}\n </Button>\n </Modal.Footer>\n </Modal.Content>\n </Modal.Root>\n </>\n );\n}\n","import React, { useEffect, useState, useCallback, useMemo } from 'react';\nimport { useNavigate } from 'react-router-dom';\nimport { Main, Box, Button, TextInput, Flex, Loader } from '@strapi/design-system';\nimport { Plus, Search } from '@strapi/icons';\nimport { useFetchClient, Layouts } from '@strapi/strapi/admin';\nimport qs from 'qs';\n\nimport { PLUGIN_ID } from '../pluginId';\nimport { EmptyState } from '../components/custom/EmptyState';\nimport { EmbeddingsTable } from '../components/custom/EmbeddingsTable';\nimport { ChatModal } from '../components/custom/ChatModal';\n\ninterface Embedding {\n id: number;\n documentId: string;\n title: string;\n content?: string;\n embeddingsId?: string;\n}\n\ninterface EmbeddingsResponse {\n data: Embedding[];\n count: number;\n totalCount: number;\n}\n\nfunction debounce<T extends (...args: any[]) => any>(\n func: T,\n wait: number\n): (...args: Parameters<T>) => void {\n let timeout: NodeJS.Timeout;\n return (...args: Parameters<T>) => {\n clearTimeout(timeout);\n timeout = setTimeout(() => func(...args), wait);\n };\n}\n\nexport function HomePage() {\n const { get } = useFetchClient();\n const navigate = useNavigate();\n\n const [embeddings, setEmbeddings] = useState<EmbeddingsResponse | null>(null);\n const [search, setSearch] = useState('');\n const [isLoading, setIsLoading] = useState(true);\n\n const buildQuery = (searchTerm: string) =>\n qs.stringify({\n filters: searchTerm\n ? {\n $or: [{ title: { $containsi: searchTerm } }, { content: { $containsi: searchTerm } }],\n }\n : undefined,\n });\n\n const fetchData = useCallback(\n async (searchTerm: string) => {\n setIsLoading(true);\n try {\n const response = await get(`/${PLUGIN_ID}/embeddings/find?${buildQuery(searchTerm)}`);\n setEmbeddings(response.data as EmbeddingsResponse);\n } catch (error) {\n console.error('Failed to fetch embeddings:', error);\n setEmbeddings({ data: [], count: 0, totalCount: 0 });\n } finally {\n setIsLoading(false);\n }\n },\n [get]\n );\n\n const debouncedFetch = useMemo(() => debounce(fetchData, 500), [fetchData]);\n\n useEffect(() => {\n debouncedFetch(search);\n }, [search, debouncedFetch]);\n\n const handleSearchChange = (e: React.ChangeEvent<HTMLInputElement>) => {\n setSearch(e.target.value);\n };\n\n const handleCreateNew = () => {\n navigate(`/plugins/${PLUGIN_ID}/embeddings`);\n };\n\n if (isLoading && !embeddings) {\n return (\n <Main>\n <Layouts.Header title={'Content Embeddings'} subtitle={'Manage your content embeddings'} />\n <Layouts.Content>\n <Flex justifyContent=\"center\" padding={8}>\n <Loader>Loading...</Loader>\n </Flex>\n </Layouts.Content>\n <ChatModal />\n </Main>\n );\n }\n\n if (embeddings?.totalCount === 0 && !search) {\n return (\n <Main>\n <Layouts.Header title={'Content Embeddings'} subtitle={'Manage your content embeddings'} />\n <Layouts.Content>\n <EmptyState />\n </Layouts.Content>\n <ChatModal />\n </Main>\n );\n }\n\n // Render embeddings content based on loading state and data\n const renderEmbeddingsContent = () => {\n if (isLoading) {\n return (\n <Flex justifyContent=\"center\" padding={8}>\n <Loader>Loading...</Loader>\n </Flex>\n );\n }\n\n if (embeddings?.data && embeddings.data.length > 0) {\n return <EmbeddingsTable data={embeddings.data} />;\n }\n\n return (\n <Box padding={8} textAlign=\"center\">\n No embeddings found matching \"{search}\"\n </Box>\n );\n };\n\n return (\n <Main>\n <Layouts.Header\n title={'Content Embeddings'}\n subtitle={`${embeddings?.count || 0} results found`}\n primaryAction={\n <Button startIcon={<Plus />} onClick={handleCreateNew}>\n Create new embedding\n </Button>\n }\n />\n <Layouts.Content>\n <Box paddingBottom={4}>\n <TextInput\n placeholder=\"Search embeddings...\"\n name=\"search\"\n value={search}\n onChange={handleSearchChange}\n startAction={<Search />}\n />\n </Box>\n {renderEmbeddingsContent()}\n </Layouts.Content>\n <ChatModal />\n </Main>\n );\n}\n","import React from 'react';\nimport { Box, Field, TextInput, Textarea } from '@strapi/design-system';\nimport { MarkdownEditor } from '../custom/MarkdownEditor';\n\ninterface CreateEmbeddingsFormProps {\n onSubmit: (e: React.FormEvent<HTMLFormElement>) => void;\n isLoading: boolean;\n input: string;\n setInput: (input: string) => void;\n markdown: string;\n handleMarkdownChange: React.Dispatch<React.SetStateAction<string>>;\n metadata: string;\n setMetadata: (metadata: string) => void;\n height?: number;\n children?: React.ReactNode;\n}\n\nexport function CreateEmbeddingsForm({\n onSubmit,\n isLoading,\n input,\n setInput,\n markdown,\n handleMarkdownChange,\n metadata,\n setMetadata,\n height,\n children,\n}: CreateEmbeddingsFormProps) {\n return (\n <form onSubmit={onSubmit}>\n <fieldset disabled={isLoading} style={{ border: 'none', padding: 0, margin: 0 }}>\n <Box marginBottom={4}>\n <Field.Root>\n <Field.Label>Title</Field.Label>\n <TextInput\n placeholder=\"Enter a title for your embedding\"\n name=\"input\"\n onChange={(e: React.ChangeEvent<HTMLInputElement>) => setInput(e.target.value)}\n value={input}\n />\n </Field.Root>\n </Box>\n\n <Box marginBottom={4}>\n <Field.Root>\n <Field.Label>Content</Field.Label>\n <MarkdownEditor\n content={markdown}\n onChange={handleMarkdownChange}\n height={height}\n />\n </Field.Root>\n </Box>\n\n <Box marginBottom={4}>\n <Field.Root>\n <Field.Label>Metadata (JSON)</Field.Label>\n <Field.Hint>Optional JSON metadata for this embedding</Field.Hint>\n <Textarea\n placeholder='{\"category\": \"docs\", \"source\": \"manual\"}'\n name=\"metadata\"\n onChange={(e: React.ChangeEvent<HTMLTextAreaElement>) => setMetadata(e.target.value)}\n value={metadata}\n />\n </Field.Root>\n </Box>\n\n {children}\n </fieldset>\n </form>\n );\n}\n","import { Link } from \"@strapi/design-system\";\nimport { ArrowLeft } from \"@strapi/icons\";\nimport { NavLink } from \"react-router-dom\";\n\ninterface BackLinkProps {\n to?: string;\n}\n\nexport function BackLink({ to }: BackLinkProps) {\n if (to) {\n return (\n <Link tag={NavLink} to={to} startIcon={<ArrowLeft />}>\n Go back\n </Link>\n );\n }\n\n return (\n <Link tag={NavLink} to=\"..\" relative=\"path\" startIcon={<ArrowLeft />}>\n Go back\n </Link>\n );\n}\n\nexport default BackLink;\n","import React, { useState } from \"react\";\nimport { useNavigate } from \"react-router-dom\";\nimport { useIntl } from \"react-intl\";\nimport { Main, Box, Button } from \"@strapi/design-system\";\nimport { useFetchClient, Layouts } from \"@strapi/strapi/admin\";\n\nimport { PLUGIN_ID } from \"../pluginId\";\nimport { CreateEmbeddingsForm } from \"../components/forms/CreateEmbeddingForm\";\nimport { BackLink } from \"../components/custom/BackLink\";\n\nconst CHUNK_SIZE = 4000; // Content over this will be auto-chunked\n\nexport default function CreateEmbeddings() {\n const { formatMessage } = useIntl();\n const navigate = useNavigate();\n const { post } = useFetchClient();\n\n const [isLoading, setIsLoading] = useState(false);\n const [title, setTitle] = useState(\"\");\n const [content, setContent] = useState(\"\");\n const [metadata, setMetadata] = useState(\"\");\n const [error, setError] = useState<string | null>(null);\n\n const isValid = title.trim() && content.trim();\n const contentLength = content.length;\n const willChunk = contentLength > CHUNK_SIZE;\n const estimatedChunks = willChunk ? Math.ceil(contentLength / (CHUNK_SIZE - 200)) : 1;\n\n function parseMetadata(): Record<string, any> | null {\n if (!metadata.trim()) return null;\n try {\n return JSON.parse(metadata);\n } catch {\n return null;\n }\n }\n\n async function handleSubmit(e: React.FormEvent<HTMLFormElement>) {\n e.preventDefault();\n e.stopPropagation();\n\n if (!isValid) {\n setError(\"Please provide a title and content\");\n return;\n }\n\n // Validate metadata JSON if provided\n if (metadata.trim()) {\n const parsedMetadata = parseMetadata();\n if (parsedMetadata === null) {\n setError(\"Invalid JSON in metadata field\");\n return;\n }\n }\n\n setIsLoading(true);\n setError(null);\n\n try {\n await post(`/${PLUGIN_ID}/embeddings/create-embedding`, {\n data: {\n title: title.trim(),\n content: content.trim(),\n metadata: parseMetadata(),\n },\n });\n navigate(`/plugins/${PLUGIN_ID}`);\n } catch (err: any) {\n console.error(\"Failed to create embedding:\", err);\n setError(err.message || \"Failed to create embedding. Please try again.\");\n setIsLoading(false);\n }\n }\n\n return (\n <Main>\n <Layouts.Header\n title={formatMessage({\n id: \"CreateEmbeddings.header.title\",\n defaultMessage: \"Create Embedding\",\n })}\n subtitle={\n willChunk\n ? formatMessage(\n {\n id: \"CreateEmbeddings.header.subtitle.chunked\",\n defaultMessage: \"Content: {length} characters (will create ~{chunks} embeddings)\",\n },\n { length: contentLength, chunks: estimatedChunks }\n )\n : formatMessage(\n {\n id: \"CreateEmbeddings.header.subtitle\",\n defaultMessage: \"Content: {length} characters\",\n },\n { length: contentLength }\n )\n }\n primaryAction={\n <Button\n type=\"submit\"\n disabled={isLoading || !isValid}\n loading={isLoading}\n onClick={(e: React.MouseEvent) => {\n e.preventDefault();\n const form = document.querySelector(\"form\");\n if (form) {\n form.dispatchEvent(\n new Event(\"submit\", { cancelable: true, bubbles: true })\n );\n }\n }}\n >\n {isLoading ? \"Creating...\" : \"Create Embedding\"}\n </Button>\n }\n navigationAction={<BackLink to={`/plugins/${PLUGIN_ID}`} />}\n />\n <Layouts.Content>\n <Box>\n {error && (\n <Box\n padding={4}\n marginBottom={4}\n background=\"danger100\"\n borderColor=\"danger600\"\n hasRadius\n >\n {error}\n </Box>\n )}\n <CreateEmbeddingsForm\n onSubmit={handleSubmit}\n isLoading={isLoading}\n input={title}\n setInput={setTitle}\n markdown={content}\n handleMarkdownChange={setContent}\n metadata={metadata}\n setMetadata={setMetadata}\n />\n </Box>\n </Layouts.Content>\n </Main>\n );\n}\n","import React, { useEffect, useState } from \"react\";\nimport { useNavigate, useParams } from \"react-router-dom\";\nimport styled from \"styled-components\";\nimport {\n Main,\n Box,\n Flex,\n Button,\n Typography,\n Grid,\n Dialog,\n Loader,\n Field,\n TextInput,\n Textarea,\n} from \"@strapi/design-system\";\nimport { Trash, Pencil, Check, Cross } from \"@strapi/icons\";\nimport { useFetchClient, Layouts, useNotification } from \"@strapi/strapi/admin\";\n\nimport { PLUGIN_ID } from \"../pluginId\";\nimport { BackLink } from \"../components/custom/BackLink\";\nimport { Markdown } from \"../components/custom/Markdown\";\nimport { MarkdownEditor } from \"../components/custom/MarkdownEditor\";\n\nconst StyledTypography = styled(Typography)`\n display: block;\n margin-bottom: 1rem;\n`;\n\ninterface EmbeddingData {\n id: number;\n documentId: string;\n title: string;\n content?: string;\n embeddingId?: string;\n embedding?: number[];\n collectionType?: string;\n fieldName?: string;\n metadata?: Record<string, any>;\n}\n\ninterface MetadataProps {\n data: EmbeddingData;\n}\n\nfunction Metadata({ data }: MetadataProps) {\n const metadata = {\n id: data.documentId,\n title: data.title,\n collectionType: data.collectionType || \"standalone\",\n fieldName: data.fieldName || \"content\",\n embeddingId: data.embeddingId || \"N/A\",\n vectorDimensions: data.embedding?.length || 0,\n };\n\n return (\n <Box padding={4} background=\"neutral0\" hasRadius>\n <StyledTypography variant=\"beta\">Metadata</StyledTypography>\n {Object.entries(metadata).map(([key, value]) => (\n <Box key={key} padding={1}>\n <Typography>\n <strong>{key}:</strong> {String(value)}\n </Typography>\n </Box>\n ))}\n {data.metadata && (\n <Box marginTop={4}>\n <Typography variant=\"sigma\">Custom Metadata</Typography>\n <Box padding={2} background=\"neutral100\" hasRadius marginTop={2}>\n <pre style={{ fontSize: \"12px\", margin: 0, whiteSpace: \"pre-wrap\" }}>\n {JSON.stringify(data.metadata, null, 2)}\n </pre>\n </Box>\n </Box>\n )}\n </Box>\n );\n}\n\ninterface ConfirmDeleteProps {\n onConfirm: () => void;\n isLoading: boolean;\n}\n\nfunction ConfirmDeleteEmbedding({ onConfirm, isLoading }: ConfirmDeleteProps) {\n return (\n <Dialog.Root>\n <Dialog.Trigger>\n <Button variant=\"danger-light\" startIcon={<Trash />}>\n Delete\n </Button>\n </Dialog.Trigger>\n <Dialog.Content>\n <Dialog.Header>Confirm Deletion</Dialog.Header>\n <Dialog.Body>\n <Flex direction=\"column\" alignItems=\"center\" gap={2}>\n <Typography>Are you sure you want to delete this embedding?</Typography>\n <Typography variant=\"pi\" textColor=\"neutral600\">\n This action cannot be undone.\n </Typography>\n </Flex>\n </Dialog.Body>\n <Dialog.Footer>\n <Dialog.Cancel>\n <Button variant=\"tertiary\">Cancel</Button>\n </Dialog.Cancel>\n <Dialog.Action>\n <Button\n variant=\"danger\"\n onClick={onConfirm}\n startIcon={<Trash />}\n loading={isLoading}\n >\n {isLoading ? \"Deleting...\" : \"Delete\"}\n </Button>\n </Dialog.Action>\n </Dialog.Footer>\n </Dialog.Content>\n </Dialog.Root>\n );\n}\n\nexport default function EmbeddingDetails() {\n const navigate = useNavigate();\n const { id } = useParams<{ id: string }>();\n const { del, get, put } = useFetchClient();\n const { toggleNotification } = useNotification();\n\n const [data, setData] = useState<EmbeddingData | null>(null);\n const [isLoading, setIsLoading] = useState(true);\n const [isDeleting, setIsDeleting] = useState(false);\n const [isEditing, setIsEditing] = useState(false);\n const [isSaving, setIsSaving] = useState(false);\n\n // Edit form state\n const [editTitle, setEditTitle] = useState(\"\");\n const [editContent, setEditContent] = useState(\"\");\n const [editMetadata, setEditMetadata] = useState(\"\");\n\n useEffect(() => {\n async function fetchData() {\n if (!id) return;\n try {\n const response = await get(`/${PLUGIN_ID}/embeddings/find/${id}`);\n const embeddingData = response.data as EmbeddingData;\n setData(embeddingData);\n // Initialize edit form\n setEditTitle(embeddingData.title || \"\");\n setEditContent(embeddingData.content || \"\");\n setEditMetadata(embeddingData.metadata ? JSON.stringify(embeddingData.metadata, null, 2) : \"\");\n } catch (error) {\n console.error(\"Failed to fetch embedding:\", error);\n } finally {\n setIsLoading(false);\n }\n }\n fetchData();\n }, [id, get]);\n\n const handleDelete = async () => {\n if (!id || isDeleting) return;\n setIsDeleting(true);\n try {\n await del(`/${PLUGIN_ID}/embeddings/delete-embedding/${id}`);\n navigate(`/plugins/${PLUGIN_ID}`);\n } catch (error) {\n console.error(\"Failed to delete embedding:\", error);\n setIsDeleting(false);\n }\n };\n\n const handleStartEdit = () => {\n if (data) {\n setEditTitle(data.title || \"\");\n setEditContent(data.content || \"\");\n setEditMetadata(data.metadata ? JSON.stringify(data.metadata, null, 2) : \"\");\n }\n setIsEditing(true);\n };\n\n const handleCancelEdit = () => {\n setIsEditing(false);\n };\n\n const handleSave = async () => {\n if (!id || isSaving) return;\n\n // Validate metadata JSON if provided\n let parsedMetadata = null;\n if (editMetadata.trim()) {\n try {\n parsedMetadata = JSON.parse(editMetadata);\n } catch {\n toggleNotification({\n type: \"warning\",\n message: \"Invalid JSON in metadata field\",\n });\n return;\n }\n }\n\n setIsSaving(true);\n try {\n const response = await put(`/${PLUGIN_ID}/embeddings/update-embedding/${id}`, {\n data: {\n title: editTitle.trim(),\n content: editContent.trim(),\n metadata: parsedMetadata,\n },\n });\n\n setData(response.data as EmbeddingData);\n setIsEditing(false);\n toggleNotification({\n type: \"success\",\n message: \"Embedding updated successfully\",\n });\n } catch (error: any) {\n console.error(\"Failed to update embedding:\", error);\n toggleNotification({\n type: \"danger\",\n message: error.message || \"Failed to update embedding\",\n });\n } finally {\n setIsSaving(false);\n }\n };\n\n if (isLoading) {\n return (\n <Main>\n <Layouts.Header\n title=\"Loading...\"\n navigationAction={<BackLink to={`/plugins/${PLUGIN_ID}`} />}\n />\n <Layouts.Content>\n <Flex justifyContent=\"center\" padding={8}>\n <Loader>Loading embedding details...</Loader>\n </Flex>\n </Layouts.Content>\n </Main>\n );\n }\n\n if (!data) {\n return (\n <Main>\n <Layouts.Header\n title=\"Embedding Not Found\"\n navigationAction={<BackLink to={`/plugins/${PLUGIN_ID}`} />}\n />\n <Layouts.Content>\n <Box padding={8} textAlign=\"center\">\n <Typography>The requested embedding could not be found.</Typography>\n </Box>\n </Layouts.Content>\n </Main>\n );\n }\n\n return (\n <Main>\n <Layouts.Header\n title={isEditing ? \"Edit Embedding\" : (data.title || \"Embedding Details\")}\n subtitle={`Embedding ID: ${data.embeddingId || \"N/A\"}`}\n primaryAction={\n isEditing ? (\n <Flex gap={2}>\n <Button variant=\"tertiary\" startIcon={<Cross />} onClick={handleCancelEdit}>\n Cancel\n </Button>\n <Button startIcon={<Check />} onClick={handleSave} loading={isSaving}>\n {isSaving ? \"Saving...\" : \"Save\"}\n </Button>\n </Flex>\n ) : (\n <Flex gap={2}>\n <Button variant=\"secondary\" startIcon={<Pencil />} onClick={handleStartEdit}>\n Edit\n </Button>\n <ConfirmDeleteEmbedding onConfirm={handleDelete} isLoading={isDeleting} />\n </Flex>\n )\n }\n navigationAction={<BackLink to={`/plugins/${PLUGIN_ID}`} />}\n />\n <Layouts.Content>\n <Box padding={8}>\n {isEditing ? (\n <Grid.Root gap={6}>\n <Grid.Item col={8} s={12}>\n <Box background=\"neutral100\" padding={1} hasRadius>\n <Box padding={4} background=\"neutral0\" hasRadius>\n <Box marginBottom={4}>\n <Field.Root>\n <Field.Label>Title</Field.Label>\n <TextInput\n value={editTitle}\n onChange={(e: React.ChangeEvent<HTMLInputElement>) =>\n setEditTitle(e.target.value)\n }\n />\n </Field.Root>\n </Box>\n <Box marginBottom={4}>\n <Field.Root>\n <Field.Label>Content</Field.Label>\n <Field.Hint>\n Changes to content will regenerate the embedding vector\n </Field.Hint>\n </Field.Root>\n <MarkdownEditor\n content={editContent}\n onChange={setEditContent}\n height={300}\n />\n </Box>\n </Box>\n </Box>\n </Grid.Item>\n <Grid.Item col={4} s={12}>\n <Box background=\"neutral100\" padding={1} hasRadius>\n <Box padding={4} background=\"neutral0\" hasRadius>\n <Box marginBottom={4}>\n <Field.Root>\n <Field.Label>Metadata (JSON)</Field.Label>\n <Field.Hint>Optional custom metadata</Field.Hint>\n <Textarea\n value={editMetadata}\n onChange={(e: React.ChangeEvent<HTMLTextAreaElement>) =>\n setEditMetadata(e.target.value)\n }\n placeholder='{\"key\": \"value\"}'\n />\n </Field.Root>\n </Box>\n </Box>\n </Box>\n </Grid.Item>\n </Grid.Root>\n ) : (\n <Grid.Root gap={6}>\n <Grid.Item col={8} s={12}>\n <Box background=\"neutral100\" padding={1} hasRadius>\n <Box padding={4} background=\"neutral0\" hasRadius>\n <StyledTypography variant=\"beta\">\n Embedding Content\n </StyledTypography>\n {data.content ? (\n <Markdown>{data.content}</Markdown>\n ) : (\n <Typography textColor=\"neutral600\">No content</Typography>\n )}\n </Box>\n </Box>\n </Grid.Item>\n <Grid.Item col={4} s={12}>\n <Box background=\"neutral100\" padding={1} hasRadius>\n <Metadata data={data} />\n </Box>\n </Grid.Item>\n </Grid.Root>\n )}\n </Box>\n </Layouts.Content>\n </Main>\n );\n}\n","import { Page } from '@strapi/strapi/admin';\nimport { Routes, Route } from 'react-router-dom';\n\nimport { HomePage } from './HomePage';\nimport CreateEmbeddings from './CreateEmbeddings';\nimport EmbeddingDetails from './EmbeddingDetails';\n\nconst App = () => {\n return (\n <Routes>\n <Route index element={<HomePage />} />\n <Route path=\"*\" element={<Page.Error />} />\n <Route path=\"/embeddings\" element={<CreateEmbeddings />} />\n <Route path=\"/embeddings/:id\" element={<EmbeddingDetails />} />\n </Routes>\n );\n};\n\nexport { App };\n"],"names":["jsxs","jsx","EmptyStateLayout","Link","PLUGIN_ID","Button","Plus","styled","Tr","useNavigate","Box","Table","Thead","Th","Typography","VisuallyHidden","Tbody","Td","Flex","IconButton","ArrowRight","ReactMarkdown","Accordion","Fragment","index","useFetchClient","useRef","useState","useEffect","qs","RobotIcon","Modal","TextInput","useCallback","useMemo","Main","Layouts","Loader","Search","Field","MarkdownEditor","Textarea","NavLink","ArrowLeft","useIntl","Dialog","Trash","useParams","useNotification","Cross","Check","Pencil","Grid","Routes","Route","Page"],"mappings":"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"}