@lumi.new/sdk 0.2.1 → 0.3.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -62,6 +62,8 @@ When using the SDK in a server-side environment (like Node.js), please note the
62
62
  - `lumi.auth.onAuthChange()`
63
63
  - **Fetching User Data**: On the server, `lumi.auth.user` is `null` by default. You must explicitly call `await lumi.auth.refreshUser()` to fetch the user's profile based on the token provided during initialization.
64
64
 
65
+ - **AI Tools**: The `lumi.tools.ai` methods are server-side only and require a `LUMI_API_KEY` environment variable to be set.
66
+
65
67
  ### Sign In
66
68
 
67
69
  To sign in a user, call the `signIn` method. This will open a popup window for the user to authenticate. Upon successful authentication, the access token and user information will be automatically stored.
@@ -298,9 +300,11 @@ deleteMultiplePosts(['post-id-1', 'post-id-2']);
298
300
 
299
301
  ## Tools
300
302
 
301
- The `tools` client provides access to various utilities, such as sending emails.
303
+ The `tools` client provides access to various utilities, such as sending emails, managing files, and AI text/image generation.
304
+
305
+ ### Email
302
306
 
303
- ### Send an Email
307
+ #### Send an Email
304
308
 
305
309
  You can send an email using the `email.send` method.
306
310
 
