@promptbook/cli 0.103.0-46 → 0.103.0-47

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (83) hide show
  1. package/apps/agents-server/config.ts.todo +4 -4
  2. package/apps/agents-server/next.config.ts +3 -0
  3. package/apps/agents-server/src/app/agents/[agentName]/api/book/route.ts +1 -1
  4. package/apps/agents-server/src/app/agents/[agentName]/api/book/test.http +2 -2
  5. package/apps/agents-server/src/app/agents/[agentName]/api/chat/route.ts +6 -4
  6. package/apps/agents-server/src/app/agents/[agentName]/api/modelRequirements/TODO.txt +1 -0
  7. package/apps/agents-server/src/app/agents/[agentName]/api/modelRequirements/route.ts +53 -0
  8. package/apps/agents-server/src/app/agents/[agentName]/api/modelRequirements/systemMessage/route.ts +45 -0
  9. package/apps/agents-server/src/app/agents/[agentName]/api/profile/route.ts +54 -0
  10. package/apps/agents-server/src/app/agents/[agentName]/book/BookEditorWrapper.tsx +83 -17
  11. package/apps/agents-server/src/app/agents/[agentName]/book/page.tsx +9 -5
  12. package/apps/agents-server/src/app/agents/[agentName]/book+chat/{SelfLearningBook.tsx → AgentBookAndChatComponent.tsx} +5 -46
  13. package/apps/agents-server/src/app/agents/[agentName]/book+chat/page.tsx +7 -4
  14. package/apps/agents-server/src/app/agents/[agentName]/chat/AgentChatWrapper.tsx +38 -0
  15. package/apps/agents-server/src/app/agents/[agentName]/chat/page.tsx +23 -0
  16. package/apps/agents-server/src/app/agents/[agentName]/page.tsx +6 -4
  17. package/apps/agents-server/src/app/agents/page.tsx +11 -0
  18. package/apps/agents-server/src/app/api/chat/route.ts +1 -1
  19. package/apps/agents-server/src/app/api/chat-streaming/route.ts +5 -1
  20. package/apps/agents-server/src/app/api/upload/route.ts +75 -0
  21. package/apps/agents-server/src/tools/$provideAgentCollectionForServer.ts +1 -0
  22. package/apps/agents-server/src/tools/$provideCdnForServer.ts +28 -0
  23. package/apps/agents-server/src/utils/cdn/classes/DigitalOceanSpaces.ts +119 -0
  24. package/apps/agents-server/src/utils/cdn/interfaces/IFilesStorage.ts +32 -0
  25. package/apps/agents-server/src/utils/cdn/interfaces/IStorage.ts +14 -0
  26. package/apps/agents-server/src/utils/cdn/utils/getUserFileCdnKey.ts +27 -0
  27. package/apps/agents-server/src/utils/cdn/utils/nameToSubfolderPath.ts +9 -0
  28. package/apps/agents-server/src/utils/cdn/utils/nextRequestToNodeRequest.ts +27 -0
  29. package/apps/agents-server/src/utils/validators/validateMimeType.ts +24 -0
  30. package/apps/agents-server/tsconfig.json +1 -1
  31. package/esm/index.es.js +154 -161
  32. package/esm/index.es.js.map +1 -1
  33. package/esm/typings/servers.d.ts +1 -7
  34. package/esm/typings/src/_packages/components.index.d.ts +4 -0
  35. package/esm/typings/src/_packages/core.index.d.ts +16 -14
  36. package/esm/typings/src/_packages/types.index.d.ts +12 -6
  37. package/esm/typings/src/book-2.0/agent-source/AgentModelRequirements.d.ts +6 -1
  38. package/esm/typings/src/book-2.0/agent-source/AgentSourceParseResult.d.ts +1 -1
  39. package/esm/typings/src/book-2.0/agent-source/createCommitmentRegex.d.ts +1 -1
  40. package/esm/typings/src/book-components/Chat/AgentChat/AgentChat.d.ts +14 -0
  41. package/esm/typings/src/book-components/Chat/AgentChat/AgentChat.test.d.ts +1 -0
  42. package/esm/typings/src/book-components/Chat/AgentChat/AgentChatProps.d.ts +13 -0
  43. package/esm/typings/src/collection/agent-collection/constructors/agent-collection-in-supabase/AgentCollectionInSupabase.d.ts +1 -60
  44. package/esm/typings/src/{book-2.0/commitments → commitments}/ACTION/ACTION.d.ts +1 -1
  45. package/esm/typings/src/{book-2.0/commitments → commitments}/DELETE/DELETE.d.ts +1 -1
  46. package/esm/typings/src/{book-2.0/commitments → commitments}/FORMAT/FORMAT.d.ts +1 -1
  47. package/esm/typings/src/{book-2.0/commitments → commitments}/GOAL/GOAL.d.ts +1 -1
  48. package/esm/typings/src/{book-2.0/commitments → commitments}/KNOWLEDGE/KNOWLEDGE.d.ts +1 -5
  49. package/esm/typings/src/{book-2.0/commitments → commitments}/MEMORY/MEMORY.d.ts +1 -1
  50. package/esm/typings/src/{book-2.0/commitments → commitments}/MESSAGE/MESSAGE.d.ts +1 -1
  51. package/esm/typings/src/{book-2.0/commitments → commitments}/META/META.d.ts +1 -1
  52. package/esm/typings/src/{book-2.0/commitments → commitments}/META_IMAGE/META_IMAGE.d.ts +1 -1
  53. package/esm/typings/src/{book-2.0/commitments → commitments}/META_LINK/META_LINK.d.ts +1 -1
  54. package/esm/typings/src/{book-2.0/commitments → commitments}/MODEL/MODEL.d.ts +1 -1
  55. package/esm/typings/src/{book-2.0/commitments → commitments}/NOTE/NOTE.d.ts +1 -1
  56. package/esm/typings/src/{book-2.0/commitments → commitments}/PERSONA/PERSONA.d.ts +1 -1
  57. package/esm/typings/src/{book-2.0/commitments → commitments}/RULE/RULE.d.ts +1 -1
  58. package/esm/typings/src/{book-2.0/commitments → commitments}/SAMPLE/SAMPLE.d.ts +1 -1
  59. package/esm/typings/src/{book-2.0/commitments → commitments}/SCENARIO/SCENARIO.d.ts +1 -1
  60. package/esm/typings/src/{book-2.0/commitments → commitments}/STYLE/STYLE.d.ts +1 -1
  61. package/esm/typings/src/{book-2.0/commitments → commitments}/_base/BaseCommitmentDefinition.d.ts +1 -1
  62. package/esm/typings/src/{book-2.0/commitments → commitments}/_base/CommitmentDefinition.d.ts +1 -1
  63. package/esm/typings/src/{book-2.0/commitments → commitments}/_base/NotYetImplementedCommitmentDefinition.d.ts +1 -1
  64. package/esm/typings/src/{book-2.0/commitments → commitments}/_base/createEmptyAgentModelRequirements.d.ts +1 -1
  65. package/esm/typings/src/execution/LlmExecutionTools.d.ts +1 -1
  66. package/esm/typings/src/llm-providers/agent/Agent.d.ts +3 -7
  67. package/esm/typings/src/llm-providers/agent/AgentLlmExecutionTools.d.ts +1 -1
  68. package/esm/typings/src/llm-providers/agent/CreateAgentLlmExecutionToolsOptions.d.ts +1 -1
  69. package/esm/typings/src/llm-providers/agent/RemoteAgent.d.ts +32 -0
  70. package/esm/typings/src/llm-providers/agent/RemoteAgentOptions.d.ts +11 -0
  71. package/esm/typings/src/llm-providers/openai/OpenAiAssistantExecutionTools.d.ts +5 -1
  72. package/esm/typings/src/storage/_common/PromptbookStorage.d.ts +1 -0
  73. package/esm/typings/src/types/typeAliases.d.ts +6 -0
  74. package/esm/typings/src/utils/color/internal-utils/checkChannelValue.d.ts +0 -3
  75. package/esm/typings/src/utils/random/$generateBookBoilerplate.d.ts +2 -2
  76. package/esm/typings/src/utils/random/$randomFullnameWithColor.d.ts +1 -1
  77. package/esm/typings/src/version.d.ts +1 -1
  78. package/package.json +1 -1
  79. package/umd/index.umd.js +154 -161
  80. package/umd/index.umd.js.map +1 -1
  81. /package/esm/typings/src/{book-2.0/commitments → commitments}/_base/BookCommitment.d.ts +0 -0
  82. /package/esm/typings/src/{book-2.0/commitments → commitments}/_base/ParsedCommitment.d.ts +0 -0
  83. /package/esm/typings/src/{book-2.0/commitments → commitments}/index.d.ts +0 -0
