mirror of
https://github.com/toeverything/AFFiNE.git
synced 2026-07-01 17:50:50 +08:00
chore: bump deps (#15059)
#### PR Dependency Tree * **PR #15059** 👈 This tree was auto-generated by [Charcoal](https://github.com/danerwilliams/charcoal) <!-- This is an auto-generated comment: release notes by coderabbit.ai --> ## Summary by CodeRabbit * **New Features** * Configurable minimum account age before new accounts can invite members or create share links (default: 24 hours). * Sign-in now returns and caches user info for improved session handling. * **Bug Fixes** * Queue handling accepts and resolves job IDs with special characters. * Improved clipboard/rich-text caret handling and nested-list paste reliability. * Calendar tests use dynamic current-month dates. * AI search returns explicit "No matching documents" when none found. * Auth session responses are explicitly non-cacheable. * **Chores** * Dependency and toolchain bumps; admin UI config/schema exposes the new account-age setting. <!-- end of auto-generated comment: release notes by coderabbit.ai -->
This commit is contained in:
@@ -175,6 +175,11 @@
|
||||
"description": "Whether require email verification before accessing restricted resources(not implemented).\n@default true",
|
||||
"default": true
|
||||
},
|
||||
"newAccountShareActionDelay": {
|
||||
"type": "number",
|
||||
"description": "Minimum account age in seconds before new accounts can invite members or create share links.\n@default 86400",
|
||||
"default": 86400
|
||||
},
|
||||
"passwordRequirements": {
|
||||
"type": "object",
|
||||
"description": "The password strength requirements when set new password.\n@default {\"min\":8,\"max\":32}",
|
||||
|
||||
@@ -65,7 +65,7 @@
|
||||
"@queuedash/api": "^3.16.0",
|
||||
"@react-email/components": "^0.5.7",
|
||||
"@socket.io/redis-adapter": "^8.3.0",
|
||||
"bullmq": "5.53.0",
|
||||
"bullmq": "5.77.6",
|
||||
"commander": "^13.1.0",
|
||||
"cookie-parser": "^1.4.7",
|
||||
"cross-env": "^10.1.0",
|
||||
|
||||
@@ -70,6 +70,14 @@ test('should be able to sign in with credential', async t => {
|
||||
t.is(session?.id, u1.id);
|
||||
});
|
||||
|
||||
test('should not cache auth session response', async t => {
|
||||
const { app } = t.context;
|
||||
|
||||
const res = await app.GET('/api/auth/session').expect(200);
|
||||
|
||||
t.is(res.headers['cache-control'], 'no-store');
|
||||
});
|
||||
|
||||
async function exchangeSession(app: TestingApp, code: string) {
|
||||
return await supertest(app.getHttpServer())
|
||||
.post('/api/auth/native/exchange')
|
||||
|
||||
@@ -104,6 +104,23 @@ test('should add job to queue', async t => {
|
||||
t.is(queuedJob!.name, job.name);
|
||||
});
|
||||
|
||||
test('should accept job id containing colon', async t => {
|
||||
const job = await queue.add(
|
||||
'nightly.__test__job',
|
||||
{ name: 'test' },
|
||||
{ jobId: 'custom:id' }
|
||||
);
|
||||
|
||||
const queuedJob = await queue.get('custom:id', 'nightly.__test__job');
|
||||
const roundTripJob = await queue.get(job.id!, 'nightly.__test__job');
|
||||
const data = await queue.remove(job.id!, 'nightly.__test__job');
|
||||
|
||||
t.is(job.id, 'custom%3Aid');
|
||||
t.is(queuedJob!.name, job.name);
|
||||
t.is(roundTripJob!.name, job.name);
|
||||
t.deepEqual(data, { name: 'test' });
|
||||
});
|
||||
|
||||
test('should remove job from queue', async t => {
|
||||
const job = await queue.add('nightly.__test__job', { name: 'test' });
|
||||
|
||||
|
||||
@@ -21,6 +21,19 @@ const removableJobStates = [
|
||||
] as const;
|
||||
const removeWhereBatchSize = 100;
|
||||
|
||||
function normalizeJobId(jobId: string) {
|
||||
return encodeURIComponent(jobId);
|
||||
}
|
||||
|
||||
function normalizedJobIds(jobId: string) {
|
||||
const normalized = normalizeJobId(jobId);
|
||||
if (jobId.includes(':')) {
|
||||
return [normalized];
|
||||
}
|
||||
|
||||
return normalized === jobId ? [jobId] : [jobId, normalized];
|
||||
}
|
||||
|
||||
@Injectable()
|
||||
export class JobQueue {
|
||||
private readonly logger = new Logger(JobQueue.name);
|
||||
@@ -30,6 +43,9 @@ export class JobQueue {
|
||||
async add<T extends JobName>(name: T, payload: Jobs[T], opts?: JobsOptions) {
|
||||
const ns = namespace(name);
|
||||
const queue = this.getQueue(ns);
|
||||
const normalizedOpts = opts?.jobId
|
||||
? { ...opts, jobId: normalizeJobId(opts.jobId) }
|
||||
: opts;
|
||||
const job = await queue.add(
|
||||
name,
|
||||
{
|
||||
@@ -37,7 +53,7 @@ export class JobQueue {
|
||||
ClsServiceManager.getClsService().getId() ?? genRequestId('job'),
|
||||
payload,
|
||||
} as JobData<T>,
|
||||
opts
|
||||
normalizedOpts
|
||||
);
|
||||
this.logger.debug(`Job [${name}] added; id=${job.id}`);
|
||||
return job;
|
||||
@@ -49,15 +65,16 @@ export class JobQueue {
|
||||
): Promise<Jobs[T] | undefined> {
|
||||
const ns = namespace(jobName);
|
||||
const queue = this.getQueue(ns);
|
||||
const job = (await queue.getJob(jobId)) as Job<JobData<T>> | undefined;
|
||||
const job = await this.get(jobId, jobName);
|
||||
|
||||
if (!job) {
|
||||
return;
|
||||
}
|
||||
|
||||
const removed = await queue.remove(jobId);
|
||||
if (!job.id) return;
|
||||
const removed = await queue.remove(job.id);
|
||||
if (removed) {
|
||||
this.logger.log(`Job ${jobName}(id=${jobId}) removed from queue ${ns}`);
|
||||
this.logger.log(`Job ${jobName}(id=${job.id}) removed from queue ${ns}`);
|
||||
return job.data.payload;
|
||||
}
|
||||
|
||||
@@ -120,7 +137,14 @@ export class JobQueue {
|
||||
async get<T extends JobName>(jobId: string, jobName: T) {
|
||||
const ns = namespace(jobName);
|
||||
const queue = this.getQueue(ns);
|
||||
return (await queue.getJob(jobId)) as Job<JobData<T>> | undefined;
|
||||
for (const id of normalizedJobIds(jobId)) {
|
||||
const job = (await queue.getJob(id)) as Job<JobData<T>> | undefined;
|
||||
if (job) {
|
||||
return job;
|
||||
}
|
||||
}
|
||||
|
||||
return undefined;
|
||||
}
|
||||
|
||||
private getQueue(ns: string): Queue {
|
||||
|
||||
@@ -11,6 +11,7 @@ export interface AuthConfig {
|
||||
allowSignupForOauth: boolean;
|
||||
requireEmailDomainVerification: boolean;
|
||||
requireEmailVerification: boolean;
|
||||
newAccountShareActionDelay: number;
|
||||
passwordRequirements: ConfigItem<{
|
||||
min: number;
|
||||
max: number;
|
||||
@@ -40,6 +41,11 @@ defineModuleConfig('auth', {
|
||||
desc: 'Whether require email verification before accessing restricted resources(not implemented).',
|
||||
default: true,
|
||||
},
|
||||
newAccountShareActionDelay: {
|
||||
desc: 'Minimum account age in seconds before new accounts can invite members or create share links.',
|
||||
default: 24 * 60 * 60,
|
||||
shape: z.number().int().min(0),
|
||||
},
|
||||
passwordRequirements: {
|
||||
desc: 'The password strength requirements when set new password.',
|
||||
default: {
|
||||
|
||||
@@ -256,6 +256,7 @@ export class AuthController {
|
||||
@Throttle('default', { limit: 1200 })
|
||||
@Public()
|
||||
@Get('/session')
|
||||
@Header('Cache-Control', 'no-store')
|
||||
async currentSessionUser(@CurrentUser() user?: CurrentUser) {
|
||||
return { user };
|
||||
}
|
||||
|
||||
@@ -1,10 +1,6 @@
|
||||
import test from 'ava';
|
||||
|
||||
import {
|
||||
containsUrlOrDomain,
|
||||
isUserOldEnoughForShareActions,
|
||||
SHARE_ACTION_ACCOUNT_AGE_MS,
|
||||
} from '../abuse';
|
||||
import { canUserExecuteLimitedActions, containsUrlOrDomain } from '../abuse';
|
||||
|
||||
test('should detect links and bare domains in workspace names', t => {
|
||||
t.true(containsUrlOrDomain('BTC https://spam.example'));
|
||||
@@ -20,10 +16,21 @@ test('should not detect email addresses or partial domain words', t => {
|
||||
});
|
||||
|
||||
test('should check account age for share actions', t => {
|
||||
t.false(isUserOldEnoughForShareActions({ createdAt: new Date() }));
|
||||
const minimumAccountAgeMs = 24 * 60 * 60 * 1000;
|
||||
|
||||
t.false(
|
||||
canUserExecuteLimitedActions({ createdAt: new Date() }, minimumAccountAgeMs)
|
||||
);
|
||||
t.true(
|
||||
isUserOldEnoughForShareActions({
|
||||
createdAt: new Date(Date.now() - SHARE_ACTION_ACCOUNT_AGE_MS - 1),
|
||||
})
|
||||
canUserExecuteLimitedActions(
|
||||
{
|
||||
createdAt: new Date(Date.now() - minimumAccountAgeMs - 1),
|
||||
},
|
||||
minimumAccountAgeMs
|
||||
)
|
||||
);
|
||||
});
|
||||
|
||||
test('should skip account age check when share action delay is disabled', t => {
|
||||
t.true(canUserExecuteLimitedActions({ createdAt: new Date() }, 0));
|
||||
});
|
||||
|
||||
@@ -1,5 +1,3 @@
|
||||
export const SHARE_ACTION_ACCOUNT_AGE_MS = 24 * 60 * 60 * 1000;
|
||||
|
||||
const URL_OR_DOMAIN_PATTERN =
|
||||
/(?:https?:\/\/|www\.|(?<![@\w-])(?:[a-z0-9](?:[a-z0-9-]{0,61}[a-z0-9])?\.)+[a-z]{2,63}(?=$|[^\p{L}\p{N}._-]))/iu;
|
||||
|
||||
@@ -7,6 +5,10 @@ export function containsUrlOrDomain(value: string | null | undefined) {
|
||||
return URL_OR_DOMAIN_PATTERN.test(value ?? '');
|
||||
}
|
||||
|
||||
export function isUserOldEnoughForShareActions(user: { createdAt: Date }) {
|
||||
return Date.now() - user.createdAt.getTime() >= SHARE_ACTION_ACCOUNT_AGE_MS;
|
||||
export function canUserExecuteLimitedActions(
|
||||
user: { createdAt: Date },
|
||||
minimumAccountAgeMs: number
|
||||
) {
|
||||
if (minimumAccountAgeMs <= 0) return true;
|
||||
return Date.now() - user.createdAt.getTime() >= minimumAccountAgeMs;
|
||||
}
|
||||
|
||||
@@ -17,6 +17,7 @@ import { SafeIntResolver } from 'graphql-scalars';
|
||||
import {
|
||||
ActionForbidden,
|
||||
Cache,
|
||||
Config,
|
||||
DocActionDenied,
|
||||
DocDefaultRoleCanNotBeOwner,
|
||||
DocNotFound,
|
||||
@@ -45,7 +46,7 @@ import {
|
||||
PermissionService,
|
||||
} from '../../permission';
|
||||
import { PublicUserType, WorkspaceUserType } from '../../user';
|
||||
import { isUserOldEnoughForShareActions } from '../abuse';
|
||||
import { canUserExecuteLimitedActions } from '../abuse';
|
||||
import { DocGrantsService } from '../doc-grants';
|
||||
import { WorkspaceType } from '../types';
|
||||
import { TimeBucket, TimeWindow } from './analytics-types';
|
||||
@@ -301,14 +302,27 @@ export class WorkspaceDocResolver {
|
||||
private readonly permission: PermissionService,
|
||||
private readonly models: Models,
|
||||
private readonly cache: Cache,
|
||||
private readonly event: EventBus
|
||||
private readonly event: EventBus,
|
||||
private readonly config: Config
|
||||
) {}
|
||||
|
||||
private async assertCanShare(userId: string) {
|
||||
private async assertCanShare(
|
||||
userId: string,
|
||||
context: { workspaceId: string; docId: string; action: 'publishDoc' }
|
||||
) {
|
||||
const user = await this.models.user.get(userId);
|
||||
if (!user || !isUserOldEnoughForShareActions(user)) {
|
||||
const newAccountAgeMs = this.config.auth.newAccountShareActionDelay * 1000;
|
||||
if (!user || !canUserExecuteLimitedActions(user, newAccountAgeMs)) {
|
||||
this.logger.warn('Share action blocked for new account', {
|
||||
userId,
|
||||
email: user?.email,
|
||||
createdAt: user?.createdAt,
|
||||
accountAgeMs: user ? Date.now() - user.createdAt.getTime() : null,
|
||||
minimumAccountAgeMs: newAccountAgeMs,
|
||||
...context,
|
||||
});
|
||||
throw new ActionForbidden(
|
||||
'Sharing links is unavailable during the first 24 hours after signup.'
|
||||
'This feature is temporarily unavailable for you.'
|
||||
);
|
||||
}
|
||||
}
|
||||
@@ -452,7 +466,11 @@ export class WorkspaceDocResolver {
|
||||
}
|
||||
|
||||
await this.ac.user(user.id).doc(workspaceId, docId).assert('Doc.Publish');
|
||||
await this.assertCanShare(user.id);
|
||||
await this.assertCanShare(user.id, {
|
||||
workspaceId,
|
||||
docId,
|
||||
action: 'publishDoc',
|
||||
});
|
||||
|
||||
const doc = await this.models.doc.publish(workspaceId, docId, mode);
|
||||
this.event.emit('doc.public_state.changed', { workspaceId, docId });
|
||||
|
||||
@@ -1,3 +1,4 @@
|
||||
import { Logger } from '@nestjs/common';
|
||||
import {
|
||||
Args,
|
||||
Int,
|
||||
@@ -21,6 +22,7 @@ import {
|
||||
AuthenticationRequired,
|
||||
Cache,
|
||||
CanNotRevokeYourself,
|
||||
Config,
|
||||
EventBus,
|
||||
InvalidInvitation,
|
||||
isValidCacheTtl,
|
||||
@@ -46,7 +48,7 @@ import {
|
||||
import { QuotaService } from '../../quota';
|
||||
import { UserType } from '../../user';
|
||||
import { validators } from '../../utils/validators';
|
||||
import { containsUrlOrDomain, isUserOldEnoughForShareActions } from '../abuse';
|
||||
import { canUserExecuteLimitedActions, containsUrlOrDomain } from '../abuse';
|
||||
import { WorkspaceService } from '../service';
|
||||
import {
|
||||
InvitationType,
|
||||
@@ -64,6 +66,8 @@ import {
|
||||
*/
|
||||
@Resolver(() => WorkspaceType)
|
||||
export class WorkspaceMemberResolver {
|
||||
private readonly logger = new Logger(WorkspaceMemberResolver.name);
|
||||
|
||||
constructor(
|
||||
private readonly cache: Cache,
|
||||
private readonly event: EventBus,
|
||||
@@ -73,14 +77,30 @@ export class WorkspaceMemberResolver {
|
||||
private readonly mutex: RequestMutex,
|
||||
private readonly policy: WorkspacePolicyService,
|
||||
private readonly workspaceService: WorkspaceService,
|
||||
private readonly quota: QuotaService
|
||||
private readonly quota: QuotaService,
|
||||
private readonly config: Config
|
||||
) {}
|
||||
|
||||
private async assertCanInviteOrShare(userId: string) {
|
||||
private async assertCanInviteOrShare(
|
||||
userId: string,
|
||||
context: {
|
||||
workspaceId: string;
|
||||
action: 'inviteMembers' | 'createInviteLink';
|
||||
}
|
||||
) {
|
||||
const user = await this.models.user.get(userId);
|
||||
if (!user || !isUserOldEnoughForShareActions(user)) {
|
||||
const newAccountAgeMs = this.config.auth.newAccountShareActionDelay * 1000;
|
||||
if (!user || !canUserExecuteLimitedActions(user, newAccountAgeMs)) {
|
||||
this.logger.warn('Share action blocked for new account', {
|
||||
userId,
|
||||
email: user?.email,
|
||||
createdAt: user?.createdAt,
|
||||
accountAgeMs: user ? Date.now() - user.createdAt.getTime() : null,
|
||||
minimumAccountAgeMs: newAccountAgeMs,
|
||||
...context,
|
||||
});
|
||||
throw new ActionForbidden(
|
||||
'Inviting members and creating share links are unavailable during the first 24 hours after signup.'
|
||||
'This feature is temporarily unavailable for you.'
|
||||
);
|
||||
}
|
||||
}
|
||||
@@ -169,7 +189,10 @@ export class WorkspaceMemberResolver {
|
||||
.user(me.id)
|
||||
.workspace(workspaceId)
|
||||
.assert('Workspace.Users.Manage');
|
||||
await this.assertCanInviteOrShare(me.id);
|
||||
await this.assertCanInviteOrShare(me.id, {
|
||||
workspaceId,
|
||||
action: 'inviteMembers',
|
||||
});
|
||||
await this.assertWorkspaceNameCanInvite(workspaceId);
|
||||
|
||||
if (emails.length > 512) {
|
||||
@@ -302,7 +325,10 @@ export class WorkspaceMemberResolver {
|
||||
.user(user.id)
|
||||
.workspace(workspaceId)
|
||||
.assert('Workspace.Users.Manage');
|
||||
await this.assertCanInviteOrShare(user.id);
|
||||
await this.assertCanInviteOrShare(user.id, {
|
||||
workspaceId,
|
||||
action: 'createInviteLink',
|
||||
});
|
||||
await this.assertWorkspaceNameCanInvite(workspaceId);
|
||||
|
||||
const cacheWorkspaceId = `workspace:inviteLink:${workspaceId}`;
|
||||
|
||||
@@ -190,11 +190,15 @@ export class WorkspaceMcpProvider {
|
||||
const abortedAfterDocs = abortIfNeeded(options.signal);
|
||||
if (abortedAfterDocs) return abortedAfterDocs;
|
||||
|
||||
if (!docs || docs.length === 0) {
|
||||
return toolText('No matching documents found.');
|
||||
}
|
||||
|
||||
return {
|
||||
content: (docs?.map(doc => ({
|
||||
content: docs.map(doc => ({
|
||||
type: 'text',
|
||||
text: clearEmbeddingChunk(doc).content,
|
||||
})) ?? []),
|
||||
})),
|
||||
};
|
||||
},
|
||||
});
|
||||
@@ -235,10 +239,10 @@ export class WorkspaceMcpProvider {
|
||||
}
|
||||
|
||||
return {
|
||||
content: (docs?.map(doc => ({
|
||||
content: docs.map(doc => ({
|
||||
type: 'text',
|
||||
text: JSON.stringify(pick(doc, 'docId', 'title', 'createdAt')),
|
||||
})) ?? []),
|
||||
})),
|
||||
};
|
||||
},
|
||||
});
|
||||
|
||||
@@ -740,10 +740,8 @@ fn parse_markdown_inner(markdown: &str) -> Result<MarkdownDocument, ParseError>
|
||||
inline.push(InlineAttr::link(dest_url.to_string()));
|
||||
}
|
||||
}
|
||||
Event::End(TagEnd::Link) => {
|
||||
if pending_bookmark.is_none() {
|
||||
inline.pop(InlineAttr::new(InlineStyle::Link));
|
||||
}
|
||||
Event::End(TagEnd::Link) if pending_bookmark.is_none() => {
|
||||
inline.pop(InlineAttr::new(InlineStyle::Link));
|
||||
}
|
||||
_ => {}
|
||||
}
|
||||
|
||||
@@ -136,11 +136,11 @@ impl YTypeRef {
|
||||
|
||||
#[allow(dead_code)]
|
||||
pub fn read(&self) -> Option<(RwLockReadGuard<'_, DocStore>, RwLockReadGuard<'_, YType>)> {
|
||||
self.store().and_then(|store| self.ty().map(|ty| (store, ty)))
|
||||
self.store().zip(self.ty())
|
||||
}
|
||||
|
||||
pub fn write(&self) -> Option<(RwLockWriteGuard<'_, DocStore>, RwLockWriteGuard<'_, YType>)> {
|
||||
self.store_mut().and_then(|store| self.ty_mut().map(|ty| (store, ty)))
|
||||
self.store_mut().zip(self.ty_mut())
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -79,6 +79,10 @@
|
||||
"type": "Boolean",
|
||||
"desc": "Whether require email verification before accessing restricted resources(not implemented)."
|
||||
},
|
||||
"newAccountShareActionDelay": {
|
||||
"type": "Number",
|
||||
"desc": "Minimum account age in seconds before new accounts can invite members or create share links."
|
||||
},
|
||||
"passwordRequirements": {
|
||||
"type": "Object",
|
||||
"desc": "The password strength requirements when set new password."
|
||||
|
||||
@@ -62,6 +62,11 @@ export const KNOWN_CONFIG_GROUPS = [
|
||||
fields: [
|
||||
'allowSignup',
|
||||
'allowSignupForOauth',
|
||||
{
|
||||
key: 'newAccountShareActionDelay',
|
||||
type: 'Number',
|
||||
desc: 'Minimum account age in seconds before new accounts can invite members or create share links.',
|
||||
},
|
||||
// nested json object
|
||||
{
|
||||
key: 'passwordRequirements',
|
||||
|
||||
@@ -8,7 +8,13 @@ import {
|
||||
setNativeAuthToken,
|
||||
} from './native-token';
|
||||
|
||||
interface SignInResponse {
|
||||
export interface SignInResponse {
|
||||
id?: string;
|
||||
email?: string;
|
||||
name?: string;
|
||||
hasPassword?: boolean | null;
|
||||
avatarUrl?: string | null;
|
||||
emailVerified?: boolean;
|
||||
exchangeCode?: string;
|
||||
redirectUri?: string;
|
||||
}
|
||||
@@ -97,7 +103,8 @@ export const authHandlers = {
|
||||
token,
|
||||
client_nonce: clientNonce,
|
||||
});
|
||||
await exchangeSession(endpoint, await readJson(response));
|
||||
const body = await readJson<SignInResponse>(response);
|
||||
await exchangeSession(endpoint, body);
|
||||
},
|
||||
|
||||
signInOauth: async (
|
||||
@@ -145,7 +152,9 @@ export const authHandlers = {
|
||||
password: credential.password,
|
||||
}),
|
||||
});
|
||||
await exchangeSession(endpoint, await readJson(response));
|
||||
const body = await readJson<SignInResponse>(response);
|
||||
await exchangeSession(endpoint, body);
|
||||
return body;
|
||||
},
|
||||
|
||||
signInOpenAppSignInCode: async (_e, endpoint: string, code: string) => {
|
||||
|
||||
@@ -74,7 +74,7 @@
|
||||
"lit": "^3.2.1",
|
||||
"lodash-es": "^4.17.23",
|
||||
"lottie-react": "^2.4.0",
|
||||
"mermaid": "^11.13.0",
|
||||
"mermaid": "^11.15.0",
|
||||
"mp4-muxer": "^5.2.2",
|
||||
"nanoid": "^5.1.6",
|
||||
"next-themes": "^0.4.4",
|
||||
|
||||
@@ -70,7 +70,7 @@ export function configureDefaultAuthProvider(framework: Framework) {
|
||||
headers['x-captcha-challenge'] = credential.challenge;
|
||||
}
|
||||
|
||||
await fetchService.fetch('/api/auth/sign-in', {
|
||||
const res = await fetchService.fetch('/api/auth/sign-in', {
|
||||
method: 'POST',
|
||||
body: JSON.stringify(credential),
|
||||
headers: {
|
||||
@@ -78,6 +78,7 @@ export function configureDefaultAuthProvider(framework: Framework) {
|
||||
...headers,
|
||||
},
|
||||
});
|
||||
return await res.json();
|
||||
},
|
||||
async signInOpenAppSignInCode(code: string) {
|
||||
await fetchService.fetch('/api/auth/open-app/sign-in', {
|
||||
|
||||
@@ -1,5 +1,14 @@
|
||||
import { createIdentifier } from '@toeverything/infra';
|
||||
|
||||
export interface SignInUserInfo {
|
||||
id: string;
|
||||
email: string;
|
||||
name: string;
|
||||
hasPassword: boolean | null;
|
||||
avatarUrl: string | null;
|
||||
emailVerified: boolean;
|
||||
}
|
||||
|
||||
export interface AuthProvider {
|
||||
signInMagicLink(
|
||||
email: string,
|
||||
@@ -19,7 +28,7 @@ export interface AuthProvider {
|
||||
password: string;
|
||||
verifyToken?: string;
|
||||
challenge?: string;
|
||||
}): Promise<void>;
|
||||
}): Promise<SignInUserInfo | void>;
|
||||
|
||||
signInOpenAppSignInCode(code: string): Promise<void>;
|
||||
|
||||
|
||||
@@ -235,7 +235,10 @@ export class AuthService extends Service {
|
||||
}) {
|
||||
track.$.$.auth.signIn({ method: 'password' });
|
||||
try {
|
||||
await this.store.signInPassword(credential);
|
||||
const user = await this.store.signInPassword(credential);
|
||||
if (user) {
|
||||
this.store.setCachedSignInUser(user);
|
||||
}
|
||||
this.session.revalidate();
|
||||
track.$.$.auth.signedIn({ method: 'password' });
|
||||
} catch (e) {
|
||||
|
||||
@@ -9,7 +9,7 @@ import { Store } from '@toeverything/infra';
|
||||
|
||||
import type { GlobalState, NbstoreService } from '../../storage';
|
||||
import type { AuthSessionInfo } from '../entities/session';
|
||||
import type { AuthProvider } from '../provider/auth';
|
||||
import type { AuthProvider, SignInUserInfo } from '../provider/auth';
|
||||
import type { FetchService } from '../services/fetch';
|
||||
import type { GraphQLService } from '../services/graphql';
|
||||
import type { ServerService } from '../services/server';
|
||||
@@ -57,6 +57,25 @@ export class AuthStore extends Store {
|
||||
this.globalState.set(`${this.serverService.server.id}-auth`, session);
|
||||
}
|
||||
|
||||
setCachedSignInUser(user: SignInUserInfo) {
|
||||
this.setCachedAuthSession({
|
||||
account: {
|
||||
id: user.id,
|
||||
email: user.email,
|
||||
label: user.name,
|
||||
avatar: user.avatarUrl,
|
||||
info: {
|
||||
id: user.id,
|
||||
email: user.email,
|
||||
name: user.name,
|
||||
hasPassword: Boolean(user.hasPassword),
|
||||
avatarUrl: user.avatarUrl,
|
||||
emailVerified: user.emailVerified ? 'true' : null,
|
||||
},
|
||||
},
|
||||
});
|
||||
}
|
||||
|
||||
getClientNonce() {
|
||||
return this.globalState.get<string>('auth-client-nonce');
|
||||
}
|
||||
@@ -67,7 +86,7 @@ export class AuthStore extends Store {
|
||||
|
||||
async fetchSession() {
|
||||
const { user } = await this.fetchService
|
||||
.fetch('/api/auth/session')
|
||||
.fetch('/api/auth/session', { cache: 'no-store' })
|
||||
.then(res => res.json());
|
||||
const authMethods = user
|
||||
? await this.fetchService
|
||||
@@ -109,7 +128,7 @@ export class AuthStore extends Store {
|
||||
verifyToken?: string;
|
||||
challenge?: string;
|
||||
}) {
|
||||
await this.authProvider.signInPassword(credential);
|
||||
return await this.authProvider.signInPassword(credential);
|
||||
}
|
||||
|
||||
async signInOpenAppSignInCode(code: string) {
|
||||
|
||||
@@ -112,7 +112,7 @@ impl SqliteDocStorage {
|
||||
return Ok(None);
|
||||
}
|
||||
|
||||
updates.sort_by(|a, b| a.timestamp.cmp(&b.timestamp));
|
||||
updates.sort_by_key(|a| a.timestamp);
|
||||
|
||||
let mut segments = Vec::with_capacity(snapshot.as_ref().map(|_| 1).unwrap_or(0) + updates.len());
|
||||
if let Some(record) = snapshot {
|
||||
|
||||
+1
-1
@@ -1,3 +1,3 @@
|
||||
[toolchain]
|
||||
channel = "1.95.0"
|
||||
channel = "1.96.0"
|
||||
profile = "default"
|
||||
|
||||
@@ -8,6 +8,7 @@ import {
|
||||
dragOverTitle,
|
||||
enterPlaygroundRoom,
|
||||
focusRichText,
|
||||
focusRichTextEnd,
|
||||
focusTitle,
|
||||
getClipboardHTML,
|
||||
getClipboardSnapshot,
|
||||
@@ -342,7 +343,7 @@ test(scoped`paste parent block`, async ({ page }) => {
|
||||
await type(page, 'This is child 2');
|
||||
await setInlineRangeInSelectedRichText(page, 0, 3);
|
||||
await copyByKeyboard(page);
|
||||
await focusRichText(page, 2);
|
||||
await focusRichTextEnd(page, 2);
|
||||
await page.keyboard.press(`${SHORT_KEY}+v`);
|
||||
await assertRichTexts(page, [
|
||||
'This is parent',
|
||||
@@ -363,7 +364,7 @@ test(scoped`clipboard copy multi selection`, async ({ page }) => {
|
||||
await waitNextFrame(page);
|
||||
await copyByKeyboard(page);
|
||||
await waitNextFrame(page);
|
||||
await focusRichText(page, 1);
|
||||
await focusRichTextEnd(page, 1);
|
||||
await pasteByKeyboard(page);
|
||||
await waitNextFrame(page);
|
||||
await type(page, 'cursor');
|
||||
|
||||
@@ -12,6 +12,7 @@ import {
|
||||
dragBetweenCoords,
|
||||
enterPlaygroundRoom,
|
||||
focusRichText,
|
||||
focusRichTextEnd,
|
||||
getAllNoteIds,
|
||||
getClipboardHTML,
|
||||
getClipboardSnapshot,
|
||||
@@ -192,7 +193,7 @@ test('paste a nested list to a nested list', async ({ page }) => {
|
||||
|
||||
// paste on end
|
||||
await undoByKeyboard(page);
|
||||
await page.keyboard.press('Control+ArrowRight');
|
||||
await focusRichTextEnd(page, 1);
|
||||
|
||||
/**
|
||||
* - aaa
|
||||
@@ -284,7 +285,7 @@ test('paste nested lists to a nested list', async ({ page }) => {
|
||||
|
||||
// paste on end
|
||||
await undoByKeyboard(page);
|
||||
await page.keyboard.press('Control+ArrowRight');
|
||||
await focusRichTextEnd(page, 1);
|
||||
|
||||
/**
|
||||
* - aaa
|
||||
|
||||
@@ -9,6 +9,28 @@ import {
|
||||
} from '../utils/actions/index.js';
|
||||
import { scoped, test } from '../utils/playwright.js';
|
||||
|
||||
const currentMonthAnchor = () => {
|
||||
const date = new Date();
|
||||
date.setHours(0, 0, 0, 0);
|
||||
date.setDate(1);
|
||||
return date.getTime();
|
||||
};
|
||||
|
||||
const timestampFromMonthAnchor = ({
|
||||
monthAnchor,
|
||||
day,
|
||||
}: {
|
||||
monthAnchor: number;
|
||||
day: number;
|
||||
}) => {
|
||||
const date = new Date(monthAnchor);
|
||||
date.setDate(day);
|
||||
return date.getTime();
|
||||
};
|
||||
|
||||
const getMonthTimestamp = (page: Page, monthAnchor: number, day: number) =>
|
||||
page.evaluate(timestampFromMonthAnchor, { monthAnchor, day });
|
||||
|
||||
const createCalendarDatabase = async (
|
||||
page: Page,
|
||||
options?: {
|
||||
@@ -20,6 +42,10 @@ const createCalendarDatabase = async (
|
||||
withEndDateColumn?: boolean;
|
||||
}
|
||||
) => {
|
||||
const monthAnchor = await page.evaluate(currentMonthAnchor);
|
||||
const dateTimestamp = await getMonthTimestamp(page, monthAnchor, 15);
|
||||
const endDateTimestamp = await getMonthTimestamp(page, monthAnchor, 17);
|
||||
|
||||
return page.evaluate(
|
||||
({
|
||||
withDateColumn,
|
||||
@@ -28,6 +54,9 @@ const createCalendarDatabase = async (
|
||||
rowCount,
|
||||
linkedDocTitle,
|
||||
withEndDateColumn,
|
||||
dateTimestamp,
|
||||
endDateTimestamp,
|
||||
monthAnchor,
|
||||
}) => {
|
||||
const { doc } = window;
|
||||
const rows = rowCount ?? 1;
|
||||
@@ -93,20 +122,12 @@ const createCalendarDatabase = async (
|
||||
name: 'End Date',
|
||||
})
|
||||
: undefined;
|
||||
if (dateColumnId) {
|
||||
for (const id of rowIds) {
|
||||
datasource.cellValueChange(
|
||||
id,
|
||||
dateColumnId,
|
||||
new Date('2026-05-15T00:00:00').getTime()
|
||||
);
|
||||
if (endDateColumnId) {
|
||||
datasource.cellValueChange(
|
||||
id,
|
||||
endDateColumnId,
|
||||
new Date('2026-05-17T00:00:00').getTime()
|
||||
);
|
||||
}
|
||||
for (const id of rowIds) {
|
||||
if (dateColumnId) {
|
||||
datasource.cellValueChange(id, dateColumnId, dateTimestamp);
|
||||
}
|
||||
if (endDateColumnId) {
|
||||
datasource.cellValueChange(id, endDateColumnId, endDateTimestamp);
|
||||
}
|
||||
}
|
||||
const viewId = datasource.viewManager.viewAdd('calendar');
|
||||
@@ -135,9 +156,10 @@ const createCalendarDatabase = async (
|
||||
endDateColumnId,
|
||||
viewId,
|
||||
linkedDocId,
|
||||
monthAnchor,
|
||||
};
|
||||
},
|
||||
options ?? {}
|
||||
{ ...options, dateTimestamp, endDateTimestamp, monthAnchor }
|
||||
);
|
||||
};
|
||||
|
||||
@@ -212,9 +234,7 @@ test(scoped`database calendar creates row from empty day`, async ({ page }) => {
|
||||
|
||||
await expect(page.locator('affine-data-view-record-detail')).toBeVisible();
|
||||
|
||||
const expectedDate = await page.evaluate(() =>
|
||||
new Date('2026-05-20T00:00:00').getTime()
|
||||
);
|
||||
const expectedDate = await getMonthTimestamp(page, ids.monthAnchor, 20);
|
||||
await expect
|
||||
.poll(() =>
|
||||
page.evaluate(
|
||||
@@ -327,9 +347,7 @@ test(
|
||||
.first();
|
||||
await entry.dragTo(targetDay);
|
||||
|
||||
const expectedDate = await page.evaluate(() =>
|
||||
new Date('2026-05-20T00:00:00').getTime()
|
||||
);
|
||||
const expectedDate = await getMonthTimestamp(page, ids.monthAnchor, 20);
|
||||
await expect
|
||||
.poll(async () =>
|
||||
page.evaluate(({ databaseId, rowId, dateColumnId }) => {
|
||||
@@ -390,12 +408,8 @@ test(
|
||||
.first();
|
||||
await entry.dragTo(targetDay);
|
||||
|
||||
const expectedStart = await page.evaluate(() =>
|
||||
new Date('2026-05-20T00:00:00').getTime()
|
||||
);
|
||||
const expectedEnd = await page.evaluate(() =>
|
||||
new Date('2026-05-22T00:00:00').getTime()
|
||||
);
|
||||
const expectedStart = await getMonthTimestamp(page, ids.monthAnchor, 20);
|
||||
const expectedEnd = await getMonthTimestamp(page, ids.monthAnchor, 22);
|
||||
await expect
|
||||
.poll(() =>
|
||||
page.evaluate(
|
||||
@@ -456,9 +470,7 @@ test(scoped`database calendar resizes row range end date`, async ({ page }) => {
|
||||
await page.mouse.move(targetBox.x + targetBox.width / 2, targetBox.y + 16);
|
||||
await page.mouse.up();
|
||||
|
||||
const expectedEnd = await page.evaluate(() =>
|
||||
new Date('2026-05-20T00:00:00').getTime()
|
||||
);
|
||||
const expectedEnd = await getMonthTimestamp(page, ids.monthAnchor, 20);
|
||||
await expect
|
||||
.poll(() =>
|
||||
page.evaluate(({ databaseId, rowId, endDateColumnId }) => {
|
||||
|
||||
@@ -667,7 +667,11 @@ export async function getInlineSelectionIndex(page: Page) {
|
||||
const selection = window.getSelection() as Selection;
|
||||
|
||||
const range = selection.getRangeAt(0);
|
||||
const component = range.startContainer.parentElement?.closest('rich-text');
|
||||
const startElement =
|
||||
range.startContainer instanceof Element
|
||||
? range.startContainer
|
||||
: range.startContainer.parentElement;
|
||||
const component = startElement?.closest('rich-text');
|
||||
const index = component?.inlineEditor?.getInlineRange()?.index;
|
||||
return index !== undefined ? index : -1;
|
||||
});
|
||||
@@ -677,7 +681,11 @@ export async function getInlineSelectionText(page: Page) {
|
||||
return page.evaluate(() => {
|
||||
const selection = window.getSelection() as Selection;
|
||||
const range = selection.getRangeAt(0);
|
||||
const component = range.startContainer.parentElement?.closest('rich-text');
|
||||
const startElement =
|
||||
range.startContainer instanceof Element
|
||||
? range.startContainer
|
||||
: range.startContainer.parentElement;
|
||||
const component = startElement?.closest('rich-text');
|
||||
return component?.inlineEditor?.yText.toString() ?? '';
|
||||
});
|
||||
}
|
||||
@@ -686,7 +694,11 @@ export async function getSelectedTextByInlineEditor(page: Page) {
|
||||
return page.evaluate(() => {
|
||||
const selection = window.getSelection() as Selection;
|
||||
const range = selection.getRangeAt(0);
|
||||
const component = range.startContainer.parentElement?.closest('rich-text');
|
||||
const startElement =
|
||||
range.startContainer instanceof Element
|
||||
? range.startContainer
|
||||
: range.startContainer.parentElement;
|
||||
const component = startElement?.closest('rich-text');
|
||||
|
||||
const inlineRange = component?.inlineEditor?.getInlineRange();
|
||||
if (!inlineRange) return '';
|
||||
@@ -736,12 +748,27 @@ export async function setInlineRangeInSelectedRichText(
|
||||
const selection = window.getSelection() as Selection;
|
||||
|
||||
const range = selection.getRangeAt(0);
|
||||
const component =
|
||||
range.startContainer.parentElement?.closest('rich-text');
|
||||
component?.inlineEditor?.setInlineRange({
|
||||
const startElement =
|
||||
range.startContainer instanceof Element
|
||||
? range.startContainer
|
||||
: range.startContainer.parentElement;
|
||||
const component = startElement?.closest('rich-text');
|
||||
const inlineEditor = component?.inlineEditor;
|
||||
if (!inlineEditor) {
|
||||
throw new Error('Cannot find inline editor from current selection');
|
||||
}
|
||||
component.focus();
|
||||
inlineEditor.setInlineRange({
|
||||
index,
|
||||
length,
|
||||
});
|
||||
const domRange = inlineEditor.toDomRange({ index, length });
|
||||
if (!domRange) {
|
||||
throw new Error('Cannot remap inline range to DOM range');
|
||||
}
|
||||
selection.removeAllRanges();
|
||||
selection.addRange(domRange);
|
||||
document.dispatchEvent(new Event('selectionchange'));
|
||||
},
|
||||
{ index, length }
|
||||
);
|
||||
@@ -946,6 +973,7 @@ export async function setSelection(
|
||||
length: 0,
|
||||
})!;
|
||||
|
||||
anchorRichText.focus();
|
||||
const sl = getSelection();
|
||||
if (!sl) throw new Error('Cannot get selection');
|
||||
const range = document.createRange();
|
||||
@@ -959,6 +987,7 @@ export async function setSelection(
|
||||
);
|
||||
sl.removeAllRanges();
|
||||
sl.addRange(range);
|
||||
document.dispatchEvent(new Event('selectionchange'));
|
||||
},
|
||||
{
|
||||
anchorBlockId,
|
||||
|
||||
@@ -449,7 +449,7 @@ __metadata:
|
||||
lit: "npm:^3.2.1"
|
||||
lodash-es: "npm:^4.17.23"
|
||||
lottie-react: "npm:^2.4.0"
|
||||
mermaid: "npm:^11.13.0"
|
||||
mermaid: "npm:^11.15.0"
|
||||
mp4-muxer: "npm:^5.2.2"
|
||||
nanoid: "npm:^5.1.6"
|
||||
next-themes: "npm:^0.4.4"
|
||||
@@ -990,7 +990,7 @@ __metadata:
|
||||
"@types/sinon": "npm:^21.0.0"
|
||||
"@types/supertest": "npm:^7.0.0"
|
||||
ava: "npm:^7.0.0"
|
||||
bullmq: "npm:5.53.0"
|
||||
bullmq: "npm:5.77.6"
|
||||
c8: "npm:^10.1.3"
|
||||
commander: "npm:^13.1.0"
|
||||
cookie-parser: "npm:^1.4.7"
|
||||
@@ -18974,13 +18974,14 @@ __metadata:
|
||||
linkType: hard
|
||||
|
||||
"axios@npm:^1.15.0":
|
||||
version: 1.15.2
|
||||
resolution: "axios@npm:1.15.2"
|
||||
version: 1.16.1
|
||||
resolution: "axios@npm:1.16.1"
|
||||
dependencies:
|
||||
follow-redirects: "npm:^1.15.11"
|
||||
follow-redirects: "npm:^1.16.0"
|
||||
form-data: "npm:^4.0.5"
|
||||
https-proxy-agent: "npm:^5.0.1"
|
||||
proxy-from-env: "npm:^2.1.0"
|
||||
checksum: 10/eebbd8cb777316d4252cd994a06ec9fb956ef519214a62dab6c5443ae8b753b5116e9a770502316789e6cdef1101e6aae53b6936d6a3791b2d66d75f4d7d2462
|
||||
checksum: 10/9b6218cf96321cfbbf8f160658d695367114bcf4fb62492bdc1ccd647f184b5c71ae400e5ecaaf41079bc561de2ecbaf1fec63f398b3ec53389beff7694df64c
|
||||
languageName: node
|
||||
linkType: hard
|
||||
|
||||
@@ -19367,18 +19368,22 @@ __metadata:
|
||||
languageName: node
|
||||
linkType: hard
|
||||
|
||||
"bullmq@npm:5.53.0":
|
||||
version: 5.53.0
|
||||
resolution: "bullmq@npm:5.53.0"
|
||||
"bullmq@npm:5.77.6":
|
||||
version: 5.77.6
|
||||
resolution: "bullmq@npm:5.77.6"
|
||||
dependencies:
|
||||
cron-parser: "npm:^4.9.0"
|
||||
ioredis: "npm:^5.4.1"
|
||||
msgpackr: "npm:^1.11.2"
|
||||
node-abort-controller: "npm:^3.1.1"
|
||||
semver: "npm:^7.5.4"
|
||||
tslib: "npm:^2.0.0"
|
||||
uuid: "npm:^9.0.0"
|
||||
checksum: 10/470a9cb63d32b6ce8fe0275a249ba983167f6d5f494b4add1fc907a0198e178ece1c8874dc59030f232dab943f4546ecdf95c2b0f1d35c655341c970cbd0d3d9
|
||||
cron-parser: "npm:4.9.0"
|
||||
ioredis: "npm:5.10.1"
|
||||
msgpackr: "npm:2.0.1"
|
||||
node-abort-controller: "npm:3.1.1"
|
||||
semver: "npm:7.8.0"
|
||||
tslib: "npm:2.8.1"
|
||||
peerDependencies:
|
||||
redis: ">=5.0.0"
|
||||
peerDependenciesMeta:
|
||||
redis:
|
||||
optional: true
|
||||
checksum: 10/6003707988972389574e24ae2685152dc68d53d9a4b51d49866f2910c56cc51dbf8ce301e90f289ee5a9e0c0b233fd96b27a0fca105144642074836e4565f8df
|
||||
languageName: node
|
||||
linkType: hard
|
||||
|
||||
@@ -20792,7 +20797,7 @@ __metadata:
|
||||
languageName: node
|
||||
linkType: hard
|
||||
|
||||
"cron-parser@npm:^4.9.0":
|
||||
"cron-parser@npm:4.9.0":
|
||||
version: 4.9.0
|
||||
resolution: "cron-parser@npm:4.9.0"
|
||||
dependencies:
|
||||
@@ -23740,7 +23745,7 @@ __metadata:
|
||||
languageName: node
|
||||
linkType: hard
|
||||
|
||||
"follow-redirects@npm:^1.0.0, follow-redirects@npm:^1.15.11":
|
||||
"follow-redirects@npm:^1.0.0, follow-redirects@npm:^1.16.0":
|
||||
version: 1.16.0
|
||||
resolution: "follow-redirects@npm:1.16.0"
|
||||
peerDependenciesMeta:
|
||||
@@ -25068,7 +25073,7 @@ __metadata:
|
||||
languageName: node
|
||||
linkType: hard
|
||||
|
||||
"https-proxy-agent@npm:^5.0.0":
|
||||
"https-proxy-agent@npm:^5.0.0, https-proxy-agent@npm:^5.0.1":
|
||||
version: 5.0.1
|
||||
resolution: "https-proxy-agent@npm:5.0.1"
|
||||
dependencies:
|
||||
@@ -27780,7 +27785,7 @@ __metadata:
|
||||
languageName: node
|
||||
linkType: hard
|
||||
|
||||
"mermaid@npm:^11.13.0":
|
||||
"mermaid@npm:^11.15.0":
|
||||
version: 11.15.0
|
||||
resolution: "mermaid@npm:11.15.0"
|
||||
dependencies:
|
||||
@@ -28545,15 +28550,15 @@ __metadata:
|
||||
languageName: node
|
||||
linkType: hard
|
||||
|
||||
"msgpackr@npm:^1.11.2":
|
||||
version: 1.11.10
|
||||
resolution: "msgpackr@npm:1.11.10"
|
||||
"msgpackr@npm:2.0.1":
|
||||
version: 2.0.1
|
||||
resolution: "msgpackr@npm:2.0.1"
|
||||
dependencies:
|
||||
msgpackr-extract: "npm:^3.0.2"
|
||||
dependenciesMeta:
|
||||
msgpackr-extract:
|
||||
optional: true
|
||||
checksum: 10/e210128fac395b6173cb6784926eec724ea5e3fca72639cd07fb0af762f4027a4026bb4096703bd3b8510eee7e9b351fd52721ddf9bc2091e354d5dff93d45dd
|
||||
checksum: 10/9b51a18832e86b27a3187cea5dfbff5d146d643f4beb24f8cf086a5c5a1ae3e4bfaa0459c7fd9b1cca6d2d68cfc3c03b2235ff63f3895d9d8a46bceda241ec4f
|
||||
languageName: node
|
||||
linkType: hard
|
||||
|
||||
@@ -28812,7 +28817,7 @@ __metadata:
|
||||
languageName: node
|
||||
linkType: hard
|
||||
|
||||
"node-abort-controller@npm:^3.1.1":
|
||||
"node-abort-controller@npm:3.1.1":
|
||||
version: 3.1.1
|
||||
resolution: "node-abort-controller@npm:3.1.1"
|
||||
checksum: 10/0a2cdb7ec0aeaf3cb31e1ca0e192f5add48f1c5c9c9ed822129f9dddbd9432f69b7425982f94ce803c56a2104884530aa67cd57696e5774b2e5b8ec2f58de042
|
||||
@@ -32503,6 +32508,15 @@ __metadata:
|
||||
languageName: node
|
||||
linkType: hard
|
||||
|
||||
"semver@npm:7.8.0, semver@npm:^7.0.0, semver@npm:^7.1.1, semver@npm:^7.1.3, semver@npm:^7.2.1, semver@npm:^7.3.2, semver@npm:^7.3.5, semver@npm:^7.3.8, semver@npm:^7.5.3, semver@npm:^7.5.4, semver@npm:^7.6.0, semver@npm:^7.6.2, semver@npm:^7.6.3, semver@npm:^7.7.1, semver@npm:^7.7.2, semver@npm:^7.7.3, semver@npm:^7.7.4":
|
||||
version: 7.8.0
|
||||
resolution: "semver@npm:7.8.0"
|
||||
bin:
|
||||
semver: bin/semver.js
|
||||
checksum: 10/039a8f68a581c03c1ac17c990316da57a79a93af9b109b712739c50cd4d464079f7e3fee31c008b472e390c7ba48a11ed2b86e91d8602bf06059d4a266db1426
|
||||
languageName: node
|
||||
linkType: hard
|
||||
|
||||
"semver@npm:^6.2.0, semver@npm:^6.3.1":
|
||||
version: 6.3.1
|
||||
resolution: "semver@npm:6.3.1"
|
||||
@@ -32512,15 +32526,6 @@ __metadata:
|
||||
languageName: node
|
||||
linkType: hard
|
||||
|
||||
"semver@npm:^7.0.0, semver@npm:^7.1.1, semver@npm:^7.1.3, semver@npm:^7.2.1, semver@npm:^7.3.2, semver@npm:^7.3.5, semver@npm:^7.3.8, semver@npm:^7.5.3, semver@npm:^7.5.4, semver@npm:^7.6.0, semver@npm:^7.6.2, semver@npm:^7.6.3, semver@npm:^7.7.1, semver@npm:^7.7.2, semver@npm:^7.7.3, semver@npm:^7.7.4":
|
||||
version: 7.8.0
|
||||
resolution: "semver@npm:7.8.0"
|
||||
bin:
|
||||
semver: bin/semver.js
|
||||
checksum: 10/039a8f68a581c03c1ac17c990316da57a79a93af9b109b712739c50cd4d464079f7e3fee31c008b472e390c7ba48a11ed2b86e91d8602bf06059d4a266db1426
|
||||
languageName: node
|
||||
linkType: hard
|
||||
|
||||
"semver@npm:~7.7.3":
|
||||
version: 7.7.4
|
||||
resolution: "semver@npm:7.7.4"
|
||||
@@ -35033,7 +35038,7 @@ __metadata:
|
||||
languageName: node
|
||||
linkType: hard
|
||||
|
||||
"uuid@npm:^9.0.0, uuid@npm:^9.0.1":
|
||||
"uuid@npm:^9.0.1":
|
||||
version: 9.0.1
|
||||
resolution: "uuid@npm:9.0.1"
|
||||
bin:
|
||||
|
||||
Reference in New Issue
Block a user