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.
- package/README.md +187 -0
- package/dist/_chunks/{App-Swmo_WMf.js → App-Rq72tIgS.js} +37 -55
- package/dist/_chunks/App-Rq72tIgS.js.map +1 -0
- package/dist/_chunks/{App-BlCKKuQN.mjs → App-j180lztd.mjs} +37 -55
- 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-CXVoFiJp.mjs → index-B3j0IFUi.mjs} +70 -27
- package/dist/_chunks/index-B3j0IFUi.mjs.map +1 -0
- package/dist/_chunks/{index-BpKkUIJY.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).
|
|
@@ -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-
|
|
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:
|
|
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
|
-
|
|
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
|
|
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()
|
|
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
|
|
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}
|
|
694
|
+
defaultMessage: "Content: {length} characters"
|
|
714
695
|
},
|
|
715
|
-
{ length: contentLength
|
|
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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"}
|