@lark-apaas/client-toolkit 1.2.33 → 1.2.35

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.
@@ -9,6 +9,10 @@ export type IUserProfile = {
9
9
  * inactive 2
10
10
  */
11
11
  status?: number;
12
+ /**
13
+ * 飞书 user_id
14
+ */
15
+ lark_user_id?: string;
12
16
  };
13
17
  export type IFileAttachment = {
14
18
  bucket_id: string;
package/lib/auth.d.ts CHANGED
@@ -1 +1,2 @@
1
- export { CanRole, AbilityContext, ROLE_SUBJECT, useAuth, } from '@lark-apaas/auth-sdk';
1
+ export { AuthProvider, useAuth, Can, CanRole, useCan, AbilityContext, useUserPermissions, PermissionClient, ROLE_SUBJECT, } from '@lark-apaas/auth-sdk';
2
+ export type { AuthProviderProps, PermissionApiResponse, PermissionApiConfig, PermissionPointData, AuthSdkConfig, } from '@lark-apaas/auth-sdk';
package/lib/auth.js CHANGED
@@ -1,2 +1,2 @@
1
- import { AbilityContext, CanRole, ROLE_SUBJECT, useAuth } from "@lark-apaas/auth-sdk";
2
- export { AbilityContext, CanRole, ROLE_SUBJECT, useAuth };
1
+ import { AbilityContext, AuthProvider, Can, CanRole, PermissionClient, ROLE_SUBJECT, useAuth, useCan, useUserPermissions } from "@lark-apaas/auth-sdk";
2
+ export { AbilityContext, AuthProvider, Can, CanRole, PermissionClient, ROLE_SUBJECT, useAuth, useCan, useUserPermissions };
@@ -7,6 +7,7 @@ import { isNewPathEnabled } from "../../utils/apiPath.js";
7
7
  import { useIsMobile } from "../../hooks/index.js";
8
8
  import { X } from "lucide-react";
9
9
  import { Sheet, SheetContent, SheetTrigger } from "../ui/drawer.js";
10
+ import { t } from "../../locales/index.js";
10
11
  const Component = ()=>{
11
12
  const HasClosedKey = `miaoda-creatByMiaoda-has-closed-${getAppId()}`;
12
13
  const [visible, setVisible] = useState(!window.localStorage?.getItem(HasClosedKey));
@@ -46,7 +47,7 @@ const Component = ()=>{
46
47
  const link = document.createElement('link');
47
48
  link.rel = 'preload';
48
49
  link.as = 'image';
49
- link.href = 'https://lf3-static.bytednsdoc.com/obj/eden-cn/LMfspH/ljhwZthlaukjlkulzlp/logo/miaodacover-mobile.png';
50
+ link.href = t('safety.cover.mobile');
50
51
  document.head.appendChild(link);
51
52
  }
52
53
  }, [
@@ -74,7 +75,7 @@ const Component = ()=>{
74
75
  }),
75
76
  /*#__PURE__*/ jsx("p", {
76
77
  className: "shrink-0 min-w-[28px] m-0! text-[#EBEBEB]",
77
- children: "妙搭"
78
+ children: t('safety.badge.label')
78
79
  })
79
80
  ]
80
81
  })
@@ -90,7 +91,7 @@ const Component = ()=>{
90
91
  onClick: ()=>setOpen(false)
91
92
  }),
92
93
  /*#__PURE__*/ jsx("img", {
93
- src: "https://lf3-static.bytednsdoc.com/obj/eden-cn/LMfspH/ljhwZthlaukjlkulzlp/logo/miaodacover-mobile.png",
94
+ src: t('safety.cover.mobile'),
94
95
  alt: "",
95
96
  className: "w-full h-full",
96
97
  onClick: ()=>{
@@ -110,12 +111,11 @@ const Component = ()=>{
110
111
  src: "https://lf3-static.bytednsdoc.com/obj/eden-cn/LMfspH/ljhwZthlaukjlkulzlp/icon/icon_company_outlined.svg",
111
112
  className: "shrink-0 w-[14px] h-[14px]"
112
113
  }),
113
- /*#__PURE__*/ jsxs("p", {
114
+ /*#__PURE__*/ jsx("p", {
114
115
  className: "shrink-0 min-w-[96px] m-0! text-[#646A73] text-sm",
115
- children: [
116
- userinfo?.name,
117
- "运营"
118
- ]
116
+ children: t('safety.tenant.operated', {
117
+ name: userinfo?.name ?? ''
118
+ })
119
119
  })
120
120
  ]
121
121
  }),
@@ -128,7 +128,7 @@ const Component = ()=>{
128
128
  }),