@@ -0,0 +1,75 @@
1
+ import { nextRequestToNodeRequest } from '@/src/utils/cdn/utils/nextRequestToNodeRequest';
2
+ import { TODO_any } from '@promptbook-local/types';
3
+ import { serializeError } from '@promptbook-local/utils';
4
+ import formidable from 'formidable';
5
+ import { readFile } from 'fs/promises';
6
+ import { NextRequest, NextResponse } from 'next/server';
7
+ import { assertsError } from '../../../../../../src/errors/assertsError';
8
+ import { string_url } from '../../../../../../src/types/typeAliases';
9
+ import { keepUnused } from '../../../../../../src/utils/organization/keepUnused';
10
+ import { $provideCdnForServer } from '../../../../src/tools/$provideCdnForServer';
11
+ import { getUserFileCdnKey } from '../../../../src/utils/cdn/utils/getUserFileCdnKey';
12
+ import { validateMimeType } from '../../../../src/utils/validators/validateMimeType';
13
+
14
+ export async function POST(request: NextRequest) {
15
+ try {
16
+ const nodeRequest = await nextRequestToNodeRequest(request);
17
+
18
+ const files = await new Promise<formidable.Files>((resolve, reject) => {
19
+ const form = formidable({});
20
+ form.parse(nodeRequest as TODO_any, (error, fields, files) => {
21
+ keepUnused(fields);
22
+
23
+ if (error) {
24
+ return reject(error);
25
+ }
26
+ resolve(files);
27
+ });
28
+ });
29
+
30
+ const uploadedFiles = files.file;
31
+
32
+ if (!uploadedFiles || uploadedFiles.length !== 1) {
33
+ return NextResponse.json(
34
+ { message: 'In form data there is not EXACTLY one "file" field' },
35
+ { status: 400 },
36
+ );
37
+ }
38
+
39
+ const uploadedFile = uploadedFiles[0]!;
40
+ const fileBuffer = await readFile(uploadedFile.filepath);
41
+ const cdn = $provideCdnForServer();
42
+ const key = getUserFileCdnKey(fileBuffer, uploadedFile.originalFilename || uploadedFile.newFilename);
43
+
44
+ await cdn.setItem(key, {
45
+ type: validateMimeType(uploadedFile.mimetype),
46
+ data: fileBuffer,
47
+ });
48
+
49
+ const fileUrl = cdn.getItemUrl(key);
50
+
51
+ return NextResponse.json({ fileUrl: fileUrl.href as string_url }, { status: 201 });
52
+ } catch (error) {
53
+ assertsError(error);
54
+
55
+ console.error(error);
56
+
57
+ return new Response(
58
+ JSON.stringify(
59
+ serializeError(error),
60
+ // <- TODO: !!! Rename `serializeError` to `errorToJson`
61
+ null,
62
+ 4,
63
+ // <- TODO: !!! Allow to configure pretty print for agent server
64
+ ),
65
+ {
66
+ status: 400, // <- TODO: !!! Make `errorToHttpStatusCode`
67
+ headers: { 'Content-Type': 'application/json' },
68
+ },
69
+ );
70
+ }
71
+ }
72
+
73
+ /**
74
+ * TODO: !!!! Is this Working on Vercel
75
+ */
@@ -6,6 +6,7 @@ import { getSupabaseForServer } from '../supabase/getSupabaseForServer';
6
6
 