@@ -323,7 +327,9 @@ async function sendWelcomeEmail(email: string) {
323
327
  sendWelcomeEmail('test@example.com');
324
328
  ```
325
329
 
326
- ### Upload Files
330
+ ### File
331
+
332
+ #### Upload Files
327
333
 
328
334
  Upload multiple files in a single request. This method returns an array of objects, each representing the result of one file upload.
329
335
 
@@ -346,7 +352,7 @@ async function uploadFiles(files: File[]) {
346
352
  // ]
347
353
  ```
348
354
 
349
- ### Delete Files
355
+ #### Delete Files
350
356
 
351
357
  Delete multiple files by their URLs in a single request.
352
358
 
@@ -364,6 +370,75 @@ const filesToDelete = [
364
370
  deleteFiles(filesToDelete);
365
371
  ```
366
372
 
373
+ ### AI
374
+
375
+ > Note:
376
+ > - AI methods are server-side only.
377
+ > - You must provide `LUMI_API_KEY` via environment variables.
378
+
379
+ #### AI: Generate Text
380
+
381
+ ```typescript
382
+ async function aiGenerateText() {
383
+ const result = await lumi.tools.ai.generateText({
384
+ // optional; defaults to 'gemini-2.5-flash'
385
+ model: 'gemini-2.5-flash',
386
+ messages: [
387
+ { role: 'system', content: 'You are a helpful assistant.' },
388
+ { role: 'user', content: 'Write a haiku about the sea.' },
389
+ ],
390
+ })
391
+ console.log(result.chatId, result.content)
392
+ }
393
+ ```
394
+
395
+ #### AI: Generate Text (Streaming)
396
+
397
+ ```typescript
398
+ async function aiGenerateTextStream() {
399
+ const stream = await lumi.tools.ai.generateTextStream({
400
+ // optional; defaults to 'gemini-2.5-flash'
401
+ model: 'gemini-2.5-flash',
402
+ messages: [
403
+ { role: 'system', content: 'You are a helpful assistant.' },
404
+ { role: 'user', content: 'Stream a short poem line by line.' },
405
+ ],
406
+ })
407
+
408
+ const reader = stream.getReader()
409
+ try {
410
+ while (true) {
411
+ const { value, done } = await reader.read()
412
+ if (done) break
413
+ // value: { event?: string, data: { chatId, content } }
414
+ process.stdout.write(value?.data?.content ?? '')
415
+ }
416
+ } finally {
417
+ reader.releaseLock()
418
+ }
419
+ }
420
+ ```
421
+
422
+ #### AI: Generate Image
423
+
424
+ ```typescript
425
+ async function aiGenerateImage() {
426
+ const result = await lumi.tools.ai.generateImage({
427
+ // optional; defaults to 'gemini-2.5-flash-image'
428
+ model: 'gemini-2.5-flash-image',
429
+ messages: [
430
+ { role: 'user', content: 'A watercolor painting of a serene mountain lake at sunrise.' },
431
+ ],
432
+ })
433
+
434
+ if (result.generationStatus === 'SUCCESS') {
435
+ console.log('Image URL:', result.imageURL)
436
+ } else {
437
+ console.log('Image generation failed:', result)
438
+ }
439
+ }
440
+ ```
441
+
367
442
  ## Functions
368
443
 
369
444
  The `functions` client allows you to invoke serverless functions deployed in your Lumi project.
@@ -651,6 +726,82 @@ Deletes one or more files based on their URLs.
651
726
 
652
727
  - `Promise<void>`: A promise that resolves when the delete operation is complete.
653
728
 
729
+ #### `ai.generateText(options)`
730
+
731
+ Generates text with the specified model and messages.
732
+
733
+ **Parameters:**
734
+
735
+ - `options` (`object`):
736
+ - `model` (`string`, optional): Model name. Defaults to `'gemini-2.5-flash'`.
737
+ - `messages` (`Message[]`): The chat history/messages.
738
+
739
+ **Returns:**
740
+
741
+ - `Promise<GenerateTextResult>`: See “Types used by AI”.
742
+
743
+ **Environment:**
744
+
745
+ - Server-side only. Requires `LUMI_API_KEY` in environment variables.
746
+
747
+ #### `ai.generateTextStream(options)`
748
+
749
+ Generates text as a stream of chunks.
750
+
751
+ **Parameters:**
752
+
753
+ - `options` (`object`):
754
+ - `model` (`string`, optional): Model name. Defaults to `'gemini-2.5-flash'`.
755
+ - `messages` (`Message[]`): The chat history/messages.
756
+
757
+ **Returns:**
758
+
759
+ - `Promise<ReadableStream<{ event?: string, data: GenerateTextResult }>>`: A readable stream producing event/data chunks.
760
+
761
+ **Environment:**
762
+
763
+ - Server-side only. Requires `LUMI_API_KEY` in environment variables.
764
+
765
+ #### `ai.generateImage(options)`
766
+
767
+ Generates an image with the specified model and messages. The call completes after the task finishes (internally polled).
768
+
769
+ **Parameters:**
770
+
771
+ - `options` (`object`):
772
+ - `model` (`string`, optional): Model name. Defaults to `'gemini-2.5-flash-image'`.
773
+ - `messages` (`Message[]`): The prompt/messages for image generation.
774
+
775
+ **Returns:**
776
+
777
+ - `Promise<GenerateImageResult>`: See “Types used by AI”. This method resolves only when the result is `'SUCCESS'` or `'FAILED'`.
778
+
779
+ **Environment:**
780
+
781
+ - Server-side only. Requires `LUMI_API_KEY` in environment variables.
782
+
783
+ #### Types used by AI
784
+
785
+ - `MessageMedia`:
786
+ - `mimeType` (`string`)
787
+ - `url` (`string`)
788
+
789
+ - `Message`:
790
+ - `role` (`'system' | 'user' | 'assistant'`)
791
+ - `content` (`string`)
792
+ - `medias?` (`MessageMedia[]`)
793
+
794
+ - `GenerateTextResult`:
795
+ - `chatId` (`string`)
796
+ - `content` (`string`)
797
+
798
+ - `GenerateImageResult`:
799
+ - `chatId` (`string`)
800
+ - `messageId` (`string`)
801
+ - `generationStatus` (`'SUCCESS' | 'FAILED'`)
802
+ - `content?` (`string`)
803
+ - `imageURL?` (`string`)
804
+
654
805
  ### Functions (`lumi.functions`)
655
806
 
656
807
  Provides a method for invoking serverless functions.
package/dist/index.d.mts CHANGED
@@ -4,6 +4,10 @@ interface PaginationData<T> {
4
4
  total: number;
5
5
  list: T[];
6
6
  }
7
+ interface StreamChunk<T> {
8
+ event?: string;
9
+ data: T;
10
+ }
7
11
 
8
12
  interface Entity extends Record<string, any> {
9
13
  id: string;
@@ -46,6 +50,47 @@ declare class FunctionsClient {
46
50
  invoke(functionName: string, options?: FetchOptions<'json'>): Promise<any>;
47
51
  }
48
52
 
53
+ interface MessageMedia {
54
+ mimeType: string;
55
+ url: string;
56
+ }
57
+ interface Message {
58
+ role: 'system' | 'user' | 'assistant';
59
+ content: string;
60
+ medias?: MessageMedia[];
61
+ }
62
+ interface GenerateTextResult {
63
+ chatId: string;
64
+ content: string;
65
+ }
66
+ interface GenerateImageResult {
67
+ chatId: string;
68
+ messageId: string;
69
+ generationStatus: 'PROCESSING' | 'SUCCESS' | 'FAILED';
70
+ content?: string;
71
+ imageURL?: string;
72
+ }
73
+ declare class AITool {
74
+ #private;
75
+ constructor(lumi: LumiClient);
76
+ /** AI 生成文本 */
77
+ generateText({ model, messages, }: {
78
+ model?: string;
79
+ messages: Message[];
80
+ }): Promise<GenerateTextResult>;
81
+ /** AI 生成文本(流式输出) */
82
+ generateTextStream({ model, messages, }: {
83
+ model?: string;
84
+ messages: Message[];
85
+ }): Promise<ReadableStream<StreamChunk<GenerateTextResult>>>;
86
+ /** AI 生成图片 */
87
+ generateImage({ model, messages, }: {
88
+ model?: string;
89
+ messages: Message[];
90
+ }): Promise<GenerateImageResult>;
91
+ private checkLumiApiKey;
92
+ }
93
+
49
94
  declare class EmailTool {
50
95
  #private;
51
96
  constructor(lumi: LumiClient);
@@ -79,6 +124,7 @@ declare class ToolsClient {
79
124
  #private;
80
125
  email: EmailTool;
81
126
  file: FileTool;
127
+ ai: AITool;
82
128
  constructor(lumi: LumiClient);
83
129
  }
84
130
 
@@ -154,4 +200,4 @@ declare class LumiAuthClient {
154
200
  }) => void): () => void;
155
201
  }
156
202
 
157
- export { EmailTool, EntitiesClient, type Entity, EntityClient, FileTool, FunctionsClient, LumiAuthClient, LumiClient, type LumiClientConfig, type MessageDataReceive, type MessageDataSend, type MessageInitData, type MessageSignInData, ToolsClient, type UploadItem, type User, createClient };
203
+ export { AITool, EmailTool, EntitiesClient, type Entity, EntityClient, FileTool, FunctionsClient, type GenerateImageResult, type GenerateTextResult, LumiAuthClient, LumiClient, type LumiClientConfig, type Message, type MessageDataReceive, type MessageDataSend, type MessageInitData, type MessageMedia, type MessageSignInData, ToolsClient, type UploadItem, type User, createClient };
package/dist/index.d.ts CHANGED
@@ -4,6 +4,10 @@ interface PaginationData<T> {
4
4
  total: number;
5
5
  list: T[];
6
6
  }
7
+ interface StreamChunk<T> {
8
+ event?: string;
9
+ data: T;
10
+ }
7
11
 
8
12
  interface Entity extends Record<string, any> {
9
13
  id: string;
@@ -46,6 +50,47 @@ declare class FunctionsClient {
46
50
  invoke(functionName: string, options?: FetchOptions<'json'>): Promise<any>;
47
51
  }
48
52
 
53
+ interface MessageMedia {
54
+ mimeType: string;
55
+ url: string;
56
+ }
57
+ interface Message {
58
+ role: 'system' | 'user' | 'assistant';
59
+ content: string;
60
+ medias?: MessageMedia[];
61
+ }
62
+ interface GenerateTextResult {
63
+ chatId: string;
64
+ content: string;
65
+ }
66
+ interface GenerateImageResult {
67
+ chatId: string;
68
+ messageId: string;
69
+ generationStatus: 'PROCESSING' | 'SUCCESS' | 'FAILED';
70
+ content?: string;
71
+ imageURL?: string;
72
+ }
73
+ declare class AITool {
74
+ #private;
75
+ constructor(lumi: LumiClient);
76
+ /** AI 生成文本 */
77
+ generateText({ model, messages, }: {
78
+ model?: string;
79
+ messages: Message[];
80
+ }): Promise<GenerateTextResult>;
81
+ /** AI 生成文本(流式输出) */
82
+ generateTextStream({ model, messages, }: {
83
+ model?: string;
84
+ messages: Message[];
85
+ }): Promise<ReadableStream<StreamChunk<GenerateTextResult>>>;
86
+ /** AI 生成图片 */
87
+ generateImage({ model, messages, }: {
88
+ model?: string;
89
+ messages: Message[];
90
+ }): Promise<GenerateImageResult>;
91
+ private checkLumiApiKey;
92
+ }
93
+
49
94
  declare class EmailTool {
50
95
  #private;
51
96
  constructor(lumi: LumiClient);
@@ -79,6 +124,7 @@ declare class ToolsClient {
79
124
  #private;
80
125
  email: EmailTool;
81
126
  file: FileTool;
127
+ ai: AITool;
82
128
  constructor(lumi: LumiClient);
83
129
  }
84
130
 
@@ -154,4 +200,4 @@ declare class LumiAuthClient {
154
200
  }) => void): () => void;
155
201
  }
156
202
 
157
- export { EmailTool, EntitiesClient, type Entity, EntityClient, FileTool, FunctionsClient, LumiAuthClient, LumiClient, type LumiClientConfig, type MessageDataReceive, type MessageDataSend, type MessageInitData, type MessageSignInData, ToolsClient, type UploadItem, type User, createClient };
203
+ export { AITool, EmailTool, EntitiesClient, type Entity, EntityClient, FileTool, FunctionsClient, type GenerateImageResult, type GenerateTextResult, LumiAuthClient, LumiClient, type LumiClientConfig, type Message, type MessageDataReceive, type MessageDataSend, type MessageInitData, type MessageMedia, type MessageSignInData, ToolsClient, type UploadItem, type User, createClient };
package/dist/index.js CHANGED
@@ -1,4 +1,4 @@
1
- "use strict";var ue=Object.create;var L=Object.defineProperty,me=Object.defineProperties,pe=Object.getOwnPropertyDescriptor,he=Object.getOwnPropertyDescriptors,de=Object.getOwnPropertyNames,K=Object.getOwnPropertySymbols,ge=Object.getPrototypeOf,J=Object.prototype.hasOwnProperty,fe=Object.prototype.propertyIsEnumerable;var G=i=>{throw TypeError(i)};var Q=(i,e,t)=>e in i?L(i,e,{enumerable:!0,configurable:!0,writable:!0,value:t}):i[e]=t,S=(i,e)=>{for(var t in e||(e={}))J.call(e,t)&&Q(i,t,e[t]);if(K)for(var t of K(e))fe.call(e,t)&&Q(i,t,e[t]);return i},H=(i,e)=>me(i,he(e));var ye=(i,e)=>{for(var t in e)L(i,t,{get:e[t],enumerable:!0})},V=(i,e,t,r)=>{if(e&&typeof e=="object"||typeof e=="function")for(let s of de(e))!J.call(i,s)&&s!==t&&L(i,s,{get:()=>e[s],enumerable:!(r=pe(e,s))||r.enumerable});return i};var x=(i,e,t)=>(t=i!=null?ue(ge(i)):{},V(e||!i||!i.__esModule?L(t,"default",{value:i,enumerable:!0}):t,i)),we=i=>V(L({},"__esModule",{value:!0}),i);var Y=(i,e,t)=>e.has(i)||G("Cannot "+t);var n=(i,e,t)=>(Y(i,e,"read from private field"),t?t.call(i):e.get(i)),a=(i,e,t)=>e.has(i)?G("Cannot add the same private member more than once"):e instanceof WeakSet?e.add(i):e.set(i,t),u=(i,e,t,r)=>(Y(i,e,"write to private field"),r?r.call(i,t):e.set(i,t),t);var m=(i,e,t)=>new Promise((r,s)=>{var o=h=>{try{f(t.next(h))}catch(y){s(y)}},l=h=>{try{f(t.throw(h))}catch(y){s(y)}},f=h=>h.done?r(h.value):Promise.resolve(h.value).then(o,l);f((t=t.apply(i,e)).next())});var Te={};ye(Te,{EmailTool:()=>U,EntitiesClient:()=>O,EntityClient:()=>P,FileTool:()=>k,FunctionsClient:()=>M,LumiAuthClient:()=>R,LumiClient:()=>B,ToolsClient:()=>$,createClient:()=>Ce});module.exports=we(Te);var ae=require("uuid");var te=x(require("crypto-js/enc-base64")),ie=x(require("crypto-js/enc-hex")),re=x(require("crypto-js/hmac-sha256")),ne=x(require("crypto-js/sha256")),oe=x(require("object-hash")),se=require("ofetch");var b=class extends Error{constructor(t,r){super(r);this.name="LumiError";this.code=t}};function Z(){var i,e;return(e=(i=document.querySelector('link[rel="icon"]'))==null?void 0:i.href)!=null?e:null}function X(){var i;return(i=document.title)!=null?i:null}var w=typeof window=="undefined";var Ee="6QrJZ7pFCmBZAeIJF7IArvkCz+EtzA0RVcpHkiQIsQyhs7QtCS9P+CueZdHfB2OtJcgX3BbqY9pfpWeAVTqCwQ==";function W(i){return encodeURIComponent(i).replace(/[!'()*]/g,e=>`%${e.charCodeAt(0).toString(16).toUpperCase()}`)}function be(i){return e=>{let{options:t}=e,r=Math.floor(Date.now()/1e3).toString(),s=Math.random().toString(36).substring(2,15),o=S({},t.query),l=Object.keys(o).sort().map(E=>`${W(E)}=${W(String(o[E]))}`).join("&"),f={"x-timestamp":r,"x-nonce":s},h=Object.keys(f).sort().map(E=>`${E}:${f[E]}`).join(`
2
- `),y=t.body&&!(t.body instanceof FormData)?JSON.stringify(t.body):"",q=(0,ne.default)(y).toString(ie.default),j=[l,h,q].join(`
3
- `),d=te.default.stringify((0,re.default)(j,i)),I=new Headers(t.headers);Object.entries(f).forEach(([E,le])=>{I.set(E,le)}),I.set("X-Sign",d),t.headers=I}}var ee=new Map;function Se(i,e){var o;e.body instanceof FormData&&(e=H(S({},e),{body:Array.from(e.body.entries())}));let t=(0,oe.default)([i,e]),r=Date.now(),s=((o=ee.get(t))==null?void 0:o.filter(l=>r-l<1e3))||[];if(s.length>=4)throw new b(429,"Too many requests");s.push(r),ee.set(t,s)}function c(i,e,t={}){i.auth.accessToken&&(t.headers=S({Authorization:`Bearer ${i.auth.accessToken}`},t.headers)),Se(e,t);let r=i.auth.isAuthenticated;return(0,se.ofetch)(e,H(S({baseURL:i.config.apiBaseUrl},t),{onRequest:be(Ee),onResponse:({response:s})=>{var o;!w&&r&&((o=s._data)==null?void 0:o.code)===2100&&i.auth.signOut()}}))}function _(i,e,t=localStorage){let r=t.getItem(i),s=e?JSON.stringify(e):null;s?t.setItem(i,s):t.removeItem(i),window.dispatchEvent(new StorageEvent("storage",{key:i,oldValue:r,newValue:s,storageArea:t}))}function z(i,e=localStorage){let t=e.getItem(i);try{return t?JSON.parse(t):null}catch(r){return null}}var g,N,v,R=class{constructor(e){a(this,g);a(this,N,`lumi-auth-${(0,ae.v4)()}`);a(this,v,null);u(this,g,e),Promise.resolve().then(()=>{!w&&this.isAuthenticated&&this.refreshUser()})}get accessToken(){if(w){let e=n(this,g).config.authorization;return e?e.replace("Bearer ",""):null}return z("lumi-access-token")}set accessToken(e){if(w){n(this,g).config.authorization=e?`Bearer ${e}`:void 0;return}_("lumi-access-token",e)}get user(){return w?n(this,v):z("lumi-user")}set user(e){if(w){u(this,v,e);return}_("lumi-user",e)}get isAuthenticated(){return!!this.accessToken}signIn(){if(w)throw new Error("auth.signIn() can only be called on the client side");let e=800,t=600,r=(window.screen.width-e)/2,s=(window.screen.height-t)/2,o=window.open(n(this,g).config.authOrigin,n(this,N),`width=${e},height=${t},left=${r},top=${s}`),l;return new Promise((f,h)=>{if(!o)return h(new Error("Open auth window failed"));let y=setInterval(()=>{o.closed&&h(new Error("Auth window closed"))},1e3),q=d=>{o.closed||(o.focus(),d.stopPropagation(),d.preventDefault())},j=({data:d,origin:I,source:E})=>{if(!(I!==n(this,g).config.authOrigin||E!==o))switch(d==null?void 0:d.type){case"lumi-ready":{o.postMessage({type:"lumi-init",data:{projectId:n(this,g).config.projectId,icon:Z(),title:X()}},n(this,g).config.authOrigin);break}case"lumi-sign-in":{if(d.data.projectId!==n(this,g).config.projectId)break;o.close(),window.focus(),this.accessToken=d.data.accessToken,this.user=d.data.user,f(d.data);break}}};window.addEventListener("message",j),document.addEventListener("click",q,!0),l=()=>{clearInterval(y),window.removeEventListener("message",j),document.removeEventListener("click",q,!0)}}).finally(()=>l==null?void 0:l())}signOut(){if(w)throw new Error("auth.signOut() can only be called on the client side");this.accessToken=null,this.user=null}refreshUser(){return m(this,null,function*(){let e=yield c(n(this,g),"/lm/user/info",{method:"POST"});if(e.code!==200)throw new Error(e.message);return this.user=e.data,e.data})}onAuthChange(e){if(w)throw new Error("auth.onAuthChange() can only be called on the client side");let t=r=>{(r.key==="lumi-access-token"||r.key==="lumi-user"||r.key===null)&&e({isAuthenticated:this.isAuthenticated,user:this.user})};return window.addEventListener("storage",t),()=>{window.removeEventListener("storage",t)}}};g=new WeakMap,N=new WeakMap,v=new WeakMap;var p,P=class{constructor(e,t){a(this,p);u(this,p,e),this.entityName=t}list(){return m(this,arguments,function*({filter:e,sort:t,limit:r,skip:s}={}){if(r){let o=yield c(n(this,p),this.uri("/find"),{method:"POST",body:{filter:e,sort:t,limit:r,skip:s}});if(o.code!==200)throw new Error(o.message);return o.data}else{let o=yield c(n(this,p),this.uri("/list"),{method:"POST",body:{filter:e,sort:t}});if(o.code!==200)throw new Error(o.message);return{total:o.data.length,list:o.data}}})}get(e){return m(this,null,function*(){let t=yield c(n(this,p),this.uri(`/${e}`),{method:"GET"});if(t.code!==200)throw new Error(t.message);return t.data})}create(e){return m(this,null,function*(){let t=yield c(n(this,p),this.uri(),{method:"POST",body:e});if(t.code!==200)throw new Error(t.message);return t.data})}createMany(e){return m(this,null,function*(){let t=yield c(n(this,p),this.uri("/batch"),{method:"POST",body:e});if(t.code!==200)throw new Error(t.message);return t.data})}update(e,t){return m(this,null,function*(){let r=yield c(n(this,p),this.uri(),{method:"PUT",body:{filter:{_id:e},update:t}});if(r.code!==200)throw new Error(r.message);return r.data})}delete(e){return m(this,null,function*(){let t=yield c(n(this,p),this.uri(`/${e}`),{method:"DELETE"});if(t.code!==200)throw new Error(t.message)})}deleteMany(e){return m(this,null,function*(){let t=yield c(n(this,p),this.uri("/batch-by-ids"),{method:"DELETE",params:{ids:e}});if(t.code!==200)throw new Error(t.message)})}uri(e=""){return`/lm/${n(this,p).config.projectId}/${this.entityName}/documents${e}`}};p=new WeakMap;var D,O=class{constructor(e){a(this,D);return u(this,D,e),new Proxy(this,{get(t,r){return r in t||(t[r]=new P(n(t,D),r)),t[r]}})}};D=new WeakMap;var ce=require("ofetch");var C,M=class{constructor(e){a(this,C);u(this,C,e)}invoke(e,t={}){return n(this,C).auth.accessToken&&(t.headers=S({Authorization:`Bearer ${n(this,C).auth.accessToken}`},t.headers)),(0,ce.ofetch)(`/v1/functions/${n(this,C).config.projectId}/${e}`,S({baseURL:n(this,C).config.apiBaseUrl},t))}};C=new WeakMap;var A,U=class{constructor(e){a(this,A);u(this,A,e)}send(h){return m(this,arguments,function*({to:e,subject:t,fromName:r,html:s,text:o="",replyTo:l,scheduledAt:f}){if(!e||!t||!s&&!o)throw new Error("Failed to send email: Missing required parameters.");typeof e=="string"&&(e=[e]),typeof l=="string"&&(l=[l]);let y=yield c(n(this,A),`/lm/${n(this,A).config.projectId}/email/send`,{method:"POST",body:{to:e,subject:t,fromName:r,html:s,text:o,replyTo:l,scheduledAt:f}});if(y.code!==200)throw new b(y.code,y.message)})}};A=new WeakMap;var T,k=class{constructor(e){a(this,T);u(this,T,e)}upload(e){return m(this,null,function*(){let t=new FormData;e.forEach(s=>{t.append("files",s)});let r=yield c(n(this,T),`/lm/${n(this,T).config.projectId}/file/batch`,{method:"POST",body:t});if(r.code!==200)throw new b(r.code,r.message);return r.data})}delete(e){return m(this,null,function*(){let t=yield c(n(this,T),`/lm/${n(this,T).config.projectId}/file/batch`,{method:"DELETE",body:{fileUrls:e}});if(t.code!==200)throw new b(t.code,t.message)})}};T=new WeakMap;var F,$=class{constructor(e){a(this,F);u(this,F,e),this.email=new U(e),this.file=new k(e)}};F=new WeakMap;var B=class{constructor(e){this.config=e,this.auth=new R(this),this.entities=new O(this),this.tools=new $(this),this.functions=new M(this)}};function Ce(i){return new B(i)}0&&(module.exports={EmailTool,EntitiesClient,EntityClient,FileTool,FunctionsClient,LumiAuthClient,LumiClient,ToolsClient,createClient});
1
+ "use strict";var fe=Object.create;var x=Object.defineProperty,ye=Object.defineProperties,we=Object.getOwnPropertyDescriptor,be=Object.getOwnPropertyDescriptors,Se=Object.getOwnPropertyNames,J=Object.getOwnPropertySymbols,Ee=Object.getPrototypeOf,Y=Object.prototype.hasOwnProperty,Te=Object.prototype.propertyIsEnumerable;var V=i=>{throw TypeError(i)};var Q=(i,e,t)=>e in i?x(i,e,{enumerable:!0,configurable:!0,writable:!0,value:t}):i[e]=t,T=(i,e)=>{for(var t in e||(e={}))Y.call(e,t)&&Q(i,t,e[t]);if(J)for(var t of J(e))Te.call(e,t)&&Q(i,t,e[t]);return i},B=(i,e)=>ye(i,be(e));var Ce=(i,e)=>{for(var t in e)x(i,t,{get:e[t],enumerable:!0})},X=(i,e,t,r)=>{if(e&&typeof e=="object"||typeof e=="function")for(let o of Se(e))!Y.call(i,o)&&o!==t&&x(i,o,{get:()=>e[o],enumerable:!(r=we(e,o))||r.enumerable});return i};var L=(i,e,t)=>(t=i!=null?fe(Ee(i)):{},X(e||!i||!i.__esModule?x(t,"default",{value:i,enumerable:!0}):t,i)),Ae=i=>X(x({},"__esModule",{value:!0}),i);var Z=(i,e,t)=>e.has(i)||V("Cannot "+t);var s=(i,e,t)=>(Z(i,e,"read from private field"),t?t.call(i):e.get(i)),p=(i,e,t)=>e.has(i)?V("Cannot add the same private member more than once"):e instanceof WeakSet?e.add(i):e.set(i,t),d=(i,e,t,r)=>(Z(i,e,"write to private field"),r?r.call(i,t):e.set(i,t),t);var c=(i,e,t)=>new Promise((r,o)=>{var n=m=>{try{h(t.next(m))}catch(g){o(g)}},a=m=>{try{h(t.throw(m))}catch(g){o(g)}},h=m=>m.done?r(m.value):Promise.resolve(m.value).then(n,a);h((t=t.apply(i,e)).next())});var ve={};Ce(ve,{AITool:()=>D,EmailTool:()=>q,EntitiesClient:()=>O,EntityClient:()=>M,FileTool:()=>$,FunctionsClient:()=>k,LumiAuthClient:()=>v,LumiClient:()=>K,ToolsClient:()=>N,createClient:()=>Le});module.exports=Ae(ve);var de=require("uuid");var ne=L(require("crypto-js/enc-base64")),oe=L(require("crypto-js/enc-hex")),ae=L(require("crypto-js/hmac-sha256")),ce=L(require("crypto-js/sha256")),me=require("eventsource-parser/stream"),le=L(require("object-hash")),_=require("ofetch");var f=class extends Error{constructor(t,r){super(r);this.name="LumiError";this.code=t}};function W(){var i,e;return(e=(i=document.querySelector('link[rel="icon"]'))==null?void 0:i.href)!=null?e:null}function ee(){var i;return(i=document.title)!=null?i:null}var y=typeof window=="undefined";function te(i){return new Promise(e=>setTimeout(e,i))}function ie(i){var e;return typeof process!="undefined"&&(e=process==null?void 0:process.env[i])!=null?e:null}var Ie="6QrJZ7pFCmBZAeIJF7IArvkCz+EtzA0RVcpHkiQIsQyhs7QtCS9P+CueZdHfB2OtJcgX3BbqY9pfpWeAVTqCwQ==";function re(i){return encodeURIComponent(i).replace(/[!'()*]/g,e=>`%${e.charCodeAt(0).toString(16).toUpperCase()}`)}function Re(i){let e=Math.floor(Date.now()/1e3).toString(),t=Math.random().toString(36).substring(2,15),r=T({},i.query),o=Object.keys(r).sort().map(l=>`${re(l)}=${re(String(r[l]))}`).join("&"),n={"x-timestamp":e,"x-nonce":t},a=Object.keys(n).sort().map(l=>`${l}:${n[l]}`).join(`
2
+ `),h=i.body&&!(i.body instanceof FormData)?JSON.stringify(i.body):"",m=(0,ce.default)(h).toString(oe.default),g=[o,a,m].join(`
3
+ `),I=ne.default.stringify((0,ae.default)(g,Ie)),E=new Headers(i.headers);Object.entries(n).forEach(([l,G])=>{E.set(l,G)}),E.set("X-Sign",I),i.headers=E}var se=new Map;function xe(i,e){var a;let t=T({},e);e.body instanceof FormData&&(t.body=Array.from(e.body.entries())),t.headers=void 0;let r=(0,le.default)([i,t]),o=Date.now(),n=((a=se.get(r))==null?void 0:a.filter(h=>o-h<1e3))||[];if(n.length>=4)throw new f(429,"Too many requests");n.push(o),se.set(r,n)}function ue(i,e,t,r=!1){t.headers=new Headers(t.headers),i.auth.accessToken&&t.headers.set("Authorization",`Bearer ${i.auth.accessToken}`),r&&(t.headers.get("Accept")||t.headers.set("Accept","text/event-stream"),t.headers.get("Cache-Control")||t.headers.set("Cache-Control","no-cache"),t.headers.get("X-Accel-Buffering")||t.headers.set("X-Accel-Buffering","no")),xe(e,t),Re(t)}function u(i,e,t={}){ue(i,e,t);let r=i.auth.isAuthenticated;return(0,_.ofetch)(e,B(T({baseURL:i.config.apiBaseUrl},t),{onResponse:({response:o})=>{var n;!y&&r&&((n=o._data)==null?void 0:n.code)===2100&&i.auth.signOut()}}))}function pe(r,o){return c(this,arguments,function*(i,e,t={}){return ue(i,e,t,!0),(yield(0,_.ofetch)(e,B(T({baseURL:i.config.apiBaseUrl},t),{responseType:"stream"}))).pipeThrough(new TextDecoderStream).pipeThrough(new me.EventSourceParserStream).pipeThrough(new TransformStream({transform:(a,h)=>{try{let m=JSON.parse(a.data);h.enqueue({event:a.event,data:m})}catch(m){}}}))})}function H(i,e,t=localStorage){let r=t.getItem(i),o=e?JSON.stringify(e):null;o?t.setItem(i,o):t.removeItem(i),window.dispatchEvent(new StorageEvent("storage",{key:i,oldValue:r,newValue:o,storageArea:t}))}function z(i,e=localStorage){let t=e.getItem(i);try{return t?JSON.parse(t):null}catch(r){return null}}var b,j,P,v=class{constructor(e){p(this,b);p(this,j,`lumi-auth-${(0,de.v4)()}`);p(this,P,null);d(this,b,e),Promise.resolve().then(()=>{!y&&this.isAuthenticated&&this.refreshUser()})}get accessToken(){if(y){let e=s(this,b).config.authorization;return e?e.replace("Bearer ",""):null}return z("lumi-access-token")}set accessToken(e){if(y){s(this,b).config.authorization=e?`Bearer ${e}`:void 0;return}H("lumi-access-token",e)}get user(){return y?s(this,P):z("lumi-user")}set user(e){if(y){d(this,P,e);return}H("lumi-user",e)}get isAuthenticated(){return!!this.accessToken}signIn(){if(y)throw new Error("auth.signIn() can only be called on the client side");let e=800,t=600,r=(window.screen.width-e)/2,o=(window.screen.height-t)/2,n=window.open(s(this,b).config.authOrigin,s(this,j),`width=${e},height=${t},left=${r},top=${o}`),a;return new Promise((h,m)=>{if(!n)return m(new Error("Open auth window failed"));let g=setInterval(()=>{n.closed&&m(new Error("Auth window closed"))},1e3),I=l=>{n.closed||(n.focus(),l.stopPropagation(),l.preventDefault())},E=({data:l,origin:G,source:ge})=>{if(!(G!==s(this,b).config.authOrigin||ge!==n))switch(l==null?void 0:l.type){case"lumi-ready":{n.postMessage({type:"lumi-init",data:{projectId:s(this,b).config.projectId,icon:W(),title:ee()}},s(this,b).config.authOrigin);break}case"lumi-sign-in":{if(l.data.projectId!==s(this,b).config.projectId)break;n.close(),window.focus(),this.accessToken=l.data.accessToken,this.user=l.data.user,h(l.data);break}}};window.addEventListener("message",E),document.addEventListener("click",I,!0),a=()=>{clearInterval(g),window.removeEventListener("message",E),document.removeEventListener("click",I,!0)}}).finally(()=>a==null?void 0:a())}signOut(){if(y)throw new Error("auth.signOut() can only be called on the client side");this.accessToken=null,this.user=null}refreshUser(){return c(this,null,function*(){let e=yield u(s(this,b),"/lm/user/info",{method:"POST"});if(e.code!==200)throw new Error(e.message);return this.user=e.data,e.data})}onAuthChange(e){if(y)throw new Error("auth.onAuthChange() can only be called on the client side");let t=r=>{(r.key==="lumi-access-token"||r.key==="lumi-user"||r.key===null)&&e({isAuthenticated:this.isAuthenticated,user:this.user})};return window.addEventListener("storage",t),()=>{window.removeEventListener("storage",t)}}};b=new WeakMap,j=new WeakMap,P=new WeakMap;var w,M=class{constructor(e,t){p(this,w);d(this,w,e),this.entityName=t}list(){return c(this,arguments,function*({filter:e,sort:t,limit:r,skip:o}={}){if(r){let n=yield u(s(this,w),this.uri("/find"),{method:"POST",body:{filter:e,sort:t,limit:r,skip:o}});if(n.code!==200)throw new Error(n.message);return n.data}else{let n=yield u(s(this,w),this.uri("/list"),{method:"POST",body:{filter:e,sort:t}});if(n.code!==200)throw new Error(n.message);return{total:n.data.length,list:n.data}}})}get(e){return c(this,null,function*(){let t=yield u(s(this,w),this.uri(`/${e}`),{method:"GET"});if(t.code!==200)throw new Error(t.message);return t.data})}create(e){return c(this,null,function*(){let t=yield u(s(this,w),this.uri(),{method:"POST",body:e});if(t.code!==200)throw new Error(t.message);return t.data})}createMany(e){return c(this,null,function*(){let t=yield u(s(this,w),this.uri("/batch"),{method:"POST",body:e});if(t.code!==200)throw new Error(t.message);return t.data})}update(e,t){return c(this,null,function*(){let r=yield u(s(this,w),this.uri(),{method:"PUT",body:{filter:{_id:e},update:t}});if(r.code!==200)throw new Error(r.message);return r.data})}delete(e){return c(this,null,function*(){let t=yield u(s(this,w),this.uri(`/${e}`),{method:"DELETE"});if(t.code!==200)throw new Error(t.message)})}deleteMany(e){return c(this,null,function*(){let t=yield u(s(this,w),this.uri("/batch-by-ids"),{method:"DELETE",params:{ids:e}});if(t.code!==200)throw new Error(t.message)})}uri(e=""){return`/lm/${s(this,w).config.projectId}/${this.entityName}/documents${e}`}};w=new WeakMap;var U,O=class{constructor(e){p(this,U);return d(this,U,e),new Proxy(this,{get(t,r){return r in t||(t[r]=new M(s(t,U),r)),t[r]}})}};U=new WeakMap;var he=require("ofetch");var C,k=class{constructor(e){p(this,C);d(this,C,e)}invoke(e,t={}){return s(this,C).auth.accessToken&&(t.headers=T({Authorization:`Bearer ${s(this,C).auth.accessToken}`},t.headers)),(0,he.ofetch)(`/v1/functions/${s(this,C).config.projectId}/${e}`,T({baseURL:s(this,C).config.apiBaseUrl},t))}};C=new WeakMap;var S,D=class{constructor(e){p(this,S);d(this,S,e)}generateText(r){return c(this,arguments,function*({model:e="gemini-2.5-flash",messages:t}){let o=this.checkLumiApiKey(),n=yield u(s(this,S),`/lm/${s(this,S).config.projectId}/ai/chat/completions`,{method:"POST",body:{lumiApiKey:o,modelName:e,chatMessages:t}});if(n.code!==200)throw new f(n.code,n.message);return n.data})}generateTextStream(r){return c(this,arguments,function*({model:e="gemini-2.5-flash",messages:t}){let o=this.checkLumiApiKey();return pe(s(this,S),`/lm/${s(this,S).config.projectId}/ai/chat/stream`,{method:"POST",body:{lumiApiKey:o,modelName:e,chatMessages:t}})})}generateImage(r){return c(this,arguments,function*({model:e="gemini-2.5-flash-image",messages:t}){let o=this.checkLumiApiKey(),{code:n,message:a,data:h}=yield u(s(this,S),`/lm/${s(this,S).config.projectId}/ai/image/task`,{method:"POST",body:{lumiApiKey:o,modelName:e,chatMessages:t}});if(n!==200)throw new f(n,a);let m=h,g=1e3;for(;m.generationStatus==="PROCESSING";){yield te(g),g<5e3&&(g+=500);let{code:I,message:E,data:l}=yield u(s(this,S),`/lm/${s(this,S).config.projectId}/ai/image/get`,{method:"POST",body:{lumiApiKey:o,messageId:m.messageId}});if(I!==200)throw new f(I,E);m=l}return m})}checkLumiApiKey(){if(!y)throw new f(400,"lumi.tools.ai is only available on the server-side");let e=ie("LUMI_API_KEY");if(!e)throw new f(400,"LUMI_API_KEY is required");return e}};S=new WeakMap;var R,q=class{constructor(e){p(this,R);d(this,R,e)}send(m){return c(this,arguments,function*({to:e,subject:t,fromName:r,html:o,text:n="",replyTo:a,scheduledAt:h}){if(!e||!t||!o&&!n)throw new Error("Failed to send email: Missing required parameters.");typeof e=="string"&&(e=[e]),typeof a=="string"&&(a=[a]);let g=yield u(s(this,R),`/lm/${s(this,R).config.projectId}/email/send`,{method:"POST",body:{to:e,subject:t,fromName:r,html:o,text:n,replyTo:a,scheduledAt:h}});if(g.code!==200)throw new f(g.code,g.message)})}};R=new WeakMap;var A,$=class{constructor(e){p(this,A);d(this,A,e)}upload(e){return c(this,null,function*(){let t=new FormData;e.forEach(o=>{t.append("files",o)});let r=yield u(s(this,A),`/lm/${s(this,A).config.projectId}/file/batch`,{method:"POST",body:t});if(r.code!==200)throw new f(r.code,r.message);return r.data})}delete(e){return c(this,null,function*(){let t=yield u(s(this,A),`/lm/${s(this,A).config.projectId}/file/batch`,{method:"DELETE",body:{fileUrls:e}});if(t.code!==200)throw new f(t.code,t.message)})}};A=new WeakMap;var F,N=class{constructor(e){p(this,F);d(this,F,e),this.email=new q(e),this.file=new $(e),this.ai=new D(e)}};F=new WeakMap;var K=class{constructor(e){this.config=e,this.auth=new v(this),this.entities=new O(this),this.tools=new N(this),this.functions=new k(this)}};function Le(i){return new K(i)}0&&(module.exports={AITool,EmailTool,EntitiesClient,EntityClient,FileTool,FunctionsClient,LumiAuthClient,LumiClient,ToolsClient,createClient});
4
4
  //# sourceMappingURL=index.js.map
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/index.ts","../src/core/auth-client.ts","../src/lib/request.ts","../src/lib/error.ts","../src/utils/common.ts","../src/utils/storage.ts","../src/core/entity-client.ts","../src/core/entities-client.ts","../src/core/functions-client.ts","../src/tools/email-tool.ts","../src/tools/file-tool.ts","../src/core/tools-client.ts","../src/core/lumi-client.ts"],"sourcesContent":["export * from './core/auth-client'\nexport * from './core/entities-client'\nexport * from './core/entity-client'\nexport * from './core/functions-client'\nexport * from './core/lumi-client'\nexport * from './core/tools-client'\nexport * from './tools'\n","import type { LumiClient } from '@/core/lumi-client'\nimport type { ApiResponse } from '@/lib/request'\nimport { v4 } from 'uuid'\nimport { MessageType, StorageKey } from '@/constants'\nimport { request } from '@/lib/request'\nimport { getIcon, getTitle, isServer } from '@/utils/common'\nimport { getStorage, setStorage } from '@/utils/storage'\n\nexport interface User {\n userId: string\n email: string\n userName: string\n userRole: 'ADMIN' | 'USER'\n createdTime: string\n}\n\nexport interface MessageSignInData {\n projectId: string\n accessToken: string\n user: User\n}\n\nexport type MessageDataReceive = {\n type: MessageType.READY\n} | {\n type: MessageType.SIGN_IN\n data: MessageSignInData\n}\n\nexport interface MessageInitData {\n projectId: string\n icon: string | null\n title: string | null\n}\n\nexport interface MessageDataSend {\n type: MessageType.INIT\n data: MessageInitData\n}\n\nexport class LumiAuthClient {\n readonly #lumi: LumiClient\n readonly #popupName: string = `lumi-auth-${v4()}`\n\n constructor(lumi: LumiClient) {\n this.#lumi = lumi\n Promise.resolve().then(() => {\n if (!isServer && this.isAuthenticated)\n void this.refreshUser()\n })\n }\n\n /** 访问令牌 */\n public get accessToken(): string | null {\n if (isServer) {\n const authorization = this.#lumi.config.authorization\n return authorization ? authorization.replace('Bearer ', '') : null\n }\n return getStorage<string>(StorageKey.ACCESS_TOKEN)\n }\n\n public set accessToken(accessToken: string | null) {\n if (isServer) {\n this.#lumi.config.authorization = accessToken ? `Bearer ${accessToken}` : undefined\n return\n }\n setStorage(StorageKey.ACCESS_TOKEN, accessToken)\n }\n\n #user: User | null = null\n\n /** 用户 */\n public get user(): User | null {\n if (isServer)\n return this.#user\n return getStorage<User>(StorageKey.USER)\n }\n\n public set user(user: User | null) {\n if (isServer) {\n this.#user = user\n return\n }\n setStorage(StorageKey.USER, user)\n }\n\n public get isAuthenticated(): boolean {\n return !!this.accessToken\n }\n\n /** 登录 */\n public signIn(): Promise<MessageSignInData> {\n if (isServer)\n throw new Error('auth.signIn() can only be called on the client side')\n\n const width = 800\n const height = 600\n const left = (window.screen.width - width) / 2\n const top = (window.screen.height - height) / 2\n const popup = window.open(this.#lumi.config.authOrigin, this.#popupName, `width=${width},height=${height},left=${left},top=${top}`)\n\n let cleanup: () => void\n return new Promise<MessageSignInData>((resolve, reject) => {\n if (!popup)\n return reject(new Error('Open auth window failed'))\n\n const timer = setInterval(() => {\n if (popup.closed)\n reject(new Error('Auth window closed'))\n }, 1000)\n\n // 全局点击事件处理函数 - 重新聚焦popup并阻止事件传播\n const handleGlobalClick = (event: MouseEvent): void => {\n if (!popup.closed) {\n popup.focus()\n event.stopPropagation()\n event.preventDefault()\n }\n }\n\n const handleMessage = ({ data, origin, source }: MessageEvent<MessageDataReceive | null>): void => {\n if (origin !== this.#lumi.config.authOrigin || source !== popup)\n return\n\n switch (data?.type) {\n case MessageType.READY: {\n popup.postMessage({\n type: MessageType.INIT,\n data: {\n projectId: this.#lumi.config.projectId,\n icon: getIcon(),\n title: getTitle(),\n },\n } satisfies MessageDataSend, this.#lumi.config.authOrigin)\n break\n }\n case MessageType.SIGN_IN: {\n if (data.data.projectId !== this.#lumi.config.projectId)\n break\n popup.close()\n window.focus()\n this.accessToken = data.data.accessToken\n this.user = data.data.user\n resolve(data.data)\n break\n }\n }\n }\n\n window.addEventListener('message', handleMessage)\n // 添加全局点击事件监听器,使用捕获阶段确保优先处理\n document.addEventListener('click', handleGlobalClick, true)\n\n cleanup = () => {\n clearInterval(timer)\n window.removeEventListener('message', handleMessage)\n document.removeEventListener('click', handleGlobalClick, true)\n }\n }).finally(() => cleanup?.())\n }\n\n /** 退出登录 */\n public signOut(): void {\n if (isServer)\n throw new Error('auth.signOut() can only be called on the client side')\n\n this.accessToken = null\n this.user = null\n }\n\n /** 获取当前用户 */\n public async refreshUser(): Promise<User> {\n const res = await request<ApiResponse<User>>(this.#lumi, '/lm/user/info', {\n method: 'POST',\n })\n if (res.code !== 200)\n throw new Error(res.message)\n this.user = res.data\n return res.data\n }\n\n /** 监听登录状态变化 */\n public onAuthChange(callback: (args: {\n isAuthenticated: boolean\n user: User | null\n }) => void): () => void {\n if (isServer)\n throw new Error('auth.onAuthChange() can only be called on the client side')\n\n const handleStorageChange = (event: StorageEvent): void => {\n if (event.key === StorageKey.ACCESS_TOKEN || event.key === StorageKey.USER || event.key === null) {\n callback({\n isAuthenticated: this.isAuthenticated,\n user: this.user,\n })\n }\n }\n\n window.addEventListener('storage', handleStorageChange)\n\n return () => {\n window.removeEventListener('storage', handleStorageChange)\n }\n }\n}\n","import type { FetchContext, FetchOptions } from 'ofetch'\nimport type { LumiClient } from '@/core/lumi-client'\nimport Base64 from 'crypto-js/enc-base64'\nimport Hex from 'crypto-js/enc-hex'\nimport HmacSHA256 from 'crypto-js/hmac-sha256'\nimport SHA256 from 'crypto-js/sha256'\nimport hash from 'object-hash'\nimport { ofetch } from 'ofetch'\nimport { LumiError } from '@/lib/error'\nimport { isServer } from '@/utils/common'\n\nconst SECRET_KEY = '6QrJZ7pFCmBZAeIJF7IArvkCz+EtzA0RVcpHkiQIsQyhs7QtCS9P+CueZdHfB2OtJcgX3BbqY9pfpWeAVTqCwQ=='\n\n/** RFC 3986 compliant URI encoding */\nfunction rfc3986Encode(str: string): string {\n return encodeURIComponent(str).replace(/[!'()*]/g, c => `%${c.charCodeAt(0).toString(16).toUpperCase()}`)\n}\n\n/** 创建签名拦截器函数 */\nexport function createSignatureInterceptor(secretKey: string) {\n return (context: FetchContext) => {\n const { options } = context\n // 生成签名\n const timestamp = Math.floor(Date.now() / 1000).toString()\n const nonce = Math.random().toString(36).substring(2, 15)\n\n // 构建查询字符串\n const queryParams: Record<string, string> = { ...options.query }\n const canonicalQueryString = Object.keys(queryParams).sort().map(key => `${rfc3986Encode(key)}=${rfc3986Encode(String(queryParams[key]))}`).join('&')\n\n // 构建签名头\n const headersToSign: Record<string, string> = {\n 'x-timestamp': timestamp,\n 'x-nonce': nonce,\n }\n const canonicalHeaders = Object.keys(headersToSign).sort().map(key => `${key}:${headersToSign[key]}`).join('\\n')\n\n // 构建载荷哈希\n const payload = (options.body && !(options.body instanceof FormData)) ? JSON.stringify(options.body) : ''\n const hashedPayload = SHA256(payload).toString(Hex)\n\n // 构建规范请求\n const canonicalRequest = [canonicalQueryString, canonicalHeaders, hashedPayload].join('\\n')\n\n // 生成签名\n const signature = Base64.stringify(HmacSHA256(canonicalRequest, secretKey))\n\n // 添加签名头到请求\n const headers = new Headers(options.headers)\n Object.entries(headersToSign).forEach(([key, value]) => {\n headers.set(key, value)\n })\n headers.set('X-Sign', signature)\n options.headers = headers\n }\n}\n\nexport interface ApiResponse<T> {\n code: number\n message: string\n data: T\n}\n\nexport interface PaginationData<T> {\n total: number\n list: T[]\n}\n\nconst requestMap = new Map<string, number[]>()\n\n/** 限制请求频率,相同参数的请求每秒只能发送4次 */\nexport function checkRateLimit(uri: string, options: FetchOptions<'json'>): void {\n if (options.body instanceof FormData) {\n options = {\n ...options,\n body: Array.from(options.body.entries()),\n }\n }\n const requestHash = hash([uri, options])\n const now = Date.now()\n const requestQueue = requestMap.get(requestHash)?.filter(time => now - time < 1000) || []\n if (requestQueue.length >= 4)\n throw new LumiError(429, 'Too many requests')\n requestQueue.push(now)\n requestMap.set(requestHash, requestQueue)\n}\n\nexport function request<T>(lumi: LumiClient, uri: string, options: FetchOptions<'json'> = {}): Promise<T> {\n // 添加认证头(如果存在)\n if (lumi.auth.accessToken) {\n options.headers = {\n Authorization: `Bearer ${lumi.auth.accessToken}`,\n ...options.headers,\n }\n }\n\n // 限制请求频率\n checkRateLimit(uri, options)\n\n const isAuthenticated = lumi.auth.isAuthenticated\n\n return ofetch<T>(uri, {\n baseURL: lumi.config.apiBaseUrl,\n ...options,\n onRequest: createSignatureInterceptor(SECRET_KEY),\n onResponse: ({ response }) => {\n // 若 Token 失效则退出登录\n if (!isServer && isAuthenticated && (response._data as ApiResponse<any>)?.code === 2100)\n lumi.auth.signOut()\n },\n })\n}\n","export class LumiError extends Error {\n name: string = 'LumiError'\n code: number\n\n constructor(code: number, message: string) {\n super(message)\n this.code = code\n }\n}\n","export function getIcon(): string | null {\n return document.querySelector<HTMLLinkElement>('link[rel=\"icon\"]')?.href ?? null\n}\n\nexport function getTitle(): string | null {\n return document.title ?? null\n}\n\nexport const isServer = typeof window === 'undefined'\n","import type { StorageKey } from '@/constants'\n\nexport function setStorage<T>(key: StorageKey, value: T, storage: Storage = localStorage): void {\n const oldValue = storage.getItem(key)\n const newValue = value ? JSON.stringify(value) : null\n if (newValue)\n storage.setItem(key, newValue)\n else\n storage.removeItem(key)\n\n window.dispatchEvent(new StorageEvent('storage', {\n key,\n oldValue,\n newValue,\n storageArea: storage,\n }))\n}\n\nexport function getStorage<T>(key: StorageKey, storage: Storage = localStorage): T | null {\n const value = storage.getItem(key)\n try {\n return value ? JSON.parse(value) : null\n }\n catch (_e) {\n return null\n }\n}\n","import type { LumiClient } from '@/core/lumi-client'\nimport type { ApiResponse, PaginationData } from '@/lib/request'\nimport { request } from '@/lib/request'\n\nexport interface Entity extends Record<string, any> {\n id: string\n}\n\nexport class EntityClient {\n readonly #lumi: LumiClient\n public readonly entityName: string\n\n constructor(lumi: LumiClient, entityName: string) {\n this.#lumi = lumi\n this.entityName = entityName\n }\n\n /** 查询文档列表 */\n public async list({ filter, sort, limit, skip }: {\n filter?: any\n sort?: Record<string, 1 | -1>\n limit?: number\n skip?: number\n } = {}): Promise<PaginationData<Entity>> {\n if (!limit) {\n const res = await request<ApiResponse<Entity[]>>(this.#lumi, this.uri('/list'), {\n method: 'POST',\n body: { filter, sort },\n })\n if (res.code !== 200)\n throw new Error(res.message)\n return {\n total: res.data.length,\n list: res.data,\n }\n }\n else {\n const res = await request<ApiResponse<PaginationData<Entity>>>(this.#lumi, this.uri('/find'), {\n method: 'POST',\n body: { filter, sort, limit, skip },\n })\n if (res.code !== 200)\n throw new Error(res.message)\n return res.data\n }\n }\n\n /** 获取单个文档 */\n public async get(id: string): Promise<Entity | null> {\n const res = await request<ApiResponse<Entity>>(this.#lumi, this.uri(`/${id}`), {\n method: 'GET',\n })\n if (res.code !== 200)\n throw new Error(res.message)\n return res.data\n }\n\n /** 创建文档 */\n public async create(data: Record<string, any>): Promise<Entity> {\n const res = await request<ApiResponse<Entity>>(this.#lumi, this.uri(), {\n method: 'POST',\n body: data,\n })\n if (res.code !== 200)\n throw new Error(res.message)\n return res.data\n }\n\n /** 批量创建文档 */\n public async createMany(data: Record<string, any>[]): Promise<Entity[]> {\n const res = await request<ApiResponse<Entity[]>>(this.#lumi, this.uri('/batch'), {\n method: 'POST',\n body: data,\n })\n if (res.code !== 200)\n throw new Error(res.message)\n return res.data\n }\n\n /** 更新文档 */\n public async update(id: string, data: Record<string, any>): Promise<Entity> {\n const res = await request<ApiResponse<Entity>>(this.#lumi, this.uri(), {\n method: 'PUT',\n body: { filter: { _id: id }, update: data },\n })\n if (res.code !== 200)\n throw new Error(res.message)\n return res.data\n }\n\n /** 删除文档 */\n public async delete(id: string): Promise<void> {\n const res = await request<ApiResponse<null>>(this.#lumi, this.uri(`/${id}`), {\n method: 'DELETE',\n })\n if (res.code !== 200)\n throw new Error(res.message)\n }\n\n /** 批量删除文档 */\n public async deleteMany(ids: string[]): Promise<void> {\n const res = await request<ApiResponse<null>>(this.#lumi, this.uri('/batch-by-ids'), {\n method: 'DELETE',\n params: { ids },\n })\n if (res.code !== 200)\n throw new Error(res.message)\n }\n\n private uri(suffix = ''): string {\n return `/lm/${this.#lumi.config.projectId}/${this.entityName}/documents${suffix}`\n }\n}\n","import type { LumiClient } from '@/core/lumi-client'\nimport { EntityClient } from '@/core/entity-client'\n\nexport class EntitiesClient {\n readonly #lumi: LumiClient\n [key: string]: EntityClient\n\n constructor(lumi: LumiClient) {\n this.#lumi = lumi\n return new Proxy(this, {\n get(target: EntitiesClient, p: string) {\n if (!(p in target))\n target[p] = new EntityClient(target.#lumi, p)\n return target[p]\n },\n }) as this\n }\n}\n","import type { FetchOptions } from 'ofetch'\nimport type { LumiClient } from '@/core/lumi-client'\nimport { ofetch } from 'ofetch'\n\nexport class FunctionsClient {\n readonly #lumi: LumiClient\n\n constructor(lumi: LumiClient) {\n this.#lumi = lumi\n }\n\n public invoke(functionName: string, options: FetchOptions<'json'> = {}): Promise<any> {\n if (this.#lumi.auth.accessToken) {\n options.headers = {\n Authorization: `Bearer ${this.#lumi.auth.accessToken}`,\n ...options.headers,\n }\n }\n\n return ofetch(`/v1/functions/${this.#lumi.config.projectId}/${functionName}`, {\n baseURL: this.#lumi.config.apiBaseUrl,\n ...options,\n })\n }\n}\n","import type { LumiClient } from '@/core/lumi-client'\nimport type { ApiResponse } from '@/lib/request'\nimport { LumiError } from '@/lib/error'\nimport { request } from '@/lib/request'\n\nexport class EmailTool {\n readonly #lumi: LumiClient\n\n constructor(lumi: LumiClient) {\n this.#lumi = lumi\n }\n\n /** 发送邮件 */\n public async send({ to, subject, fromName, html, text = '', replyTo, scheduledAt }: {\n to: string | string[]\n subject: string\n fromName?: string\n html?: string\n text?: string\n replyTo?: string | string[]\n scheduledAt?: string\n }): Promise<void> {\n if (!to || !subject || (!html && !text))\n throw new Error('Failed to send email: Missing required parameters.')\n\n if (typeof to === 'string')\n to = [to]\n if (typeof replyTo === 'string')\n replyTo = [replyTo]\n\n const res = await request<ApiResponse<null>>(this.#lumi, `/lm/${this.#lumi.config.projectId}/email/send`, {\n method: 'POST',\n body: { to, subject, fromName, html, text, replyTo, scheduledAt },\n })\n\n if (res.code !== 200)\n throw new LumiError(res.code, res.message)\n }\n}\n","import type { LumiClient } from '@/core/lumi-client'\nimport type { ApiResponse } from '@/lib/request'\nimport { LumiError } from '@/lib/error'\nimport { request } from '@/lib/request'\n\nexport interface UploadItem {\n fileName: string\n fileUrl?: string\n uploadError?: string\n}\n\nexport class FileTool {\n readonly #lumi: LumiClient\n\n constructor(lumi: LumiClient) {\n this.#lumi = lumi\n }\n\n /** 上传文件 */\n public async upload(files: File[]): Promise<UploadItem[]> {\n const formData = new FormData()\n files.forEach((file) => {\n formData.append('files', file)\n })\n\n const res = await request<ApiResponse<UploadItem[]>>(this.#lumi, `/lm/${this.#lumi.config.projectId}/file/batch`, {\n method: 'POST',\n body: formData,\n })\n\n if (res.code !== 200)\n throw new LumiError(res.code, res.message)\n\n return res.data\n }\n\n /** 批量删除文件 */\n public async delete(fileUrls: string[]): Promise<void> {\n const res = await request<ApiResponse<null>>(this.#lumi, `/lm/${this.#lumi.config.projectId}/file/batch`, {\n method: 'DELETE',\n body: { fileUrls },\n })\n\n if (res.code !== 200)\n throw new LumiError(res.code, res.message)\n }\n}\n","import type { LumiClient } from '@/core/lumi-client'\nimport { EmailTool, FileTool } from '@/tools'\n\nexport class ToolsClient {\n readonly #lumi: LumiClient\n\n public email: EmailTool\n public file: FileTool\n\n constructor(lumi: LumiClient) {\n this.#lumi = lumi\n this.email = new EmailTool(lumi)\n this.file = new FileTool(lumi)\n }\n}\n","import { LumiAuthClient } from '@/core/auth-client'\nimport { EntitiesClient } from '@/core/entities-client'\nimport { FunctionsClient } from '@/core/functions-client'\nimport { ToolsClient } from '@/core/tools-client'\n\nexport interface LumiClientConfig {\n projectId: string\n apiBaseUrl: string\n authOrigin: string\n authorization?: string\n}\n\nexport class LumiClient {\n public config: LumiClientConfig\n\n public auth: LumiAuthClient\n public entities: EntitiesClient\n public tools: ToolsClient\n public functions: FunctionsClient\n\n constructor(config: LumiClientConfig) {\n this.config = config\n this.auth = new LumiAuthClient(this)\n this.entities = new EntitiesClient(this)\n this.tools = new ToolsClient(this)\n this.functions = new FunctionsClient(this)\n }\n}\n\nexport function createClient(config: LumiClientConfig): LumiClient {\n return new LumiClient(config)\n}\n"],"mappings":"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","names":["index_exports","__export","EmailTool","EntitiesClient","EntityClient","FileTool","FunctionsClient","LumiAuthClient","LumiClient","ToolsClient","createClient","__toCommonJS","import_uuid","import_enc_base64","import_enc_hex","import_hmac_sha256","import_sha256","import_object_hash","import_ofetch","LumiError","code","message","getIcon","_a","_b","getTitle","isServer","SECRET_KEY","rfc3986Encode","str","c","createSignatureInterceptor","secretKey","context","options","timestamp","nonce","queryParams","__spreadValues","canonicalQueryString","key","headersToSign","canonicalHeaders","payload","hashedPayload","SHA256","Hex","canonicalRequest","signature","Base64","HmacSHA256","headers","value","requestMap","checkRateLimit","uri","_a","__spreadProps","requestHash","hash","now","requestQueue","time","LumiError","request","lumi","isAuthenticated","response","isServer","setStorage","key","value","storage","oldValue","newValue","getStorage","_e","_lumi","_popupName","_user","LumiAuthClient","lumi","__privateAdd","__privateSet","isServer","authorization","__privateGet","getStorage","accessToken","setStorage","user","width","height","left","top","popup","cleanup","resolve","reject","timer","handleGlobalClick","event","handleMessage","data","origin","source","getIcon","getTitle","__async","res","request","callback","handleStorageChange","_lumi","EntityClient","lumi","entityName","__privateAdd","__privateSet","__async","filter","sort","limit","skip","res","request","__privateGet","id","data","ids","suffix","_lumi","EntitiesClient","lumi","__privateAdd","__privateSet","target","p","EntityClient","__privateGet","import_ofetch","_lumi","FunctionsClient","lumi","__privateAdd","__privateSet","functionName","options","__privateGet","__spreadValues","_lumi","EmailTool","lumi","__privateAdd","__privateSet","_0","__async","to","subject","fromName","html","text","replyTo","scheduledAt","res","request","__privateGet","LumiError","_lumi","FileTool","lumi","__privateAdd","__privateSet","files","__async","formData","file","res","request","__privateGet","LumiError","fileUrls","_lumi","ToolsClient","lumi","__privateAdd","__privateSet","EmailTool","FileTool","LumiClient","config","LumiAuthClient","EntitiesClient","ToolsClient","FunctionsClient","createClient"]}
1
+ {"version":3,"sources":["../src/index.ts","../src/core/auth-client.ts","../src/lib/request.ts","../src/lib/error.ts","../src/utils/common.ts","../src/utils/storage.ts","../src/core/entity-client.ts","../src/core/entities-client.ts","../src/core/functions-client.ts","../src/tools/ai-tool.ts","../src/tools/email-tool.ts","../src/tools/file-tool.ts","../src/core/tools-client.ts","../src/core/lumi-client.ts"],"sourcesContent":["export * from './core/auth-client'\nexport * from './core/entities-client'\nexport * from './core/entity-client'\nexport * from './core/functions-client'\nexport * from './core/lumi-client'\nexport * from './core/tools-client'\nexport * from './tools'\n","import type { LumiClient } from '@/core/lumi-client'\nimport type { ApiResponse } from '@/lib/request'\nimport { v4 } from 'uuid'\nimport { MessageType, StorageKey } from '@/constants'\nimport { request } from '@/lib/request'\nimport { getIcon, getTitle, isServer } from '@/utils/common'\nimport { getStorage, setStorage } from '@/utils/storage'\n\nexport interface User {\n userId: string\n email: string\n userName: string\n userRole: 'ADMIN' | 'USER'\n createdTime: string\n}\n\nexport interface MessageSignInData {\n projectId: string\n accessToken: string\n user: User\n}\n\nexport type MessageDataReceive = {\n type: MessageType.READY\n} | {\n type: MessageType.SIGN_IN\n data: MessageSignInData\n}\n\nexport interface MessageInitData {\n projectId: string\n icon: string | null\n title: string | null\n}\n\nexport interface MessageDataSend {\n type: MessageType.INIT\n data: MessageInitData\n}\n\nexport class LumiAuthClient {\n readonly #lumi: LumiClient\n readonly #popupName: string = `lumi-auth-${v4()}`\n\n constructor(lumi: LumiClient) {\n this.#lumi = lumi\n Promise.resolve().then(() => {\n if (!isServer && this.isAuthenticated)\n void this.refreshUser()\n })\n }\n\n /** 访问令牌 */\n public get accessToken(): string | null {\n if (isServer) {\n const authorization = this.#lumi.config.authorization\n return authorization ? authorization.replace('Bearer ', '') : null\n }\n return getStorage<string>(StorageKey.ACCESS_TOKEN)\n }\n\n public set accessToken(accessToken: string | null) {\n if (isServer) {\n this.#lumi.config.authorization = accessToken ? `Bearer ${accessToken}` : undefined\n return\n }\n setStorage(StorageKey.ACCESS_TOKEN, accessToken)\n }\n\n #user: User | null = null\n\n /** 用户 */\n public get user(): User | null {\n if (isServer)\n return this.#user\n return getStorage<User>(StorageKey.USER)\n }\n\n public set user(user: User | null) {\n if (isServer) {\n this.#user = user\n return\n }\n setStorage(StorageKey.USER, user)\n }\n\n public get isAuthenticated(): boolean {\n return !!this.accessToken\n }\n\n /** 登录 */\n public signIn(): Promise<MessageSignInData> {\n if (isServer)\n throw new Error('auth.signIn() can only be called on the client side')\n\n const width = 800\n const height = 600\n const left = (window.screen.width - width) / 2\n const top = (window.screen.height - height) / 2\n const popup = window.open(this.#lumi.config.authOrigin, this.#popupName, `width=${width},height=${height},left=${left},top=${top}`)\n\n let cleanup: () => void\n return new Promise<MessageSignInData>((resolve, reject) => {\n if (!popup)\n return reject(new Error('Open auth window failed'))\n\n const timer = setInterval(() => {\n if (popup.closed)\n reject(new Error('Auth window closed'))\n }, 1000)\n\n // 全局点击事件处理函数 - 重新聚焦popup并阻止事件传播\n const handleGlobalClick = (event: MouseEvent): void => {\n if (!popup.closed) {\n popup.focus()\n event.stopPropagation()\n event.preventDefault()\n }\n }\n\n const handleMessage = ({ data, origin, source }: MessageEvent<MessageDataReceive | null>): void => {\n if (origin !== this.#lumi.config.authOrigin || source !== popup)\n return\n\n switch (data?.type) {\n case MessageType.READY: {\n popup.postMessage({\n type: MessageType.INIT,\n data: {\n projectId: this.#lumi.config.projectId,\n icon: getIcon(),\n title: getTitle(),\n },\n } satisfies MessageDataSend, this.#lumi.config.authOrigin)\n break\n }\n case MessageType.SIGN_IN: {\n if (data.data.projectId !== this.#lumi.config.projectId)\n break\n popup.close()\n window.focus()\n this.accessToken = data.data.accessToken\n this.user = data.data.user\n resolve(data.data)\n break\n }\n }\n }\n\n window.addEventListener('message', handleMessage)\n // 添加全局点击事件监听器,使用捕获阶段确保优先处理\n document.addEventListener('click', handleGlobalClick, true)\n\n cleanup = () => {\n clearInterval(timer)\n window.removeEventListener('message', handleMessage)\n document.removeEventListener('click', handleGlobalClick, true)\n }\n }).finally(() => cleanup?.())\n }\n\n /** 退出登录 */\n public signOut(): void {\n if (isServer)\n throw new Error('auth.signOut() can only be called on the client side')\n\n this.accessToken = null\n this.user = null\n }\n\n /** 获取当前用户 */\n public async refreshUser(): Promise<User> {\n const res = await request<ApiResponse<User>>(this.#lumi, '/lm/user/info', {\n method: 'POST',\n })\n if (res.code !== 200)\n throw new Error(res.message)\n this.user = res.data\n return res.data\n }\n\n /** 监听登录状态变化 */\n public onAuthChange(callback: (args: {\n isAuthenticated: boolean\n user: User | null\n }) => void): () => void {\n if (isServer)\n throw new Error('auth.onAuthChange() can only be called on the client side')\n\n const handleStorageChange = (event: StorageEvent): void => {\n if (event.key === StorageKey.ACCESS_TOKEN || event.key === StorageKey.USER || event.key === null) {\n callback({\n isAuthenticated: this.isAuthenticated,\n user: this.user,\n })\n }\n }\n\n window.addEventListener('storage', handleStorageChange)\n\n return () => {\n window.removeEventListener('storage', handleStorageChange)\n }\n }\n}\n","import type { FetchOptions, ResponseType } from 'ofetch'\nimport type { LumiClient } from '@/core/lumi-client'\nimport Base64 from 'crypto-js/enc-base64'\nimport Hex from 'crypto-js/enc-hex'\nimport HmacSHA256 from 'crypto-js/hmac-sha256'\nimport SHA256 from 'crypto-js/sha256'\nimport { EventSourceParserStream } from 'eventsource-parser/stream'\nimport hash from 'object-hash'\nimport { ofetch } from 'ofetch'\nimport { LumiError } from '@/lib/error'\nimport { isServer } from '@/utils/common'\n\nconst SECRET_KEY = '6QrJZ7pFCmBZAeIJF7IArvkCz+EtzA0RVcpHkiQIsQyhs7QtCS9P+CueZdHfB2OtJcgX3BbqY9pfpWeAVTqCwQ=='\n\n/** RFC 3986 compliant URI encoding */\nfunction rfc3986Encode(str: string): string {\n return encodeURIComponent(str).replace(/[!'()*]/g, c => `%${c.charCodeAt(0).toString(16).toUpperCase()}`)\n}\n\n/** 添加签名 */\nfunction addSignature<T extends ResponseType>(options: FetchOptions<T>): void {\n // 生成签名\n const timestamp = Math.floor(Date.now() / 1000).toString()\n const nonce = Math.random().toString(36).substring(2, 15)\n\n // 构建查询字符串\n const queryParams: Record<string, string> = { ...options.query }\n const canonicalQueryString = Object.keys(queryParams).sort().map(key => `${rfc3986Encode(key)}=${rfc3986Encode(String(queryParams[key]))}`).join('&')\n\n // 构建签名头\n const headersToSign: Record<string, string> = {\n 'x-timestamp': timestamp,\n 'x-nonce': nonce,\n }\n const canonicalHeaders = Object.keys(headersToSign).sort().map(key => `${key}:${headersToSign[key]}`).join('\\n')\n\n // 构建载荷哈希\n const payload = (options.body && !(options.body instanceof FormData)) ? JSON.stringify(options.body) : ''\n const hashedPayload = SHA256(payload).toString(Hex)\n\n // 构建规范请求\n const canonicalRequest = [canonicalQueryString, canonicalHeaders, hashedPayload].join('\\n')\n\n // 生成签名\n const signature = Base64.stringify(HmacSHA256(canonicalRequest, SECRET_KEY))\n\n // 添加签名头到请求\n const headers = new Headers(options.headers)\n Object.entries(headersToSign).forEach(([key, value]) => {\n headers.set(key, value)\n })\n headers.set('X-Sign', signature)\n options.headers = headers\n}\n\nexport interface ApiResponse<T> {\n code: number\n message: string\n data: T\n}\n\nexport interface PaginationData<T> {\n total: number\n list: T[]\n}\n\nconst requestMap = new Map<string, number[]>()\n\n/** 限制请求频率,相同参数的请求每秒只能发送4次 */\nfunction checkRateLimit<T extends ResponseType>(uri: string, options: FetchOptions<T>): void {\n const clonedOptions = { ...options }\n\n if (options.body instanceof FormData)\n clonedOptions.body = Array.from(options.body.entries())\n\n clonedOptions.headers = undefined\n\n const requestHash = hash([uri, clonedOptions])\n const now = Date.now()\n const requestQueue = requestMap.get(requestHash)?.filter(time => now - time < 1000) || []\n if (requestQueue.length >= 4)\n throw new LumiError(429, 'Too many requests')\n requestQueue.push(now)\n requestMap.set(requestHash, requestQueue)\n}\n\nfunction beforeRequest<T extends ResponseType>(lumi: LumiClient, uri: string, options: FetchOptions<T>, stream: boolean = false): void {\n options.headers = new Headers(options.headers)\n\n // 添加认证头(如果存在)\n if (lumi.auth.accessToken)\n options.headers.set('Authorization', `Bearer ${lumi.auth.accessToken}`)\n\n // 添加流式请求头\n if (stream) {\n if (!options.headers.get('Accept'))\n options.headers.set('Accept', 'text/event-stream')\n if (!options.headers.get('Cache-Control'))\n options.headers.set('Cache-Control', 'no-cache')\n if (!options.headers.get('X-Accel-Buffering'))\n options.headers.set('X-Accel-Buffering', 'no')\n }\n\n // 限制请求频率\n checkRateLimit(uri, options)\n\n // 添加签名\n addSignature(options)\n}\n\n/** 网络请求 */\nexport function request<T>(lumi: LumiClient, uri: string, options: FetchOptions<'json'> = {}): Promise<T> {\n beforeRequest(lumi, uri, options)\n\n const isAuthenticated = lumi.auth.isAuthenticated\n\n return ofetch<T>(uri, {\n baseURL: lumi.config.apiBaseUrl,\n ...options,\n onResponse: ({ response }) => {\n // 若 Token 失效则退出登录\n if (!isServer && isAuthenticated && (response._data as ApiResponse<any>)?.code === 2100)\n lumi.auth.signOut()\n },\n })\n}\n\nexport interface StreamChunk<T> {\n event?: string\n data: T\n}\n\n/** 网络请求(流式) */\nexport async function requestStream<T>(lumi: LumiClient, uri: string, options: FetchOptions<'stream'> = {}): Promise<ReadableStream<StreamChunk<T>>> {\n beforeRequest(lumi, uri, options, true)\n\n const stream = await ofetch(uri, {\n baseURL: lumi.config.apiBaseUrl,\n ...options,\n responseType: 'stream',\n })\n\n return stream\n .pipeThrough(new TextDecoderStream())\n .pipeThrough(new EventSourceParserStream())\n .pipeThrough<StreamChunk<T>>(new TransformStream({\n transform: (chunk, controller) => {\n try {\n const data = JSON.parse(chunk.data)\n controller.enqueue({\n event: chunk.event,\n data,\n })\n }\n catch {}\n },\n }))\n}\n","export class LumiError extends Error {\n name: string = 'LumiError'\n code: number\n\n constructor(code: number, message: string) {\n super(message)\n this.code = code\n }\n}\n","export function getIcon(): string | null {\n return document.querySelector<HTMLLinkElement>('link[rel=\"icon\"]')?.href ?? null\n}\n\nexport function getTitle(): string | null {\n return document.title ?? null\n}\n\nexport const isServer = typeof window === 'undefined'\n\nexport function delay(ms: number): Promise<void> {\n return new Promise(resolve => setTimeout(resolve, ms))\n}\n\nexport function getEnv(key: string): string | null {\n return typeof process !== 'undefined' ? process?.env[key] ?? null : null\n}\n","import type { StorageKey } from '@/constants'\n\nexport function setStorage<T>(key: StorageKey, value: T, storage: Storage = localStorage): void {\n const oldValue = storage.getItem(key)\n const newValue = value ? JSON.stringify(value) : null\n if (newValue)\n storage.setItem(key, newValue)\n else\n storage.removeItem(key)\n\n window.dispatchEvent(new StorageEvent('storage', {\n key,\n oldValue,\n newValue,\n storageArea: storage,\n }))\n}\n\nexport function getStorage<T>(key: StorageKey, storage: Storage = localStorage): T | null {\n const value = storage.getItem(key)\n try {\n return value ? JSON.parse(value) : null\n }\n catch (_e) {\n return null\n }\n}\n","import type { LumiClient } from '@/core/lumi-client'\nimport type { ApiResponse, PaginationData } from '@/lib/request'\nimport { request } from '@/lib/request'\n\nexport interface Entity extends Record<string, any> {\n id: string\n}\n\nexport class EntityClient {\n readonly #lumi: LumiClient\n public readonly entityName: string\n\n constructor(lumi: LumiClient, entityName: string) {\n this.#lumi = lumi\n this.entityName = entityName\n }\n\n /** 查询文档列表 */\n public async list({ filter, sort, limit, skip }: {\n filter?: any\n sort?: Record<string, 1 | -1>\n limit?: number\n skip?: number\n } = {}): Promise<PaginationData<Entity>> {\n if (!limit) {\n const res = await request<ApiResponse<Entity[]>>(this.#lumi, this.uri('/list'), {\n method: 'POST',\n body: { filter, sort },\n })\n if (res.code !== 200)\n throw new Error(res.message)\n return {\n total: res.data.length,\n list: res.data,\n }\n }\n else {\n const res = await request<ApiResponse<PaginationData<Entity>>>(this.#lumi, this.uri('/find'), {\n method: 'POST',\n body: { filter, sort, limit, skip },\n })\n if (res.code !== 200)\n throw new Error(res.message)\n return res.data\n }\n }\n\n /** 获取单个文档 */\n public async get(id: string): Promise<Entity | null> {\n const res = await request<ApiResponse<Entity>>(this.#lumi, this.uri(`/${id}`), {\n method: 'GET',\n })\n if (res.code !== 200)\n throw new Error(res.message)\n return res.data\n }\n\n /** 创建文档 */\n public async create(data: Record<string, any>): Promise<Entity> {\n const res = await request<ApiResponse<Entity>>(this.#lumi, this.uri(), {\n method: 'POST',\n body: data,\n })\n if (res.code !== 200)\n throw new Error(res.message)\n return res.data\n }\n\n /** 批量创建文档 */\n public async createMany(data: Record<string, any>[]): Promise<Entity[]> {\n const res = await request<ApiResponse<Entity[]>>(this.#lumi, this.uri('/batch'), {\n method: 'POST',\n body: data,\n })\n if (res.code !== 200)\n throw new Error(res.message)\n return res.data\n }\n\n /** 更新文档 */\n public async update(id: string, data: Record<string, any>): Promise<Entity> {\n const res = await request<ApiResponse<Entity>>(this.#lumi, this.uri(), {\n method: 'PUT',\n body: { filter: { _id: id }, update: data },\n })\n if (res.code !== 200)\n throw new Error(res.message)\n return res.data\n }\n\n /** 删除文档 */\n public async delete(id: string): Promise<void> {\n const res = await request<ApiResponse<null>>(this.#lumi, this.uri(`/${id}`), {\n method: 'DELETE',\n })\n if (res.code !== 200)\n throw new Error(res.message)\n }\n\n /** 批量删除文档 */\n public async deleteMany(ids: string[]): Promise<void> {\n const res = await request<ApiResponse<null>>(this.#lumi, this.uri('/batch-by-ids'), {\n method: 'DELETE',\n params: { ids },\n })\n if (res.code !== 200)\n throw new Error(res.message)\n }\n\n private uri(suffix = ''): string {\n return `/lm/${this.#lumi.config.projectId}/${this.entityName}/documents${suffix}`\n }\n}\n","import type { LumiClient } from '@/core/lumi-client'\nimport { EntityClient } from '@/core/entity-client'\n\nexport class EntitiesClient {\n readonly #lumi: LumiClient\n [key: string]: EntityClient\n\n constructor(lumi: LumiClient) {\n this.#lumi = lumi\n return new Proxy(this, {\n get(target: EntitiesClient, p: string) {\n if (!(p in target))\n target[p] = new EntityClient(target.#lumi, p)\n return target[p]\n },\n }) as this\n }\n}\n","import type { FetchOptions } from 'ofetch'\nimport type { LumiClient } from '@/core/lumi-client'\nimport { ofetch } from 'ofetch'\n\nexport class FunctionsClient {\n readonly #lumi: LumiClient\n\n constructor(lumi: LumiClient) {\n this.#lumi = lumi\n }\n\n public invoke(functionName: string, options: FetchOptions<'json'> = {}): Promise<any> {\n if (this.#lumi.auth.accessToken) {\n options.headers = {\n Authorization: `Bearer ${this.#lumi.auth.accessToken}`,\n ...options.headers,\n }\n }\n\n return ofetch(`/v1/functions/${this.#lumi.config.projectId}/${functionName}`, {\n baseURL: this.#lumi.config.apiBaseUrl,\n ...options,\n })\n }\n}\n","import type { LumiClient } from '@/core/lumi-client'\nimport type { ApiResponse, StreamChunk } from '@/lib/request'\nimport { LumiError } from '@/lib/error'\nimport { requestStream } from '@/lib/request'\nimport { request } from '@/lib/request'\nimport { delay, getEnv, isServer } from '@/utils/common'\n\nexport interface MessageMedia {\n mimeType: string\n url: string\n}\n\nexport interface Message {\n role: 'system' | 'user' | 'assistant'\n content: string\n medias?: MessageMedia[]\n}\n\nexport interface GenerateTextResult {\n chatId: string\n content: string\n}\n\nexport interface GenerateImageResult {\n chatId: string\n messageId: string\n generationStatus: 'PROCESSING' | 'SUCCESS' | 'FAILED'\n content?: string\n imageURL?: string\n}\n\nexport class AITool {\n readonly #lumi: LumiClient\n\n constructor(lumi: LumiClient) {\n this.#lumi = lumi\n }\n\n /** AI 生成文本 */\n public async generateText({\n model = 'gemini-2.5-flash',\n messages,\n }: {\n model?: string\n messages: Message[]\n }): Promise<GenerateTextResult> {\n const lumiApiKey = this.checkLumiApiKey()\n\n const res = await request<ApiResponse<GenerateTextResult>>(this.#lumi, `/lm/${this.#lumi.config.projectId}/ai/chat/completions`, {\n method: 'POST',\n body: {\n lumiApiKey,\n modelName: model,\n chatMessages: messages,\n },\n })\n\n if (res.code !== 200)\n throw new LumiError(res.code, res.message)\n\n return res.data\n }\n\n /** AI 生成文本(流式输出) */\n public async generateTextStream({\n model = 'gemini-2.5-flash',\n messages,\n }: {\n model?: string\n messages: Message[]\n }): Promise<ReadableStream<StreamChunk<GenerateTextResult>>> {\n const lumiApiKey = this.checkLumiApiKey()\n\n return requestStream<GenerateTextResult>(this.#lumi, `/lm/${this.#lumi.config.projectId}/ai/chat/stream`, {\n method: 'POST',\n body: {\n lumiApiKey,\n modelName: model,\n chatMessages: messages,\n },\n })\n }\n\n /** AI 生成图片 */\n public async generateImage({\n model = 'gemini-2.5-flash-image',\n messages,\n }: {\n model?: string\n messages: Message[]\n }): Promise<GenerateImageResult> {\n const lumiApiKey = this.checkLumiApiKey()\n\n const { code, message, data } = await request<ApiResponse<GenerateImageResult>>(this.#lumi, `/lm/${this.#lumi.config.projectId}/ai/image/task`, {\n method: 'POST',\n body: {\n lumiApiKey,\n modelName: model,\n chatMessages: messages,\n },\n })\n if (code !== 200)\n throw new LumiError(code, message)\n\n let result = data\n let delayMs = 1000\n\n // 轮询生成结果\n while (result.generationStatus === 'PROCESSING') {\n await delay(delayMs)\n if (delayMs < 5000)\n delayMs += 500\n\n const { code, message, data } = await request<ApiResponse<GenerateImageResult>>(this.#lumi, `/lm/${this.#lumi.config.projectId}/ai/image/get`, {\n method: 'POST',\n body: {\n lumiApiKey,\n messageId: result.messageId,\n },\n })\n if (code !== 200)\n throw new LumiError(code, message)\n result = data\n }\n\n return result\n }\n\n private checkLumiApiKey(): string {\n if (!isServer)\n throw new LumiError(400, 'lumi.tools.ai is only available on the server-side')\n\n const lumiApiKey = getEnv('LUMI_API_KEY')\n if (!lumiApiKey)\n throw new LumiError(400, 'LUMI_API_KEY is required')\n\n return lumiApiKey\n }\n}\n","import type { LumiClient } from '@/core/lumi-client'\nimport type { ApiResponse } from '@/lib/request'\nimport { LumiError } from '@/lib/error'\nimport { request } from '@/lib/request'\n\nexport class EmailTool {\n readonly #lumi: LumiClient\n\n constructor(lumi: LumiClient) {\n this.#lumi = lumi\n }\n\n /** 发送邮件 */\n public async send({ to, subject, fromName, html, text = '', replyTo, scheduledAt }: {\n to: string | string[]\n subject: string\n fromName?: string\n html?: string\n text?: string\n replyTo?: string | string[]\n scheduledAt?: string\n }): Promise<void> {\n if (!to || !subject || (!html && !text))\n throw new Error('Failed to send email: Missing required parameters.')\n\n if (typeof to === 'string')\n to = [to]\n if (typeof replyTo === 'string')\n replyTo = [replyTo]\n\n const res = await request<ApiResponse<null>>(this.#lumi, `/lm/${this.#lumi.config.projectId}/email/send`, {\n method: 'POST',\n body: { to, subject, fromName, html, text, replyTo, scheduledAt },\n })\n\n if (res.code !== 200)\n throw new LumiError(res.code, res.message)\n }\n}\n","import type { LumiClient } from '@/core/lumi-client'\nimport type { ApiResponse } from '@/lib/request'\nimport { LumiError } from '@/lib/error'\nimport { request } from '@/lib/request'\n\nexport interface UploadItem {\n fileName: string\n fileUrl?: string\n uploadError?: string\n}\n\nexport class FileTool {\n readonly #lumi: LumiClient\n\n constructor(lumi: LumiClient) {\n this.#lumi = lumi\n }\n\n /** 上传文件 */\n public async upload(files: File[]): Promise<UploadItem[]> {\n const formData = new FormData()\n files.forEach((file) => {\n formData.append('files', file)\n })\n\n const res = await request<ApiResponse<UploadItem[]>>(this.#lumi, `/lm/${this.#lumi.config.projectId}/file/batch`, {\n method: 'POST',\n body: formData,\n })\n\n if (res.code !== 200)\n throw new LumiError(res.code, res.message)\n\n return res.data\n }\n\n /** 批量删除文件 */\n public async delete(fileUrls: string[]): Promise<void> {\n const res = await request<ApiResponse<null>>(this.#lumi, `/lm/${this.#lumi.config.projectId}/file/batch`, {\n method: 'DELETE',\n body: { fileUrls },\n })\n\n if (res.code !== 200)\n throw new LumiError(res.code, res.message)\n }\n}\n","import type { LumiClient } from '@/core/lumi-client'\nimport { AITool, EmailTool, FileTool } from '@/tools'\n\nexport class ToolsClient {\n readonly #lumi: LumiClient\n\n public email: EmailTool\n public file: FileTool\n public ai: AITool\n\n constructor(lumi: LumiClient) {\n this.#lumi = lumi\n this.email = new EmailTool(lumi)\n this.file = new FileTool(lumi)\n this.ai = new AITool(lumi)\n }\n}\n","import { LumiAuthClient } from '@/core/auth-client'\nimport { EntitiesClient } from '@/core/entities-client'\nimport { FunctionsClient } from '@/core/functions-client'\nimport { ToolsClient } from '@/core/tools-client'\n\nexport interface LumiClientConfig {\n projectId: string\n apiBaseUrl: string\n authOrigin: string\n authorization?: string\n}\n\nexport class LumiClient {\n public config: LumiClientConfig\n\n public auth: LumiAuthClient\n public entities: EntitiesClient\n public tools: ToolsClient\n public functions: FunctionsClient\n\n constructor(config: LumiClientConfig) {\n this.config = config\n this.auth = new LumiAuthClient(this)\n this.entities = new EntitiesClient(this)\n this.tools = new ToolsClient(this)\n this.functions = new FunctionsClient(this)\n }\n}\n\nexport function createClient(config: LumiClientConfig): LumiClient {\n return new LumiClient(config)\n}\n"],"mappings":"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","names":["index_exports","__export","AITool","EmailTool","EntitiesClient","EntityClient","FileTool","FunctionsClient","LumiAuthClient","LumiClient","ToolsClient","createClient","__toCommonJS","import_uuid","import_enc_base64","import_enc_hex","import_hmac_sha256","import_sha256","import_stream","import_object_hash","import_ofetch","LumiError","code","message","getIcon","_a","_b","getTitle","isServer","delay","ms","resolve","getEnv","key","SECRET_KEY","rfc3986Encode","str","c","addSignature","options","timestamp","nonce","queryParams","__spreadValues","canonicalQueryString","key","headersToSign","canonicalHeaders","payload","hashedPayload","SHA256","Hex","canonicalRequest","signature","Base64","HmacSHA256","headers","value","requestMap","checkRateLimit","uri","_a","clonedOptions","requestHash","hash","now","requestQueue","time","LumiError","beforeRequest","lumi","stream","request","isAuthenticated","__spreadProps","response","isServer","requestStream","_0","_1","__async","chunk","controller","data","e","setStorage","key","value","storage","oldValue","newValue","getStorage","_e","_lumi","_popupName","_user","LumiAuthClient","lumi","__privateAdd","__privateSet","isServer","authorization","__privateGet","getStorage","accessToken","setStorage","user","width","height","left","top","popup","cleanup","resolve","reject","timer","handleGlobalClick","event","handleMessage","data","origin","source","getIcon","getTitle","__async","res","request","callback","handleStorageChange","_lumi","EntityClient","lumi","entityName","__privateAdd","__privateSet","__async","filter","sort","limit","skip","res","request","__privateGet","id","data","ids","suffix","_lumi","EntitiesClient","lumi","__privateAdd","__privateSet","target","p","EntityClient","__privateGet","import_ofetch","_lumi","FunctionsClient","lumi","__privateAdd","__privateSet","functionName","options","__privateGet","__spreadValues","_lumi","AITool","lumi","__privateAdd","__privateSet","_0","__async","model","messages","lumiApiKey","res","request","__privateGet","LumiError","requestStream","code","message","data","result","delayMs","delay","isServer","getEnv","_lumi","EmailTool","lumi","__privateAdd","__privateSet","_0","__async","to","subject","fromName","html","text","replyTo","scheduledAt","res","request","__privateGet","LumiError","_lumi","FileTool","lumi","__privateAdd","__privateSet","files","__async","formData","file","res","request","__privateGet","LumiError","fileUrls","_lumi","ToolsClient","lumi","__privateAdd","__privateSet","EmailTool","FileTool","AITool","LumiClient","config","LumiAuthClient","EntitiesClient","ToolsClient","FunctionsClient","createClient"]}
package/dist/index.mjs CHANGED
@@ -1,4 +1,4 @@
1
- var X=Object.defineProperty,W=Object.defineProperties;var ee=Object.getOwnPropertyDescriptors;var _=Object.getOwnPropertySymbols;var te=Object.prototype.hasOwnProperty,ie=Object.prototype.propertyIsEnumerable;var K=i=>{throw TypeError(i)};var z=(i,e,t)=>e in i?X(i,e,{enumerable:!0,configurable:!0,writable:!0,value:t}):i[e]=t,S=(i,e)=>{for(var t in e||(e={}))te.call(e,t)&&z(i,t,e[t]);if(_)for(var t of _(e))ie.call(e,t)&&z(i,t,e[t]);return i},N=(i,e)=>W(i,ee(e));var Q=(i,e,t)=>e.has(i)||K("Cannot "+t);var n=(i,e,t)=>(Q(i,e,"read from private field"),t?t.call(i):e.get(i)),a=(i,e,t)=>e.has(i)?K("Cannot add the same private member more than once"):e instanceof WeakSet?e.add(i):e.set(i,t),u=(i,e,t,r)=>(Q(i,e,"write to private field"),r?r.call(i,t):e.set(i,t),t);var m=(i,e,t)=>new Promise((r,s)=>{var o=h=>{try{f(t.next(h))}catch(y){s(y)}},l=h=>{try{f(t.throw(h))}catch(y){s(y)}},f=h=>h.done?r(h.value):Promise.resolve(h.value).then(o,l);f((t=t.apply(i,e)).next())});import{v4 as pe}from"uuid";import re from"crypto-js/enc-base64";import ne from"crypto-js/enc-hex";import oe from"crypto-js/hmac-sha256";import se from"crypto-js/sha256";import ae from"object-hash";import{ofetch as ce}from"ofetch";var b=class extends Error{constructor(t,r){super(r);this.name="LumiError";this.code=t}};function J(){var i,e;return(e=(i=document.querySelector('link[rel="icon"]'))==null?void 0:i.href)!=null?e:null}function G(){var i;return(i=document.title)!=null?i:null}var w=typeof window=="undefined";var le="6QrJZ7pFCmBZAeIJF7IArvkCz+EtzA0RVcpHkiQIsQyhs7QtCS9P+CueZdHfB2OtJcgX3BbqY9pfpWeAVTqCwQ==";function V(i){return encodeURIComponent(i).replace(/[!'()*]/g,e=>`%${e.charCodeAt(0).toString(16).toUpperCase()}`)}function ue(i){return e=>{let{options:t}=e,r=Math.floor(Date.now()/1e3).toString(),s=Math.random().toString(36).substring(2,15),o=S({},t.query),l=Object.keys(o).sort().map(E=>`${V(E)}=${V(String(o[E]))}`).join("&"),f={"x-timestamp":r,"x-nonce":s},h=Object.keys(f).sort().map(E=>`${E}:${f[E]}`).join(`
2
- `),y=t.body&&!(t.body instanceof FormData)?JSON.stringify(t.body):"",R=se(y).toString(ne),v=[l,h,R].join(`
3
- `),d=re.stringify(oe(v,i)),I=new Headers(t.headers);Object.entries(f).forEach(([E,Z])=>{I.set(E,Z)}),I.set("X-Sign",d),t.headers=I}}var Y=new Map;function me(i,e){var o;e.body instanceof FormData&&(e=N(S({},e),{body:Array.from(e.body.entries())}));let t=ae([i,e]),r=Date.now(),s=((o=Y.get(t))==null?void 0:o.filter(l=>r-l<1e3))||[];if(s.length>=4)throw new b(429,"Too many requests");s.push(r),Y.set(t,s)}function c(i,e,t={}){i.auth.accessToken&&(t.headers=S({Authorization:`Bearer ${i.auth.accessToken}`},t.headers)),me(e,t);let r=i.auth.isAuthenticated;return ce(e,N(S({baseURL:i.config.apiBaseUrl},t),{onRequest:ue(le),onResponse:({response:s})=>{var o;!w&&r&&((o=s._data)==null?void 0:o.code)===2100&&i.auth.signOut()}}))}function F(i,e,t=localStorage){let r=t.getItem(i),s=e?JSON.stringify(e):null;s?t.setItem(i,s):t.removeItem(i),window.dispatchEvent(new StorageEvent("storage",{key:i,oldValue:r,newValue:s,storageArea:t}))}function B(i,e=localStorage){let t=e.getItem(i);try{return t?JSON.parse(t):null}catch(r){return null}}var g,O,L,P=class{constructor(e){a(this,g);a(this,O,`lumi-auth-${pe()}`);a(this,L,null);u(this,g,e),Promise.resolve().then(()=>{!w&&this.isAuthenticated&&this.refreshUser()})}get accessToken(){if(w){let e=n(this,g).config.authorization;return e?e.replace("Bearer ",""):null}return B("lumi-access-token")}set accessToken(e){if(w){n(this,g).config.authorization=e?`Bearer ${e}`:void 0;return}F("lumi-access-token",e)}get user(){return w?n(this,L):B("lumi-user")}set user(e){if(w){u(this,L,e);return}F("lumi-user",e)}get isAuthenticated(){return!!this.accessToken}signIn(){if(w)throw new Error("auth.signIn() can only be called on the client side");let e=800,t=600,r=(window.screen.width-e)/2,s=(window.screen.height-t)/2,o=window.open(n(this,g).config.authOrigin,n(this,O),`width=${e},height=${t},left=${r},top=${s}`),l;return new Promise((f,h)=>{if(!o)return h(new Error("Open auth window failed"));let y=setInterval(()=>{o.closed&&h(new Error("Auth window closed"))},1e3),R=d=>{o.closed||(o.focus(),d.stopPropagation(),d.preventDefault())},v=({data:d,origin:I,source:E})=>{if(!(I!==n(this,g).config.authOrigin||E!==o))switch(d==null?void 0:d.type){case"lumi-ready":{o.postMessage({type:"lumi-init",data:{projectId:n(this,g).config.projectId,icon:J(),title:G()}},n(this,g).config.authOrigin);break}case"lumi-sign-in":{if(d.data.projectId!==n(this,g).config.projectId)break;o.close(),window.focus(),this.accessToken=d.data.accessToken,this.user=d.data.user,f(d.data);break}}};window.addEventListener("message",v),document.addEventListener("click",R,!0),l=()=>{clearInterval(y),window.removeEventListener("message",v),document.removeEventListener("click",R,!0)}}).finally(()=>l==null?void 0:l())}signOut(){if(w)throw new Error("auth.signOut() can only be called on the client side");this.accessToken=null,this.user=null}refreshUser(){return m(this,null,function*(){let e=yield c(n(this,g),"/lm/user/info",{method:"POST"});if(e.code!==200)throw new Error(e.message);return this.user=e.data,e.data})}onAuthChange(e){if(w)throw new Error("auth.onAuthChange() can only be called on the client side");let t=r=>{(r.key==="lumi-access-token"||r.key==="lumi-user"||r.key===null)&&e({isAuthenticated:this.isAuthenticated,user:this.user})};return window.addEventListener("storage",t),()=>{window.removeEventListener("storage",t)}}};g=new WeakMap,O=new WeakMap,L=new WeakMap;var p,D=class{constructor(e,t){a(this,p);u(this,p,e),this.entityName=t}list(){return m(this,arguments,function*({filter:e,sort:t,limit:r,skip:s}={}){if(r){let o=yield c(n(this,p),this.uri("/find"),{method:"POST",body:{filter:e,sort:t,limit:r,skip:s}});if(o.code!==200)throw new Error(o.message);return o.data}else{let o=yield c(n(this,p),this.uri("/list"),{method:"POST",body:{filter:e,sort:t}});if(o.code!==200)throw new Error(o.message);return{total:o.data.length,list:o.data}}})}get(e){return m(this,null,function*(){let t=yield c(n(this,p),this.uri(`/${e}`),{method:"GET"});if(t.code!==200)throw new Error(t.message);return t.data})}create(e){return m(this,null,function*(){let t=yield c(n(this,p),this.uri(),{method:"POST",body:e});if(t.code!==200)throw new Error(t.message);return t.data})}createMany(e){return m(this,null,function*(){let t=yield c(n(this,p),this.uri("/batch"),{method:"POST",body:e});if(t.code!==200)throw new Error(t.message);return t.data})}update(e,t){return m(this,null,function*(){let r=yield c(n(this,p),this.uri(),{method:"PUT",body:{filter:{_id:e},update:t}});if(r.code!==200)throw new Error(r.message);return r.data})}delete(e){return m(this,null,function*(){let t=yield c(n(this,p),this.uri(`/${e}`),{method:"DELETE"});if(t.code!==200)throw new Error(t.message)})}deleteMany(e){return m(this,null,function*(){let t=yield c(n(this,p),this.uri("/batch-by-ids"),{method:"DELETE",params:{ids:e}});if(t.code!==200)throw new Error(t.message)})}uri(e=""){return`/lm/${n(this,p).config.projectId}/${this.entityName}/documents${e}`}};p=new WeakMap;var x,M=class{constructor(e){a(this,x);return u(this,x,e),new Proxy(this,{get(t,r){return r in t||(t[r]=new D(n(t,x),r)),t[r]}})}};x=new WeakMap;import{ofetch as he}from"ofetch";var C,U=class{constructor(e){a(this,C);u(this,C,e)}invoke(e,t={}){return n(this,C).auth.accessToken&&(t.headers=S({Authorization:`Bearer ${n(this,C).auth.accessToken}`},t.headers)),he(`/v1/functions/${n(this,C).config.projectId}/${e}`,S({baseURL:n(this,C).config.apiBaseUrl},t))}};C=new WeakMap;var A,k=class{constructor(e){a(this,A);u(this,A,e)}send(h){return m(this,arguments,function*({to:e,subject:t,fromName:r,html:s,text:o="",replyTo:l,scheduledAt:f}){if(!e||!t||!s&&!o)throw new Error("Failed to send email: Missing required parameters.");typeof e=="string"&&(e=[e]),typeof l=="string"&&(l=[l]);let y=yield c(n(this,A),`/lm/${n(this,A).config.projectId}/email/send`,{method:"POST",body:{to:e,subject:t,fromName:r,html:s,text:o,replyTo:l,scheduledAt:f}});if(y.code!==200)throw new b(y.code,y.message)})}};A=new WeakMap;var T,$=class{constructor(e){a(this,T);u(this,T,e)}upload(e){return m(this,null,function*(){let t=new FormData;e.forEach(s=>{t.append("files",s)});let r=yield c(n(this,T),`/lm/${n(this,T).config.projectId}/file/batch`,{method:"POST",body:t});if(r.code!==200)throw new b(r.code,r.message);return r.data})}delete(e){return m(this,null,function*(){let t=yield c(n(this,T),`/lm/${n(this,T).config.projectId}/file/batch`,{method:"DELETE",body:{fileUrls:e}});if(t.code!==200)throw new b(t.code,t.message)})}};T=new WeakMap;var j,q=class{constructor(e){a(this,j);u(this,j,e),this.email=new k(e),this.file=new $(e)}};j=new WeakMap;var H=class{constructor(e){this.config=e,this.auth=new P(this),this.entities=new M(this),this.tools=new q(this),this.functions=new U(this)}};function nt(i){return new H(i)}export{k as EmailTool,M as EntitiesClient,D as EntityClient,$ as FileTool,U as FunctionsClient,P as LumiAuthClient,H as LumiClient,q as ToolsClient,nt as createClient};
1
+ var se=Object.defineProperty,ne=Object.defineProperties;var oe=Object.getOwnPropertyDescriptors;var _=Object.getOwnPropertySymbols;var ae=Object.prototype.hasOwnProperty,ce=Object.prototype.propertyIsEnumerable;var z=i=>{throw TypeError(i)};var H=(i,e,t)=>e in i?se(i,e,{enumerable:!0,configurable:!0,writable:!0,value:t}):i[e]=t,T=(i,e)=>{for(var t in e||(e={}))ae.call(e,t)&&H(i,t,e[t]);if(_)for(var t of _(e))ce.call(e,t)&&H(i,t,e[t]);return i},F=(i,e)=>ne(i,oe(e));var J=(i,e,t)=>e.has(i)||z("Cannot "+t);var r=(i,e,t)=>(J(i,e,"read from private field"),t?t.call(i):e.get(i)),p=(i,e,t)=>e.has(i)?z("Cannot add the same private member more than once"):e instanceof WeakSet?e.add(i):e.set(i,t),d=(i,e,t,s)=>(J(i,e,"write to private field"),s?s.call(i,t):e.set(i,t),t);var c=(i,e,t)=>new Promise((s,o)=>{var n=m=>{try{h(t.next(m))}catch(g){o(g)}},a=m=>{try{h(t.throw(m))}catch(g){o(g)}},h=m=>m.done?s(m.value):Promise.resolve(m.value).then(n,a);h((t=t.apply(i,e)).next())});import{v4 as we}from"uuid";import me from"crypto-js/enc-base64";import le from"crypto-js/enc-hex";import ue from"crypto-js/hmac-sha256";import pe from"crypto-js/sha256";import{EventSourceParserStream as de}from"eventsource-parser/stream";import he from"object-hash";import{ofetch as ee}from"ofetch";var f=class extends Error{constructor(t,s){super(s);this.name="LumiError";this.code=t}};function Q(){var i,e;return(e=(i=document.querySelector('link[rel="icon"]'))==null?void 0:i.href)!=null?e:null}function Y(){var i;return(i=document.title)!=null?i:null}var y=typeof window=="undefined";function V(i){return new Promise(e=>setTimeout(e,i))}function X(i){var e;return typeof process!="undefined"&&(e=process==null?void 0:process.env[i])!=null?e:null}var ge="6QrJZ7pFCmBZAeIJF7IArvkCz+EtzA0RVcpHkiQIsQyhs7QtCS9P+CueZdHfB2OtJcgX3BbqY9pfpWeAVTqCwQ==";function Z(i){return encodeURIComponent(i).replace(/[!'()*]/g,e=>`%${e.charCodeAt(0).toString(16).toUpperCase()}`)}function fe(i){let e=Math.floor(Date.now()/1e3).toString(),t=Math.random().toString(36).substring(2,15),s=T({},i.query),o=Object.keys(s).sort().map(l=>`${Z(l)}=${Z(String(s[l]))}`).join("&"),n={"x-timestamp":e,"x-nonce":t},a=Object.keys(n).sort().map(l=>`${l}:${n[l]}`).join(`
2
+ `),h=i.body&&!(i.body instanceof FormData)?JSON.stringify(i.body):"",m=pe(h).toString(le),g=[o,a,m].join(`
3
+ `),I=me.stringify(ue(g,ge)),E=new Headers(i.headers);Object.entries(n).forEach(([l,j])=>{E.set(l,j)}),E.set("X-Sign",I),i.headers=E}var W=new Map;function ye(i,e){var a;let t=T({},e);e.body instanceof FormData&&(t.body=Array.from(e.body.entries())),t.headers=void 0;let s=he([i,t]),o=Date.now(),n=((a=W.get(s))==null?void 0:a.filter(h=>o-h<1e3))||[];if(n.length>=4)throw new f(429,"Too many requests");n.push(o),W.set(s,n)}function te(i,e,t,s=!1){t.headers=new Headers(t.headers),i.auth.accessToken&&t.headers.set("Authorization",`Bearer ${i.auth.accessToken}`),s&&(t.headers.get("Accept")||t.headers.set("Accept","text/event-stream"),t.headers.get("Cache-Control")||t.headers.set("Cache-Control","no-cache"),t.headers.get("X-Accel-Buffering")||t.headers.set("X-Accel-Buffering","no")),ye(e,t),fe(t)}function u(i,e,t={}){te(i,e,t);let s=i.auth.isAuthenticated;return ee(e,F(T({baseURL:i.config.apiBaseUrl},t),{onResponse:({response:o})=>{var n;!y&&s&&((n=o._data)==null?void 0:n.code)===2100&&i.auth.signOut()}}))}function ie(s,o){return c(this,arguments,function*(i,e,t={}){return te(i,e,t,!0),(yield ee(e,F(T({baseURL:i.config.apiBaseUrl},t),{responseType:"stream"}))).pipeThrough(new TextDecoderStream).pipeThrough(new de).pipeThrough(new TransformStream({transform:(a,h)=>{try{let m=JSON.parse(a.data);h.enqueue({event:a.event,data:m})}catch(m){}}}))})}function K(i,e,t=localStorage){let s=t.getItem(i),o=e?JSON.stringify(e):null;o?t.setItem(i,o):t.removeItem(i),window.dispatchEvent(new StorageEvent("storage",{key:i,oldValue:s,newValue:o,storageArea:t}))}function G(i,e=localStorage){let t=e.getItem(i);try{return t?JSON.parse(t):null}catch(s){return null}}var b,P,x,v=class{constructor(e){p(this,b);p(this,P,`lumi-auth-${we()}`);p(this,x,null);d(this,b,e),Promise.resolve().then(()=>{!y&&this.isAuthenticated&&this.refreshUser()})}get accessToken(){if(y){let e=r(this,b).config.authorization;return e?e.replace("Bearer ",""):null}return G("lumi-access-token")}set accessToken(e){if(y){r(this,b).config.authorization=e?`Bearer ${e}`:void 0;return}K("lumi-access-token",e)}get user(){return y?r(this,x):G("lumi-user")}set user(e){if(y){d(this,x,e);return}K("lumi-user",e)}get isAuthenticated(){return!!this.accessToken}signIn(){if(y)throw new Error("auth.signIn() can only be called on the client side");let e=800,t=600,s=(window.screen.width-e)/2,o=(window.screen.height-t)/2,n=window.open(r(this,b).config.authOrigin,r(this,P),`width=${e},height=${t},left=${s},top=${o}`),a;return new Promise((h,m)=>{if(!n)return m(new Error("Open auth window failed"));let g=setInterval(()=>{n.closed&&m(new Error("Auth window closed"))},1e3),I=l=>{n.closed||(n.focus(),l.stopPropagation(),l.preventDefault())},E=({data:l,origin:j,source:re})=>{if(!(j!==r(this,b).config.authOrigin||re!==n))switch(l==null?void 0:l.type){case"lumi-ready":{n.postMessage({type:"lumi-init",data:{projectId:r(this,b).config.projectId,icon:Q(),title:Y()}},r(this,b).config.authOrigin);break}case"lumi-sign-in":{if(l.data.projectId!==r(this,b).config.projectId)break;n.close(),window.focus(),this.accessToken=l.data.accessToken,this.user=l.data.user,h(l.data);break}}};window.addEventListener("message",E),document.addEventListener("click",I,!0),a=()=>{clearInterval(g),window.removeEventListener("message",E),document.removeEventListener("click",I,!0)}}).finally(()=>a==null?void 0:a())}signOut(){if(y)throw new Error("auth.signOut() can only be called on the client side");this.accessToken=null,this.user=null}refreshUser(){return c(this,null,function*(){let e=yield u(r(this,b),"/lm/user/info",{method:"POST"});if(e.code!==200)throw new Error(e.message);return this.user=e.data,e.data})}onAuthChange(e){if(y)throw new Error("auth.onAuthChange() can only be called on the client side");let t=s=>{(s.key==="lumi-access-token"||s.key==="lumi-user"||s.key===null)&&e({isAuthenticated:this.isAuthenticated,user:this.user})};return window.addEventListener("storage",t),()=>{window.removeEventListener("storage",t)}}};b=new WeakMap,P=new WeakMap,x=new WeakMap;var w,M=class{constructor(e,t){p(this,w);d(this,w,e),this.entityName=t}list(){return c(this,arguments,function*({filter:e,sort:t,limit:s,skip:o}={}){if(s){let n=yield u(r(this,w),this.uri("/find"),{method:"POST",body:{filter:e,sort:t,limit:s,skip:o}});if(n.code!==200)throw new Error(n.message);return n.data}else{let n=yield u(r(this,w),this.uri("/list"),{method:"POST",body:{filter:e,sort:t}});if(n.code!==200)throw new Error(n.message);return{total:n.data.length,list:n.data}}})}get(e){return c(this,null,function*(){let t=yield u(r(this,w),this.uri(`/${e}`),{method:"GET"});if(t.code!==200)throw new Error(t.message);return t.data})}create(e){return c(this,null,function*(){let t=yield u(r(this,w),this.uri(),{method:"POST",body:e});if(t.code!==200)throw new Error(t.message);return t.data})}createMany(e){return c(this,null,function*(){let t=yield u(r(this,w),this.uri("/batch"),{method:"POST",body:e});if(t.code!==200)throw new Error(t.message);return t.data})}update(e,t){return c(this,null,function*(){let s=yield u(r(this,w),this.uri(),{method:"PUT",body:{filter:{_id:e},update:t}});if(s.code!==200)throw new Error(s.message);return s.data})}delete(e){return c(this,null,function*(){let t=yield u(r(this,w),this.uri(`/${e}`),{method:"DELETE"});if(t.code!==200)throw new Error(t.message)})}deleteMany(e){return c(this,null,function*(){let t=yield u(r(this,w),this.uri("/batch-by-ids"),{method:"DELETE",params:{ids:e}});if(t.code!==200)throw new Error(t.message)})}uri(e=""){return`/lm/${r(this,w).config.projectId}/${this.entityName}/documents${e}`}};w=new WeakMap;var L,O=class{constructor(e){p(this,L);return d(this,L,e),new Proxy(this,{get(t,s){return s in t||(t[s]=new M(r(t,L),s)),t[s]}})}};L=new WeakMap;import{ofetch as be}from"ofetch";var C,U=class{constructor(e){p(this,C);d(this,C,e)}invoke(e,t={}){return r(this,C).auth.accessToken&&(t.headers=T({Authorization:`Bearer ${r(this,C).auth.accessToken}`},t.headers)),be(`/v1/functions/${r(this,C).config.projectId}/${e}`,T({baseURL:r(this,C).config.apiBaseUrl},t))}};C=new WeakMap;var S,k=class{constructor(e){p(this,S);d(this,S,e)}generateText(s){return c(this,arguments,function*({model:e="gemini-2.5-flash",messages:t}){let o=this.checkLumiApiKey(),n=yield u(r(this,S),`/lm/${r(this,S).config.projectId}/ai/chat/completions`,{method:"POST",body:{lumiApiKey:o,modelName:e,chatMessages:t}});if(n.code!==200)throw new f(n.code,n.message);return n.data})}generateTextStream(s){return c(this,arguments,function*({model:e="gemini-2.5-flash",messages:t}){let o=this.checkLumiApiKey();return ie(r(this,S),`/lm/${r(this,S).config.projectId}/ai/chat/stream`,{method:"POST",body:{lumiApiKey:o,modelName:e,chatMessages:t}})})}generateImage(s){return c(this,arguments,function*({model:e="gemini-2.5-flash-image",messages:t}){let o=this.checkLumiApiKey(),{code:n,message:a,data:h}=yield u(r(this,S),`/lm/${r(this,S).config.projectId}/ai/image/task`,{method:"POST",body:{lumiApiKey:o,modelName:e,chatMessages:t}});if(n!==200)throw new f(n,a);let m=h,g=1e3;for(;m.generationStatus==="PROCESSING";){yield V(g),g<5e3&&(g+=500);let{code:I,message:E,data:l}=yield u(r(this,S),`/lm/${r(this,S).config.projectId}/ai/image/get`,{method:"POST",body:{lumiApiKey:o,messageId:m.messageId}});if(I!==200)throw new f(I,E);m=l}return m})}checkLumiApiKey(){if(!y)throw new f(400,"lumi.tools.ai is only available on the server-side");let e=X("LUMI_API_KEY");if(!e)throw new f(400,"LUMI_API_KEY is required");return e}};S=new WeakMap;var R,D=class{constructor(e){p(this,R);d(this,R,e)}send(m){return c(this,arguments,function*({to:e,subject:t,fromName:s,html:o,text:n="",replyTo:a,scheduledAt:h}){if(!e||!t||!o&&!n)throw new Error("Failed to send email: Missing required parameters.");typeof e=="string"&&(e=[e]),typeof a=="string"&&(a=[a]);let g=yield u(r(this,R),`/lm/${r(this,R).config.projectId}/email/send`,{method:"POST",body:{to:e,subject:t,fromName:s,html:o,text:n,replyTo:a,scheduledAt:h}});if(g.code!==200)throw new f(g.code,g.message)})}};R=new WeakMap;var A,q=class{constructor(e){p(this,A);d(this,A,e)}upload(e){return c(this,null,function*(){let t=new FormData;e.forEach(o=>{t.append("files",o)});let s=yield u(r(this,A),`/lm/${r(this,A).config.projectId}/file/batch`,{method:"POST",body:t});if(s.code!==200)throw new f(s.code,s.message);return s.data})}delete(e){return c(this,null,function*(){let t=yield u(r(this,A),`/lm/${r(this,A).config.projectId}/file/batch`,{method:"DELETE",body:{fileUrls:e}});if(t.code!==200)throw new f(t.code,t.message)})}};A=new WeakMap;var N,$=class{constructor(e){p(this,N);d(this,N,e),this.email=new D(e),this.file=new q(e),this.ai=new k(e)}};N=new WeakMap;var B=class{constructor(e){this.config=e,this.auth=new v(this),this.entities=new O(this),this.tools=new $(this),this.functions=new U(this)}};function yt(i){return new B(i)}export{k as AITool,D as EmailTool,O as EntitiesClient,M as EntityClient,q as FileTool,U as FunctionsClient,v as LumiAuthClient,B as LumiClient,$ as ToolsClient,yt as createClient};
4
4
  //# sourceMappingURL=index.mjs.map
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/core/auth-client.ts","../src/lib/request.ts","../src/lib/error.ts","../src/utils/common.ts","../src/utils/storage.ts","../src/core/entity-client.ts","../src/core/entities-client.ts","../src/core/functions-client.ts","../src/tools/email-tool.ts","../src/tools/file-tool.ts","../src/core/tools-client.ts","../src/core/lumi-client.ts"],"sourcesContent":["import type { LumiClient } from '@/core/lumi-client'\nimport type { ApiResponse } from '@/lib/request'\nimport { v4 } from 'uuid'\nimport { MessageType, StorageKey } from '@/constants'\nimport { request } from '@/lib/request'\nimport { getIcon, getTitle, isServer } from '@/utils/common'\nimport { getStorage, setStorage } from '@/utils/storage'\n\nexport interface User {\n userId: string\n email: string\n userName: string\n userRole: 'ADMIN' | 'USER'\n createdTime: string\n}\n\nexport interface MessageSignInData {\n projectId: string\n accessToken: string\n user: User\n}\n\nexport type MessageDataReceive = {\n type: MessageType.READY\n} | {\n type: MessageType.SIGN_IN\n data: MessageSignInData\n}\n\nexport interface MessageInitData {\n projectId: string\n icon: string | null\n title: string | null\n}\n\nexport interface MessageDataSend {\n type: MessageType.INIT\n data: MessageInitData\n}\n\nexport class LumiAuthClient {\n readonly #lumi: LumiClient\n readonly #popupName: string = `lumi-auth-${v4()}`\n\n constructor(lumi: LumiClient) {\n this.#lumi = lumi\n Promise.resolve().then(() => {\n if (!isServer && this.isAuthenticated)\n void this.refreshUser()\n })\n }\n\n /** 访问令牌 */\n public get accessToken(): string | null {\n if (isServer) {\n const authorization = this.#lumi.config.authorization\n return authorization ? authorization.replace('Bearer ', '') : null\n }\n return getStorage<string>(StorageKey.ACCESS_TOKEN)\n }\n\n public set accessToken(accessToken: string | null) {\n if (isServer) {\n this.#lumi.config.authorization = accessToken ? `Bearer ${accessToken}` : undefined\n return\n }\n setStorage(StorageKey.ACCESS_TOKEN, accessToken)\n }\n\n #user: User | null = null\n\n /** 用户 */\n public get user(): User | null {\n if (isServer)\n return this.#user\n return getStorage<User>(StorageKey.USER)\n }\n\n public set user(user: User | null) {\n if (isServer) {\n this.#user = user\n return\n }\n setStorage(StorageKey.USER, user)\n }\n\n public get isAuthenticated(): boolean {\n return !!this.accessToken\n }\n\n /** 登录 */\n public signIn(): Promise<MessageSignInData> {\n if (isServer)\n throw new Error('auth.signIn() can only be called on the client side')\n\n const width = 800\n const height = 600\n const left = (window.screen.width - width) / 2\n const top = (window.screen.height - height) / 2\n const popup = window.open(this.#lumi.config.authOrigin, this.#popupName, `width=${width},height=${height},left=${left},top=${top}`)\n\n let cleanup: () => void\n return new Promise<MessageSignInData>((resolve, reject) => {\n if (!popup)\n return reject(new Error('Open auth window failed'))\n\n const timer = setInterval(() => {\n if (popup.closed)\n reject(new Error('Auth window closed'))\n }, 1000)\n\n // 全局点击事件处理函数 - 重新聚焦popup并阻止事件传播\n const handleGlobalClick = (event: MouseEvent): void => {\n if (!popup.closed) {\n popup.focus()\n event.stopPropagation()\n event.preventDefault()\n }\n }\n\n const handleMessage = ({ data, origin, source }: MessageEvent<MessageDataReceive | null>): void => {\n if (origin !== this.#lumi.config.authOrigin || source !== popup)\n return\n\n switch (data?.type) {\n case MessageType.READY: {\n popup.postMessage({\n type: MessageType.INIT,\n data: {\n projectId: this.#lumi.config.projectId,\n icon: getIcon(),\n title: getTitle(),\n },\n } satisfies MessageDataSend, this.#lumi.config.authOrigin)\n break\n }\n case MessageType.SIGN_IN: {\n if (data.data.projectId !== this.#lumi.config.projectId)\n break\n popup.close()\n window.focus()\n this.accessToken = data.data.accessToken\n this.user = data.data.user\n resolve(data.data)\n break\n }\n }\n }\n\n window.addEventListener('message', handleMessage)\n // 添加全局点击事件监听器,使用捕获阶段确保优先处理\n document.addEventListener('click', handleGlobalClick, true)\n\n cleanup = () => {\n clearInterval(timer)\n window.removeEventListener('message', handleMessage)\n document.removeEventListener('click', handleGlobalClick, true)\n }\n }).finally(() => cleanup?.())\n }\n\n /** 退出登录 */\n public signOut(): void {\n if (isServer)\n throw new Error('auth.signOut() can only be called on the client side')\n\n this.accessToken = null\n this.user = null\n }\n\n /** 获取当前用户 */\n public async refreshUser(): Promise<User> {\n const res = await request<ApiResponse<User>>(this.#lumi, '/lm/user/info', {\n method: 'POST',\n })\n if (res.code !== 200)\n throw new Error(res.message)\n this.user = res.data\n return res.data\n }\n\n /** 监听登录状态变化 */\n public onAuthChange(callback: (args: {\n isAuthenticated: boolean\n user: User | null\n }) => void): () => void {\n if (isServer)\n throw new Error('auth.onAuthChange() can only be called on the client side')\n\n const handleStorageChange = (event: StorageEvent): void => {\n if (event.key === StorageKey.ACCESS_TOKEN || event.key === StorageKey.USER || event.key === null) {\n callback({\n isAuthenticated: this.isAuthenticated,\n user: this.user,\n })\n }\n }\n\n window.addEventListener('storage', handleStorageChange)\n\n return () => {\n window.removeEventListener('storage', handleStorageChange)\n }\n }\n}\n","import type { FetchContext, FetchOptions } from 'ofetch'\nimport type { LumiClient } from '@/core/lumi-client'\nimport Base64 from 'crypto-js/enc-base64'\nimport Hex from 'crypto-js/enc-hex'\nimport HmacSHA256 from 'crypto-js/hmac-sha256'\nimport SHA256 from 'crypto-js/sha256'\nimport hash from 'object-hash'\nimport { ofetch } from 'ofetch'\nimport { LumiError } from '@/lib/error'\nimport { isServer } from '@/utils/common'\n\nconst SECRET_KEY = '6QrJZ7pFCmBZAeIJF7IArvkCz+EtzA0RVcpHkiQIsQyhs7QtCS9P+CueZdHfB2OtJcgX3BbqY9pfpWeAVTqCwQ=='\n\n/** RFC 3986 compliant URI encoding */\nfunction rfc3986Encode(str: string): string {\n return encodeURIComponent(str).replace(/[!'()*]/g, c => `%${c.charCodeAt(0).toString(16).toUpperCase()}`)\n}\n\n/** 创建签名拦截器函数 */\nexport function createSignatureInterceptor(secretKey: string) {\n return (context: FetchContext) => {\n const { options } = context\n // 生成签名\n const timestamp = Math.floor(Date.now() / 1000).toString()\n const nonce = Math.random().toString(36).substring(2, 15)\n\n // 构建查询字符串\n const queryParams: Record<string, string> = { ...options.query }\n const canonicalQueryString = Object.keys(queryParams).sort().map(key => `${rfc3986Encode(key)}=${rfc3986Encode(String(queryParams[key]))}`).join('&')\n\n // 构建签名头\n const headersToSign: Record<string, string> = {\n 'x-timestamp': timestamp,\n 'x-nonce': nonce,\n }\n const canonicalHeaders = Object.keys(headersToSign).sort().map(key => `${key}:${headersToSign[key]}`).join('\\n')\n\n // 构建载荷哈希\n const payload = (options.body && !(options.body instanceof FormData)) ? JSON.stringify(options.body) : ''\n const hashedPayload = SHA256(payload).toString(Hex)\n\n // 构建规范请求\n const canonicalRequest = [canonicalQueryString, canonicalHeaders, hashedPayload].join('\\n')\n\n // 生成签名\n const signature = Base64.stringify(HmacSHA256(canonicalRequest, secretKey))\n\n // 添加签名头到请求\n const headers = new Headers(options.headers)\n Object.entries(headersToSign).forEach(([key, value]) => {\n headers.set(key, value)\n })\n headers.set('X-Sign', signature)\n options.headers = headers\n }\n}\n\nexport interface ApiResponse<T> {\n code: number\n message: string\n data: T\n}\n\nexport interface PaginationData<T> {\n total: number\n list: T[]\n}\n\nconst requestMap = new Map<string, number[]>()\n\n/** 限制请求频率,相同参数的请求每秒只能发送4次 */\nexport function checkRateLimit(uri: string, options: FetchOptions<'json'>): void {\n if (options.body instanceof FormData) {\n options = {\n ...options,\n body: Array.from(options.body.entries()),\n }\n }\n const requestHash = hash([uri, options])\n const now = Date.now()\n const requestQueue = requestMap.get(requestHash)?.filter(time => now - time < 1000) || []\n if (requestQueue.length >= 4)\n throw new LumiError(429, 'Too many requests')\n requestQueue.push(now)\n requestMap.set(requestHash, requestQueue)\n}\n\nexport function request<T>(lumi: LumiClient, uri: string, options: FetchOptions<'json'> = {}): Promise<T> {\n // 添加认证头(如果存在)\n if (lumi.auth.accessToken) {\n options.headers = {\n Authorization: `Bearer ${lumi.auth.accessToken}`,\n ...options.headers,\n }\n }\n\n // 限制请求频率\n checkRateLimit(uri, options)\n\n const isAuthenticated = lumi.auth.isAuthenticated\n\n return ofetch<T>(uri, {\n baseURL: lumi.config.apiBaseUrl,\n ...options,\n onRequest: createSignatureInterceptor(SECRET_KEY),\n onResponse: ({ response }) => {\n // 若 Token 失效则退出登录\n if (!isServer && isAuthenticated && (response._data as ApiResponse<any>)?.code === 2100)\n lumi.auth.signOut()\n },\n })\n}\n","export class LumiError extends Error {\n name: string = 'LumiError'\n code: number\n\n constructor(code: number, message: string) {\n super(message)\n this.code = code\n }\n}\n","export function getIcon(): string | null {\n return document.querySelector<HTMLLinkElement>('link[rel=\"icon\"]')?.href ?? null\n}\n\nexport function getTitle(): string | null {\n return document.title ?? null\n}\n\nexport const isServer = typeof window === 'undefined'\n","import type { StorageKey } from '@/constants'\n\nexport function setStorage<T>(key: StorageKey, value: T, storage: Storage = localStorage): void {\n const oldValue = storage.getItem(key)\n const newValue = value ? JSON.stringify(value) : null\n if (newValue)\n storage.setItem(key, newValue)\n else\n storage.removeItem(key)\n\n window.dispatchEvent(new StorageEvent('storage', {\n key,\n oldValue,\n newValue,\n storageArea: storage,\n }))\n}\n\nexport function getStorage<T>(key: StorageKey, storage: Storage = localStorage): T | null {\n const value = storage.getItem(key)\n try {\n return value ? JSON.parse(value) : null\n }\n catch (_e) {\n return null\n }\n}\n","import type { LumiClient } from '@/core/lumi-client'\nimport type { ApiResponse, PaginationData } from '@/lib/request'\nimport { request } from '@/lib/request'\n\nexport interface Entity extends Record<string, any> {\n id: string\n}\n\nexport class EntityClient {\n readonly #lumi: LumiClient\n public readonly entityName: string\n\n constructor(lumi: LumiClient, entityName: string) {\n this.#lumi = lumi\n this.entityName = entityName\n }\n\n /** 查询文档列表 */\n public async list({ filter, sort, limit, skip }: {\n filter?: any\n sort?: Record<string, 1 | -1>\n limit?: number\n skip?: number\n } = {}): Promise<PaginationData<Entity>> {\n if (!limit) {\n const res = await request<ApiResponse<Entity[]>>(this.#lumi, this.uri('/list'), {\n method: 'POST',\n body: { filter, sort },\n })\n if (res.code !== 200)\n throw new Error(res.message)\n return {\n total: res.data.length,\n list: res.data,\n }\n }\n else {\n const res = await request<ApiResponse<PaginationData<Entity>>>(this.#lumi, this.uri('/find'), {\n method: 'POST',\n body: { filter, sort, limit, skip },\n })\n if (res.code !== 200)\n throw new Error(res.message)\n return res.data\n }\n }\n\n /** 获取单个文档 */\n public async get(id: string): Promise<Entity | null> {\n const res = await request<ApiResponse<Entity>>(this.#lumi, this.uri(`/${id}`), {\n method: 'GET',\n })\n if (res.code !== 200)\n throw new Error(res.message)\n return res.data\n }\n\n /** 创建文档 */\n public async create(data: Record<string, any>): Promise<Entity> {\n const res = await request<ApiResponse<Entity>>(this.#lumi, this.uri(), {\n method: 'POST',\n body: data,\n })\n if (res.code !== 200)\n throw new Error(res.message)\n return res.data\n }\n\n /** 批量创建文档 */\n public async createMany(data: Record<string, any>[]): Promise<Entity[]> {\n const res = await request<ApiResponse<Entity[]>>(this.#lumi, this.uri('/batch'), {\n method: 'POST',\n body: data,\n })\n if (res.code !== 200)\n throw new Error(res.message)\n return res.data\n }\n\n /** 更新文档 */\n public async update(id: string, data: Record<string, any>): Promise<Entity> {\n const res = await request<ApiResponse<Entity>>(this.#lumi, this.uri(), {\n method: 'PUT',\n body: { filter: { _id: id }, update: data },\n })\n if (res.code !== 200)\n throw new Error(res.message)\n return res.data\n }\n\n /** 删除文档 */\n public async delete(id: string): Promise<void> {\n const res = await request<ApiResponse<null>>(this.#lumi, this.uri(`/${id}`), {\n method: 'DELETE',\n })\n if (res.code !== 200)\n throw new Error(res.message)\n }\n\n /** 批量删除文档 */\n public async deleteMany(ids: string[]): Promise<void> {\n const res = await request<ApiResponse<null>>(this.#lumi, this.uri('/batch-by-ids'), {\n method: 'DELETE',\n params: { ids },\n })\n if (res.code !== 200)\n throw new Error(res.message)\n }\n\n private uri(suffix = ''): string {\n return `/lm/${this.#lumi.config.projectId}/${this.entityName}/documents${suffix}`\n }\n}\n","import type { LumiClient } from '@/core/lumi-client'\nimport { EntityClient } from '@/core/entity-client'\n\nexport class EntitiesClient {\n readonly #lumi: LumiClient\n [key: string]: EntityClient\n\n constructor(lumi: LumiClient) {\n this.#lumi = lumi\n return new Proxy(this, {\n get(target: EntitiesClient, p: string) {\n if (!(p in target))\n target[p] = new EntityClient(target.#lumi, p)\n return target[p]\n },\n }) as this\n }\n}\n","import type { FetchOptions } from 'ofetch'\nimport type { LumiClient } from '@/core/lumi-client'\nimport { ofetch } from 'ofetch'\n\nexport class FunctionsClient {\n readonly #lumi: LumiClient\n\n constructor(lumi: LumiClient) {\n this.#lumi = lumi\n }\n\n public invoke(functionName: string, options: FetchOptions<'json'> = {}): Promise<any> {\n if (this.#lumi.auth.accessToken) {\n options.headers = {\n Authorization: `Bearer ${this.#lumi.auth.accessToken}`,\n ...options.headers,\n }\n }\n\n return ofetch(`/v1/functions/${this.#lumi.config.projectId}/${functionName}`, {\n baseURL: this.#lumi.config.apiBaseUrl,\n ...options,\n })\n }\n}\n","import type { LumiClient } from '@/core/lumi-client'\nimport type { ApiResponse } from '@/lib/request'\nimport { LumiError } from '@/lib/error'\nimport { request } from '@/lib/request'\n\nexport class EmailTool {\n readonly #lumi: LumiClient\n\n constructor(lumi: LumiClient) {\n this.#lumi = lumi\n }\n\n /** 发送邮件 */\n public async send({ to, subject, fromName, html, text = '', replyTo, scheduledAt }: {\n to: string | string[]\n subject: string\n fromName?: string\n html?: string\n text?: string\n replyTo?: string | string[]\n scheduledAt?: string\n }): Promise<void> {\n if (!to || !subject || (!html && !text))\n throw new Error('Failed to send email: Missing required parameters.')\n\n if (typeof to === 'string')\n to = [to]\n if (typeof replyTo === 'string')\n replyTo = [replyTo]\n\n const res = await request<ApiResponse<null>>(this.#lumi, `/lm/${this.#lumi.config.projectId}/email/send`, {\n method: 'POST',\n body: { to, subject, fromName, html, text, replyTo, scheduledAt },\n })\n\n if (res.code !== 200)\n throw new LumiError(res.code, res.message)\n }\n}\n","import type { LumiClient } from '@/core/lumi-client'\nimport type { ApiResponse } from '@/lib/request'\nimport { LumiError } from '@/lib/error'\nimport { request } from '@/lib/request'\n\nexport interface UploadItem {\n fileName: string\n fileUrl?: string\n uploadError?: string\n}\n\nexport class FileTool {\n readonly #lumi: LumiClient\n\n constructor(lumi: LumiClient) {\n this.#lumi = lumi\n }\n\n /** 上传文件 */\n public async upload(files: File[]): Promise<UploadItem[]> {\n const formData = new FormData()\n files.forEach((file) => {\n formData.append('files', file)\n })\n\n const res = await request<ApiResponse<UploadItem[]>>(this.#lumi, `/lm/${this.#lumi.config.projectId}/file/batch`, {\n method: 'POST',\n body: formData,\n })\n\n if (res.code !== 200)\n throw new LumiError(res.code, res.message)\n\n return res.data\n }\n\n /** 批量删除文件 */\n public async delete(fileUrls: string[]): Promise<void> {\n const res = await request<ApiResponse<null>>(this.#lumi, `/lm/${this.#lumi.config.projectId}/file/batch`, {\n method: 'DELETE',\n body: { fileUrls },\n })\n\n if (res.code !== 200)\n throw new LumiError(res.code, res.message)\n }\n}\n","import type { LumiClient } from '@/core/lumi-client'\nimport { EmailTool, FileTool } from '@/tools'\n\nexport class ToolsClient {\n readonly #lumi: LumiClient\n\n public email: EmailTool\n public file: FileTool\n\n constructor(lumi: LumiClient) {\n this.#lumi = lumi\n this.email = new EmailTool(lumi)\n this.file = new FileTool(lumi)\n }\n}\n","import { LumiAuthClient } from '@/core/auth-client'\nimport { EntitiesClient } from '@/core/entities-client'\nimport { FunctionsClient } from '@/core/functions-client'\nimport { ToolsClient } from '@/core/tools-client'\n\nexport interface LumiClientConfig {\n projectId: string\n apiBaseUrl: string\n authOrigin: string\n authorization?: string\n}\n\nexport class LumiClient {\n public config: LumiClientConfig\n\n public auth: LumiAuthClient\n public entities: EntitiesClient\n public tools: ToolsClient\n public functions: FunctionsClient\n\n constructor(config: LumiClientConfig) {\n this.config = config\n this.auth = new LumiAuthClient(this)\n this.entities = new EntitiesClient(this)\n this.tools = new ToolsClient(this)\n this.functions = new FunctionsClient(this)\n }\n}\n\nexport function createClient(config: LumiClientConfig): LumiClient {\n return new LumiClient(config)\n}\n"],"mappings":"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","names":["v4","Base64","Hex","HmacSHA256","SHA256","hash","ofetch","LumiError","code","message","getIcon","_a","_b","getTitle","isServer","SECRET_KEY","rfc3986Encode","str","c","createSignatureInterceptor","secretKey","context","options","timestamp","nonce","queryParams","__spreadValues","canonicalQueryString","key","headersToSign","canonicalHeaders","payload","hashedPayload","SHA256","Hex","canonicalRequest","signature","Base64","HmacSHA256","headers","value","requestMap","checkRateLimit","uri","_a","__spreadProps","requestHash","hash","now","requestQueue","time","LumiError","request","lumi","isAuthenticated","ofetch","response","isServer","setStorage","key","value","storage","oldValue","newValue","getStorage","_e","_lumi","_popupName","_user","LumiAuthClient","lumi","__privateAdd","v4","__privateSet","isServer","authorization","__privateGet","getStorage","accessToken","setStorage","user","width","height","left","top","popup","cleanup","resolve","reject","timer","handleGlobalClick","event","handleMessage","data","origin","source","getIcon","getTitle","__async","res","request","callback","handleStorageChange","_lumi","EntityClient","lumi","entityName","__privateAdd","__privateSet","__async","filter","sort","limit","skip","res","request","__privateGet","id","data","ids","suffix","_lumi","EntitiesClient","lumi","__privateAdd","__privateSet","target","p","EntityClient","__privateGet","ofetch","_lumi","FunctionsClient","lumi","__privateAdd","__privateSet","functionName","options","__privateGet","__spreadValues","ofetch","_lumi","EmailTool","lumi","__privateAdd","__privateSet","_0","__async","to","subject","fromName","html","text","replyTo","scheduledAt","res","request","__privateGet","LumiError","_lumi","FileTool","lumi","__privateAdd","__privateSet","files","__async","formData","file","res","request","__privateGet","LumiError","fileUrls","_lumi","ToolsClient","lumi","__privateAdd","__privateSet","EmailTool","FileTool","LumiClient","config","LumiAuthClient","EntitiesClient","ToolsClient","FunctionsClient","createClient"]}
1
+ {"version":3,"sources":["../src/core/auth-client.ts","../src/lib/request.ts","../src/lib/error.ts","../src/utils/common.ts","../src/utils/storage.ts","../src/core/entity-client.ts","../src/core/entities-client.ts","../src/core/functions-client.ts","../src/tools/ai-tool.ts","../src/tools/email-tool.ts","../src/tools/file-tool.ts","../src/core/tools-client.ts","../src/core/lumi-client.ts"],"sourcesContent":["import type { LumiClient } from '@/core/lumi-client'\nimport type { ApiResponse } from '@/lib/request'\nimport { v4 } from 'uuid'\nimport { MessageType, StorageKey } from '@/constants'\nimport { request } from '@/lib/request'\nimport { getIcon, getTitle, isServer } from '@/utils/common'\nimport { getStorage, setStorage } from '@/utils/storage'\n\nexport interface User {\n userId: string\n email: string\n userName: string\n userRole: 'ADMIN' | 'USER'\n createdTime: string\n}\n\nexport interface MessageSignInData {\n projectId: string\n accessToken: string\n user: User\n}\n\nexport type MessageDataReceive = {\n type: MessageType.READY\n} | {\n type: MessageType.SIGN_IN\n data: MessageSignInData\n}\n\nexport interface MessageInitData {\n projectId: string\n icon: string | null\n title: string | null\n}\n\nexport interface MessageDataSend {\n type: MessageType.INIT\n data: MessageInitData\n}\n\nexport class LumiAuthClient {\n readonly #lumi: LumiClient\n readonly #popupName: string = `lumi-auth-${v4()}`\n\n constructor(lumi: LumiClient) {\n this.#lumi = lumi\n Promise.resolve().then(() => {\n if (!isServer && this.isAuthenticated)\n void this.refreshUser()\n })\n }\n\n /** 访问令牌 */\n public get accessToken(): string | null {\n if (isServer) {\n const authorization = this.#lumi.config.authorization\n return authorization ? authorization.replace('Bearer ', '') : null\n }\n return getStorage<string>(StorageKey.ACCESS_TOKEN)\n }\n\n public set accessToken(accessToken: string | null) {\n if (isServer) {\n this.#lumi.config.authorization = accessToken ? `Bearer ${accessToken}` : undefined\n return\n }\n setStorage(StorageKey.ACCESS_TOKEN, accessToken)\n }\n\n #user: User | null = null\n\n /** 用户 */\n public get user(): User | null {\n if (isServer)\n return this.#user\n return getStorage<User>(StorageKey.USER)\n }\n\n public set user(user: User | null) {\n if (isServer) {\n this.#user = user\n return\n }\n setStorage(StorageKey.USER, user)\n }\n\n public get isAuthenticated(): boolean {\n return !!this.accessToken\n }\n\n /** 登录 */\n public signIn(): Promise<MessageSignInData> {\n if (isServer)\n throw new Error('auth.signIn() can only be called on the client side')\n\n const width = 800\n const height = 600\n const left = (window.screen.width - width) / 2\n const top = (window.screen.height - height) / 2\n const popup = window.open(this.#lumi.config.authOrigin, this.#popupName, `width=${width},height=${height},left=${left},top=${top}`)\n\n let cleanup: () => void\n return new Promise<MessageSignInData>((resolve, reject) => {\n if (!popup)\n return reject(new Error('Open auth window failed'))\n\n const timer = setInterval(() => {\n if (popup.closed)\n reject(new Error('Auth window closed'))\n }, 1000)\n\n // 全局点击事件处理函数 - 重新聚焦popup并阻止事件传播\n const handleGlobalClick = (event: MouseEvent): void => {\n if (!popup.closed) {\n popup.focus()\n event.stopPropagation()\n event.preventDefault()\n }\n }\n\n const handleMessage = ({ data, origin, source }: MessageEvent<MessageDataReceive | null>): void => {\n if (origin !== this.#lumi.config.authOrigin || source !== popup)\n return\n\n switch (data?.type) {\n case MessageType.READY: {\n popup.postMessage({\n type: MessageType.INIT,\n data: {\n projectId: this.#lumi.config.projectId,\n icon: getIcon(),\n title: getTitle(),\n },\n } satisfies MessageDataSend, this.#lumi.config.authOrigin)\n break\n }\n case MessageType.SIGN_IN: {\n if (data.data.projectId !== this.#lumi.config.projectId)\n break\n popup.close()\n window.focus()\n this.accessToken = data.data.accessToken\n this.user = data.data.user\n resolve(data.data)\n break\n }\n }\n }\n\n window.addEventListener('message', handleMessage)\n // 添加全局点击事件监听器,使用捕获阶段确保优先处理\n document.addEventListener('click', handleGlobalClick, true)\n\n cleanup = () => {\n clearInterval(timer)\n window.removeEventListener('message', handleMessage)\n document.removeEventListener('click', handleGlobalClick, true)\n }\n }).finally(() => cleanup?.())\n }\n\n /** 退出登录 */\n public signOut(): void {\n if (isServer)\n throw new Error('auth.signOut() can only be called on the client side')\n\n this.accessToken = null\n this.user = null\n }\n\n /** 获取当前用户 */\n public async refreshUser(): Promise<User> {\n const res = await request<ApiResponse<User>>(this.#lumi, '/lm/user/info', {\n method: 'POST',\n })\n if (res.code !== 200)\n throw new Error(res.message)\n this.user = res.data\n return res.data\n }\n\n /** 监听登录状态变化 */\n public onAuthChange(callback: (args: {\n isAuthenticated: boolean\n user: User | null\n }) => void): () => void {\n if (isServer)\n throw new Error('auth.onAuthChange() can only be called on the client side')\n\n const handleStorageChange = (event: StorageEvent): void => {\n if (event.key === StorageKey.ACCESS_TOKEN || event.key === StorageKey.USER || event.key === null) {\n callback({\n isAuthenticated: this.isAuthenticated,\n user: this.user,\n })\n }\n }\n\n window.addEventListener('storage', handleStorageChange)\n\n return () => {\n window.removeEventListener('storage', handleStorageChange)\n }\n }\n}\n","import type { FetchOptions, ResponseType } from 'ofetch'\nimport type { LumiClient } from '@/core/lumi-client'\nimport Base64 from 'crypto-js/enc-base64'\nimport Hex from 'crypto-js/enc-hex'\nimport HmacSHA256 from 'crypto-js/hmac-sha256'\nimport SHA256 from 'crypto-js/sha256'\nimport { EventSourceParserStream } from 'eventsource-parser/stream'\nimport hash from 'object-hash'\nimport { ofetch } from 'ofetch'\nimport { LumiError } from '@/lib/error'\nimport { isServer } from '@/utils/common'\n\nconst SECRET_KEY = '6QrJZ7pFCmBZAeIJF7IArvkCz+EtzA0RVcpHkiQIsQyhs7QtCS9P+CueZdHfB2OtJcgX3BbqY9pfpWeAVTqCwQ=='\n\n/** RFC 3986 compliant URI encoding */\nfunction rfc3986Encode(str: string): string {\n return encodeURIComponent(str).replace(/[!'()*]/g, c => `%${c.charCodeAt(0).toString(16).toUpperCase()}`)\n}\n\n/** 添加签名 */\nfunction addSignature<T extends ResponseType>(options: FetchOptions<T>): void {\n // 生成签名\n const timestamp = Math.floor(Date.now() / 1000).toString()\n const nonce = Math.random().toString(36).substring(2, 15)\n\n // 构建查询字符串\n const queryParams: Record<string, string> = { ...options.query }\n const canonicalQueryString = Object.keys(queryParams).sort().map(key => `${rfc3986Encode(key)}=${rfc3986Encode(String(queryParams[key]))}`).join('&')\n\n // 构建签名头\n const headersToSign: Record<string, string> = {\n 'x-timestamp': timestamp,\n 'x-nonce': nonce,\n }\n const canonicalHeaders = Object.keys(headersToSign).sort().map(key => `${key}:${headersToSign[key]}`).join('\\n')\n\n // 构建载荷哈希\n const payload = (options.body && !(options.body instanceof FormData)) ? JSON.stringify(options.body) : ''\n const hashedPayload = SHA256(payload).toString(Hex)\n\n // 构建规范请求\n const canonicalRequest = [canonicalQueryString, canonicalHeaders, hashedPayload].join('\\n')\n\n // 生成签名\n const signature = Base64.stringify(HmacSHA256(canonicalRequest, SECRET_KEY))\n\n // 添加签名头到请求\n const headers = new Headers(options.headers)\n Object.entries(headersToSign).forEach(([key, value]) => {\n headers.set(key, value)\n })\n headers.set('X-Sign', signature)\n options.headers = headers\n}\n\nexport interface ApiResponse<T> {\n code: number\n message: string\n data: T\n}\n\nexport interface PaginationData<T> {\n total: number\n list: T[]\n}\n\nconst requestMap = new Map<string, number[]>()\n\n/** 限制请求频率,相同参数的请求每秒只能发送4次 */\nfunction checkRateLimit<T extends ResponseType>(uri: string, options: FetchOptions<T>): void {\n const clonedOptions = { ...options }\n\n if (options.body instanceof FormData)\n clonedOptions.body = Array.from(options.body.entries())\n\n clonedOptions.headers = undefined\n\n const requestHash = hash([uri, clonedOptions])\n const now = Date.now()\n const requestQueue = requestMap.get(requestHash)?.filter(time => now - time < 1000) || []\n if (requestQueue.length >= 4)\n throw new LumiError(429, 'Too many requests')\n requestQueue.push(now)\n requestMap.set(requestHash, requestQueue)\n}\n\nfunction beforeRequest<T extends ResponseType>(lumi: LumiClient, uri: string, options: FetchOptions<T>, stream: boolean = false): void {\n options.headers = new Headers(options.headers)\n\n // 添加认证头(如果存在)\n if (lumi.auth.accessToken)\n options.headers.set('Authorization', `Bearer ${lumi.auth.accessToken}`)\n\n // 添加流式请求头\n if (stream) {\n if (!options.headers.get('Accept'))\n options.headers.set('Accept', 'text/event-stream')\n if (!options.headers.get('Cache-Control'))\n options.headers.set('Cache-Control', 'no-cache')\n if (!options.headers.get('X-Accel-Buffering'))\n options.headers.set('X-Accel-Buffering', 'no')\n }\n\n // 限制请求频率\n checkRateLimit(uri, options)\n\n // 添加签名\n addSignature(options)\n}\n\n/** 网络请求 */\nexport function request<T>(lumi: LumiClient, uri: string, options: FetchOptions<'json'> = {}): Promise<T> {\n beforeRequest(lumi, uri, options)\n\n const isAuthenticated = lumi.auth.isAuthenticated\n\n return ofetch<T>(uri, {\n baseURL: lumi.config.apiBaseUrl,\n ...options,\n onResponse: ({ response }) => {\n // 若 Token 失效则退出登录\n if (!isServer && isAuthenticated && (response._data as ApiResponse<any>)?.code === 2100)\n lumi.auth.signOut()\n },\n })\n}\n\nexport interface StreamChunk<T> {\n event?: string\n data: T\n}\n\n/** 网络请求(流式) */\nexport async function requestStream<T>(lumi: LumiClient, uri: string, options: FetchOptions<'stream'> = {}): Promise<ReadableStream<StreamChunk<T>>> {\n beforeRequest(lumi, uri, options, true)\n\n const stream = await ofetch(uri, {\n baseURL: lumi.config.apiBaseUrl,\n ...options,\n responseType: 'stream',\n })\n\n return stream\n .pipeThrough(new TextDecoderStream())\n .pipeThrough(new EventSourceParserStream())\n .pipeThrough<StreamChunk<T>>(new TransformStream({\n transform: (chunk, controller) => {\n try {\n const data = JSON.parse(chunk.data)\n controller.enqueue({\n event: chunk.event,\n data,\n })\n }\n catch {}\n },\n }))\n}\n","export class LumiError extends Error {\n name: string = 'LumiError'\n code: number\n\n constructor(code: number, message: string) {\n super(message)\n this.code = code\n }\n}\n","export function getIcon(): string | null {\n return document.querySelector<HTMLLinkElement>('link[rel=\"icon\"]')?.href ?? null\n}\n\nexport function getTitle(): string | null {\n return document.title ?? null\n}\n\nexport const isServer = typeof window === 'undefined'\n\nexport function delay(ms: number): Promise<void> {\n return new Promise(resolve => setTimeout(resolve, ms))\n}\n\nexport function getEnv(key: string): string | null {\n return typeof process !== 'undefined' ? process?.env[key] ?? null : null\n}\n","import type { StorageKey } from '@/constants'\n\nexport function setStorage<T>(key: StorageKey, value: T, storage: Storage = localStorage): void {\n const oldValue = storage.getItem(key)\n const newValue = value ? JSON.stringify(value) : null\n if (newValue)\n storage.setItem(key, newValue)\n else\n storage.removeItem(key)\n\n window.dispatchEvent(new StorageEvent('storage', {\n key,\n oldValue,\n newValue,\n storageArea: storage,\n }))\n}\n\nexport function getStorage<T>(key: StorageKey, storage: Storage = localStorage): T | null {\n const value = storage.getItem(key)\n try {\n return value ? JSON.parse(value) : null\n }\n catch (_e) {\n return null\n }\n}\n","import type { LumiClient } from '@/core/lumi-client'\nimport type { ApiResponse, PaginationData } from '@/lib/request'\nimport { request } from '@/lib/request'\n\nexport interface Entity extends Record<string, any> {\n id: string\n}\n\nexport class EntityClient {\n readonly #lumi: LumiClient\n public readonly entityName: string\n\n constructor(lumi: LumiClient, entityName: string) {\n this.#lumi = lumi\n this.entityName = entityName\n }\n\n /** 查询文档列表 */\n public async list({ filter, sort, limit, skip }: {\n filter?: any\n sort?: Record<string, 1 | -1>\n limit?: number\n skip?: number\n } = {}): Promise<PaginationData<Entity>> {\n if (!limit) {\n const res = await request<ApiResponse<Entity[]>>(this.#lumi, this.uri('/list'), {\n method: 'POST',\n body: { filter, sort },\n })\n if (res.code !== 200)\n throw new Error(res.message)\n return {\n total: res.data.length,\n list: res.data,\n }\n }\n else {\n const res = await request<ApiResponse<PaginationData<Entity>>>(this.#lumi, this.uri('/find'), {\n method: 'POST',\n body: { filter, sort, limit, skip },\n })\n if (res.code !== 200)\n throw new Error(res.message)\n return res.data\n }\n }\n\n /** 获取单个文档 */\n public async get(id: string): Promise<Entity | null> {\n const res = await request<ApiResponse<Entity>>(this.#lumi, this.uri(`/${id}`), {\n method: 'GET',\n })\n if (res.code !== 200)\n throw new Error(res.message)\n return res.data\n }\n\n /** 创建文档 */\n public async create(data: Record<string, any>): Promise<Entity> {\n const res = await request<ApiResponse<Entity>>(this.#lumi, this.uri(), {\n method: 'POST',\n body: data,\n })\n if (res.code !== 200)\n throw new Error(res.message)\n return res.data\n }\n\n /** 批量创建文档 */\n public async createMany(data: Record<string, any>[]): Promise<Entity[]> {\n const res = await request<ApiResponse<Entity[]>>(this.#lumi, this.uri('/batch'), {\n method: 'POST',\n body: data,\n })\n if (res.code !== 200)\n throw new Error(res.message)\n return res.data\n }\n\n /** 更新文档 */\n public async update(id: string, data: Record<string, any>): Promise<Entity> {\n const res = await request<ApiResponse<Entity>>(this.#lumi, this.uri(), {\n method: 'PUT',\n body: { filter: { _id: id }, update: data },\n })\n if (res.code !== 200)\n throw new Error(res.message)\n return res.data\n }\n\n /** 删除文档 */\n public async delete(id: string): Promise<void> {\n const res = await request<ApiResponse<null>>(this.#lumi, this.uri(`/${id}`), {\n method: 'DELETE',\n })\n if (res.code !== 200)\n throw new Error(res.message)\n }\n\n /** 批量删除文档 */\n public async deleteMany(ids: string[]): Promise<void> {\n const res = await request<ApiResponse<null>>(this.#lumi, this.uri('/batch-by-ids'), {\n method: 'DELETE',\n params: { ids },\n })\n if (res.code !== 200)\n throw new Error(res.message)\n }\n\n private uri(suffix = ''): string {\n return `/lm/${this.#lumi.config.projectId}/${this.entityName}/documents${suffix}`\n }\n}\n","import type { LumiClient } from '@/core/lumi-client'\nimport { EntityClient } from '@/core/entity-client'\n\nexport class EntitiesClient {\n readonly #lumi: LumiClient\n [key: string]: EntityClient\n\n constructor(lumi: LumiClient) {\n this.#lumi = lumi\n return new Proxy(this, {\n get(target: EntitiesClient, p: string) {\n if (!(p in target))\n target[p] = new EntityClient(target.#lumi, p)\n return target[p]\n },\n }) as this\n }\n}\n","import type { FetchOptions } from 'ofetch'\nimport type { LumiClient } from '@/core/lumi-client'\nimport { ofetch } from 'ofetch'\n\nexport class FunctionsClient {\n readonly #lumi: LumiClient\n\n constructor(lumi: LumiClient) {\n this.#lumi = lumi\n }\n\n public invoke(functionName: string, options: FetchOptions<'json'> = {}): Promise<any> {\n if (this.#lumi.auth.accessToken) {\n options.headers = {\n Authorization: `Bearer ${this.#lumi.auth.accessToken}`,\n ...options.headers,\n }\n }\n\n return ofetch(`/v1/functions/${this.#lumi.config.projectId}/${functionName}`, {\n baseURL: this.#lumi.config.apiBaseUrl,\n ...options,\n })\n }\n}\n","import type { LumiClient } from '@/core/lumi-client'\nimport type { ApiResponse, StreamChunk } from '@/lib/request'\nimport { LumiError } from '@/lib/error'\nimport { requestStream } from '@/lib/request'\nimport { request } from '@/lib/request'\nimport { delay, getEnv, isServer } from '@/utils/common'\n\nexport interface MessageMedia {\n mimeType: string\n url: string\n}\n\nexport interface Message {\n role: 'system' | 'user' | 'assistant'\n content: string\n medias?: MessageMedia[]\n}\n\nexport interface GenerateTextResult {\n chatId: string\n content: string\n}\n\nexport interface GenerateImageResult {\n chatId: string\n messageId: string\n generationStatus: 'PROCESSING' | 'SUCCESS' | 'FAILED'\n content?: string\n imageURL?: string\n}\n\nexport class AITool {\n readonly #lumi: LumiClient\n\n constructor(lumi: LumiClient) {\n this.#lumi = lumi\n }\n\n /** AI 生成文本 */\n public async generateText({\n model = 'gemini-2.5-flash',\n messages,\n }: {\n model?: string\n messages: Message[]\n }): Promise<GenerateTextResult> {\n const lumiApiKey = this.checkLumiApiKey()\n\n const res = await request<ApiResponse<GenerateTextResult>>(this.#lumi, `/lm/${this.#lumi.config.projectId}/ai/chat/completions`, {\n method: 'POST',\n body: {\n lumiApiKey,\n modelName: model,\n chatMessages: messages,\n },\n })\n\n if (res.code !== 200)\n throw new LumiError(res.code, res.message)\n\n return res.data\n }\n\n /** AI 生成文本(流式输出) */\n public async generateTextStream({\n model = 'gemini-2.5-flash',\n messages,\n }: {\n model?: string\n messages: Message[]\n }): Promise<ReadableStream<StreamChunk<GenerateTextResult>>> {\n const lumiApiKey = this.checkLumiApiKey()\n\n return requestStream<GenerateTextResult>(this.#lumi, `/lm/${this.#lumi.config.projectId}/ai/chat/stream`, {\n method: 'POST',\n body: {\n lumiApiKey,\n modelName: model,\n chatMessages: messages,\n },\n })\n }\n\n /** AI 生成图片 */\n public async generateImage({\n model = 'gemini-2.5-flash-image',\n messages,\n }: {\n model?: string\n messages: Message[]\n }): Promise<GenerateImageResult> {\n const lumiApiKey = this.checkLumiApiKey()\n\n const { code, message, data } = await request<ApiResponse<GenerateImageResult>>(this.#lumi, `/lm/${this.#lumi.config.projectId}/ai/image/task`, {\n method: 'POST',\n body: {\n lumiApiKey,\n modelName: model,\n chatMessages: messages,\n },\n })\n if (code !== 200)\n throw new LumiError(code, message)\n\n let result = data\n let delayMs = 1000\n\n // 轮询生成结果\n while (result.generationStatus === 'PROCESSING') {\n await delay(delayMs)\n if (delayMs < 5000)\n delayMs += 500\n\n const { code, message, data } = await request<ApiResponse<GenerateImageResult>>(this.#lumi, `/lm/${this.#lumi.config.projectId}/ai/image/get`, {\n method: 'POST',\n body: {\n lumiApiKey,\n messageId: result.messageId,\n },\n })\n if (code !== 200)\n throw new LumiError(code, message)\n result = data\n }\n\n return result\n }\n\n private checkLumiApiKey(): string {\n if (!isServer)\n throw new LumiError(400, 'lumi.tools.ai is only available on the server-side')\n\n const lumiApiKey = getEnv('LUMI_API_KEY')\n if (!lumiApiKey)\n throw new LumiError(400, 'LUMI_API_KEY is required')\n\n return lumiApiKey\n }\n}\n","import type { LumiClient } from '@/core/lumi-client'\nimport type { ApiResponse } from '@/lib/request'\nimport { LumiError } from '@/lib/error'\nimport { request } from '@/lib/request'\n\nexport class EmailTool {\n readonly #lumi: LumiClient\n\n constructor(lumi: LumiClient) {\n this.#lumi = lumi\n }\n\n /** 发送邮件 */\n public async send({ to, subject, fromName, html, text = '', replyTo, scheduledAt }: {\n to: string | string[]\n subject: string\n fromName?: string\n html?: string\n text?: string\n replyTo?: string | string[]\n scheduledAt?: string\n }): Promise<void> {\n if (!to || !subject || (!html && !text))\n throw new Error('Failed to send email: Missing required parameters.')\n\n if (typeof to === 'string')\n to = [to]\n if (typeof replyTo === 'string')\n replyTo = [replyTo]\n\n const res = await request<ApiResponse<null>>(this.#lumi, `/lm/${this.#lumi.config.projectId}/email/send`, {\n method: 'POST',\n body: { to, subject, fromName, html, text, replyTo, scheduledAt },\n })\n\n if (res.code !== 200)\n throw new LumiError(res.code, res.message)\n }\n}\n","import type { LumiClient } from '@/core/lumi-client'\nimport type { ApiResponse } from '@/lib/request'\nimport { LumiError } from '@/lib/error'\nimport { request } from '@/lib/request'\n\nexport interface UploadItem {\n fileName: string\n fileUrl?: string\n uploadError?: string\n}\n\nexport class FileTool {\n readonly #lumi: LumiClient\n\n constructor(lumi: LumiClient) {\n this.#lumi = lumi\n }\n\n /** 上传文件 */\n public async upload(files: File[]): Promise<UploadItem[]> {\n const formData = new FormData()\n files.forEach((file) => {\n formData.append('files', file)\n })\n\n const res = await request<ApiResponse<UploadItem[]>>(this.#lumi, `/lm/${this.#lumi.config.projectId}/file/batch`, {\n method: 'POST',\n body: formData,\n })\n\n if (res.code !== 200)\n throw new LumiError(res.code, res.message)\n\n return res.data\n }\n\n /** 批量删除文件 */\n public async delete(fileUrls: string[]): Promise<void> {\n const res = await request<ApiResponse<null>>(this.#lumi, `/lm/${this.#lumi.config.projectId}/file/batch`, {\n method: 'DELETE',\n body: { fileUrls },\n })\n\n if (res.code !== 200)\n throw new LumiError(res.code, res.message)\n }\n}\n","import type { LumiClient } from '@/core/lumi-client'\nimport { AITool, EmailTool, FileTool } from '@/tools'\n\nexport class ToolsClient {\n readonly #lumi: LumiClient\n\n public email: EmailTool\n public file: FileTool\n public ai: AITool\n\n constructor(lumi: LumiClient) {\n this.#lumi = lumi\n this.email = new EmailTool(lumi)\n this.file = new FileTool(lumi)\n this.ai = new AITool(lumi)\n }\n}\n","import { LumiAuthClient } from '@/core/auth-client'\nimport { EntitiesClient } from '@/core/entities-client'\nimport { FunctionsClient } from '@/core/functions-client'\nimport { ToolsClient } from '@/core/tools-client'\n\nexport interface LumiClientConfig {\n projectId: string\n apiBaseUrl: string\n authOrigin: string\n authorization?: string\n}\n\nexport class LumiClient {\n public config: LumiClientConfig\n\n public auth: LumiAuthClient\n public entities: EntitiesClient\n public tools: ToolsClient\n public functions: FunctionsClient\n\n constructor(config: LumiClientConfig) {\n this.config = config\n this.auth = new LumiAuthClient(this)\n this.entities = new EntitiesClient(this)\n this.tools = new ToolsClient(this)\n this.functions = new FunctionsClient(this)\n }\n}\n\nexport function createClient(config: LumiClientConfig): LumiClient {\n return new LumiClient(config)\n}\n"],"mappings":"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","names":["v4","Base64","Hex","HmacSHA256","SHA256","EventSourceParserStream","hash","ofetch","LumiError","code","message","getIcon","_a","_b","getTitle","isServer","delay","ms","resolve","getEnv","key","SECRET_KEY","rfc3986Encode","str","c","addSignature","options","timestamp","nonce","queryParams","__spreadValues","canonicalQueryString","key","headersToSign","canonicalHeaders","payload","hashedPayload","SHA256","Hex","canonicalRequest","signature","Base64","HmacSHA256","headers","value","requestMap","checkRateLimit","uri","_a","clonedOptions","requestHash","hash","now","requestQueue","time","LumiError","beforeRequest","lumi","stream","request","isAuthenticated","ofetch","__spreadProps","response","isServer","requestStream","_0","_1","__async","EventSourceParserStream","chunk","controller","data","e","setStorage","key","value","storage","oldValue","newValue","getStorage","_e","_lumi","_popupName","_user","LumiAuthClient","lumi","__privateAdd","v4","__privateSet","isServer","authorization","__privateGet","getStorage","accessToken","setStorage","user","width","height","left","top","popup","cleanup","resolve","reject","timer","handleGlobalClick","event","handleMessage","data","origin","source","getIcon","getTitle","__async","res","request","callback","handleStorageChange","_lumi","EntityClient","lumi","entityName","__privateAdd","__privateSet","__async","filter","sort","limit","skip","res","request","__privateGet","id","data","ids","suffix","_lumi","EntitiesClient","lumi","__privateAdd","__privateSet","target","p","EntityClient","__privateGet","ofetch","_lumi","FunctionsClient","lumi","__privateAdd","__privateSet","functionName","options","__privateGet","__spreadValues","ofetch","_lumi","AITool","lumi","__privateAdd","__privateSet","_0","__async","model","messages","lumiApiKey","res","request","__privateGet","LumiError","requestStream","code","message","data","result","delayMs","delay","isServer","getEnv","_lumi","EmailTool","lumi","__privateAdd","__privateSet","_0","__async","to","subject","fromName","html","text","replyTo","scheduledAt","res","request","__privateGet","LumiError","_lumi","FileTool","lumi","__privateAdd","__privateSet","files","__async","formData","file","res","request","__privateGet","LumiError","fileUrls","_lumi","ToolsClient","lumi","__privateAdd","__privateSet","EmailTool","FileTool","AITool","LumiClient","config","LumiAuthClient","EntitiesClient","ToolsClient","FunctionsClient","createClient"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@lumi.new/sdk",
3
- "version": "0.2.1",
3
+ "version": "0.3.1",
4
4
  "description": "SDK for Lumi.new",
5
5
  "sideEffects": false,
6
6
  "main": "dist/index.js",
@@ -9,8 +9,12 @@
9
9
  "files": [
10
10
  "dist"
11
11
  ],
12
+ "engines": {
13
+ "node": ">=18"
14
+ },
12
15
  "dependencies": {
13
16
  "crypto-js": "^4.2.0",
17
+ "eventsource-parser": "^3.0.6",
14
18
  "object-hash": "^3.0.0",
15
19
  "ofetch": "^1.4.1",
16
20
  "uuid": "^11.1.0"