129
129
  /*#__PURE__*/ jsx("p", {
130
130
  className: "shrink-0 min-w-[163px] m-0! text-[#646A73] text-sm",
131
- children: "包含 AI 生成内容,请注意甄别"
131
+ children: t('safety.ai.disclaimer')
132
132
  })
133
133
  ]
134
134
  })
@@ -146,14 +146,14 @@ const Component = ()=>{
146
146
  setTimeout(()=>setVisible(false), 200);
147
147
  window.localStorage?.setItem(HasClosedKey, 'true');
148
148
  },
149
- children: "不再展示"
149
+ children: t('safety.button.dontShowAgain')
150
150
  }),
151
151
  /*#__PURE__*/ jsx("div", {
152
152
  className: "flex-1 flex rounded-[99px] items-center justify-center border-[0.5px] border-black bg-black text-white cursor-pointer text-lg py-[12px]",
153
153
  onClick: ()=>{
154
154
  window.open('https://miaoda.feishu.cn/landing', '_blank');
155
155
  },
156
- children: "了解更多"
156
+ children: t('safety.button.learnMore')
157
157
  })
158
158
  ]
159
159
  })
@@ -186,7 +186,7 @@ const Component = ()=>{
186
186
  }),
187
187
  /*#__PURE__*/ jsx("p", {
188
188
  className: "shrink-0 min-w-[60px] m-0! text-[#EBEBEB]",
189
- children: "由妙搭搭建"
189
+ children: t('safety.badge.builtWith')
190
190
  })
191
191
  ]
192
192
  })
@@ -210,7 +210,7 @@ const Component = ()=>{
210
210
  className: "flex flex-col bg-[#1A1A1A]",
211
211
  children: [
212
212
  /*#__PURE__*/ jsx("img", {
213
- src: "https://lf3-static.bytednsdoc.com/obj/eden-cn/LMfspH/ljhwZthlaukjlkulzlp/logo/miaodacover.png",
213
+ src: t('safety.cover.pc'),
214
214
  alt: "",
215
215
  className: "w-[286px] h-[128px] cursor-pointer",
216
216
  onClick: ()=>{
@@ -230,12 +230,11 @@ const Component = ()=>{
230
230
  src: "https://lf3-static.bytednsdoc.com/obj/eden-cn/LMfspH/ljhwZthlaukjlkulzlp/icon/icon_company_outlined.svg",
231
231
  className: "shrink-0 w-[12px] h-[12px]"
232
232
  }),
233
- /*#__PURE__*/ jsxs("p", {
233
+ /*#__PURE__*/ jsx("p", {
234
234
  className: "shrink-0 min-w-[96px] m-0! text-[#a6a6a6]",
235
- children: [
236
- userinfo?.name,
237
- "运营"
238
- ]
235
+ children: t('safety.tenant.operated', {
236
+ name: userinfo?.name ?? ''
237
+ })
239
238
  })
240
239
  ]
241
240
  }),
@@ -248,7 +247,7 @@ const Component = ()=>{
248
247
  }),
249
248
  /*#__PURE__*/ jsx("p", {
250
249
  className: "shrink-0 min-w-[163px] m-0! text-[#a6a6a6]",
251
- children: "包含 AI 生成内容,请注意甄别"
250
+ children: t('safety.ai.disclaimer')
252
251
  })
253
252
  ]
254
253
  })
@@ -266,7 +265,7 @@ const Component = ()=>{
266
265
  setVisible(false);
267
266
  window.localStorage?.setItem(HasClosedKey, 'true');
268
267
  },
269
- children: "不再展示"
268
+ children: t('safety.button.dontShowAgain')
270
269
  }),
271
270
  /*#__PURE__*/ jsx("div", {
272
271
  className: "flex-1 flex rounded-[8px] items-center justify-center h-[34px] border-[0.5px] border-solid border-[#ffffff1c] hover:border-[#ffffff33] bg-[#ffffff08] hover:bg-[#ffffff14] cursor-pointer text-[#ebebeb]",
@@ -274,7 +273,7 @@ const Component = ()=>{
274
273
  onClick: ()=>{
275
274
  window.open('https://miaoda.feishu.cn/landing', '_blank');
276
275
  },
277
- children: "了解更多"
276
+ children: t('safety.button.learnMore')
278
277
  })
279
278
  ]
