@abtnode/blocklet-services 1.16.42-beta-20250412-083409-53f46a85 → 1.16.42-beta-20250413-121549-22e9a196
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/api/index.js +32 -2
- package/api/libs/push-kit/index.js +16 -11
- package/api/routes/user-session.js +49 -40
- package/api/routes/user.js +16 -4
- package/api/services/auth/session.js +37 -10
- package/api/util/attach-shared-utils.js +12 -5
- package/dist/assets/{ArrowDropDown-C_E3qkPd.js → ArrowDropDown-a-C4j7Lx.js} +1 -1
- package/dist/assets/{CheckCircle-D7Ntmb2o.js → CheckCircle-CuYzdCtm.js} +1 -1
- package/dist/assets/{ChevronLeft-BIgT65EY.js → ChevronLeft-CnflXey6.js} +1 -1
- package/dist/assets/{ChevronRight-foZmiP4J.js → ChevronRight-DE7hYCb3.js} +1 -1
- package/dist/assets/{Community-Ds1NG5Zj.js → Community-C8NtUXNH.js} +1 -1
- package/dist/assets/{DeleteOutline-CKVA3PXs.js → DeleteOutline-PGhagisl.js} +1 -1
- package/dist/assets/{Done-BDrYSkZM.js → Done-BVaQae1-.js} +1 -1
- package/dist/assets/{Download-BQiSNQBP.js → Download-DNNh4N1T.js} +1 -1
- package/dist/assets/{EditIcon-Bt91o1l6.js → EditIcon-Y1U-3M09.js} +1 -1
- package/dist/assets/{Email-C7SKOpM8.js → Email-BYzCrU9G.js} +1 -1
- package/dist/assets/{Error-CK4gTMoT.js → Error-C4dVUMw4.js} +1 -1
- package/dist/assets/{ExpandLess-1B6PdCYC.js → ExpandLess-Blh1x-4z.js} +1 -1
- package/dist/assets/{Google-DimUFzIe.js → Google-CAujGJ1A.js} +1 -1
- package/dist/assets/{Holiday-Dfj3kBBI.js → Holiday-DxDxhx5X.js} +1 -1
- package/dist/assets/{InfoOutlined-CEHciuDy.js → InfoOutlined-DF9JwavD.js} +1 -1
- package/dist/assets/{Launch-DX76v5dw.js → Launch-gOz_lwLg.js} +1 -1
- package/dist/assets/{LaunchOutlined-CnOQGhu-.js → LaunchOutlined-MbSPR9qn.js} +1 -1
- package/dist/assets/{Location--4ueDivQ.js → Location-CEwgGcA5.js} +1 -1
- package/dist/assets/{LockIcon-B84lMZao.js → LockIcon-D7BH0t_p.js} +1 -1
- package/dist/assets/{Meeting-CVVnP-bH.js → Meeting-BJz5QiFc.js} +1 -1
- package/dist/assets/{MoreHoriz-THEF6D0C.js → MoreHoriz-CGWVyYh4.js} +1 -1
- package/dist/assets/{OffSick-C_2x7yDI.js → OffSick--msvqLGm.js} +1 -1
- package/dist/assets/{Phone-CaKOX2Yu.js → Phone-CDl3oDYG.js} +1 -1
- package/dist/assets/{PlayArrow-DpcMtsY5.js → PlayArrow-CxWt6459.js} +1 -1
- package/dist/assets/{QuestionMarkCircle-5K6WR8Pc.js → QuestionMarkCircle-CPHZnM6u.js} +1 -1
- package/dist/assets/{ServerLogo-C4P-qXGV.js → ServerLogo-vX_nf8Sc.js} +1 -1
- package/dist/assets/{Timezone-Bgwxnj29.js → Timezone-uXi4_8BX.js} +1 -1
- package/dist/assets/{TuneOutlined-DHqZ7jGz.js → TuneOutlined-DcqnBMqE.js} +1 -1
- package/dist/assets/{ViewList-DOu7Co_L.js → ViewList-B6eveRY-.js} +1 -1
- package/dist/assets/{WorkingRemotely-Dm2BOs5c.js → WorkingRemotely-CK4z1-Sd.js} +1 -1
- package/dist/assets/{access-control-wqGz9JCo.js → access-control-Dxyg5dpc.js} +1 -1
- package/dist/assets/{actions-CAL1KFv7.js → actions-BAjt7G_t.js} +1 -1
- package/dist/assets/{add-component-core-DqvSoT6a.js → add-component-core-BXowXTJz.js} +1 -1
- package/dist/assets/{add-resource-DmN1IY_o.js → add-resource-BetUBTvt.js} +1 -1
- package/dist/assets/{addon-D1_etOjl.js → addon-B1aBsfOX.js} +1 -1
- package/dist/assets/{advanced-CzJgEcda.js → advanced-C8t1232G.js} +1 -1
- package/dist/assets/{appearance-Bn0kYJHu.js → appearance-BIOmTg1q.js} +1 -1
- package/dist/assets/{audit-logs-DU66CSpw.js → audit-logs-SSAqJt40.js} +1 -1
- package/dist/assets/{base32-CLGITfm-.js → base32-iVyUiizz.js} +1 -1
- package/dist/assets/{branding-C5rNAfwW.js → branding-BDuOcSfH.js} +1 -1
- package/dist/assets/{branding-CiE421Dn.js → branding-NmIAi0kv.js} +2 -2
- package/dist/assets/{bundle-avatar-vSofRY4G.js → bundle-avatar-DfAWUEqX.js} +1 -1
- package/dist/assets/button-Kk1bqv_H.js +1 -0
- package/dist/assets/{click-to-copy-C9iEVQIO.js → click-to-copy-DCUroWft.js} +1 -1
- package/dist/assets/{complete-BeWyfTmK.js → complete-R99mlHBI.js} +1 -1
- package/dist/assets/{component-CZ948vh4.js → component-By64LaLD.js} +1 -1
- package/dist/assets/{config-DzZULB-q.js → config-BHBxeyYh.js} +1 -1
- package/dist/assets/{config-CFbM7U_5.js → config-BvNNOdw8.js} +1 -1
- package/dist/assets/config-navigation-CPd_4svD.js +26 -0
- package/dist/assets/{config-space-Drp9NI7a.js → config-space-BgWEAM0m.js} +1 -1
- package/dist/assets/{confirm-B9OqaJpT.js → confirm-92J4n1RP.js} +1 -1
- package/dist/assets/{connect-CpUC5BAX.js → connect-Bq-p3hnS.js} +1 -1
- package/dist/assets/connect-D20f2K4u.js +5 -0
- package/dist/assets/{connect-to-BasQxweP.js → connect-to-bQVCrkhh.js} +1 -1
- package/dist/assets/{content-layout-Bs-vdQ_U.js → content-layout-CFHoSupA.js} +1 -1
- package/dist/assets/{dashboard-BYUKOMAe.js → dashboard-BgQFc2Pc.js} +11 -11
- package/dist/assets/{delete-confirm-CM4f-ajO.js → delete-confirm-AkwIQ5yF.js} +1 -1
- package/dist/assets/{did-address-B_VfxEC5.js → did-address-D4v6iN2l.js} +1 -1
- package/dist/assets/domain-CBdLs2D9.js +9 -0
- package/dist/assets/{domain-action-card-pHkmKFE3.js → domain-action-card-DN3KjFwP.js} +2 -2
- package/dist/assets/domains-CMJbalsn.js +1 -0
- package/dist/assets/{email-C5iUsSjP.js → email-JviRloBo.js} +1 -1
- package/dist/assets/{empty-spinner-Cia24BVr.js → empty-spinner-DlzuJxRt.js} +1 -1
- package/dist/assets/{exchange-passport-CyGvLDal.js → exchange-passport-3aXbJr-q.js} +1 -1
- package/dist/assets/{fuel-C5H_z1Ay.js → fuel-DOs3bqYm.js} +1 -1
- package/dist/assets/{get-safe-url-BOY9UAaI.js → get-safe-url-CWgEUVPT.js} +1 -1
- package/dist/assets/{get-safe-url-D_Puq_oh.js → get-safe-url-eATw6wn7.js} +1 -1
- package/dist/assets/{home-CjHQSa_8.js → home-a0RwGUp7.js} +1 -1
- package/dist/assets/{iframe-BhRKCdFb.js → iframe-DVBEjxC1.js} +1 -1
- package/dist/assets/{index-Kaq7tC6c.js → index-3q6oLa2e.js} +1 -1
- package/dist/assets/{index-C5lvlGzp.js → index-BBl5mftj.js} +3 -3
- package/dist/assets/{index-d34Q6nXR.js → index-BGeUo-5w.js} +1 -1
- package/dist/assets/{index-Cp_gD4ZQ.js → index-BJYrBMyS.js} +1 -1
- package/dist/assets/{index-B8FtPEFx.js → index-BVOsgGKh.js} +1 -1
- package/dist/assets/{index-DUU3hQpo.js → index-BxTrfRkG.js} +1 -1
- package/dist/assets/index-C3-7zmeY.js +346 -0
- package/dist/assets/{index-BTOIZL6e.js → index-CCwUSMuP.js} +1 -1
- package/dist/assets/{index-CjmXl7jW.js → index-CwdXL2a0.js} +1 -1
- package/dist/assets/{index-Ctl8J5Z5.js → index-DBLdE2lQ.js} +1 -1
- package/dist/assets/{index-CVIeQOoE.js → index-DQIxR9j5.js} +1 -1
- package/dist/assets/index-DWuuU3ta.js +113 -0
- package/dist/assets/{index-Cuoxr9xB.js → index-DjIOxw8x.js} +1 -1
- package/dist/assets/{index-CYAgDp3G.js → index-DrUxhMM5.js} +1 -1
- package/dist/assets/index-K5iSHoYm.js +224 -0
- package/dist/assets/{index-DR9nQ2tJ.js → index-KLUHbsxg.js} +1 -1
- package/dist/assets/{index-BYsXlcQC.js → index-MqiIu4kd.js} +1 -1
- package/dist/assets/{index-BjYech25.js → index-Th4BUzmb.js} +23 -23
- package/dist/assets/{index-Du2DtxVt.js → index-VvWvVQT8.js} +1 -1
- package/dist/assets/{index-DkZfhapD.js → index-sv3a1ZUw.js} +1 -1
- package/dist/assets/{invitation-C9hFp_wD.js → invitation-CwtXOXZc.js} +4 -4
- package/dist/assets/{invite-CzfqL5Tx.js → invite-CMj5995Z.js} +1 -1
- package/dist/assets/{issue-passport-D3F63x7z.js → issue-passport-DtEbcgX6.js} +1 -1
- package/dist/assets/{item-C_TigVeC.js → item-zVdGBfMR.js} +1 -1
- package/dist/assets/{landing-page-B5nKEn_3.js → landing-page-BM1TmagN.js} +1 -1
- package/dist/assets/{launch-result-message-C0Ao1hE-.js → launch-result-message-DOa3SWMR.js} +1 -1
- package/dist/assets/{layout-D5ALyxXA.js → layout-DCzN0ui5.js} +1 -1
- package/dist/assets/{list-DuzdV0Wa.js → list-BuXPrN1N.js} +8 -8
- package/dist/assets/{list-header-BibBP5t1.js → list-header-BLGbkNad.js} +1 -1
- package/dist/assets/localization-qd6O0FX9.js +1 -0
- package/dist/assets/{log-Dg8IZoeo.js → log-C0ajhie8.js} +1 -1
- package/dist/assets/{logger-Bd77RetM.js → logger-D6n4nbwy.js} +1 -1
- package/dist/assets/{login-B9XiO8eS.js → login-f1AXbIm0.js} +1 -1
- package/dist/assets/{login-oauth-callback-5AlLzjZ2.js → login-oauth-callback-C5CFUlUR.js} +1 -1
- package/dist/assets/{logo-uploader-ChABQ_cS.js → logo-uploader-DbAVW3BS.js} +3 -3
- package/dist/assets/{lost-passport-gGzRW--h.js → lost-passport-Dlmv8D-L.js} +2 -2
- package/dist/assets/{open-window-CX_mRFPs.js → open-window-DskgWqba.js} +1 -1
- package/dist/assets/{overview-ClfPLbRa.js → overview-DY9-9zrL.js} +1 -1
- package/dist/assets/{page-header-DWf5vODI.js → page-header-XbzfRkA7.js} +1 -1
- package/dist/assets/{permission-CmHMB8NK.js → permission-Bi46fqEy.js} +1 -1
- package/dist/assets/{preferences-COMN8FgJ.js → preferences-DPpwnIAs.js} +1 -1
- package/dist/assets/{profile-embed-Crj1xM6Z.js → profile-embed-DV5vno0C.js} +1 -1
- package/dist/assets/publish-resource-z2_THSz6.js +1 -0
- package/dist/assets/{react-beautiful-dnd.esm-B3HYEQUw.js → react-beautiful-dnd.esm-B68lWn23.js} +1 -1
- package/dist/assets/{relative-time-CE47D6s7.js → relative-time-BeR_4mfM.js} +1 -1
- package/dist/assets/{runtime-CdKbOwQM.js → runtime-D6tESY33.js} +1 -1
- package/dist/assets/sdk-DNr-nKBq.js +1 -0
- package/dist/assets/{security-DmjkCoJ_.js → security-DmR541pi.js} +1 -1
- package/dist/assets/{session-0rF5SuCk.js → session-CJFbMNLl.js} +1 -1
- package/dist/assets/setup-BSsqBrsc.js +19 -0
- package/dist/assets/{shorten-label-DDBwv81M.js → shorten-label-BuH-7PI8.js} +1 -1
- package/dist/assets/{simple-select-vCKkiHhG.js → simple-select-DHvjqsw4.js} +1 -1
- package/dist/assets/{slicedToArray-Da5X2Yv5.js → slicedToArray-C0MW3KIe.js} +1 -1
- package/dist/assets/{spaces-CyFGYf34.js → spaces-gLUqr6Uq.js} +1 -1
- package/dist/assets/{start-CwSXw9Xs.js → start-XbVMHGiE.js} +1 -1
- package/dist/assets/{status-CgfGREVl.js → status-DmyvEkKL.js} +1 -1
- package/dist/assets/{step-actions-BNW8I1sI.js → step-actions-brdohYRv.js} +1 -1
- package/dist/assets/{studio-cLKWVbyy.js → studio-DJeZlIfF.js} +1 -1
- package/dist/assets/{switch-control-DpvMHPLC.js → switch-control-Br1YNOSB.js} +1 -1
- package/dist/assets/{traffic-ChJb5Oyl.js → traffic-LuIcK-L1.js} +1 -1
- package/dist/assets/{transfer-BHifjdxl.js → transfer-DboKwiA_.js} +1 -1
- package/dist/assets/{unsubscribe-CcyOqX8E.js → unsubscribe-C9k_L3g5.js} +1 -1
- package/dist/assets/{use-mobile-DjnNmGRG.js → use-mobile-C98e2dYy.js} +1 -1
- package/dist/assets/{use-mobile-DpXDDjat.js → use-mobile-CqLLcWbr.js} +1 -1
- package/dist/assets/useAsync-BxB2TpDl.js +1 -0
- package/dist/assets/useAsync-L7_fHBH5.js +1 -0
- package/dist/assets/useAsyncRetry-CEouAsle.js +7 -0
- package/dist/assets/{useLocalStorage-Cq9THoCr.js → useLocalStorage-0mjZfnua.js} +1 -1
- package/dist/assets/user-center-BaKv3X4J.js +77 -0
- package/dist/assets/{util-1n1Ulk5w.js → util-DLK92HF8.js} +1 -1
- package/dist/assets/{util-BmbbisPE.js → util-YnaMqM3B.js} +1 -1
- package/dist/assets/{vendor-arcblock-V3HYELzG.js → vendor-arcblock-DPMnO1AO.js} +83 -83
- package/dist/assets/{vendor-mui-core-1jJmAutq.js → vendor-mui-core-x3zpe_j_.js} +1 -1
- package/dist/assets/{vendor-mui-x-CPwlD3uT.js → vendor-mui-x-CPOc4sOK.js} +1 -1
- package/dist/assets/{vendor-ux-did-connect-DEbq97Xz.js → vendor-ux-did-connect-B1u6nvmg.js} +7 -7
- package/dist/assets/{wrap-locale-5LSB6OAr.js → wrap-locale-m_NRjq-h.js} +1 -1
- package/dist/index.html +4 -4
- package/dist/service-worker.js +1 -1
- package/package.json +30 -30
- package/dist/assets/button-BSJ2z68-.js +0 -1
- package/dist/assets/config-navigation-hrdLDyFZ.js +0 -26
- package/dist/assets/connect-0zFqdTRm.js +0 -5
- package/dist/assets/domain-D0xqhxvZ.js +0 -9
- package/dist/assets/domains-BrNRyjCn.js +0 -1
- package/dist/assets/index-Bee_YcJC.js +0 -113
- package/dist/assets/index-D3YZfNxX.js +0 -5
- package/dist/assets/index-DFoYVoMG.js +0 -346
- package/dist/assets/index-Dzvf3kMf.js +0 -220
- package/dist/assets/localization-WUgLuLwU.js +0 -1
- package/dist/assets/publish-resource-C6TOyrHC.js +0 -1
- package/dist/assets/sdk-j4w1A_X4.js +0 -1
- package/dist/assets/setup-DLCw2h80.js +0 -19
- package/dist/assets/useAsync-BENTIUEf.js +0 -1
- package/dist/assets/useAsyncRetry-DSYkC9r5.js +0 -7
- package/dist/assets/user-center-DfDG_giT.js +0 -77
package/api/index.js
CHANGED
|
@@ -234,6 +234,36 @@ module.exports = function createServer(node, serverOptions = {}) {
|
|
|
234
234
|
}
|
|
235
235
|
});
|
|
236
236
|
});
|
|
237
|
+
[BlockletEvents.removed].forEach((name) => {
|
|
238
|
+
eventHub.on(name, (data) => {
|
|
239
|
+
const did = get(data, 'appDid') || get(data, 'meta.did');
|
|
240
|
+
if (did) {
|
|
241
|
+
node
|
|
242
|
+
.destroyTeamStates(did)
|
|
243
|
+
.then(() => {
|
|
244
|
+
logger.info('destroy team states on blocklet removed', { name, did });
|
|
245
|
+
})
|
|
246
|
+
.catch((error) => {
|
|
247
|
+
logger.error('Failed to destroy team states on blocklet removed', { name, did, error });
|
|
248
|
+
});
|
|
249
|
+
}
|
|
250
|
+
});
|
|
251
|
+
});
|
|
252
|
+
[BlockletEvents.installed, BlockletInternalEvents.componentInstalled].forEach((name) => {
|
|
253
|
+
eventHub.on(name, (data) => {
|
|
254
|
+
const did = get(data, 'appDid') || get(data, 'meta.did');
|
|
255
|
+
if (did) {
|
|
256
|
+
node
|
|
257
|
+
.createTeamStates(did)
|
|
258
|
+
.then(() => {
|
|
259
|
+
logger.info('create team states on blocklet installed', { name, did });
|
|
260
|
+
})
|
|
261
|
+
.catch((error) => {
|
|
262
|
+
logger.error('Failed to create team states on blocklet installed', { name, did, error });
|
|
263
|
+
});
|
|
264
|
+
}
|
|
265
|
+
});
|
|
266
|
+
});
|
|
237
267
|
eventHub.on(BlockletEvents.securityConfigUpdated, ({ did }) => {
|
|
238
268
|
// 检测到安全配置更新后,需要批量删除指定前缀的 cache(会包含整个 blocklet 所有的 security cache)
|
|
239
269
|
logger.info('blocklet securityConfig update', { did, pid: process.pid });
|
|
@@ -362,8 +392,7 @@ module.exports = function createServer(node, serverOptions = {}) {
|
|
|
362
392
|
...whitelist.official,
|
|
363
393
|
...iconifyDomains,
|
|
364
394
|
].filter(Boolean),
|
|
365
|
-
|
|
366
|
-
fontSrc: ["'self'", 'https://fonts.gstatic.com'],
|
|
395
|
+
fontSrc: ["'self'", 'data:', 'https://fonts.gstatic.com'],
|
|
367
396
|
connectSrc: uniq(
|
|
368
397
|
[
|
|
369
398
|
"'self'",
|
|
@@ -599,6 +628,7 @@ self.blocklet = {
|
|
|
599
628
|
|
|
600
629
|
// Auth: login token and user info
|
|
601
630
|
server.use(authMiddlewares.sessionBearerToken);
|
|
631
|
+
// FIXME: @zhanghan 这里的逻辑会导致对当前会话的校验是否强制注销失败,暂时先只在 req.ensureUser 进行增强处理。后续需要考虑完整的 auth 鉴权层
|
|
602
632
|
server.use(authMiddlewares.userInfo);
|
|
603
633
|
|
|
604
634
|
// API: gql
|
|
@@ -3,7 +3,7 @@ const { getSignData } = require('@blocklet/sdk/lib/util/verify-sign');
|
|
|
3
3
|
const getBlockletInfo = require('@blocklet/meta/lib/info');
|
|
4
4
|
const { joinURL } = require('ufo');
|
|
5
5
|
const pRetry = require('p-retry');
|
|
6
|
-
|
|
6
|
+
const pMap = require('p-map');
|
|
7
7
|
const { api } = require('../api');
|
|
8
8
|
const logger = require('../logger')('blocklet-services:notification');
|
|
9
9
|
|
|
@@ -42,18 +42,23 @@ async function sendPush(receiver, notification, { node, teamDid }) {
|
|
|
42
42
|
throw new Error(`Invalid push kit endpoint: ${config.endpoint}`);
|
|
43
43
|
}
|
|
44
44
|
|
|
45
|
-
const
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
45
|
+
const userSessionList = await pMap(
|
|
46
|
+
receiver,
|
|
47
|
+
async (x) => {
|
|
48
|
+
const result = await node.getUserSessions({
|
|
49
|
+
teamDid,
|
|
50
|
+
query: {
|
|
51
|
+
userDid: x,
|
|
52
|
+
},
|
|
53
|
+
});
|
|
54
|
+
return result.list;
|
|
51
55
|
},
|
|
52
|
-
|
|
56
|
+
{ concurrency: 10 }
|
|
57
|
+
);
|
|
53
58
|
|
|
54
|
-
const targets =
|
|
55
|
-
if (
|
|
56
|
-
|
|
59
|
+
const targets = userSessionList.reduce((acc, userSessions) => {
|
|
60
|
+
if (userSessions?.length > 0) {
|
|
61
|
+
userSessions.forEach((x) => {
|
|
57
62
|
// NOTICE: 这里需要转为小写来判断
|
|
58
63
|
const platform = x?.extra?.walletOS?.toLowerCase();
|
|
59
64
|
// 兼容处理,支持读取 walletDeviceMessageToken
|
|
@@ -1,9 +1,9 @@
|
|
|
1
1
|
/* eslint-disable no-await-in-loop */
|
|
2
2
|
const {
|
|
3
3
|
WELLKNOWN_SERVICE_PATH_PREFIX,
|
|
4
|
-
SESSION_TTL,
|
|
5
4
|
PASSPORT_LOG_ACTION,
|
|
6
5
|
PASSPORT_STATUS,
|
|
6
|
+
SESSION_TTL,
|
|
7
7
|
} = require('@abtnode/constant');
|
|
8
8
|
const { LOGIN_PROVIDER } = require('@blocklet/constant');
|
|
9
9
|
const pick = require('lodash/pick');
|
|
@@ -13,7 +13,7 @@ const omit = require('lodash/omit');
|
|
|
13
13
|
const pLimit = require('p-limit');
|
|
14
14
|
const { getSourceProvider } = require('@blocklet/meta/lib/did-utils');
|
|
15
15
|
const getRequestIP = require('@abtnode/util/lib/get-request-ip');
|
|
16
|
-
const { getFederatedMembers, getFederatedMaster } = require('@abtnode/auth/lib/util/federated');
|
|
16
|
+
const { getFederatedMembers, getFederatedMaster, findFederatedSite } = require('@abtnode/auth/lib/util/federated');
|
|
17
17
|
const { messages } = require('@abtnode/auth/lib/auth');
|
|
18
18
|
const { Joi } = require('@arcblock/validator');
|
|
19
19
|
const { getDeviceData } = require('@abtnode/util/lib/device');
|
|
@@ -51,12 +51,16 @@ async function getPassportFromFederatedSite(passportId, { appPid, node, teamDid,
|
|
|
51
51
|
return undefined;
|
|
52
52
|
}
|
|
53
53
|
|
|
54
|
-
|
|
54
|
+
function patchUserSessionStatus(userSession, { blocklet }) {
|
|
55
55
|
const sessionTtl = blocklet.settings?.session?.ttl || SESSION_TTL;
|
|
56
56
|
// 修正当前 session 的 status,如果超出时间了,则应该显示为过期
|
|
57
57
|
if (Date.now() - new Date(userSession.updatedAt).getTime() > sessionTtl * 1000) {
|
|
58
58
|
userSession.status = 'expired';
|
|
59
59
|
}
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
async function patchUserSessionData(userSession, { blocklet, appPid, teamDid, node }) {
|
|
63
|
+
patchUserSessionStatus(userSession, { blocklet });
|
|
60
64
|
// 修正 avatar 地址,从 bn:// 转换为 http
|
|
61
65
|
if (userSession.user?.avatar) {
|
|
62
66
|
userSession.user.avatar = getUserAvatarUrl(userSession.user.avatar, blocklet);
|
|
@@ -154,9 +158,8 @@ module.exports = {
|
|
|
154
158
|
}
|
|
155
159
|
|
|
156
160
|
const teamDid = blocklet.appPid;
|
|
157
|
-
const
|
|
161
|
+
const validSession = await node.getUserSession({
|
|
158
162
|
teamDid,
|
|
159
|
-
userDid: validUserSession.userDid,
|
|
160
163
|
id: validUserSession.id,
|
|
161
164
|
});
|
|
162
165
|
|
|
@@ -338,35 +341,31 @@ module.exports = {
|
|
|
338
341
|
|
|
339
342
|
app.get(`${prefix}/myself`, ensureBlocklet(), checkUser, async (req, res) => {
|
|
340
343
|
const { blocklet } = req;
|
|
341
|
-
const {
|
|
342
|
-
const teamDid = appPid;
|
|
344
|
+
const { page, pageSize, status } = req.query;
|
|
345
|
+
const teamDid = blocklet.appPid;
|
|
343
346
|
const userDid = req.user.did;
|
|
344
347
|
// 用户管理自己所有的登录会话,不限制 visitorId
|
|
345
|
-
const userSessions = await node.
|
|
348
|
+
const { list: userSessions, paging } = await node.getUserSessions({
|
|
346
349
|
teamDid,
|
|
347
|
-
|
|
348
|
-
|
|
350
|
+
query: {
|
|
351
|
+
appPid: teamDid, // 不需要向主站查询 member 的会话列表,所以固定为 teamDid 即可
|
|
352
|
+
userDid,
|
|
353
|
+
status,
|
|
354
|
+
},
|
|
355
|
+
paging: {
|
|
356
|
+
page: parseInt(page, 10),
|
|
357
|
+
pageSize: parseInt(pageSize, 10),
|
|
358
|
+
},
|
|
349
359
|
});
|
|
350
360
|
|
|
351
|
-
const
|
|
352
|
-
|
|
353
|
-
|
|
354
|
-
|
|
355
|
-
|
|
356
|
-
|
|
357
|
-
|
|
358
|
-
|
|
359
|
-
)
|
|
360
|
-
);
|
|
361
|
-
await Promise.all(pendingList);
|
|
362
|
-
|
|
363
|
-
const result = userSessions
|
|
364
|
-
.filter((x) => {
|
|
365
|
-
return x?.user?.approved;
|
|
366
|
-
})
|
|
367
|
-
.map((x) => omit(x, userInfoBlackList));
|
|
368
|
-
|
|
369
|
-
res.json(result);
|
|
361
|
+
const list = userSessions.map((x) => {
|
|
362
|
+
patchUserSessionStatus(x, { blocklet });
|
|
363
|
+
return omit(x, userInfoBlackList);
|
|
364
|
+
});
|
|
365
|
+
res.json({
|
|
366
|
+
list,
|
|
367
|
+
paging,
|
|
368
|
+
});
|
|
370
369
|
});
|
|
371
370
|
|
|
372
371
|
/**
|
|
@@ -381,8 +380,6 @@ module.exports = {
|
|
|
381
380
|
const { appPid } = blocklet;
|
|
382
381
|
const teamDid = appPid;
|
|
383
382
|
|
|
384
|
-
const federatedSites = blocklet?.settings?.federated?.sites || [];
|
|
385
|
-
|
|
386
383
|
const visitorId = req.get('x-blocklet-visitor-id');
|
|
387
384
|
|
|
388
385
|
if (!visitorId) {
|
|
@@ -390,19 +387,34 @@ module.exports = {
|
|
|
390
387
|
return;
|
|
391
388
|
}
|
|
392
389
|
|
|
393
|
-
const userSessions = await node.
|
|
390
|
+
const { list: userSessions } = await node.getUserSessions({
|
|
394
391
|
teamDid,
|
|
395
|
-
|
|
392
|
+
query: {
|
|
393
|
+
visitorId,
|
|
394
|
+
includeUser: true,
|
|
395
|
+
},
|
|
396
|
+
paging: {
|
|
397
|
+
page: 1,
|
|
398
|
+
// 模拟获取所有登录会话
|
|
399
|
+
pageSize: 100,
|
|
400
|
+
},
|
|
396
401
|
});
|
|
397
402
|
|
|
398
403
|
const validUserSessions = userSessions.filter((x) => {
|
|
399
|
-
|
|
400
|
-
if (federatedSite?.appPid === teamDid) {
|
|
404
|
+
if (x.appPid === teamDid) {
|
|
401
405
|
return true;
|
|
402
406
|
}
|
|
403
|
-
|
|
404
|
-
|
|
407
|
+
|
|
408
|
+
const federatedSite = findFederatedSite(blocklet, x.appPid);
|
|
409
|
+
if (federatedSite) {
|
|
410
|
+
if (federatedSite.appPid === teamDid) {
|
|
411
|
+
return true;
|
|
412
|
+
}
|
|
413
|
+
if (federatedSite.status === 'approved') {
|
|
414
|
+
return true;
|
|
415
|
+
}
|
|
405
416
|
}
|
|
417
|
+
|
|
406
418
|
return false;
|
|
407
419
|
});
|
|
408
420
|
const pendingList = validUserSessions.map((item) =>
|
|
@@ -419,9 +431,6 @@ module.exports = {
|
|
|
419
431
|
|
|
420
432
|
const result = validUserSessions
|
|
421
433
|
.filter((x) => {
|
|
422
|
-
if (x.status === 'expired') {
|
|
423
|
-
return false;
|
|
424
|
-
}
|
|
425
434
|
if (!x?.user?.approved) {
|
|
426
435
|
return false;
|
|
427
436
|
}
|
package/api/routes/user.js
CHANGED
|
@@ -25,6 +25,7 @@ const { withQuery, joinURL } = require('ufo');
|
|
|
25
25
|
const cors = require('cors');
|
|
26
26
|
const createTranslator = require('@abtnode/util/lib/translate');
|
|
27
27
|
const { getDeviceData } = require('@abtnode/util/lib/device');
|
|
28
|
+
const { Op } = require('sequelize');
|
|
28
29
|
|
|
29
30
|
const { createTokenFn, getDidConnectVersion } = require('../util');
|
|
30
31
|
const initJwt = require('../libs/jwt');
|
|
@@ -921,18 +922,29 @@ module.exports = {
|
|
|
921
922
|
const { blocklet } = req;
|
|
922
923
|
const teamDid = blocklet.appPid;
|
|
923
924
|
// NOTICE: 此处需要保留从 body 携带 visitorId 的功能,用于已登录用户注销自己指定的登录会话
|
|
924
|
-
|
|
925
|
+
// eslint-disable-next-line prefer-const
|
|
926
|
+
let { status, visitorId } = req.body;
|
|
927
|
+
const params = {};
|
|
925
928
|
if (!visitorId) {
|
|
926
929
|
visitorId = req.get('x-blocklet-visitor-id');
|
|
927
930
|
}
|
|
928
|
-
if (
|
|
929
|
-
|
|
931
|
+
if (status) {
|
|
932
|
+
params.status = status;
|
|
933
|
+
if (visitorId && status === 'online') {
|
|
934
|
+
// HACK: 用户使用当前登录会话注销所有会话时,要排除当前登录会话
|
|
935
|
+
params.visitorId = { [Op.ne]: visitorId };
|
|
936
|
+
}
|
|
937
|
+
} else if (visitorId) {
|
|
938
|
+
params.visitorId = visitorId;
|
|
939
|
+
}
|
|
940
|
+
if (!params.visitorId && !params.status) {
|
|
941
|
+
res.status(400).json({ error: 'visitorId or status is required' });
|
|
930
942
|
return;
|
|
931
943
|
}
|
|
932
944
|
|
|
933
945
|
await node.logoutUser({
|
|
946
|
+
...params,
|
|
934
947
|
userDid: req.user.did,
|
|
935
|
-
visitorId,
|
|
936
948
|
appPid: teamDid,
|
|
937
949
|
teamDid,
|
|
938
950
|
});
|
|
@@ -17,10 +17,32 @@ const getRequestIP = require('@abtnode/util/lib/get-request-ip');
|
|
|
17
17
|
const { getDeviceData } = require('@abtnode/util/lib/device');
|
|
18
18
|
const isUrl = require('is-url');
|
|
19
19
|
const { LRUCache } = require('lru-cache');
|
|
20
|
+
const createTranslator = require('@abtnode/util/lib/translate');
|
|
20
21
|
|
|
21
22
|
const { createTokenFn, getDidConnectVersion } = require('../../util');
|
|
22
23
|
const checkUser = require('../../middlewares/check-user');
|
|
23
24
|
|
|
25
|
+
const translations = {
|
|
26
|
+
zh: {
|
|
27
|
+
userSessionLogout: '用户会话已退出',
|
|
28
|
+
invalidWebhookType: '无效的 webhook 类型',
|
|
29
|
+
invalidWebhookUrl: '无效的 webhook 地址',
|
|
30
|
+
testWebhookMessage: '这是来自 {did} 的测试消息',
|
|
31
|
+
invalidToken: '无效的 token',
|
|
32
|
+
emptyRefreshToken: 'RefreshToken 不能为空',
|
|
33
|
+
},
|
|
34
|
+
en: {
|
|
35
|
+
userSessionLogout: 'User session is logout',
|
|
36
|
+
invalidWebhookType: 'Invalid webhook type',
|
|
37
|
+
invalidWebhookUrl: 'Invalid webhook url',
|
|
38
|
+
testWebhookMessage: 'This is a test message from user {did}',
|
|
39
|
+
invalidToken: 'Invalid token',
|
|
40
|
+
emptyRefreshToken: 'RefreshToken should not be empty',
|
|
41
|
+
},
|
|
42
|
+
};
|
|
43
|
+
|
|
44
|
+
const t = createTranslator({ translations });
|
|
45
|
+
|
|
24
46
|
const sessionRBACCache = new LRUCache({ max: 3000, ttl: 1000 * 30 });
|
|
25
47
|
const unReadCountCache = new LRUCache({ max: 3000, ttl: 1000 * 5 });
|
|
26
48
|
|
|
@@ -176,13 +198,14 @@ module.exports = {
|
|
|
176
198
|
|
|
177
199
|
const sessionApi = `${WELLKNOWN_SERVICE_PATH_PREFIX}/api/did/session`;
|
|
178
200
|
router.get(sessionApi, nocache(), sessionBearerToken, handleSession);
|
|
201
|
+
router.post(sessionApi, nocache(), sessionBearerToken, handleSession);
|
|
202
|
+
|
|
179
203
|
router.get(
|
|
180
204
|
`${WELLKNOWN_SERVICE_PATH_PREFIX}/api/notifications/unread-count`,
|
|
181
205
|
nocache(),
|
|
182
206
|
sessionBearerToken,
|
|
183
207
|
handleGetUnreadCount
|
|
184
208
|
);
|
|
185
|
-
router.post(sessionApi, nocache(), sessionBearerToken, handleSession);
|
|
186
209
|
|
|
187
210
|
// update user extra: settings, webhooks
|
|
188
211
|
const extraApi = `${WELLKNOWN_SERVICE_PATH_PREFIX}/api/user/extra`;
|
|
@@ -207,17 +230,17 @@ module.exports = {
|
|
|
207
230
|
});
|
|
208
231
|
router.put(extraApi, nocache(), sessionBearerToken, checkUser, async (req, res) => {
|
|
209
232
|
if (['slack', 'api'].includes(req.body.type) === false) {
|
|
210
|
-
res.status(400).send({ error: '
|
|
233
|
+
res.status(400).send({ error: t('invalidWebhookType', req.blockletLocale) });
|
|
211
234
|
return;
|
|
212
235
|
}
|
|
213
236
|
|
|
214
237
|
if (isUrl(req.body.url) === false) {
|
|
215
|
-
res.status(400).send({ error: '
|
|
238
|
+
res.status(400).send({ error: t('invalidWebhookUrl', req.blockletLocale) });
|
|
216
239
|
}
|
|
217
240
|
|
|
218
241
|
await node.sendTestMessage({
|
|
219
242
|
webhook: { type: req.body.type, params: [{ name: 'url', value: req.body.url }] },
|
|
220
|
-
message:
|
|
243
|
+
message: t('testWebhookMessage', req.blockletLocale, { did: req.user.did }),
|
|
221
244
|
});
|
|
222
245
|
|
|
223
246
|
res.json({ success: true });
|
|
@@ -227,7 +250,7 @@ module.exports = {
|
|
|
227
250
|
const { wallet } = await req.getBlockletInfo();
|
|
228
251
|
const { token } = req.body;
|
|
229
252
|
if (!verify(token, wallet.publicKey)) {
|
|
230
|
-
res.status(403).send('
|
|
253
|
+
res.status(403).send(t('invalidToken', req.blockletLocale));
|
|
231
254
|
return;
|
|
232
255
|
}
|
|
233
256
|
|
|
@@ -258,10 +281,14 @@ module.exports = {
|
|
|
258
281
|
if (token) {
|
|
259
282
|
const teamDid = req.getBlockletDid();
|
|
260
283
|
const visitorId = req.get('x-blocklet-visitor-id');
|
|
284
|
+
if (!visitorId) {
|
|
285
|
+
res.status(400).send(t('userSessionLogout', req.blockletLocale));
|
|
286
|
+
return;
|
|
287
|
+
}
|
|
261
288
|
const { appPid = teamDid } = req.query;
|
|
262
289
|
await req.ensureUser({ token, visitorId, appPid });
|
|
263
290
|
if (!req.user) {
|
|
264
|
-
res.status(400).send('
|
|
291
|
+
res.status(400).send(t('userSessionLogout', req.blockletLocale));
|
|
265
292
|
return;
|
|
266
293
|
}
|
|
267
294
|
|
|
@@ -279,9 +306,9 @@ module.exports = {
|
|
|
279
306
|
} = await verifySessionToken(token, secret, {
|
|
280
307
|
checkFromDb: true,
|
|
281
308
|
teamDid,
|
|
282
|
-
checkToken: (
|
|
283
|
-
if (
|
|
284
|
-
throw new Error(`invalid token type ${
|
|
309
|
+
checkToken: (_token) => {
|
|
310
|
+
if (_token.tokenType !== refreshTokenType) {
|
|
311
|
+
throw new Error(`invalid token type ${_token.tokenType}`);
|
|
285
312
|
}
|
|
286
313
|
},
|
|
287
314
|
locale: req.blockletLocale,
|
|
@@ -368,7 +395,7 @@ module.exports = {
|
|
|
368
395
|
res.status(400).send(err.message);
|
|
369
396
|
}
|
|
370
397
|
} else {
|
|
371
|
-
res.status(400).send('
|
|
398
|
+
res.status(400).send(t('emptyRefreshToken', req.blockletLocale));
|
|
372
399
|
}
|
|
373
400
|
});
|
|
374
401
|
},
|
|
@@ -171,15 +171,22 @@ module.exports = ({ node, req, options }) => {
|
|
|
171
171
|
const user = await verifySessionToken(token, secret, opt);
|
|
172
172
|
|
|
173
173
|
if (user && visitorId) {
|
|
174
|
-
|
|
174
|
+
// 这里只取第一个有效的会话
|
|
175
|
+
const { list: userSessions } = await node.getUserSessions({
|
|
175
176
|
teamDid,
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
177
|
+
query: {
|
|
178
|
+
appPid,
|
|
179
|
+
userDid: user.did,
|
|
180
|
+
visitorId,
|
|
181
|
+
},
|
|
179
182
|
});
|
|
180
|
-
|
|
183
|
+
// 获取用户会话
|
|
184
|
+
if (userSessions.length > 0) {
|
|
185
|
+
const [userSession] = userSessions;
|
|
181
186
|
req.userSession = userSession;
|
|
182
187
|
} else {
|
|
188
|
+
// HACK: 需要强制设置为 null,因为在 core/blocklet-services/api/index.js L603 中会给 req.user 赋值
|
|
189
|
+
req.user = null;
|
|
183
190
|
return;
|
|
184
191
|
}
|
|
185
192
|
}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
import{a2 as e,j as t}from"./vendor-mui-core-
|
|
1
|
+
import{a2 as e,j as t}from"./vendor-mui-core-x3zpe_j_.js";import{r as a}from"./vendor-ux-did-connect-B1u6nvmg.js";var r={},o=e;Object.defineProperty(r,"__esModule",{value:!0});var u=r.default=void 0,i=o(a()),p=t;u=r.default=(0,i.default)((0,p.jsx)("path",{d:"m7 10 5 5 5-5z"}),"ArrowDropDown");export{u as d};
|
|
@@ -1 +1 @@
|
|
|
1
|
-
import{a2 as r,j as t}from"./vendor-mui-core-
|
|
1
|
+
import{a2 as r,j as t}from"./vendor-mui-core-x3zpe_j_.js";import{r as a}from"./vendor-ux-did-connect-B1u6nvmg.js";var e={},o=r;Object.defineProperty(e,"__esModule",{value:!0});var u=e.default=void 0,i=o(a()),l=t;u=e.default=(0,i.default)((0,l.jsx)("path",{d:"M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2m-2 15-5-5 1.41-1.41L10 14.17l7.59-7.59L19 8z"}),"CheckCircle");export{u as d};
|
|
@@ -1 +1 @@
|
|
|
1
|
-
import{a2 as r,j as t}from"./vendor-mui-core-
|
|
1
|
+
import{a2 as r,j as t}from"./vendor-mui-core-x3zpe_j_.js";import{r as a}from"./vendor-ux-did-connect-B1u6nvmg.js";var e={},o=r;Object.defineProperty(e,"__esModule",{value:!0});var u=e.default=void 0,f=o(a()),i=t;u=e.default=(0,f.default)((0,i.jsx)("path",{d:"M15.41 7.41 14 6l-6 6 6 6 1.41-1.41L10.83 12z"}),"ChevronLeft");export{u as d};
|
|
@@ -1 +1 @@
|
|
|
1
|
-
import{a2 as r,j as t}from"./vendor-mui-core-
|
|
1
|
+
import{a2 as r,j as t}from"./vendor-mui-core-x3zpe_j_.js";import{r as a}from"./vendor-ux-did-connect-B1u6nvmg.js";var e={},o=r;Object.defineProperty(e,"__esModule",{value:!0});var u=e.default=void 0,i=o(a()),v=t;u=e.default=(0,i.default)((0,v.jsx)("path",{d:"M10 6 8.59 7.41 13.17 12l-4.58 4.59L10 18l6-6z"}),"ChevronRight");export{u as d};
|
|
@@ -1 +1 @@
|
|
|
1
|
-
import"./vendor-react-DEoibe3W.js";import{j as h}from"./vendor-mui-core-1jJmAutq.js";var H=function(Y){return h.jsxs("svg",{...Y,children:[h.jsx("rect",{width:"25",height:"25",fill:"url(#pattern0_68_4738)"}),h.jsxs("defs",{children:[h.jsx("pattern",{id:"pattern0_68_4738",patternContentUnits:"objectBoundingBox",width:"1",height:"1",children:h.jsx("use",{xlinkHref:"#image0_68_4738",transform:"scale(0.00833333)"})}),h.jsx("image",{id:"image0_68_4738",width:"120",height:"120",preserveAspectRatio:"none",xlinkHref:""})]})]})};H.defaultProps={width:"25",height:"25",viewBox:"0 0 25 25",fill:"none",xmlns:"http://www.w3.org/2000/svg",xmlnsXlink:"http://www.w3.org/1999/xlink"};H.displayName="commuting";export{H as default};
|
|
1
|
+
import"./vendor-react-DEoibe3W.js";import{j as h}from"./vendor-mui-core-x3zpe_j_.js";var H=function(Y){return h.jsxs("svg",{...Y,children:[h.jsx("rect",{width:"25",height:"25",fill:"url(#pattern0_68_4738)"}),h.jsxs("defs",{children:[h.jsx("pattern",{id:"pattern0_68_4738",patternContentUnits:"objectBoundingBox",width:"1",height:"1",children:h.jsx("use",{xlinkHref:"#image0_68_4738",transform:"scale(0.00833333)"})}),h.jsx("image",{id:"image0_68_4738",width:"120",height:"120",preserveAspectRatio:"none",xlinkHref:""})]})]})};H.defaultProps={width:"25",height:"25",viewBox:"0 0 25 25",fill:"none",xmlns:"http://www.w3.org/2000/svg",xmlnsXlink:"http://www.w3.org/1999/xlink"};H.displayName="commuting";export{H as default};
|
|
@@ -1 +1 @@
|
|
|
1
|
-
import{a2 as t,j as r}from"./vendor-mui-core-
|
|
1
|
+
import{a2 as t,j as r}from"./vendor-mui-core-x3zpe_j_.js";import{r as a}from"./vendor-ux-did-connect-B1u6nvmg.js";var e={},u=t;Object.defineProperty(e,"__esModule",{value:!0});var o=e.default=void 0,i=u(a()),l=r;o=e.default=(0,i.default)((0,l.jsx)("path",{d:"M6 19c0 1.1.9 2 2 2h8c1.1 0 2-.9 2-2V7H6zM8 9h8v10H8zm7.5-5-1-1h-5l-1 1H5v2h14V4z"}),"DeleteOutline");export{o as d};
|
|
@@ -1 +1 @@
|
|
|
1
|
-
import{a2 as r,j as t}from"./vendor-mui-core-
|
|
1
|
+
import{a2 as r,j as t}from"./vendor-mui-core-x3zpe_j_.js";import{r as a}from"./vendor-ux-did-connect-B1u6nvmg.js";var e={},o=r;Object.defineProperty(e,"__esModule",{value:!0});var u=e.default=void 0,i=o(a()),l=t;u=e.default=(0,i.default)((0,l.jsx)("path",{d:"M9 16.2 4.8 12l-1.4 1.4L9 19 21 7l-1.4-1.4z"}),"Done");export{u as d};
|
|
@@ -1 +1 @@
|
|
|
1
|
-
import{a2 as r,j as a}from"./vendor-mui-core-
|
|
1
|
+
import{a2 as r,j as a}from"./vendor-mui-core-x3zpe_j_.js";import{r as t}from"./vendor-ux-did-connect-B1u6nvmg.js";var e={},o=r;Object.defineProperty(e,"__esModule",{value:!0});var u=e.default=void 0,d=o(t()),i=a;u=e.default=(0,d.default)((0,i.jsx)("path",{d:"M5 20h14v-2H5zM19 9h-4V3H9v6H5l7 7z"}),"Download");export{u as d};
|