7
7
  /**
8
8
  * Cache of provided agent collection
9
+ *
9
10
  * @private internal cache for `$provideAgentCollectionForServer`
10
11
  */
11
12
  let agentCollection: null | AgentCollection = null;
@@ -0,0 +1,28 @@
1
+ import { DigitalOceanSpaces } from '../utils/cdn/classes/DigitalOceanSpaces';
2
+ import { IIFilesStorageWithCdn } from '../utils/cdn/interfaces/IFilesStorage';
3
+
4
+ /**
5
+ * Cache of CDN instance
6
+ *
7
+ * @private internal cache for `$provideCdnForServer`
8
+ */
9
+ let cdn: IIFilesStorageWithCdn | null = null;
10
+
11
+ /**
12
+ * !!!
13
+ */
14
+ export function $provideCdnForServer(): IIFilesStorageWithCdn {
15
+ if (!cdn) {
16
+ cdn = new DigitalOceanSpaces({
17
+ bucket: process.env.CDN_BUCKET!,
18
+ pathPrefix: process.env.NEXT_PUBLIC_CDN_PATH_PREFIX!,
19
+ endpoint: process.env.CDN_ENDPOINT!,
20
+ accessKeyId: process.env.CDN_ACCESS_KEY_ID!,
21
+ secretAccessKey: process.env.CDN_SECRET_ACCESS_KEY!,
22
+ cdnPublicUrl: new URL(process.env.NEXT_PUBLIC_CDN_PUBLIC_URL!),
23
+ gzip: true,
24
+ });
25
+ }
26
+
27
+ return cdn;
28
+ }
@@ -0,0 +1,119 @@
1
+ import { GetObjectCommand, PutObjectCommand, PutObjectCommandInput, S3Client } from '@aws-sdk/client-s3';
2
+ import { NotYetImplementedError } from '@promptbook-local/core';
3
+ import { gzip, ungzip } from 'node-gzip';
4
+ import { TODO_USE } from '../../../../../../src/utils/organization/TODO_USE';
5
+ import { validateMimeType } from '../../validators/validateMimeType';
6
+ import type { IFile, IIFilesStorageWithCdn } from '../interfaces/IFilesStorage';
7
+
8
+ type IDigitalOceanSpacesConfig = {
9
+ readonly bucket: string;
10
+ readonly pathPrefix: string;
11
+ readonly endpoint: string;
12
+ readonly accessKeyId: string;
13
+ readonly secretAccessKey: string;
14
+ readonly cdnPublicUrl: URL;
15
+ readonly gzip: boolean;
16
+
17
+ // TODO: [⛳️] Probbably prefix should be in this config not on the consumer side
18
+ };
19
+
20
+ export class DigitalOceanSpaces implements IIFilesStorageWithCdn {
21
+ public get cdnPublicUrl() {
22
+ return this.config.cdnPublicUrl;
23
+ }
24
+
25
+ private s3: S3Client;
26
+
27
+ public constructor(private readonly config: IDigitalOceanSpacesConfig) {
28
+ this.s3 = new S3Client({
29
+ region: 'auto',
30
+ endpoint: 'https://' + config.endpoint,
31
+ credentials: {
32
+ accessKeyId: config.accessKeyId,
33
+ secretAccessKey: config.secretAccessKey,
34
+ },
35
+ });
36
+ }
37
+
38
+ public getItemUrl(key: string): URL {
39
+ return new URL(this.config.pathPrefix + '/' + key, this.cdnPublicUrl);
40
+ }
41
+
42
+ public async getItem(key: string): Promise<IFile | null> {
43
+ const parameters = {
44
+ Bucket: this.config.bucket,
45
+ Key: this.config.pathPrefix + '/' + key,
46
+ };
47
+
48
+ try {
49
+ const { Body, ContentType, ContentEncoding } = await this.s3.send(new GetObjectCommand(parameters));
50
+
51
+ // const blob = new Blob([await Body?.transformToByteArray()!]);
52
+
53
+ if (ContentEncoding === 'gzip') {
54
+ return {
55
+ type: validateMimeType(ContentType),
56
+ data: await ungzip(await Body!.transformToByteArray()),
57
+ };
58
+ } else {
59
+ return {
60
+ type: validateMimeType(ContentType),
61
+ data: (await Body!.transformToByteArray()) as Buffer,
62
+ };
63
+ }
64
+ } catch (error) {
65
+ if (error instanceof Error && error.name.match(/^NoSuchKey/)) {
66
+ return null;
67
+ } else {
68
+ throw error;
69
+ }
70
+ }
71
+ }
72
+
73
+ public async removeItem(key: string): Promise<void> {
74
+ TODO_USE(key);
75
+ throw new NotYetImplementedError(`DigitalOceanSpaces.removeItem is not implemented yet`);
76
+ }
77
+
78
+ public async setItem(key: string, file: IFile): Promise<void> {
79
+ // TODO: Put putObjectRequestAdditional into processedFile
80
+ const putObjectRequestAdditional: Partial<PutObjectCommandInput> = {};
81
+
82
+ let processedFile: IFile;
83
+ if (this.config.gzip) {
84
+ const gzipped = await gzip(file.data);
85
+ const sizePercentageAfterCompression = gzipped.byteLength / file.data.byteLength;
86
+ if (sizePercentageAfterCompression < 0.7) {
87
+ // consolex.log(`Gzipping ${key} (${Math.floor(sizePercentageAfterCompression * 100)}%)`);
88
+ processedFile = { ...file, data: gzipped };
89
+ putObjectRequestAdditional.ContentEncoding = 'gzip';
90
+ } else {
91
+ processedFile = file;
92
+ // consolex.log(`NOT Gzipping ${key} (${Math.floor(sizePercentageAfterCompression * 100)}%)`);
93
+ }
94
+ } else {
95
+ processedFile = file;
96
+ }
97
+
98
+ const uploadResult = await this.s3.send(
99
+ new PutObjectCommand({
100
+ Bucket: this.config.bucket,
101
+ Key: this.config.pathPrefix + '/' + key,
102
+ ContentType: processedFile.type,
103
+ ...putObjectRequestAdditional,
104
+ Body: processedFile.data,
105
+ // TODO: Public read access / just private to extending class
106
+ ACL: 'public-read',
107
+ }),
108
+ );
109
+
110
+ if (!uploadResult.ETag) {
111
+ throw new Error(`Upload result does not contain ETag`);
112
+ }
113
+ }
114
+ }
115
+
116
+ /**
117
+ * TODO: Implement Read-only mode
118
+ * TODO: [☹️] Unite with `PromptbookStorage` and move to `/src/...`
119
+ */
@@ -0,0 +1,32 @@
1
+ import type { string_mime_type } from '../../../../../../src/types/typeAliases';
2
+ import type { IStorage } from './IStorage';
3
+
4
+ export type IFile = {
5
+ // Maybe TODO name: string_name;
6
+ type: string_mime_type;
7
+ data: Buffer;
8
+ };
9
+
10
+ /**
11
+ * Represents storage that will store each keypair in a separate file.
12
+ */
13
+ export type IFilesStorage = Omit<IStorage<IFile>, 'length' | 'clear' | 'key'>;
14
+
15
+ /**
16
+ * Represents storage that can give public deterministic URL for each file
17
+ */
18
+ export type IIFilesStorageWithCdn = IFilesStorage & {
19
+ readonly cdnPublicUrl: URL;
20
+ getItemUrl(key: string): URL;
21
+ };
22
+
23
+ /**
24
+ * TODO: Probably not deterministic and async getItemUrl
25
+ * TODO: Probably just createUrlMaker
26
+ * TODO: List method
27
+ * TODO: Glob method
28
+ * TODO: Subfolder (similar to PrefixStorage) method
29
+ * TODO: Subscribe, list, sub(folder) should be part of LIB everstorage
30
+ * TODO: Probably implement observe through RxJS
31
+ * TODO: [☹️] Unite with `PromptbookStorage` and move to `/src/...`
32
+ */
@@ -0,0 +1,14 @@
1
+ // Note: This is a simplified version of the IStorage interface based on the usage in the project.
2
+ export type IStorage<T> = {
3
+ readonly length: Promise<number>;
4
+ clear(): Promise<void>;
5
+ getItem(key: string): Promise<T | null>;
6
+ key(index: number): Promise<string | null>;
7
+ removeItem(key: string): Promise<void>;
8
+ setItem(key: string, value: T): Promise<void>;
9
+ };
10
+
11
+
12
+ /**
13
+ * TODO: [☹️] Unite with `PromptbookStorage` and move to `/src/...`
14
+ */
@@ -0,0 +1,27 @@
1
+ import { titleToName } from '../../../../../../src/utils/normalization/titleToName';
2
+ import hexEncoder from 'crypto-js/enc-hex';
3
+ import sha256 from 'crypto-js/sha256';
4
+ import type { string_uri } from '../../../../../../src/types/typeAliases';
5
+ import { nameToSubfolderPath } from './nameToSubfolderPath';
6
+
7
+ /**
8
+ * Generates a path for the user content
9
+ */
10
+ export function getUserFileCdnKey(file: Buffer, originalFilename: string): string_uri {
11
+ const hash = sha256(hexEncoder.parse(file.toString('hex'))).toString(/* hex */);
12
+
13
+ const originalFilenameParts = originalFilename.split('.');
14
+ const extension = originalFilenameParts.pop();
15
+ const name = titleToName(originalFilenameParts.join('.'));
16
+
17
+ const filename = name + '.' + extension;
18
+ // <- Note: [⛳️] Preserving original file name
19
+
20
+ return `user/files/${nameToSubfolderPath(hash).join('/')}/${filename}`;
21
+ }
22
+
23
+ /**
24
+ * TODO: [🌍] Unite this logic in one place
25
+ * TODO: Way to garbage unused uploaded files
26
+ * TODO: Probably separate util countBufferHash
27
+ */
@@ -0,0 +1,9 @@
1
+ import type { string_name } from '../../../../../../src/types/typeAliases';
2
+
3
+ export function nameToSubfolderPath(name: string_name): Array<string> {
4
+ return [name.substr(0, 2).toLowerCase(), name.substr(2, 5).toLowerCase()];
5
+ }
6
+
7
+ /**
8
+ * TODO: !!! Use `nameToSubfolderPath` from src
9
+ */
@@ -0,0 +1,27 @@
1
+ import { TODO_any } from '@promptbook-local/types';
2
+ import { NextRequest } from 'next/server';
3
+ import { Readable } from 'node:stream';
4
+
5
+ export async function nextRequestToNodeRequest(nextRequest: NextRequest): Promise<Readable> {
6
+ const reader = nextRequest.body?.getReader();
7
+
8
+ if (!reader) {
9
+ throw new Error(`Can not get nextRequest.body.getReader()`);
10
+ }
11
+
12
+ const nodeStream = new Readable({
13
+ async read() {
14
+ const { done, value } = await reader.read();
15
+ if (done) this.push(null);
16
+ else this.push(Buffer.from(value));
17
+ },
18
+ });
19
+
20
+ // Fake IncomingMessage with headers
21
+ (nodeStream as TODO_any).headers = Object.fromEntries(nextRequest.headers.entries());
22
+ (nodeStream as TODO_any).method = nextRequest.method;
23
+ (nodeStream as TODO_any).url = nextRequest.url;
24
+ (nodeStream as TODO_any).socket = {}; // required by formidable
25
+
26
+ return nodeStream;
27
+ }
@@ -0,0 +1,24 @@
1
+ import type { string_mime_type } from '../../../../../src/types/typeAliases';
2
+
3
+ /**
4
+ * Checks if the value is valid mime-type
5
+ *
6
+ * @param value candidate for mime-type
7
+ * @returns the value if it is valid mime-type
8
+ * @throws TypeError if the value is not valid mime-type
9
+ */
10
+ export function validateMimeType(value: unknown): string_mime_type {
11
+ if (typeof value !== 'string') {
12
+ throw new TypeError(`Mime-type must be string, but it is ${typeof value}`);
13
+ }
14
+
15
+ if (!/^[a-z]+\/(?:[a-z0-9]+[.-])*[a-z0-9]+$/i.test(value)) {
16
+ throw new TypeError(`Invalid mime-type "${value}"`);
17
+ }
18
+
19
+ return value as string_mime_type;
20
+ }
21
+
22
+ /**
23
+ * TODO: [🧠] Move to main Promptbook utils
24
+ */
@@ -24,6 +24,6 @@
24
24
  "@promptbook-local/*": ["../../src/_packages/*.index"]
25
25
  }
26
26
  },
27
- "include": ["next-env.d.ts", "**/*.ts", "**/*.tsx", ".next/types/**/*.ts"],
27
+ "include": ["next-env.d.ts", "**/*.ts", "**/*.tsx", ".next/types/**/*.ts", "../_common/hooks/usePromise.ts"],
28
28
  "exclude": ["node_modules"]
29
29
  }