280
279
  })
@@ -3,11 +3,25 @@ import { logger } from "../logger/index.js";
3
3
  import { getCurrentUserProfile } from "../integrations/getCurrentUserProfile.js";
4
4
  import { getDataloom } from "../integrations/dataloom.js";
5
5
  import { isSparkRuntime } from "../utils/utils.js";
6
+ import { getAxiosForBackend } from "../utils/getAxiosForBackend.js";
6
7
  function getNameFromArray(nameArray) {
7
8
  if (!nameArray || 0 === nameArray.length) return;
8
9
  const chineseName = nameArray.find((item)=>2052 === item.language_code);
9
10
  return chineseName?.text ?? nameArray[0]?.text;
10
11
  }
12
+ let _larkUserIdCache = null;
13
+ let _larkUserIdPromise = null;
14
+ function fetchLarkUserId() {
15
+ if (null !== _larkUserIdCache) return Promise.resolve(_larkUserIdCache);
16
+ if (_larkUserIdPromise) return _larkUserIdPromise;
17
+ _larkUserIdPromise = getAxiosForBackend().get('/api/authnpaas/lark-user-id').then(({ data })=>{
18
+ _larkUserIdCache = data?.lark_user_id || null;
19
+ return _larkUserIdCache;
20
+ }).catch(()=>null).finally(()=>{
21
+ _larkUserIdPromise = null;
22
+ });
23
+ return _larkUserIdPromise;
24
+ }
11
25
  function getCompatibilityUserProfile() {
12
26
  const userInfo = getCurrentUserProfile();
13
27
  return {
@@ -34,6 +48,8 @@ const useCurrentUserProfile = ()=>{
34
48
  userName: userName,
35
49
  userAvatar: info?.avatar?.image?.large
36
50
  };
51
+ const larkUserId = await fetchLarkUserId();
52
+ if (larkUserId) newUserInfo.lark_user_id = larkUserId;
37
53
  if ('development' === process.env.NODE_ENV) logger.info('MiaoDaMetaInfoChanged', newUserInfo);
38
54
  setUserInfo(newUserInfo);
39
55
  };
@@ -0,0 +1,15 @@
1
+ /**
2
+ * 获取当前语言环境下的翻译文案。
3
+ *
4
+ * 支持 `{key}` 形式的变量替换:
5
+ * ```ts
6
+ * t('safety.tenant.operated', { name: '字节' })
7
+ * // zh → "字节运营"
8
+ * // en → "Operated by 字节"
9
+ * ```
10
+ *
11
+ * 如果 key 不存在,fallback 到中文;中文也没有则返回 key 本身。
12
+ */
13
+ export declare function t(key: string, params?: Record<string, string>): string;
14
+ export { getLocale } from '../utils/locale';
15
+ export type { Locale } from '../utils/locale';
@@ -0,0 +1,10 @@
1
+ import messages from "./messages.js";
2
+ import { getLocale } from "../utils/locale.js";
3
+ function t(key, params) {
4
+ const locale = getLocale();
5
+ const entry = messages[key];
6
+ let text = entry?.[locale] ?? entry?.zh ?? key;
7
+ if (params) for (const [k, v] of Object.entries(params))text = text.replace(`{${k}}`, v);
8
+ return text;
9
+ }
10
+ export { getLocale, t };
@@ -0,0 +1,12 @@
1
+ /**
2
+ * 中英文文案源(统一管理)。
3
+ *
4
+ * 每条文案同时包含 zh 和 en,方便对照和维护。
5
+ * Key 规则:`{组件}.{区域}.{标识}`
6
+ * 文案来源:https://bytedance.larkoffice.com/wiki/BJCQwv9auiP9erkudMfcVIIGnRg
7
+ */
8
+ declare const messages: Record<string, {
9
+ zh: string;
10
+ en: string;
11
+ }>;
12
+ export default messages;
@@ -0,0 +1,36 @@
1
+ const messages_messages = {
2
+ 'safety.badge.label': {
3
+ zh: '妙搭',
4
+ en: 'Spark'
5
+ },
6
+ 'safety.badge.builtWith': {
7
+ zh: '由妙搭搭建',
8
+ en: 'Built with Spark'
9
+ },
10
+ 'safety.ai.disclaimer': {
11
+ zh: '包含 AI 生成内容,请注意甄别',
12
+ en: 'AI-generated content. Use with care.'
13
+ },
14
+ 'safety.tenant.operated': {
15
+ zh: '{name}运营',
16
+ en: 'Operated by {name}'
17
+ },
18
+ 'safety.button.dontShowAgain': {
19
+ zh: '不再展示',
20
+ en: "Don't show again"
21
+ },
22
+ 'safety.button.learnMore': {
23
+ zh: '了解更多',
24
+ en: 'Learn more'
25
+ },
26
+ 'safety.cover.pc': {
27
+ zh: 'https://lf3-static.bytednsdoc.com/obj/eden-cn/LMfspH/ljhwZthlaukjlkulzlp/logo/miaodacover.png',
28
+ en: 'https://lf3-static.bytednsdoc.com/obj/eden-cn/LMfspH/ljhwZthlaukjlkulzlp/logo/miaodacover-pcen.png'
29
+ },
30
+ 'safety.cover.mobile': {
31
+ zh: 'https://lf3-static.bytednsdoc.com/obj/eden-cn/LMfspH/ljhwZthlaukjlkulzlp/logo/miaodacover-mobile.png',
32
+ en: 'https://lf3-static.bytednsdoc.com/obj/eden-cn/LMfspH/ljhwZthlaukjlkulzlp/logo/miaodacover-weben.png'
33
+ }
34
+ };
35
+ const messages = messages_messages;
36
+ export { messages as default };
@@ -0,0 +1,9 @@
1
+ /**
2
+ * 统一的语言判断方法。
3
+ *
4
+ * 基于浏览器 Accept-Language(navigator.language)判断。
5
+ * 返回标准化的 locale code:'zh' | 'en'。
6
+ * 目前只区分中英文,未来需要更多语言时在此扩展。
7
+ */
8
+ export type Locale = 'zh' | 'en';
9
+ export declare function getLocale(): Locale;
@@ -0,0 +1,5 @@
1
+ function getLocale() {
2
+ const lang = navigator.language || 'zh';
3
+ return lang.startsWith('zh') ? 'zh' : 'en';
4
+ }
5
+ export { getLocale };
@@ -12,6 +12,11 @@ function safeStringify(obj) {
12
12
  if (value instanceof Set) return Array.from(value);
13
13
  if (void 0 === value) return 'undefined';
14
14
  if ('symbol' == typeof value) return value.toString();
15
+ if (value instanceof Error) return {
16
+ name: value.name,
17
+ message: value.message,
18
+ stack: value.stack
19
+ };
15
20
  return value;
16
21
  });
