strapi-content-embeddings 0.1.4 → 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.
- package/README.md +187 -0
- package/dist/_chunks/{App-CnXhqiao.js → App-Rq72tIgS.js} +35 -37
- package/dist/_chunks/App-Rq72tIgS.js.map +1 -0
- package/dist/_chunks/{App-4UwemHRe.mjs → App-j180lztd.mjs} +35 -37
- package/dist/_chunks/App-j180lztd.mjs.map +1 -0
- package/dist/_chunks/en-B4KWt_jN.js +1 -0
- package/dist/_chunks/en-B4KWt_jN.js.map +1 -0
- package/dist/_chunks/en-Byx4XI2L.mjs +1 -0
- package/dist/_chunks/en-Byx4XI2L.mjs.map +1 -0
- package/dist/_chunks/{index-BWSiu_nE.mjs → index-B3j0IFUi.mjs} +70 -27
- package/dist/_chunks/index-B3j0IFUi.mjs.map +1 -0
- package/dist/_chunks/{index-BaPVw3mi.js → index-jf6vikTZ.js} +70 -27
- package/dist/_chunks/index-jf6vikTZ.js.map +1 -0
- package/dist/admin/index.js +2 -1
- package/dist/admin/index.js.map +1 -0
- package/dist/admin/index.mjs +2 -1
- package/dist/admin/index.mjs.map +1 -0
- package/dist/admin/src/components/custom/MarkdownEditor.d.ts +1 -1
- package/dist/server/index.js +850 -57
- package/dist/server/index.js.map +1 -0
- package/dist/server/index.mjs +850 -57
- package/dist/server/index.mjs.map +1 -0
- package/dist/server/src/config/index.d.ts +9 -0
- package/dist/server/src/controllers/controller.d.ts +14 -0
- package/dist/server/src/controllers/index.d.ts +2 -0
- package/dist/server/src/index.d.ts +38 -2
- package/dist/server/src/mcp/tools/create-embedding.d.ts +6 -0
- package/dist/server/src/mcp/tools/index.d.ts +4 -0
- package/dist/server/src/plugin-manager.d.ts +16 -0
- package/dist/server/src/routes/content-api.d.ts +10 -0
- package/dist/server/src/routes/index.d.ts +10 -0
- package/dist/server/src/services/embeddings.d.ts +43 -2
- package/dist/server/src/services/index.d.ts +23 -2
- package/dist/server/src/services/sync.d.ts +48 -0
- package/dist/server/src/utils/chunking.d.ts +44 -0
- 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).
|
|
@@ -7,7 +7,7 @@ const react = require("react");
|
|
|
7
7
|
const designSystem = require("@strapi/design-system");
|
|
8
8
|
const icons = require("@strapi/icons");
|
|
9
9
|
const qs = require("qs");
|
|
10
|
-
const index = require("./index-
|
|
10
|
+
const index = require("./index-jf6vikTZ.js");
|
|
11
11
|
const styled = require("styled-components");
|
|
12
12
|
const ReactMarkdown = require("react-markdown");
|
|
13
13
|
const reactIntl = require("react-intl");
|
|
@@ -483,19 +483,14 @@ function HomePage() {
|
|
|
483
483
|
const [isLoading, setIsLoading] = react.useState(true);
|
|
484
484
|
const buildQuery = (searchTerm) => qs__default.default.stringify({
|
|
485
485
|
filters: searchTerm ? {
|
|
486
|
-
$or: [
|
|
487
|
-
{ title: { $containsi: searchTerm } },
|
|
488
|
-
{ content: { $containsi: searchTerm } }
|
|
489
|
-
]
|
|
486
|
+
$or: [{ title: { $containsi: searchTerm } }, { content: { $containsi: searchTerm } }]
|
|
490
487
|
} : void 0
|
|
491
488
|
});
|
|
492
489
|
const fetchData = react.useCallback(
|
|
493
490
|
async (searchTerm) => {
|
|
494
491
|
setIsLoading(true);
|
|
495
492
|
try {
|
|
496
|
-
const response = await get(
|
|
497
|
-
`/${index.PLUGIN_ID}/embeddings/find?${buildQuery(searchTerm)}`
|
|
498
|
-
);
|
|
493
|
+
const response = await get(`/${index.PLUGIN_ID}/embeddings/find?${buildQuery(searchTerm)}`);
|
|
499
494
|
setEmbeddings(response.data);
|
|
500
495
|
} catch (error) {
|
|
501
496
|
console.error("Failed to fetch embeddings:", error);
|
|
@@ -506,10 +501,7 @@ function HomePage() {
|
|
|
506
501
|
},
|
|
507
502
|
[get]
|
|
508
503
|
);
|
|
509
|
-
const debouncedFetch = react.useMemo(
|
|
510
|
-
() => debounce(fetchData, 500),
|
|
511
|
-
[fetchData]
|
|
512
|
-
);
|
|
504
|
+
const debouncedFetch = react.useMemo(() => debounce(fetchData, 500), [fetchData]);
|
|
513
505
|
react.useEffect(() => {
|
|
514
506
|
debouncedFetch(search);
|
|
515
507
|
}, [search, debouncedFetch]);
|
|
@@ -521,30 +513,31 @@ function HomePage() {
|
|
|
521
513
|
};
|
|
522
514
|
if (isLoading && !embeddings) {
|
|
523
515
|
return /* @__PURE__ */ jsxRuntime.jsxs(designSystem.Main, { children: [
|
|
524
|
-
/* @__PURE__ */ jsxRuntime.jsx(
|
|
525
|
-
admin.Layouts.Header,
|
|
526
|
-
{
|
|
527
|
-
title: "Content Embeddings",
|
|
528
|
-
subtitle: "Manage your content embeddings"
|
|
529
|
-
}
|
|
530
|
-
),
|
|
516
|
+
/* @__PURE__ */ jsxRuntime.jsx(admin.Layouts.Header, { title: "Content Embeddings", subtitle: "Manage your content embeddings" }),
|
|
531
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..." }) }) }),
|
|
532
518
|
/* @__PURE__ */ jsxRuntime.jsx(ChatModal, {})
|
|
533
519
|
] });
|
|
534
520
|
}
|
|
535
521
|
if (embeddings?.totalCount === 0 && !search) {
|
|
536
522
|
return /* @__PURE__ */ jsxRuntime.jsxs(designSystem.Main, { children: [
|
|
537
|
-
/* @__PURE__ */ jsxRuntime.jsx(
|
|
538
|
-
admin.Layouts.Header,
|
|
539
|
-
{
|
|
540
|
-
title: "Content Embeddings",
|
|
541
|
-
subtitle: "Manage your content embeddings"
|
|
542
|
-
}
|
|
543
|
-
),
|
|
523
|
+
/* @__PURE__ */ jsxRuntime.jsx(admin.Layouts.Header, { title: "Content Embeddings", subtitle: "Manage your content embeddings" }),
|
|
544
524
|
/* @__PURE__ */ jsxRuntime.jsx(admin.Layouts.Content, { children: /* @__PURE__ */ jsxRuntime.jsx(EmptyState, {}) }),
|
|
545
525
|
/* @__PURE__ */ jsxRuntime.jsx(ChatModal, {})
|
|
546
526
|
] });
|
|
547
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
|
+
};
|
|
548
541
|
return /* @__PURE__ */ jsxRuntime.jsxs(designSystem.Main, { children: [
|
|
549
542
|
/* @__PURE__ */ jsxRuntime.jsx(
|
|
550
543
|
admin.Layouts.Header,
|
|
@@ -565,11 +558,7 @@ function HomePage() {
|
|
|
565
558
|
startAction: /* @__PURE__ */ jsxRuntime.jsx(icons.Search, {})
|
|
566
559
|
}
|
|
567
560
|
) }),
|
|
568
|
-
|
|
569
|
-
'No embeddings found matching "',
|
|
570
|
-
search,
|
|
571
|
-
'"'
|
|
572
|
-
] })
|
|
561
|
+
renderEmbeddingsContent()
|
|
573
562
|
] }),
|
|
574
563
|
/* @__PURE__ */ jsxRuntime.jsx(ChatModal, {})
|
|
575
564
|
] });
|
|
@@ -632,7 +621,7 @@ function BackLink({ to }) {
|
|
|
632
621
|
}
|
|
633
622
|
return /* @__PURE__ */ jsxRuntime.jsx(designSystem.Link, { tag: reactRouterDom.NavLink, to: "..", relative: "path", startIcon: /* @__PURE__ */ jsxRuntime.jsx(icons.ArrowLeft, {}), children: "Go back" });
|
|
634
623
|
}
|
|
635
|
-
const
|
|
624
|
+
const CHUNK_SIZE = 4e3;
|
|
636
625
|
function CreateEmbeddings() {
|
|
637
626
|
const { formatMessage } = reactIntl.useIntl();
|
|
638
627
|
const navigate = reactRouterDom.useNavigate();
|
|
@@ -642,8 +631,10 @@ function CreateEmbeddings() {
|
|
|
642
631
|
const [content, setContent] = react.useState("");
|
|
643
632
|
const [metadata, setMetadata] = react.useState("");
|
|
644
633
|
const [error, setError] = react.useState(null);
|
|
645
|
-
const isValid = title.trim() && content.trim()
|
|
634
|
+
const isValid = title.trim() && content.trim();
|
|
646
635
|
const contentLength = content.length;
|
|
636
|
+
const willChunk = contentLength > CHUNK_SIZE;
|
|
637
|
+
const estimatedChunks = willChunk ? Math.ceil(contentLength / (CHUNK_SIZE - 200)) : 1;
|
|
647
638
|
function parseMetadata() {
|
|
648
639
|
if (!metadata.trim()) return null;
|
|
649
640
|
try {
|
|
@@ -656,7 +647,7 @@ function CreateEmbeddings() {
|
|
|
656
647
|
e.preventDefault();
|
|
657
648
|
e.stopPropagation();
|
|
658
649
|
if (!isValid) {
|
|
659
|
-
setError("Please provide a title and content
|
|
650
|
+
setError("Please provide a title and content");
|
|
660
651
|
return;
|
|
661
652
|
}
|
|
662
653
|
if (metadata.trim()) {
|
|
@@ -691,12 +682,18 @@ function CreateEmbeddings() {
|
|
|
691
682
|
id: "CreateEmbeddings.header.title",
|
|
692
683
|
defaultMessage: "Create Embedding"
|
|
693
684
|
}),
|
|
694
|
-
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(
|
|
695
692
|
{
|
|
696
693
|
id: "CreateEmbeddings.header.subtitle",
|
|
697
|
-
defaultMessage: "Content: {length}
|
|
694
|
+
defaultMessage: "Content: {length} characters"
|
|
698
695
|
},
|
|
699
|
-
{ length: contentLength
|
|
696
|
+
{ length: contentLength }
|
|
700
697
|
),
|
|
701
698
|
primaryAction: /* @__PURE__ */ jsxRuntime.jsx(
|
|
702
699
|
designSystem.Button,
|
|
@@ -990,3 +987,4 @@ const App = () => {
|
|
|
990
987
|
] });
|
|
991
988
|
};
|
|
992
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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"}
|