@lumi.new/sdk 0.4.0 → 0.4.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/dist/index.d.mts +47 -3
- package/dist/index.d.ts +47 -3
- package/dist/index.js +4 -4
- package/dist/index.js.map +1 -1
- package/dist/index.mjs +4 -4
- package/dist/index.mjs.map +1 -1
- package/package.json +1 -1
package/dist/index.d.mts
CHANGED
|
@@ -1,5 +1,32 @@
|
|
|
1
1
|
import { FetchOptions } from 'ofetch';
|
|
2
2
|
|
|
3
|
+
/** popup 句柄 */
|
|
4
|
+
interface AuthPopupHandle {
|
|
5
|
+
/** 向 popup 发送消息 */
|
|
6
|
+
postMessage: (data: any) => void;
|
|
7
|
+
/** 关闭 popup */
|
|
8
|
+
close: () => void;
|
|
9
|
+
/** 聚焦 popup(可选) */
|
|
10
|
+
focus?: () => void;
|
|
11
|
+
/** 检查 popup 是否已关闭 */
|
|
12
|
+
closed: boolean;
|
|
13
|
+
}
|
|
14
|
+
/** window 适配器 */
|
|
15
|
+
interface AuthAdapter {
|
|
16
|
+
/** 打开认证弹窗。成功返回 popup 句柄,失败返回 null */
|
|
17
|
+
open: (url: string, name?: string) => AuthPopupHandle | null;
|
|
18
|
+
/** 监听来自指定 popup 的消息。返回取消监听函数 */
|
|
19
|
+
listenMessage: (callback: (data: any, origin: string) => void) => () => void;
|
|
20
|
+
/** 监听全局点击。返回取消监听函数 */
|
|
21
|
+
listenGlobalClick?: (callback: (e: MouseEvent) => void) => () => void;
|
|
22
|
+
/** 获取当前页面的 icon */
|
|
23
|
+
getIcon: () => string | null;
|
|
24
|
+
/** 获取当前页面的 title */
|
|
25
|
+
getTitle: () => string | null;
|
|
26
|
+
}
|
|
27
|
+
/** 创建默认的 window 适配器 */
|
|
28
|
+
declare function createDefaultAuthAdapter(): AuthAdapter;
|
|
29
|
+
|
|
3
30
|
interface StorageAdapter {
|
|
4
31
|
storage: {
|
|
5
32
|
getItem: (key: string) => string | null;
|
|
@@ -8,9 +35,15 @@ interface StorageAdapter {
|
|
|
8
35
|
};
|
|
9
36
|
listen: (callback: (key: string | null | undefined, value: string | null) => void) => () => void;
|
|
10
37
|
}
|
|
38
|
+
declare function createDefaultStorageAdapter(): StorageAdapter;
|
|
11
39
|
|
|
12
40
|
type Optional<T, K extends keyof T> = Omit<T, K> & Partial<Pick<T, K>>;
|
|
13
41
|
|
|
42
|
+
interface ApiResponse<T> {
|
|
43
|
+
code: number;
|
|
44
|
+
message: string;
|
|
45
|
+
data: T;
|
|
46
|
+
}
|
|
14
47
|
interface PaginationData<T> {
|
|
15
48
|
total: number;
|
|
16
49
|
list: T[];
|
|
@@ -147,6 +180,7 @@ interface LumiClientConfig {
|
|
|
147
180
|
apiBaseUrl: string;
|
|
148
181
|
authOrigin: string;
|
|
149
182
|
storageAdapter: StorageAdapter;
|
|
183
|
+
authAdapter: AuthAdapter;
|
|
150
184
|
authorization?: string;
|
|
151
185
|
}
|
|
152
186
|
declare class LumiClient {
|
|
@@ -155,10 +189,14 @@ declare class LumiClient {
|
|
|
155
189
|
entities: EntitiesClient;
|
|
156
190
|
tools: ToolsClient;
|
|
157
191
|
functions: FunctionsClient;
|
|
158
|
-
constructor(config: Optional<LumiClientConfig, 'storageAdapter'>);
|
|
192
|
+
constructor(config: Optional<LumiClientConfig, 'storageAdapter' | 'authAdapter'>);
|
|
159
193
|
}
|
|
160
|
-
declare function createClient(config: Optional<LumiClientConfig, 'storageAdapter'>): LumiClient;
|
|
194
|
+
declare function createClient(config: Optional<LumiClientConfig, 'storageAdapter' | 'authAdapter'>): LumiClient;
|
|
161
195
|
|
|
196
|
+
declare enum StorageKey {
|
|
197
|
+
ACCESS_TOKEN = "lumi-access-token",
|
|
198
|
+
USER = "lumi-user"
|
|
199
|
+
}
|
|
162
200
|
declare enum MessageType {
|
|
163
201
|
READY = "lumi-ready",
|
|
164
202
|
INIT = "lumi-init",
|
|
@@ -215,4 +253,10 @@ declare class LumiAuthClient {
|
|
|
215
253
|
}) => void): () => void;
|
|
216
254
|
}
|
|
217
255
|
|
|
218
|
-
|
|
256
|
+
declare class LumiError extends Error {
|
|
257
|
+
name: string;
|
|
258
|
+
code: number;
|
|
259
|
+
constructor(code: number, message: string);
|
|
260
|
+
}
|
|
261
|
+
|
|
262
|
+
export { AITool, type ApiResponse, type ApiStreamResponse, type AuthAdapter, type AuthPopupHandle, EmailTool, EntitiesClient, type Entity, EntityClient, FileTool, FunctionsClient, type GenerateImageResult, type GenerateTextResult, LumiAuthClient, LumiClient, type LumiClientConfig, LumiError, type Message, type MessageDataReceive, type MessageDataSend, type MessageInitData, type MessageMedia, type MessageSignInData, MessageType, type Optional, type PaginationData, type StorageAdapter, StorageKey, type StreamChunk, ToolsClient, type UploadItem, type User, createClient, createDefaultAuthAdapter, createDefaultStorageAdapter };
|
package/dist/index.d.ts
CHANGED
|
@@ -1,5 +1,32 @@
|
|
|
1
1
|
import { FetchOptions } from 'ofetch';
|
|
2
2
|
|
|
3
|
+
/** popup 句柄 */
|
|
4
|
+
interface AuthPopupHandle {
|
|
5
|
+
/** 向 popup 发送消息 */
|
|
6
|
+
postMessage: (data: any) => void;
|
|
7
|
+
/** 关闭 popup */
|
|
8
|
+
close: () => void;
|
|
9
|
+
/** 聚焦 popup(可选) */
|
|
10
|
+
focus?: () => void;
|
|
11
|
+
/** 检查 popup 是否已关闭 */
|
|
12
|
+
closed: boolean;
|
|
13
|
+
}
|
|
14
|
+
/** window 适配器 */
|
|
15
|
+
interface AuthAdapter {
|
|
16
|
+
/** 打开认证弹窗。成功返回 popup 句柄,失败返回 null */
|
|
17
|
+
open: (url: string, name?: string) => AuthPopupHandle | null;
|
|
18
|
+
/** 监听来自指定 popup 的消息。返回取消监听函数 */
|
|
19
|
+
listenMessage: (callback: (data: any, origin: string) => void) => () => void;
|
|
20
|
+
/** 监听全局点击。返回取消监听函数 */
|
|
21
|
+
listenGlobalClick?: (callback: (e: MouseEvent) => void) => () => void;
|
|
22
|
+
/** 获取当前页面的 icon */
|
|
23
|
+
getIcon: () => string | null;
|
|
24
|
+
/** 获取当前页面的 title */
|
|
25
|
+
getTitle: () => string | null;
|
|
26
|
+
}
|
|
27
|
+
/** 创建默认的 window 适配器 */
|
|
28
|
+
declare function createDefaultAuthAdapter(): AuthAdapter;
|
|
29
|
+
|
|
3
30
|
interface StorageAdapter {
|
|
4
31
|
storage: {
|
|
5
32
|
getItem: (key: string) => string | null;
|
|
@@ -8,9 +35,15 @@ interface StorageAdapter {
|
|
|
8
35
|
};
|
|
9
36
|
listen: (callback: (key: string | null | undefined, value: string | null) => void) => () => void;
|
|
10
37
|
}
|
|
38
|
+
declare function createDefaultStorageAdapter(): StorageAdapter;
|
|
11
39
|
|
|
12
40
|
type Optional<T, K extends keyof T> = Omit<T, K> & Partial<Pick<T, K>>;
|
|
13
41
|
|
|
42
|
+
interface ApiResponse<T> {
|
|
43
|
+
code: number;
|
|
44
|
+
message: string;
|
|
45
|
+
data: T;
|
|
46
|
+
}
|
|
14
47
|
interface PaginationData<T> {
|
|
15
48
|
total: number;
|
|
16
49
|
list: T[];
|
|
@@ -147,6 +180,7 @@ interface LumiClientConfig {
|
|
|
147
180
|
apiBaseUrl: string;
|
|
148
181
|
authOrigin: string;
|
|
149
182
|
storageAdapter: StorageAdapter;
|
|
183
|
+
authAdapter: AuthAdapter;
|
|
150
184
|
authorization?: string;
|
|
151
185
|
}
|
|
152
186
|
declare class LumiClient {
|
|
@@ -155,10 +189,14 @@ declare class LumiClient {
|
|
|
155
189
|
entities: EntitiesClient;
|
|
156
190
|
tools: ToolsClient;
|
|
157
191
|
functions: FunctionsClient;
|
|
158
|
-
constructor(config: Optional<LumiClientConfig, 'storageAdapter'>);
|
|
192
|
+
constructor(config: Optional<LumiClientConfig, 'storageAdapter' | 'authAdapter'>);
|
|
159
193
|
}
|
|
160
|
-
declare function createClient(config: Optional<LumiClientConfig, 'storageAdapter'>): LumiClient;
|
|
194
|
+
declare function createClient(config: Optional<LumiClientConfig, 'storageAdapter' | 'authAdapter'>): LumiClient;
|
|
161
195
|
|
|
196
|
+
declare enum StorageKey {
|
|
197
|
+
ACCESS_TOKEN = "lumi-access-token",
|
|
198
|
+
USER = "lumi-user"
|
|
199
|
+
}
|
|
162
200
|
declare enum MessageType {
|
|
163
201
|
READY = "lumi-ready",
|
|
164
202
|
INIT = "lumi-init",
|
|
@@ -215,4 +253,10 @@ declare class LumiAuthClient {
|
|
|
215
253
|
}) => void): () => void;
|
|
216
254
|
}
|
|
217
255
|
|
|
218
|
-
|
|
256
|
+
declare class LumiError extends Error {
|
|
257
|
+
name: string;
|
|
258
|
+
code: number;
|
|
259
|
+
constructor(code: number, message: string);
|
|
260
|
+
}
|
|
261
|
+
|
|
262
|
+
export { AITool, type ApiResponse, type ApiStreamResponse, type AuthAdapter, type AuthPopupHandle, EmailTool, EntitiesClient, type Entity, EntityClient, FileTool, FunctionsClient, type GenerateImageResult, type GenerateTextResult, LumiAuthClient, LumiClient, type LumiClientConfig, LumiError, type Message, type MessageDataReceive, type MessageDataSend, type MessageInitData, type MessageMedia, type MessageSignInData, MessageType, type Optional, type PaginationData, type StorageAdapter, StorageKey, type StreamChunk, ToolsClient, type UploadItem, type User, createClient, createDefaultAuthAdapter, createDefaultStorageAdapter };
|
package/dist/index.js
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
|
-
"use strict";var we=Object.create;var
|
|
2
|
-
`),
|
|
3
|
-
`),
|
|
4
|
-
${t.data}`)}}}))})}checkLumiApiKey(){if(!w)throw new
|
|
1
|
+
"use strict";var we=Object.create;var R=Object.defineProperty,Ae=Object.defineProperties,Se=Object.getOwnPropertyDescriptor,be=Object.getOwnPropertyDescriptors,Ee=Object.getOwnPropertyNames,X=Object.getOwnPropertySymbols,Ie=Object.getPrototypeOf,W=Object.prototype.hasOwnProperty,Te=Object.prototype.propertyIsEnumerable;var ee=r=>{throw TypeError(r)};var Z=(r,e,t)=>e in r?R(r,e,{enumerable:!0,configurable:!0,writable:!0,value:t}):r[e]=t,I=(r,e)=>{for(var t in e||(e={}))W.call(e,t)&&Z(r,t,e[t]);if(X)for(var t of X(e))Te.call(e,t)&&Z(r,t,e[t]);return r},_=(r,e)=>Ae(r,be(e));var Ce=(r,e)=>{for(var t in e)R(r,t,{get:e[t],enumerable:!0})},te=(r,e,t,i)=>{if(e&&typeof e=="object"||typeof e=="function")for(let o of Ee(e))!W.call(r,o)&&o!==t&&R(r,o,{get:()=>e[o],enumerable:!(i=Se(e,o))||i.enumerable});return r};var L=(r,e,t)=>(t=r!=null?we(Ie(r)):{},te(e||!r||!r.__esModule?R(t,"default",{value:r,enumerable:!0}):t,r)),xe=r=>te(R({},"__esModule",{value:!0}),r);var re=(r,e,t)=>e.has(r)||ee("Cannot "+t);var s=(r,e,t)=>(re(r,e,"read from private field"),t?t.call(r):e.get(r)),p=(r,e,t)=>e.has(r)?ee("Cannot add the same private member more than once"):e instanceof WeakSet?e.add(r):e.set(r,t),d=(r,e,t,i)=>(re(r,e,"write to private field"),i?i.call(r,t):e.set(r,t),t);var a=(r,e,t)=>new Promise((i,o)=>{var n=h=>{try{f(t.next(h))}catch(y){o(y)}},c=h=>{try{f(t.throw(h))}catch(y){o(y)}},f=h=>h.done?i(h.value):Promise.resolve(h.value).then(n,c);f((t=t.apply(r,e)).next())});var Me={};Ce(Me,{AITool:()=>N,EmailTool:()=>q,EntitiesClient:()=>D,EntityClient:()=>O,FileTool:()=>$,FunctionsClient:()=>U,LumiAuthClient:()=>P,LumiClient:()=>B,LumiError:()=>u,MessageType:()=>J,StorageKey:()=>z,ToolsClient:()=>j,createClient:()=>Pe,createDefaultAuthAdapter:()=>F,createDefaultStorageAdapter:()=>K});module.exports=xe(Me);var he=require("uuid");var z=(t=>(t.ACCESS_TOKEN="lumi-access-token",t.USER="lumi-user",t))(z||{}),J=(i=>(i.READY="lumi-ready",i.INIT="lumi-init",i.SIGN_IN="lumi-sign-in",i))(J||{});var u=class extends Error{constructor(t,i){super(i);this.name="LumiError";this.code=t}};var ce=L(require("crypto-js/enc-base64")),le=L(require("crypto-js/enc-hex")),me=L(require("crypto-js/hmac-sha256")),ue=L(require("crypto-js/sha256")),pe=L(require("object-hash")),Q=require("ofetch");var ie,w=typeof window=="undefined"&&((ie=globalThis.navigator)==null?void 0:ie.product)!=="ReactNative";function se(r){return new Promise(e=>setTimeout(e,r))}function ne(r){var e;return typeof process!="undefined"&&(e=process==null?void 0:process.env[r])!=null?e:null}var ve="6QrJZ7pFCmBZAeIJF7IArvkCz+EtzA0RVcpHkiQIsQyhs7QtCS9P+CueZdHfB2OtJcgX3BbqY9pfpWeAVTqCwQ==";function oe(r){return encodeURIComponent(r).replace(/[!'()*]/g,e=>`%${e.charCodeAt(0).toString(16).toUpperCase()}`)}function Re(r){let e=Math.floor(Date.now()/1e3).toString(),t=Math.random().toString(36).substring(2,15),i=I({},r.query),o=Object.keys(i).sort().map(m=>`${oe(m)}=${oe(String(i[m]))}`).join("&"),n={"x-timestamp":e,"x-nonce":t},c=Object.keys(n).sort().map(m=>`${m}:${n[m]}`).join(`
|
|
2
|
+
`),f=r.body&&!(r.body instanceof FormData)?JSON.stringify(r.body):"",h=(0,ue.default)(f).toString(le.default),y=[o,c,h].join(`
|
|
3
|
+
`),b=ce.default.stringify((0,me.default)(y,ve)),E=new Headers(r.headers);Object.entries(n).forEach(([m,x])=>{E.set(m,x)}),E.set("X-Sign",b),r.headers=E}var ae=new Map;function Le(r,e){var c;let t=I({},e);e.body instanceof FormData&&(t.body=Array.from(e.body.entries())),t.headers=void 0;let i=(0,pe.default)([r,t]),o=Date.now(),n=((c=ae.get(i))==null?void 0:c.filter(f=>o-f<1e3))||[];if(n.length>=4)throw new u(429,"Too many requests");n.push(o),ae.set(i,n)}function de(r,e,t,i=!1){t.headers=new Headers(t.headers),r.auth.accessToken&&t.headers.set("Authorization",`Bearer ${r.auth.accessToken}`),i&&(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")),Le(e,t),Re(t)}function l(r,e,t={}){de(r,e,t);let i=r.auth.isAuthenticated;return(0,Q.ofetch)(e,_(I({baseURL:r.config.apiBaseUrl},t),{onResponse:({response:o})=>{var n;!w&&i&&((n=o._data)==null?void 0:n.code)===2100&&r.auth.signOut()}}))}function ge(i,o){return a(this,arguments,function*(r,e,t={}){return de(r,e,t,!0),yield(0,Q.ofetch)(e,_(I({baseURL:r.config.apiBaseUrl},t),{responseType:"stream"}))})}function Y(r,e,t){let i=r.config.storageAdapter.storage;t!==null?i.setItem(e,JSON.stringify(t)):i.removeItem(e)}function V(r,e){let i=r.config.storageAdapter.storage.getItem(e);try{return i?JSON.parse(i):null}catch(o){return null}}var g,G,M,P=class{constructor(e){p(this,g);p(this,G,`lumi-auth-${(0,he.v4)()}`);p(this,M,null);d(this,g,e),Promise.resolve().then(()=>{!w&&this.isAuthenticated&&this.refreshUser()})}get accessToken(){if(w){let e=s(this,g).config.authorization;return e?e.replace("Bearer ",""):null}return V(s(this,g),"lumi-access-token")}set accessToken(e){if(w){s(this,g).config.authorization=e?`Bearer ${e}`:void 0;return}Y(s(this,g),"lumi-access-token",e)}get user(){return w?s(this,M):V(s(this,g),"lumi-user")}set user(e){if(w){d(this,M,e);return}Y(s(this,g),"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{authAdapter:e}=s(this,g).config,t=e.open(s(this,g).config.authOrigin,s(this,G)),i;return new Promise((o,n)=>{var E;if(!t)return n(new Error("Open auth window failed"));let c=setInterval(()=>{t.closed&&n(new u(499,"Auth window closed"))},1e3),f=m=>{var x;t.closed||((x=t.focus)==null||x.call(t),m.stopPropagation(),m.preventDefault())},h=(m,x)=>{if(!(x!==s(this,g).config.authOrigin||!m))switch(m.type){case"lumi-ready":{t.postMessage({type:"lumi-init",data:{projectId:s(this,g).config.projectId,icon:e.getIcon(),title:e.getTitle()}});break}case"lumi-sign-in":{if(m.data.projectId!==s(this,g).config.projectId)break;t.close(),this.accessToken=m.data.accessToken,this.user=m.data.user,o(m.data);break}}},y=e.listenMessage(h),b=(E=e.listenGlobalClick)==null?void 0:E.call(e,f);i=()=>{clearInterval(c),y(),b==null||b()}}).finally(()=>i==null?void 0:i())}signOut(){if(w)throw new Error("auth.signOut() can only be called on the client side");this.accessToken=null,this.user=null}refreshUser(){return a(this,null,function*(){let e=yield l(s(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=s(this,g).config.storageAdapter.listen(i=>{(i==="lumi-access-token"||i==="lumi-user"||!i)&&e({isAuthenticated:this.isAuthenticated,user:this.user})});return()=>t()}};g=new WeakMap,G=new WeakMap,M=new WeakMap;var A,O=class{constructor(e,t){p(this,A);d(this,A,e),this.entityName=t}list(){return a(this,arguments,function*({filter:e,sort:t,limit:i,skip:o}={}){if(i){let n=yield l(s(this,A),this.uri("/find"),{method:"POST",body:{filter:e,sort:t,limit:i,skip:o}});if(n.code!==200)throw new Error(n.message);return n.data}else{let n=yield l(s(this,A),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 a(this,null,function*(){let t=yield l(s(this,A),this.uri(`/${e}`),{method:"GET"});if(t.code!==200)throw new Error(t.message);return t.data})}create(e){return a(this,null,function*(){let t=yield l(s(this,A),this.uri(),{method:"POST",body:e});if(t.code!==200)throw new Error(t.message);return t.data})}createMany(e){return a(this,null,function*(){let t=yield l(s(this,A),this.uri("/batch"),{method:"POST",body:e});if(t.code!==200)throw new Error(t.message);return t.data})}update(e,t){return a(this,null,function*(){let i=yield l(s(this,A),this.uri(),{method:"PUT",body:{filter:{_id:e},update:t}});if(i.code!==200)throw new Error(i.message);return i.data})}delete(e){return a(this,null,function*(){let t=yield l(s(this,A),this.uri(`/${e}`),{method:"DELETE"});if(t.code!==200)throw new Error(t.message)})}deleteMany(e){return a(this,null,function*(){let t=yield l(s(this,A),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,A).config.projectId}/${this.entityName}/documents${e}`}};A=new WeakMap;var k,D=class{constructor(e){p(this,k);return d(this,k,e),new Proxy(this,{get(t,i){return i in t||(t[i]=new O(s(t,k),i)),t[i]}})}};k=new WeakMap;var fe=require("ofetch");var T,U=class{constructor(e){p(this,T);d(this,T,e)}invoke(e,t={}){return s(this,T).auth.accessToken&&(t.headers=I({Authorization:`Bearer ${s(this,T).auth.accessToken}`},t.headers)),(0,fe.ofetch)(`/v1/functions/${s(this,T).config.projectId}/${e}`,I({baseURL:s(this,T).config.apiBaseUrl},t))}};T=new WeakMap;function F(){return{open:(r,e)=>{let o=(window.screen.width-800)/2,n=(window.screen.height-600)/2,c=window.open(r,e,`width=800,height=600,left=${o},top=${n}`);return c?{postMessage:f=>c.postMessage(f,r),close:()=>c.close(),focus:()=>c.focus(),get closed(){return c.closed}}:null},listenMessage:r=>{let e=t=>{r(t.data,t.origin)};return window.addEventListener("message",e),()=>window.removeEventListener("message",e)},listenGlobalClick:r=>(window.addEventListener("click",r,!0),()=>window.removeEventListener("click",r,!0)),getIcon:()=>{var r,e;return(e=(r=document.querySelector('link[rel="icon"]'))==null?void 0:r.href)!=null?e:null},getTitle:()=>document.title}}function K(){return{storage:{getItem:localStorage.getItem,setItem:(r,e)=>{localStorage.setItem(r,e),window.dispatchEvent(new StorageEvent("storage",{key:r,oldValue:null,newValue:e,storageArea:localStorage}))},removeItem:localStorage.removeItem},listen:r=>{let e=t=>{r(t.key,t.newValue)};return window.addEventListener("storage",e),()=>window.removeEventListener("storage",e)}}}var ye=require("eventsource-parser/stream");var S,N=class{constructor(e){p(this,S);d(this,S,e)}generateText(i){return a(this,arguments,function*({model:e="gemini-2.5-flash",messages:t}){let o=this.checkLumiApiKey(),n=yield l(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 u(n.code,n.message);return n.data})}generateTextStream(i){return a(this,arguments,function*({model:e="gemini-2.5-flash",messages:t}){let o=this.checkLumiApiKey();return ge(s(this,S),`/lm/${s(this,S).config.projectId}/ai/chat/stream`,{method:"POST",body:{lumiApiKey:o,modelName:e,chatMessages:t}})})}generateImage(i){return a(this,arguments,function*({model:e="gemini-2.5-flash-image",messages:t}){let o=this.checkLumiApiKey(),{code:n,message:c,data:f}=yield l(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 u(n,c);let h=f,y=1e3;for(;h.generationStatus==="PROCESSING";){yield se(y),y<5e3&&(y+=500);let{code:b,message:E,data:m}=yield l(s(this,S),`/lm/${s(this,S).config.projectId}/ai/image/get`,{method:"POST",body:{lumiApiKey:o,messageId:h.messageId}});if(b!==200)throw new u(b,E);h=m}return h})}parseStream(e){return a(this,null,function*(){return e.pipeThrough(new TextDecoderStream).pipeThrough(new ye.EventSourceParserStream).pipeThrough(new TransformStream({transform:(t,i)=>{try{let o=JSON.parse(t.data);i.enqueue({event:t.event,data:o})}catch(o){console.error(`Parse stream chunk failed: Invalid JSON.
|
|
4
|
+
${t.data}`)}}}))})}checkLumiApiKey(){if(!w)throw new u(400,"lumi.tools.ai is only available on the server-side");let e=ne("LUMI_API_KEY");if(!e)throw new u(400,"LUMI_API_KEY is required");return e}};S=new WeakMap;var v,q=class{constructor(e){p(this,v);d(this,v,e)}send(h){return a(this,arguments,function*({to:e,subject:t,fromName:i,html:o,text:n="",replyTo:c,scheduledAt:f}){if(!e||!t||!o&&!n)throw new Error("Failed to send email: Missing required parameters.");typeof e=="string"&&(e=[e]),typeof c=="string"&&(c=[c]);let y=yield l(s(this,v),`/lm/${s(this,v).config.projectId}/email/send`,{method:"POST",body:{to:e,subject:t,fromName:i,html:o,text:n,replyTo:c,scheduledAt:f}});if(y.code!==200)throw new u(y.code,y.message)})}};v=new WeakMap;var C,$=class{constructor(e){p(this,C);d(this,C,e)}upload(e){return a(this,null,function*(){let t=new FormData;e.forEach(o=>{t.append("files",o)});let i=yield l(s(this,C),`/lm/${s(this,C).config.projectId}/file/batch`,{method:"POST",body:t});if(i.code!==200)throw new u(i.code,i.message);return i.data})}delete(e){return a(this,null,function*(){let t=yield l(s(this,C),`/lm/${s(this,C).config.projectId}/file/batch`,{method:"DELETE",body:{fileUrls:e}});if(t.code!==200)throw new u(t.code,t.message)})}};C=new WeakMap;var H,j=class{constructor(e){p(this,H);d(this,H,e),this.email=new q(e),this.file=new $(e),this.ai=new N(e)}};H=new WeakMap;var B=class{constructor(e){var t,i;(t=e.storageAdapter)!=null||(e.storageAdapter=K()),(i=e.authAdapter)!=null||(e.authAdapter=F()),this.config=e,this.auth=new P(this),this.entities=new D(this),this.tools=new j(this),this.functions=new U(this)}};function Pe(r){return new B(r)}0&&(module.exports={AITool,EmailTool,EntitiesClient,EntityClient,FileTool,FunctionsClient,LumiAuthClient,LumiClient,LumiError,MessageType,StorageKey,ToolsClient,createClient,createDefaultAuthAdapter,createDefaultStorageAdapter});
|
|
5
5
|
//# 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/adapters/storage.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>(this.#lumi, 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(this.#lumi, 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>(this.#lumi, StorageKey.USER)\n }\n\n public set user(user: User | null) {\n if (isServer) {\n this.#user = user\n return\n }\n setStorage(this.#lumi, 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 unlisten = this.#lumi.config.storageAdapter.listen((key): void => {\n if (key === StorageKey.ACCESS_TOKEN || key === StorageKey.USER || !key) {\n callback({\n isAuthenticated: this.isAuthenticated,\n user: this.user,\n })\n }\n })\n\n return () => unlisten()\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 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 type ApiStreamResponse = ReadableStream<Uint8Array>\n\n/** 网络请求(流式) */\nexport async function requestStream(lumi: LumiClient, uri: string, options: FetchOptions<'stream'> = {}): Promise<ApiStreamResponse> {\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}\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' && globalThis.navigator?.product !== 'ReactNative'\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'\nimport type { LumiClient } from '@/core/lumi-client'\n\nexport function setStorage<T>(lumi: LumiClient, key: StorageKey, value: T): void {\n const storage = lumi.config.storageAdapter.storage\n if (value !== null)\n storage.setItem(key, JSON.stringify(value))\n else\n storage.removeItem(key)\n}\n\nexport function getStorage<T>(lumi: LumiClient, key: StorageKey): T | null {\n const storage = lumi.config.storageAdapter.storage\n const value = storage.getItem(key)\n try {\n return value ? JSON.parse(value) : null\n }\n catch {\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 = {}): 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","export interface StorageAdapter {\n storage: {\n getItem: (key: string) => string | null\n setItem: (key: string, value: string) => void\n removeItem: (key: string) => void\n }\n listen: (callback: (key: string | null | undefined, value: string | null) => void) => () => void\n}\n\nexport function createDefaultStorageAdapter(): StorageAdapter {\n return {\n storage: {\n getItem: localStorage.getItem,\n setItem: (key, value) => {\n localStorage.setItem(key, value)\n window.dispatchEvent(new StorageEvent('storage', {\n key,\n oldValue: null,\n newValue: value,\n storageArea: localStorage,\n }))\n },\n removeItem: localStorage.removeItem,\n },\n listen: (callback) => {\n const handler = (e: StorageEvent): void => {\n callback(e.key, e.newValue)\n }\n window.addEventListener('storage', handler)\n return () => window.removeEventListener('storage', handler)\n },\n }\n}\n","import type { LumiClient } from '@/core/lumi-client'\nimport type { ApiResponse, ApiStreamResponse } from '@/lib/request'\nimport { EventSourceParserStream } from 'eventsource-parser/stream'\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 interface StreamChunk<T> {\n event?: string\n data: T\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<ApiStreamResponse> {\n const lumiApiKey = this.checkLumiApiKey()\n\n return requestStream(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 /** 解析流式输出(将原始 SSE 流解析为 GenerateTextResult 块) */\n public async parseStream(stream: ApiStreamResponse): Promise<ReadableStream<StreamChunk<GenerateTextResult>>> {\n return stream\n .pipeThrough(new TextDecoderStream())\n .pipeThrough(new EventSourceParserStream())\n .pipeThrough<StreamChunk<GenerateTextResult>>(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 console.error(`Parse stream chunk failed: Invalid JSON.\\n${chunk.data}`)\n }\n },\n }))\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 type { StorageAdapter } from '@/adapters/storage'\nimport type { Optional } from '@/types/utils'\nimport { createDefaultStorageAdapter } from '@/adapters/storage'\nimport { 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 storageAdapter: StorageAdapter\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: Optional<LumiClientConfig, 'storageAdapter'>) {\n config.storageAdapter ??= createDefaultStorageAdapter()\n\n this.config = config as LumiClientConfig\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: Optional<LumiClientConfig, 'storageAdapter'>): 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_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","setStorage","lumi","key","value","storage","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","unlisten","key","_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","createDefaultStorageAdapter","key","value","callback","handler","e","import_stream","_lumi","AITool","lumi","__privateAdd","__privateSet","_0","__async","model","messages","lumiApiKey","res","request","__privateGet","LumiError","requestStream","code","message","data","result","delayMs","delay","stream","chunk","controller","e","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","_a","createDefaultStorageAdapter","LumiAuthClient","EntitiesClient","ToolsClient","FunctionsClient","createClient"]}
|
|
1
|
+
{"version":3,"sources":["../src/index.ts","../src/core/auth-client.ts","../src/constants.ts","../src/lib/error.ts","../src/lib/request.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/adapters/auth.ts","../src/adapters/storage.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'\nexport * from './types'\n","import type { LumiClient } from '@/core/lumi-client'\nimport type { ApiResponse } from '@/lib/request'\nimport { v4 } from 'uuid'\nimport { MessageType, StorageKey } from '@/constants'\nimport { LumiError } from '@/lib/error'\nimport { request } from '@/lib/request'\nimport { 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>(this.#lumi, 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(this.#lumi, 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>(this.#lumi, StorageKey.USER)\n }\n\n public set user(user: User | null) {\n if (isServer) {\n this.#user = user\n return\n }\n setStorage(this.#lumi, 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 { authAdapter } = this.#lumi.config\n\n const popup = authAdapter.open(this.#lumi.config.authOrigin, this.#popupName)\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 LumiError(499, 'Auth window closed'))\n }, 1000)\n\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: MessageDataReceive, origin: string): void => {\n if (origin !== this.#lumi.config.authOrigin || !data)\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: authAdapter.getIcon(),\n title: authAdapter.getTitle(),\n },\n } satisfies MessageDataSend)\n break\n }\n case MessageType.SIGN_IN: {\n if (data.data.projectId !== this.#lumi.config.projectId)\n break\n popup.close()\n this.accessToken = data.data.accessToken\n this.user = data.data.user\n resolve(data.data)\n break\n }\n }\n }\n\n const unlistenMessage = authAdapter.listenMessage(handleMessage)\n const unlistenGlobalClick = authAdapter.listenGlobalClick?.(handleGlobalClick)\n\n cleanup = () => {\n clearInterval(timer)\n unlistenMessage()\n unlistenGlobalClick?.()\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 unlisten = this.#lumi.config.storageAdapter.listen((key): void => {\n if (key === StorageKey.ACCESS_TOKEN || key === StorageKey.USER || !key) {\n callback({\n isAuthenticated: this.isAuthenticated,\n user: this.user,\n })\n }\n })\n\n return () => unlisten()\n }\n}\n","export enum StorageKey {\n ACCESS_TOKEN = 'lumi-access-token',\n USER = 'lumi-user',\n}\n\nexport enum MessageType {\n READY = 'lumi-ready',\n INIT = 'lumi-init',\n SIGN_IN = 'lumi-sign-in',\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","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 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 type ApiStreamResponse = ReadableStream<Uint8Array>\n\n/** 网络请求(流式) */\nexport async function requestStream(lumi: LumiClient, uri: string, options: FetchOptions<'stream'> = {}): Promise<ApiStreamResponse> {\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}\n","export const isServer = typeof window === 'undefined' && globalThis.navigator?.product !== 'ReactNative'\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'\nimport type { LumiClient } from '@/core/lumi-client'\n\nexport function setStorage<T>(lumi: LumiClient, key: StorageKey, value: T): void {\n const storage = lumi.config.storageAdapter.storage\n if (value !== null)\n storage.setItem(key, JSON.stringify(value))\n else\n storage.removeItem(key)\n}\n\nexport function getStorage<T>(lumi: LumiClient, key: StorageKey): T | null {\n const storage = lumi.config.storageAdapter.storage\n const value = storage.getItem(key)\n try {\n return value ? JSON.parse(value) : null\n }\n catch {\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 = {}): 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","/** popup 句柄 */\nexport interface AuthPopupHandle {\n /** 向 popup 发送消息 */\n postMessage: (data: any) => void\n /** 关闭 popup */\n close: () => void\n /** 聚焦 popup(可选) */\n focus?: () => void\n /** 检查 popup 是否已关闭 */\n closed: boolean\n}\n\n/** window 适配器 */\nexport interface AuthAdapter {\n /** 打开认证弹窗。成功返回 popup 句柄,失败返回 null */\n open: (url: string, name?: string) => AuthPopupHandle | null\n /** 监听来自指定 popup 的消息。返回取消监听函数 */\n listenMessage: (callback: (data: any, origin: string) => void) => () => void\n /** 监听全局点击。返回取消监听函数 */\n listenGlobalClick?: (callback: (e: MouseEvent) => void) => () => void\n /** 获取当前页面的 icon */\n getIcon: () => string | null\n /** 获取当前页面的 title */\n getTitle: () => string | null\n}\n\n/** 创建默认的 window 适配器 */\nexport function createDefaultAuthAdapter(): AuthAdapter {\n return {\n open: (url: string, name?: string) => {\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(url, name, `width=${width},height=${height},left=${left},top=${top}`)\n\n if (!popup)\n return null\n\n return {\n postMessage: data => popup.postMessage(data, url),\n close: () => popup.close(),\n focus: () => popup.focus(),\n get closed() { return popup.closed },\n } satisfies AuthPopupHandle\n },\n\n listenMessage: (callback) => {\n const handler = (e: MessageEvent): void => {\n callback(e.data, e.origin)\n }\n window.addEventListener('message', handler)\n return () => window.removeEventListener('message', handler)\n },\n\n listenGlobalClick: (callback) => {\n window.addEventListener('click', callback, true)\n return () => window.removeEventListener('click', callback, true)\n },\n\n getIcon: () => document.querySelector<HTMLLinkElement>('link[rel=\"icon\"]')?.href ?? null,\n\n getTitle: () => document.title,\n }\n}\n","export interface StorageAdapter {\n storage: {\n getItem: (key: string) => string | null\n setItem: (key: string, value: string) => void\n removeItem: (key: string) => void\n }\n listen: (callback: (key: string | null | undefined, value: string | null) => void) => () => void\n}\n\nexport function createDefaultStorageAdapter(): StorageAdapter {\n return {\n storage: {\n getItem: localStorage.getItem,\n setItem: (key, value) => {\n localStorage.setItem(key, value)\n window.dispatchEvent(new StorageEvent('storage', {\n key,\n oldValue: null,\n newValue: value,\n storageArea: localStorage,\n }))\n },\n removeItem: localStorage.removeItem,\n },\n listen: (callback) => {\n const handler = (e: StorageEvent): void => {\n callback(e.key, e.newValue)\n }\n window.addEventListener('storage', handler)\n return () => window.removeEventListener('storage', handler)\n },\n }\n}\n","import type { LumiClient } from '@/core/lumi-client'\nimport type { ApiResponse, ApiStreamResponse } from '@/lib/request'\nimport { EventSourceParserStream } from 'eventsource-parser/stream'\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 interface StreamChunk<T> {\n event?: string\n data: T\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<ApiStreamResponse> {\n const lumiApiKey = this.checkLumiApiKey()\n\n return requestStream(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 /** 解析流式输出(将原始 SSE 流解析为 GenerateTextResult 块) */\n public async parseStream(stream: ApiStreamResponse): Promise<ReadableStream<StreamChunk<GenerateTextResult>>> {\n return stream\n .pipeThrough(new TextDecoderStream())\n .pipeThrough(new EventSourceParserStream())\n .pipeThrough<StreamChunk<GenerateTextResult>>(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 console.error(`Parse stream chunk failed: Invalid JSON.\\n${chunk.data}`)\n }\n },\n }))\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 type { AuthAdapter } from '@/adapters/auth'\nimport type { StorageAdapter } from '@/adapters/storage'\nimport type { Optional } from '@/types/utils'\nimport { createDefaultAuthAdapter } from '@/adapters/auth'\nimport { createDefaultStorageAdapter } from '@/adapters/storage'\nimport { 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 storageAdapter: StorageAdapter\n authAdapter: AuthAdapter\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: Optional<LumiClientConfig, 'storageAdapter' | 'authAdapter'>) {\n config.storageAdapter ??= createDefaultStorageAdapter()\n config.authAdapter ??= createDefaultAuthAdapter()\n\n this.config = config as LumiClientConfig\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: Optional<LumiClientConfig, 'storageAdapter' | 'authAdapter'>): 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","LumiError","MessageType","StorageKey","ToolsClient","createClient","createDefaultAuthAdapter","createDefaultStorageAdapter","__toCommonJS","import_uuid","StorageKey","MessageType","LumiError","code","message","import_enc_base64","import_enc_hex","import_hmac_sha256","import_sha256","import_object_hash","import_ofetch","_a","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","setStorage","lumi","key","value","storage","getStorage","e","_lumi","_popupName","_user","LumiAuthClient","lumi","__privateAdd","__privateSet","isServer","authorization","__privateGet","getStorage","accessToken","setStorage","user","authAdapter","popup","cleanup","resolve","reject","_a","timer","LumiError","handleGlobalClick","event","handleMessage","data","origin","unlistenMessage","unlistenGlobalClick","__async","res","request","callback","unlisten","key","_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","createDefaultAuthAdapter","url","name","left","top","popup","data","callback","handler","e","_a","_b","createDefaultStorageAdapter","key","value","callback","handler","e","import_stream","_lumi","AITool","lumi","__privateAdd","__privateSet","_0","__async","model","messages","lumiApiKey","res","request","__privateGet","LumiError","requestStream","code","message","data","result","delayMs","delay","stream","chunk","controller","e","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","_a","_b","createDefaultStorageAdapter","createDefaultAuthAdapter","LumiAuthClient","EntitiesClient","ToolsClient","FunctionsClient","createClient"]}
|
package/dist/index.mjs
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
|
-
var oe=Object.defineProperty,ae=Object.defineProperties;var ce=Object.getOwnPropertyDescriptors;var
|
|
2
|
-
`),
|
|
3
|
-
`),
|
|
4
|
-
${t.data}`)}}}))})}checkLumiApiKey(){if(!w)throw new
|
|
1
|
+
var oe=Object.defineProperty,ae=Object.defineProperties;var ce=Object.getOwnPropertyDescriptors;var z=Object.getOwnPropertySymbols;var le=Object.prototype.hasOwnProperty,me=Object.prototype.propertyIsEnumerable;var Q=r=>{throw TypeError(r)};var J=(r,e,t)=>e in r?oe(r,e,{enumerable:!0,configurable:!0,writable:!0,value:t}):r[e]=t,I=(r,e)=>{for(var t in e||(e={}))le.call(e,t)&&J(r,t,e[t]);if(z)for(var t of z(e))me.call(e,t)&&J(r,t,e[t]);return r},G=(r,e)=>ae(r,ce(e));var Y=(r,e,t)=>e.has(r)||Q("Cannot "+t);var s=(r,e,t)=>(Y(r,e,"read from private field"),t?t.call(r):e.get(r)),u=(r,e,t)=>e.has(r)?Q("Cannot add the same private member more than once"):e instanceof WeakSet?e.add(r):e.set(r,t),d=(r,e,t,i)=>(Y(r,e,"write to private field"),i?i.call(r,t):e.set(r,t),t);var a=(r,e,t)=>new Promise((i,o)=>{var n=h=>{try{f(t.next(h))}catch(y){o(y)}},c=h=>{try{f(t.throw(h))}catch(y){o(y)}},f=h=>h.done?i(h.value):Promise.resolve(h.value).then(n,c);f((t=t.apply(r,e)).next())});import{v4 as Ae}from"uuid";var V=(t=>(t.ACCESS_TOKEN="lumi-access-token",t.USER="lumi-user",t))(V||{}),X=(i=>(i.READY="lumi-ready",i.INIT="lumi-init",i.SIGN_IN="lumi-sign-in",i))(X||{});var p=class extends Error{constructor(t,i){super(i);this.name="LumiError";this.code=t}};import ue from"crypto-js/enc-base64";import pe from"crypto-js/enc-hex";import de from"crypto-js/hmac-sha256";import ge from"crypto-js/sha256";import he from"object-hash";import{ofetch as ie}from"ofetch";var Z,w=typeof window=="undefined"&&((Z=globalThis.navigator)==null?void 0:Z.product)!=="ReactNative";function W(r){return new Promise(e=>setTimeout(e,r))}function ee(r){var e;return typeof process!="undefined"&&(e=process==null?void 0:process.env[r])!=null?e:null}var fe="6QrJZ7pFCmBZAeIJF7IArvkCz+EtzA0RVcpHkiQIsQyhs7QtCS9P+CueZdHfB2OtJcgX3BbqY9pfpWeAVTqCwQ==";function te(r){return encodeURIComponent(r).replace(/[!'()*]/g,e=>`%${e.charCodeAt(0).toString(16).toUpperCase()}`)}function ye(r){let e=Math.floor(Date.now()/1e3).toString(),t=Math.random().toString(36).substring(2,15),i=I({},r.query),o=Object.keys(i).sort().map(m=>`${te(m)}=${te(String(i[m]))}`).join("&"),n={"x-timestamp":e,"x-nonce":t},c=Object.keys(n).sort().map(m=>`${m}:${n[m]}`).join(`
|
|
2
|
+
`),f=r.body&&!(r.body instanceof FormData)?JSON.stringify(r.body):"",h=ge(f).toString(pe),y=[o,c,h].join(`
|
|
3
|
+
`),b=ue.stringify(de(y,fe)),E=new Headers(r.headers);Object.entries(n).forEach(([m,x])=>{E.set(m,x)}),E.set("X-Sign",b),r.headers=E}var re=new Map;function we(r,e){var c;let t=I({},e);e.body instanceof FormData&&(t.body=Array.from(e.body.entries())),t.headers=void 0;let i=he([r,t]),o=Date.now(),n=((c=re.get(i))==null?void 0:c.filter(f=>o-f<1e3))||[];if(n.length>=4)throw new p(429,"Too many requests");n.push(o),re.set(i,n)}function se(r,e,t,i=!1){t.headers=new Headers(t.headers),r.auth.accessToken&&t.headers.set("Authorization",`Bearer ${r.auth.accessToken}`),i&&(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")),we(e,t),ye(t)}function l(r,e,t={}){se(r,e,t);let i=r.auth.isAuthenticated;return ie(e,G(I({baseURL:r.config.apiBaseUrl},t),{onResponse:({response:o})=>{var n;!w&&i&&((n=o._data)==null?void 0:n.code)===2100&&r.auth.signOut()}}))}function ne(i,o){return a(this,arguments,function*(r,e,t={}){return se(r,e,t,!0),yield ie(e,G(I({baseURL:r.config.apiBaseUrl},t),{responseType:"stream"}))})}function F(r,e,t){let i=r.config.storageAdapter.storage;t!==null?i.setItem(e,JSON.stringify(t)):i.removeItem(e)}function K(r,e){let i=r.config.storageAdapter.storage.getItem(e);try{return i?JSON.parse(i):null}catch(o){return null}}var g,M,R,P=class{constructor(e){u(this,g);u(this,M,`lumi-auth-${Ae()}`);u(this,R,null);d(this,g,e),Promise.resolve().then(()=>{!w&&this.isAuthenticated&&this.refreshUser()})}get accessToken(){if(w){let e=s(this,g).config.authorization;return e?e.replace("Bearer ",""):null}return K(s(this,g),"lumi-access-token")}set accessToken(e){if(w){s(this,g).config.authorization=e?`Bearer ${e}`:void 0;return}F(s(this,g),"lumi-access-token",e)}get user(){return w?s(this,R):K(s(this,g),"lumi-user")}set user(e){if(w){d(this,R,e);return}F(s(this,g),"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{authAdapter:e}=s(this,g).config,t=e.open(s(this,g).config.authOrigin,s(this,M)),i;return new Promise((o,n)=>{var E;if(!t)return n(new Error("Open auth window failed"));let c=setInterval(()=>{t.closed&&n(new p(499,"Auth window closed"))},1e3),f=m=>{var x;t.closed||((x=t.focus)==null||x.call(t),m.stopPropagation(),m.preventDefault())},h=(m,x)=>{if(!(x!==s(this,g).config.authOrigin||!m))switch(m.type){case"lumi-ready":{t.postMessage({type:"lumi-init",data:{projectId:s(this,g).config.projectId,icon:e.getIcon(),title:e.getTitle()}});break}case"lumi-sign-in":{if(m.data.projectId!==s(this,g).config.projectId)break;t.close(),this.accessToken=m.data.accessToken,this.user=m.data.user,o(m.data);break}}},y=e.listenMessage(h),b=(E=e.listenGlobalClick)==null?void 0:E.call(e,f);i=()=>{clearInterval(c),y(),b==null||b()}}).finally(()=>i==null?void 0:i())}signOut(){if(w)throw new Error("auth.signOut() can only be called on the client side");this.accessToken=null,this.user=null}refreshUser(){return a(this,null,function*(){let e=yield l(s(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=s(this,g).config.storageAdapter.listen(i=>{(i==="lumi-access-token"||i==="lumi-user"||!i)&&e({isAuthenticated:this.isAuthenticated,user:this.user})});return()=>t()}};g=new WeakMap,M=new WeakMap,R=new WeakMap;var A,O=class{constructor(e,t){u(this,A);d(this,A,e),this.entityName=t}list(){return a(this,arguments,function*({filter:e,sort:t,limit:i,skip:o}={}){if(i){let n=yield l(s(this,A),this.uri("/find"),{method:"POST",body:{filter:e,sort:t,limit:i,skip:o}});if(n.code!==200)throw new Error(n.message);return n.data}else{let n=yield l(s(this,A),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 a(this,null,function*(){let t=yield l(s(this,A),this.uri(`/${e}`),{method:"GET"});if(t.code!==200)throw new Error(t.message);return t.data})}create(e){return a(this,null,function*(){let t=yield l(s(this,A),this.uri(),{method:"POST",body:e});if(t.code!==200)throw new Error(t.message);return t.data})}createMany(e){return a(this,null,function*(){let t=yield l(s(this,A),this.uri("/batch"),{method:"POST",body:e});if(t.code!==200)throw new Error(t.message);return t.data})}update(e,t){return a(this,null,function*(){let i=yield l(s(this,A),this.uri(),{method:"PUT",body:{filter:{_id:e},update:t}});if(i.code!==200)throw new Error(i.message);return i.data})}delete(e){return a(this,null,function*(){let t=yield l(s(this,A),this.uri(`/${e}`),{method:"DELETE"});if(t.code!==200)throw new Error(t.message)})}deleteMany(e){return a(this,null,function*(){let t=yield l(s(this,A),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,A).config.projectId}/${this.entityName}/documents${e}`}};A=new WeakMap;var L,D=class{constructor(e){u(this,L);return d(this,L,e),new Proxy(this,{get(t,i){return i in t||(t[i]=new O(s(t,L),i)),t[i]}})}};L=new WeakMap;import{ofetch as Se}from"ofetch";var T,k=class{constructor(e){u(this,T);d(this,T,e)}invoke(e,t={}){return s(this,T).auth.accessToken&&(t.headers=I({Authorization:`Bearer ${s(this,T).auth.accessToken}`},t.headers)),Se(`/v1/functions/${s(this,T).config.projectId}/${e}`,I({baseURL:s(this,T).config.apiBaseUrl},t))}};T=new WeakMap;function H(){return{open:(r,e)=>{let o=(window.screen.width-800)/2,n=(window.screen.height-600)/2,c=window.open(r,e,`width=800,height=600,left=${o},top=${n}`);return c?{postMessage:f=>c.postMessage(f,r),close:()=>c.close(),focus:()=>c.focus(),get closed(){return c.closed}}:null},listenMessage:r=>{let e=t=>{r(t.data,t.origin)};return window.addEventListener("message",e),()=>window.removeEventListener("message",e)},listenGlobalClick:r=>(window.addEventListener("click",r,!0),()=>window.removeEventListener("click",r,!0)),getIcon:()=>{var r,e;return(e=(r=document.querySelector('link[rel="icon"]'))==null?void 0:r.href)!=null?e:null},getTitle:()=>document.title}}function B(){return{storage:{getItem:localStorage.getItem,setItem:(r,e)=>{localStorage.setItem(r,e),window.dispatchEvent(new StorageEvent("storage",{key:r,oldValue:null,newValue:e,storageArea:localStorage}))},removeItem:localStorage.removeItem},listen:r=>{let e=t=>{r(t.key,t.newValue)};return window.addEventListener("storage",e),()=>window.removeEventListener("storage",e)}}}import{EventSourceParserStream as be}from"eventsource-parser/stream";var S,U=class{constructor(e){u(this,S);d(this,S,e)}generateText(i){return a(this,arguments,function*({model:e="gemini-2.5-flash",messages:t}){let o=this.checkLumiApiKey(),n=yield l(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 p(n.code,n.message);return n.data})}generateTextStream(i){return a(this,arguments,function*({model:e="gemini-2.5-flash",messages:t}){let o=this.checkLumiApiKey();return ne(s(this,S),`/lm/${s(this,S).config.projectId}/ai/chat/stream`,{method:"POST",body:{lumiApiKey:o,modelName:e,chatMessages:t}})})}generateImage(i){return a(this,arguments,function*({model:e="gemini-2.5-flash-image",messages:t}){let o=this.checkLumiApiKey(),{code:n,message:c,data:f}=yield l(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 p(n,c);let h=f,y=1e3;for(;h.generationStatus==="PROCESSING";){yield W(y),y<5e3&&(y+=500);let{code:b,message:E,data:m}=yield l(s(this,S),`/lm/${s(this,S).config.projectId}/ai/image/get`,{method:"POST",body:{lumiApiKey:o,messageId:h.messageId}});if(b!==200)throw new p(b,E);h=m}return h})}parseStream(e){return a(this,null,function*(){return e.pipeThrough(new TextDecoderStream).pipeThrough(new be).pipeThrough(new TransformStream({transform:(t,i)=>{try{let o=JSON.parse(t.data);i.enqueue({event:t.event,data:o})}catch(o){console.error(`Parse stream chunk failed: Invalid JSON.
|
|
4
|
+
${t.data}`)}}}))})}checkLumiApiKey(){if(!w)throw new p(400,"lumi.tools.ai is only available on the server-side");let e=ee("LUMI_API_KEY");if(!e)throw new p(400,"LUMI_API_KEY is required");return e}};S=new WeakMap;var v,N=class{constructor(e){u(this,v);d(this,v,e)}send(h){return a(this,arguments,function*({to:e,subject:t,fromName:i,html:o,text:n="",replyTo:c,scheduledAt:f}){if(!e||!t||!o&&!n)throw new Error("Failed to send email: Missing required parameters.");typeof e=="string"&&(e=[e]),typeof c=="string"&&(c=[c]);let y=yield l(s(this,v),`/lm/${s(this,v).config.projectId}/email/send`,{method:"POST",body:{to:e,subject:t,fromName:i,html:o,text:n,replyTo:c,scheduledAt:f}});if(y.code!==200)throw new p(y.code,y.message)})}};v=new WeakMap;var C,q=class{constructor(e){u(this,C);d(this,C,e)}upload(e){return a(this,null,function*(){let t=new FormData;e.forEach(o=>{t.append("files",o)});let i=yield l(s(this,C),`/lm/${s(this,C).config.projectId}/file/batch`,{method:"POST",body:t});if(i.code!==200)throw new p(i.code,i.message);return i.data})}delete(e){return a(this,null,function*(){let t=yield l(s(this,C),`/lm/${s(this,C).config.projectId}/file/batch`,{method:"DELETE",body:{fileUrls:e}});if(t.code!==200)throw new p(t.code,t.message)})}};C=new WeakMap;var j,$=class{constructor(e){u(this,j);d(this,j,e),this.email=new N(e),this.file=new q(e),this.ai=new U(e)}};j=new WeakMap;var _=class{constructor(e){var t,i;(t=e.storageAdapter)!=null||(e.storageAdapter=B()),(i=e.authAdapter)!=null||(e.authAdapter=H()),this.config=e,this.auth=new P(this),this.entities=new D(this),this.tools=new $(this),this.functions=new k(this)}};function Ct(r){return new _(r)}export{U as AITool,N as EmailTool,D as EntitiesClient,O as EntityClient,q as FileTool,k as FunctionsClient,P as LumiAuthClient,_ as LumiClient,p as LumiError,X as MessageType,V as StorageKey,$ as ToolsClient,Ct as createClient,H as createDefaultAuthAdapter,B as createDefaultStorageAdapter};
|
|
5
5
|
//# sourceMappingURL=index.mjs.map
|
package/dist/index.mjs.map
CHANGED
|
@@ -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/adapters/storage.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>(this.#lumi, 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(this.#lumi, 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>(this.#lumi, StorageKey.USER)\n }\n\n public set user(user: User | null) {\n if (isServer) {\n this.#user = user\n return\n }\n setStorage(this.#lumi, 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 unlisten = this.#lumi.config.storageAdapter.listen((key): void => {\n if (key === StorageKey.ACCESS_TOKEN || key === StorageKey.USER || !key) {\n callback({\n isAuthenticated: this.isAuthenticated,\n user: this.user,\n })\n }\n })\n\n return () => unlisten()\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 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 type ApiStreamResponse = ReadableStream<Uint8Array>\n\n/** 网络请求(流式) */\nexport async function requestStream(lumi: LumiClient, uri: string, options: FetchOptions<'stream'> = {}): Promise<ApiStreamResponse> {\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}\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' && globalThis.navigator?.product !== 'ReactNative'\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'\nimport type { LumiClient } from '@/core/lumi-client'\n\nexport function setStorage<T>(lumi: LumiClient, key: StorageKey, value: T): void {\n const storage = lumi.config.storageAdapter.storage\n if (value !== null)\n storage.setItem(key, JSON.stringify(value))\n else\n storage.removeItem(key)\n}\n\nexport function getStorage<T>(lumi: LumiClient, key: StorageKey): T | null {\n const storage = lumi.config.storageAdapter.storage\n const value = storage.getItem(key)\n try {\n return value ? JSON.parse(value) : null\n }\n catch {\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 = {}): 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","export interface StorageAdapter {\n storage: {\n getItem: (key: string) => string | null\n setItem: (key: string, value: string) => void\n removeItem: (key: string) => void\n }\n listen: (callback: (key: string | null | undefined, value: string | null) => void) => () => void\n}\n\nexport function createDefaultStorageAdapter(): StorageAdapter {\n return {\n storage: {\n getItem: localStorage.getItem,\n setItem: (key, value) => {\n localStorage.setItem(key, value)\n window.dispatchEvent(new StorageEvent('storage', {\n key,\n oldValue: null,\n newValue: value,\n storageArea: localStorage,\n }))\n },\n removeItem: localStorage.removeItem,\n },\n listen: (callback) => {\n const handler = (e: StorageEvent): void => {\n callback(e.key, e.newValue)\n }\n window.addEventListener('storage', handler)\n return () => window.removeEventListener('storage', handler)\n },\n }\n}\n","import type { LumiClient } from '@/core/lumi-client'\nimport type { ApiResponse, ApiStreamResponse } from '@/lib/request'\nimport { EventSourceParserStream } from 'eventsource-parser/stream'\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 interface StreamChunk<T> {\n event?: string\n data: T\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<ApiStreamResponse> {\n const lumiApiKey = this.checkLumiApiKey()\n\n return requestStream(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 /** 解析流式输出(将原始 SSE 流解析为 GenerateTextResult 块) */\n public async parseStream(stream: ApiStreamResponse): Promise<ReadableStream<StreamChunk<GenerateTextResult>>> {\n return stream\n .pipeThrough(new TextDecoderStream())\n .pipeThrough(new EventSourceParserStream())\n .pipeThrough<StreamChunk<GenerateTextResult>>(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 console.error(`Parse stream chunk failed: Invalid JSON.\\n${chunk.data}`)\n }\n },\n }))\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 type { StorageAdapter } from '@/adapters/storage'\nimport type { Optional } from '@/types/utils'\nimport { createDefaultStorageAdapter } from '@/adapters/storage'\nimport { 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 storageAdapter: StorageAdapter\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: Optional<LumiClientConfig, 'storageAdapter'>) {\n config.storageAdapter ??= createDefaultStorageAdapter()\n\n this.config = config as LumiClientConfig\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: Optional<LumiClientConfig, 'storageAdapter'>): 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","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","setStorage","lumi","key","value","storage","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","unlisten","key","_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","createDefaultStorageAdapter","key","value","callback","handler","e","EventSourceParserStream","_lumi","AITool","lumi","__privateAdd","__privateSet","_0","__async","model","messages","lumiApiKey","res","request","__privateGet","LumiError","requestStream","code","message","data","result","delayMs","delay","stream","EventSourceParserStream","chunk","controller","e","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","_a","createDefaultStorageAdapter","LumiAuthClient","EntitiesClient","ToolsClient","FunctionsClient","createClient"]}
|
|
1
|
+
{"version":3,"sources":["../src/core/auth-client.ts","../src/constants.ts","../src/lib/error.ts","../src/lib/request.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/adapters/auth.ts","../src/adapters/storage.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 { LumiError } from '@/lib/error'\nimport { request } from '@/lib/request'\nimport { 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>(this.#lumi, 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(this.#lumi, 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>(this.#lumi, StorageKey.USER)\n }\n\n public set user(user: User | null) {\n if (isServer) {\n this.#user = user\n return\n }\n setStorage(this.#lumi, 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 { authAdapter } = this.#lumi.config\n\n const popup = authAdapter.open(this.#lumi.config.authOrigin, this.#popupName)\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 LumiError(499, 'Auth window closed'))\n }, 1000)\n\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: MessageDataReceive, origin: string): void => {\n if (origin !== this.#lumi.config.authOrigin || !data)\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: authAdapter.getIcon(),\n title: authAdapter.getTitle(),\n },\n } satisfies MessageDataSend)\n break\n }\n case MessageType.SIGN_IN: {\n if (data.data.projectId !== this.#lumi.config.projectId)\n break\n popup.close()\n this.accessToken = data.data.accessToken\n this.user = data.data.user\n resolve(data.data)\n break\n }\n }\n }\n\n const unlistenMessage = authAdapter.listenMessage(handleMessage)\n const unlistenGlobalClick = authAdapter.listenGlobalClick?.(handleGlobalClick)\n\n cleanup = () => {\n clearInterval(timer)\n unlistenMessage()\n unlistenGlobalClick?.()\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 unlisten = this.#lumi.config.storageAdapter.listen((key): void => {\n if (key === StorageKey.ACCESS_TOKEN || key === StorageKey.USER || !key) {\n callback({\n isAuthenticated: this.isAuthenticated,\n user: this.user,\n })\n }\n })\n\n return () => unlisten()\n }\n}\n","export enum StorageKey {\n ACCESS_TOKEN = 'lumi-access-token',\n USER = 'lumi-user',\n}\n\nexport enum MessageType {\n READY = 'lumi-ready',\n INIT = 'lumi-init',\n SIGN_IN = 'lumi-sign-in',\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","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 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 type ApiStreamResponse = ReadableStream<Uint8Array>\n\n/** 网络请求(流式) */\nexport async function requestStream(lumi: LumiClient, uri: string, options: FetchOptions<'stream'> = {}): Promise<ApiStreamResponse> {\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}\n","export const isServer = typeof window === 'undefined' && globalThis.navigator?.product !== 'ReactNative'\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'\nimport type { LumiClient } from '@/core/lumi-client'\n\nexport function setStorage<T>(lumi: LumiClient, key: StorageKey, value: T): void {\n const storage = lumi.config.storageAdapter.storage\n if (value !== null)\n storage.setItem(key, JSON.stringify(value))\n else\n storage.removeItem(key)\n}\n\nexport function getStorage<T>(lumi: LumiClient, key: StorageKey): T | null {\n const storage = lumi.config.storageAdapter.storage\n const value = storage.getItem(key)\n try {\n return value ? JSON.parse(value) : null\n }\n catch {\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 = {}): 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","/** popup 句柄 */\nexport interface AuthPopupHandle {\n /** 向 popup 发送消息 */\n postMessage: (data: any) => void\n /** 关闭 popup */\n close: () => void\n /** 聚焦 popup(可选) */\n focus?: () => void\n /** 检查 popup 是否已关闭 */\n closed: boolean\n}\n\n/** window 适配器 */\nexport interface AuthAdapter {\n /** 打开认证弹窗。成功返回 popup 句柄,失败返回 null */\n open: (url: string, name?: string) => AuthPopupHandle | null\n /** 监听来自指定 popup 的消息。返回取消监听函数 */\n listenMessage: (callback: (data: any, origin: string) => void) => () => void\n /** 监听全局点击。返回取消监听函数 */\n listenGlobalClick?: (callback: (e: MouseEvent) => void) => () => void\n /** 获取当前页面的 icon */\n getIcon: () => string | null\n /** 获取当前页面的 title */\n getTitle: () => string | null\n}\n\n/** 创建默认的 window 适配器 */\nexport function createDefaultAuthAdapter(): AuthAdapter {\n return {\n open: (url: string, name?: string) => {\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(url, name, `width=${width},height=${height},left=${left},top=${top}`)\n\n if (!popup)\n return null\n\n return {\n postMessage: data => popup.postMessage(data, url),\n close: () => popup.close(),\n focus: () => popup.focus(),\n get closed() { return popup.closed },\n } satisfies AuthPopupHandle\n },\n\n listenMessage: (callback) => {\n const handler = (e: MessageEvent): void => {\n callback(e.data, e.origin)\n }\n window.addEventListener('message', handler)\n return () => window.removeEventListener('message', handler)\n },\n\n listenGlobalClick: (callback) => {\n window.addEventListener('click', callback, true)\n return () => window.removeEventListener('click', callback, true)\n },\n\n getIcon: () => document.querySelector<HTMLLinkElement>('link[rel=\"icon\"]')?.href ?? null,\n\n getTitle: () => document.title,\n }\n}\n","export interface StorageAdapter {\n storage: {\n getItem: (key: string) => string | null\n setItem: (key: string, value: string) => void\n removeItem: (key: string) => void\n }\n listen: (callback: (key: string | null | undefined, value: string | null) => void) => () => void\n}\n\nexport function createDefaultStorageAdapter(): StorageAdapter {\n return {\n storage: {\n getItem: localStorage.getItem,\n setItem: (key, value) => {\n localStorage.setItem(key, value)\n window.dispatchEvent(new StorageEvent('storage', {\n key,\n oldValue: null,\n newValue: value,\n storageArea: localStorage,\n }))\n },\n removeItem: localStorage.removeItem,\n },\n listen: (callback) => {\n const handler = (e: StorageEvent): void => {\n callback(e.key, e.newValue)\n }\n window.addEventListener('storage', handler)\n return () => window.removeEventListener('storage', handler)\n },\n }\n}\n","import type { LumiClient } from '@/core/lumi-client'\nimport type { ApiResponse, ApiStreamResponse } from '@/lib/request'\nimport { EventSourceParserStream } from 'eventsource-parser/stream'\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 interface StreamChunk<T> {\n event?: string\n data: T\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<ApiStreamResponse> {\n const lumiApiKey = this.checkLumiApiKey()\n\n return requestStream(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 /** 解析流式输出(将原始 SSE 流解析为 GenerateTextResult 块) */\n public async parseStream(stream: ApiStreamResponse): Promise<ReadableStream<StreamChunk<GenerateTextResult>>> {\n return stream\n .pipeThrough(new TextDecoderStream())\n .pipeThrough(new EventSourceParserStream())\n .pipeThrough<StreamChunk<GenerateTextResult>>(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 console.error(`Parse stream chunk failed: Invalid JSON.\\n${chunk.data}`)\n }\n },\n }))\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 type { AuthAdapter } from '@/adapters/auth'\nimport type { StorageAdapter } from '@/adapters/storage'\nimport type { Optional } from '@/types/utils'\nimport { createDefaultAuthAdapter } from '@/adapters/auth'\nimport { createDefaultStorageAdapter } from '@/adapters/storage'\nimport { 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 storageAdapter: StorageAdapter\n authAdapter: AuthAdapter\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: Optional<LumiClientConfig, 'storageAdapter' | 'authAdapter'>) {\n config.storageAdapter ??= createDefaultStorageAdapter()\n config.authAdapter ??= createDefaultAuthAdapter()\n\n this.config = config as LumiClientConfig\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: Optional<LumiClientConfig, 'storageAdapter' | 'authAdapter'>): LumiClient {\n return new LumiClient(config)\n}\n"],"mappings":"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","names":["v4","StorageKey","MessageType","LumiError","code","message","Base64","Hex","HmacSHA256","SHA256","hash","ofetch","_a","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","setStorage","lumi","key","value","storage","getStorage","e","_lumi","_popupName","_user","LumiAuthClient","lumi","__privateAdd","v4","__privateSet","isServer","authorization","__privateGet","getStorage","accessToken","setStorage","user","authAdapter","popup","cleanup","resolve","reject","_a","timer","LumiError","handleGlobalClick","event","handleMessage","data","origin","unlistenMessage","unlistenGlobalClick","__async","res","request","callback","unlisten","key","_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","createDefaultAuthAdapter","url","name","left","top","popup","data","callback","handler","e","_a","_b","createDefaultStorageAdapter","key","value","callback","handler","e","EventSourceParserStream","_lumi","AITool","lumi","__privateAdd","__privateSet","_0","__async","model","messages","lumiApiKey","res","request","__privateGet","LumiError","requestStream","code","message","data","result","delayMs","delay","stream","EventSourceParserStream","chunk","controller","e","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","_a","_b","createDefaultStorageAdapter","createDefaultAuthAdapter","LumiAuthClient","EntitiesClient","ToolsClient","FunctionsClient","createClient"]}
|