17
22
  } catch {
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,125 @@
1
+ import { describe, expect, it } from "vitest";
2
+ import { mapLogLevel, processLogParams, safeStringify } from "./safeStringify.js";
3
+ describe('safeStringify', ()=>{
4
+ it('should serialize a plain string', ()=>{
5
+ expect(safeStringify('hello')).toBe('"hello"');
6
+ });
7
+ it('should serialize a plain object', ()=>{
8
+ expect(safeStringify({
9
+ a: 1
10
+ })).toBe('{"a":1}');
11
+ });
12
+ it('should handle circular references', ()=>{
13
+ const obj = {
14
+ a: 1
15
+ };
16
+ obj.self = obj;
17
+ expect(safeStringify(obj)).toBe('{"a":1,"self":"[Circular]"}');
18
+ });
19
+ it('should serialize BigInt as string', ()=>{
20
+ expect(safeStringify(BigInt(123))).toBe('"123"');
21
+ });
22
+ it('should serialize Date as ISO string', ()=>{
23
+ const date = new Date('2024-01-01T00:00:00.000Z');
24
+ expect(safeStringify(date)).toBe('"2024-01-01T00:00:00.000Z"');
25
+ });
26
+ it('should serialize Map as object', ()=>{
27
+ const map = new Map([
28
+ [
29
+ 'key',
30
+ 'value'
31
+ ]
32
+ ]);
33
+ expect(safeStringify(map)).toBe('{"key":"value"}');
34
+ });
35
+ it('should serialize Set as array', ()=>{
36
+ const set = new Set([
37
+ 1,
38
+ 2,
39
+ 3
40
+ ]);
41
+ expect(safeStringify(set)).toBe('[1,2,3]');
42
+ });
43
+ it('should serialize undefined as string', ()=>{
44
+ expect(safeStringify({
45
+ a: void 0
46
+ })).toBe('{"a":"undefined"}');
47
+ });
48
+ it('should serialize Symbol as string', ()=>{
49
+ expect(safeStringify({
50
+ s: Symbol('test')
51
+ })).toBe('{"s":"Symbol(test)"}');
52
+ });
53
+ it('should serialize Error with name, message and stack', ()=>{
54
+ const err = new Error('something went wrong');
55
+ const result = JSON.parse(safeStringify(err));
56
+ expect(result.name).toBe('Error');
57
+ expect(result.message).toBe('something went wrong');
58
+ expect(result.stack).toContain('something went wrong');
59
+ });
60
+ it('should serialize Error subclass with correct name', ()=>{
61
+ const err = new TypeError('bad type');
62
+ const result = JSON.parse(safeStringify(err));
63
+ expect(result.name).toBe('TypeError');
64
+ expect(result.message).toBe('bad type');
65
+ });
66
+ it('should serialize Error nested in object', ()=>{
67
+ const err = new Error('inner error');
68
+ const result = JSON.parse(safeStringify({
69
+ err
70
+ }));
71
+ expect(result.err.name).toBe('Error');
72
+ expect(result.err.message).toBe('inner error');
73
+ });
74
+ it('should return empty string on unexpected failure', ()=>{
75
+ const bad = {
76
+ toJSON () {
77
+ throw new Error('fail');
78
+ }
79
+ };
80
+ expect(safeStringify(bad)).toBe('');
81
+ });
82
+ });
83
+ describe('processLogParams', ()=>{
84
+ it('should serialize single argument directly', ()=>{
85
+ expect(processLogParams([
86
+ 'hello'
87
+ ])).toBe('"hello"');
88
+ });
89
+ it('should serialize single object argument', ()=>{
90
+ expect(processLogParams([
91
+ {
92
+ a: 1
93
+ }
94
+ ])).toBe('{"a":1}');
95
+ });
96
+ it('should serialize multiple arguments as indexed object', ()=>{
97
+ const result = JSON.parse(processLogParams([
98
+ 'msg',
99
+ {
100
+ key: 'val'
101
+ }
102
+ ]));
103
+ expect(result['0']).toBe('msg');
104
+ expect(result['1']).toEqual({
105
+ key: 'val'
106
+ });
107
+ });
108
+ });
109
+ describe('mapLogLevel', ()=>{
110
+ it('should map error to ERROR', ()=>{
111
+ expect(mapLogLevel('error')).toBe('ERROR');
112
+ });
113
+ it('should map info to INFO', ()=>{
114
+ expect(mapLogLevel('info')).toBe('INFO');
115
+ });
116
+ it('should map success to INFO', ()=>{
117
+ expect(mapLogLevel('success')).toBe('INFO');
118
+ });
119
+ it('should map warn to WARN', ()=>{
120
+ expect(mapLogLevel('warn')).toBe('WARN');
121
+ });
122
+ it('should default unknown level to INFO', ()=>{
123
+ expect(mapLogLevel('debug')).toBe('INFO');
124
+ });
125
+ });
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@lark-apaas/client-toolkit",
3
- "version": "1.2.33",
3
+ "version": "1.2.35",
4
4
  "types": "./lib/index.d.ts",
5
5
  "main": "./lib/index.js",
6
6
  "files": [
@@ -99,12 +99,12 @@
99
99
  "@ant-design/colors": "^7.2.1",
100
100
  "@ant-design/cssinjs": "^1.24.0",
101
101
  "@data-loom/js": "0.4.13",
102
- "@lark-apaas/aily-web-sdk": "^0.0.8",
103
- "@lark-apaas/auth-sdk": "^0.1.3",
102
+ "@lark-apaas/aily-web-sdk": "^0.0.9",
103
+ "@lark-apaas/auth-sdk": "^0.1.4",
104
104
  "@lark-apaas/client-capability": "^0.1.6",
105
105
  "@lark-apaas/internal-slardar": "^0.0.3",
106
106
  "@lark-apaas/miaoda-inspector": "^1.0.23",
107
- "@lark-apaas/observable-web": "^1.0.5",
107
+ "@lark-apaas/observable-web": "^1.0.6",
108
108
  "@radix-ui/react-avatar": "^1.1.10",
109
109
  "@radix-ui/react-popover": "^1.1.15",
110
110
  "@radix-ui/react-slot": "^1.2.3",