mirror of
https://github.com/outline/outline.git
synced 2026-06-13 03:14:59 +03:00
ecafd5f32a
* chore: Update JSON importer to use zip streaming, new importer flow * chore: Drop teamId from import urlId collision check and remove unused internal-id scaffolding urlId is globally unique on Document/Collection so the team scope was wrong. Also removes leftover internal-id generation in JSONAPIImportTask that was never used in task input/output. * Restore classes used upstream
1003 lines
24 KiB
TypeScript
1003 lines
24 KiB
TypeScript
import { faker } from "@faker-js/faker";
|
|
import { isNil, isNull } from "es-toolkit/compat";
|
|
import { Node } from "prosemirror-model";
|
|
import type { InferCreationAttributes } from "sequelize";
|
|
import type { DeepPartial } from "utility-types";
|
|
import { randomUUID } from "node:crypto";
|
|
import { randomString } from "@shared/random";
|
|
import type { ProsemirrorData, ReactionSummary } from "@shared/types";
|
|
import {
|
|
CollectionPermission,
|
|
FileOperationState,
|
|
FileOperationType,
|
|
ImportState,
|
|
IntegrationService,
|
|
IntegrationType,
|
|
MentionType,
|
|
NotificationEventType,
|
|
SubscriptionType,
|
|
UserRole,
|
|
} from "@shared/types";
|
|
import { parser, schema } from "@server/editor";
|
|
import type { AuthenticationProvider } from "@server/models";
|
|
import {
|
|
Share,
|
|
Team,
|
|
User,
|
|
UserPasskey,
|
|
Event,
|
|
Document,
|
|
Emoji,
|
|
Star,
|
|
Collection,
|
|
Group,
|
|
GroupUser,
|
|
Attachment,
|
|
IntegrationAuthentication,
|
|
Integration,
|
|
FileOperation,
|
|
WebhookSubscription,
|
|
WebhookDelivery,
|
|
ApiKey,
|
|
Subscription,
|
|
Notification,
|
|
SearchQuery,
|
|
Pin,
|
|
Comment,
|
|
Import,
|
|
OAuthAuthorizationCode,
|
|
OAuthClient,
|
|
OAuthAuthentication,
|
|
Relationship,
|
|
Template,
|
|
} from "@server/models";
|
|
import { RelationshipType } from "@server/models/Relationship";
|
|
import AttachmentHelper from "@server/models/helpers/AttachmentHelper";
|
|
import { hash } from "@server/utils/crypto";
|
|
import { OAuthInterface } from "@server/utils/oauth/OAuthInterface";
|
|
|
|
export async function buildApiKey(overrides: Partial<ApiKey> = {}) {
|
|
if (!overrides.userId) {
|
|
const user = await buildUser();
|
|
overrides.userId = user.id;
|
|
}
|
|
|
|
return ApiKey.create({
|
|
name: faker.lorem.words(3),
|
|
...overrides,
|
|
});
|
|
}
|
|
|
|
export async function buildShare(overrides: Partial<Share> = {}) {
|
|
if (!overrides.teamId) {
|
|
const team = await buildTeam();
|
|
overrides.teamId = team.id;
|
|
}
|
|
|
|
if (!overrides.userId) {
|
|
const user = await buildUser({
|
|
teamId: overrides.teamId,
|
|
});
|
|
overrides.userId = user.id;
|
|
}
|
|
|
|
if (!overrides.documentId && !overrides.collectionId) {
|
|
const document = await buildDocument({
|
|
createdById: overrides.userId,
|
|
teamId: overrides.teamId,
|
|
});
|
|
overrides.documentId = document.id;
|
|
}
|
|
|
|
return Share.create({
|
|
published: true,
|
|
...overrides,
|
|
});
|
|
}
|
|
|
|
export async function buildStar(overrides: Partial<Star> = {}) {
|
|
let user;
|
|
|
|
if (overrides.userId) {
|
|
user = await User.findByPk(overrides.userId, {
|
|
rejectOnEmpty: true,
|
|
});
|
|
} else {
|
|
user = await buildUser();
|
|
overrides.userId = user.id;
|
|
}
|
|
|
|
if (!overrides.documentId) {
|
|
const document = await buildDocument({
|
|
createdById: overrides.userId,
|
|
teamId: user.teamId,
|
|
});
|
|
overrides.documentId = document.id;
|
|
}
|
|
|
|
return Star.create({
|
|
index: "h",
|
|
...overrides,
|
|
});
|
|
}
|
|
|
|
export async function buildSubscription(overrides: Partial<Subscription> = {}) {
|
|
let user;
|
|
|
|
if (overrides.userId) {
|
|
user = await User.findByPk(overrides.userId, {
|
|
rejectOnEmpty: true,
|
|
});
|
|
} else {
|
|
user = await buildUser();
|
|
overrides.userId = user.id;
|
|
}
|
|
|
|
if (!overrides.documentId && !overrides.collectionId) {
|
|
const document = await buildDocument({
|
|
createdById: overrides.userId,
|
|
teamId: user.teamId,
|
|
});
|
|
overrides.documentId = document.id;
|
|
}
|
|
|
|
return Subscription.create({
|
|
event: SubscriptionType.Document,
|
|
...overrides,
|
|
});
|
|
}
|
|
|
|
export function buildTeam(
|
|
overrides: Omit<Partial<Team>, "authenticationProviders"> & {
|
|
authenticationProviders?: Partial<AuthenticationProvider>[];
|
|
} = {}
|
|
) {
|
|
return Team.create(
|
|
{
|
|
name: faker.company.name(),
|
|
passkeysEnabled: false,
|
|
authenticationProviders: [
|
|
{
|
|
name: "slack",
|
|
providerId: randomString(32),
|
|
},
|
|
],
|
|
...overrides,
|
|
} as Partial<InferCreationAttributes<Team>>,
|
|
{
|
|
include: "authenticationProviders",
|
|
}
|
|
);
|
|
}
|
|
|
|
export function buildEvent(overrides: Partial<Event> = {}) {
|
|
return Event.create({
|
|
name: "documents.publish",
|
|
ip: "127.0.0.1",
|
|
...overrides,
|
|
});
|
|
}
|
|
|
|
export async function buildGuestUser(overrides: Partial<User> = {}) {
|
|
if (!overrides.teamId) {
|
|
const team = await buildTeam();
|
|
overrides.teamId = team.id;
|
|
}
|
|
|
|
return User.create({
|
|
email: faker.internet.email().toLowerCase(),
|
|
name: faker.person.fullName(),
|
|
createdAt: new Date("2018-01-01T00:00:00.000Z"),
|
|
lastActiveAt: new Date("2018-01-01T00:00:00.000Z"),
|
|
role: UserRole.Guest,
|
|
...overrides,
|
|
});
|
|
}
|
|
|
|
export async function buildUser(overrides: Partial<User> = {}) {
|
|
let team;
|
|
|
|
if (!overrides.teamId) {
|
|
team = await buildTeam();
|
|
overrides.teamId = team.id;
|
|
} else {
|
|
team = await Team.findByPk(overrides.teamId, {
|
|
include: "authenticationProviders",
|
|
rejectOnEmpty: true,
|
|
paranoid: false,
|
|
});
|
|
}
|
|
|
|
const authenticationProvider = team.authenticationProviders[0];
|
|
const user = await User.create(
|
|
{
|
|
email: faker.internet.email().toLowerCase(),
|
|
name: faker.person.fullName(),
|
|
createdAt: new Date("2018-01-01T00:00:00.000Z"),
|
|
updatedAt: new Date("2018-01-02T00:00:00.000Z"),
|
|
lastActiveAt: new Date("2018-01-03T00:00:00.000Z"),
|
|
authentications: authenticationProvider
|
|
? [
|
|
{
|
|
authenticationProviderId: authenticationProvider.id,
|
|
providerId: randomString(32),
|
|
},
|
|
]
|
|
: [],
|
|
...overrides,
|
|
} as Partial<InferCreationAttributes<User>>,
|
|
{
|
|
include: "authentications",
|
|
}
|
|
);
|
|
|
|
if (team) {
|
|
user.team = team;
|
|
}
|
|
return user;
|
|
}
|
|
|
|
export async function buildAdmin(overrides: Partial<User> = {}) {
|
|
return buildUser({ ...overrides, role: UserRole.Admin });
|
|
}
|
|
|
|
export async function buildViewer(overrides: Partial<User> = {}) {
|
|
return buildUser({ ...overrides, role: UserRole.Viewer });
|
|
}
|
|
|
|
export async function buildInvite(overrides: Partial<User> = {}) {
|
|
if (!overrides.teamId) {
|
|
const team = await buildTeam();
|
|
overrides.teamId = team.id;
|
|
}
|
|
|
|
const actor = await buildUser({ teamId: overrides.teamId });
|
|
|
|
return User.create({
|
|
email: faker.internet.email().toLowerCase(),
|
|
name: faker.person.fullName(),
|
|
createdAt: new Date("2018-01-01T00:00:00.000Z"),
|
|
invitedById: actor.id,
|
|
authentications: [],
|
|
...overrides,
|
|
lastActiveAt: null,
|
|
});
|
|
}
|
|
|
|
export async function buildIntegration(overrides: Partial<Integration> = {}) {
|
|
if (!overrides.teamId) {
|
|
const team = await buildTeam();
|
|
overrides.teamId = team.id;
|
|
}
|
|
|
|
const user = await buildUser({
|
|
teamId: overrides.teamId,
|
|
});
|
|
const authentication = await IntegrationAuthentication.create({
|
|
service: IntegrationService.Slack,
|
|
userId: user.id,
|
|
teamId: user.teamId,
|
|
token: randomString(32),
|
|
scopes: ["example", "scopes", "here"],
|
|
});
|
|
return Integration.create({
|
|
service: IntegrationService.Slack,
|
|
type: IntegrationType.Post,
|
|
events: ["documents.update", "documents.publish"],
|
|
settings: {
|
|
serviceTeamId: randomUUID(),
|
|
},
|
|
authenticationId: authentication.id,
|
|
...overrides,
|
|
});
|
|
}
|
|
|
|
export async function buildCollection(
|
|
overrides: Partial<Collection> & { userId?: string } = {}
|
|
) {
|
|
if (!overrides.teamId) {
|
|
const team = await buildTeam();
|
|
overrides.teamId = team.id;
|
|
}
|
|
|
|
if (!overrides.userId) {
|
|
const user = await buildUser({
|
|
teamId: overrides.teamId,
|
|
});
|
|
overrides.userId = user.id;
|
|
}
|
|
|
|
if (overrides.archivedAt && !overrides.archivedById) {
|
|
overrides.archivedById = overrides.userId;
|
|
}
|
|
|
|
if (overrides.permission === undefined) {
|
|
overrides.permission = CollectionPermission.ReadWrite;
|
|
}
|
|
|
|
return Collection.scope("withDocumentStructure").create({
|
|
name: faker.lorem.words(2),
|
|
description: faker.lorem.words(4),
|
|
createdById: overrides.userId,
|
|
...overrides,
|
|
});
|
|
}
|
|
|
|
export async function buildGroup(
|
|
overrides: Partial<Group> & { userId?: string } = {}
|
|
) {
|
|
if (!overrides.teamId) {
|
|
const team = await buildTeam();
|
|
overrides.teamId = team.id;
|
|
}
|
|
|
|
if (!overrides.userId) {
|
|
const user = await buildUser({
|
|
teamId: overrides.teamId,
|
|
});
|
|
overrides.userId = user.id;
|
|
}
|
|
|
|
return Group.create({
|
|
name: faker.lorem.words(2),
|
|
createdById: overrides.userId,
|
|
...overrides,
|
|
});
|
|
}
|
|
|
|
export async function buildGroupUser(
|
|
overrides: Partial<GroupUser> & { userId?: string; teamId?: string } = {}
|
|
) {
|
|
if (!overrides.teamId) {
|
|
const team = await buildTeam();
|
|
overrides.teamId = team.id;
|
|
}
|
|
|
|
if (!overrides.userId) {
|
|
const user = await buildUser({
|
|
teamId: overrides.teamId,
|
|
});
|
|
overrides.userId = user.id;
|
|
}
|
|
|
|
return GroupUser.create({
|
|
createdById: overrides.userId,
|
|
...overrides,
|
|
});
|
|
}
|
|
|
|
export async function buildDraftDocument(
|
|
overrides: Partial<Document> & { userId?: string } = {}
|
|
) {
|
|
return buildDocument({ ...overrides, publishedAt: null });
|
|
}
|
|
|
|
export async function buildDocument(
|
|
// Omission first, addition later?
|
|
// This is actually a workaround to allow
|
|
// passing collectionId as null. Ideally, it
|
|
// should be updated in the Document model itself
|
|
// but that'd cascade and require further changes
|
|
// beyond the scope of what's required now
|
|
overrides: Omit<Partial<Document>, "collectionId"> & { userId?: string } & {
|
|
collectionId?: string | null;
|
|
} = {}
|
|
) {
|
|
if (!overrides.teamId) {
|
|
const team = await buildTeam();
|
|
overrides.teamId = team.id;
|
|
}
|
|
|
|
if (!overrides.userId) {
|
|
const user = await buildUser();
|
|
overrides.userId = user.id;
|
|
}
|
|
|
|
let collection;
|
|
if (overrides.collectionId === undefined) {
|
|
collection = await buildCollection({
|
|
teamId: overrides.teamId,
|
|
userId: overrides.userId,
|
|
});
|
|
overrides.collectionId = collection.id;
|
|
}
|
|
|
|
const text = overrides.text ?? "This is the text in an example document";
|
|
const document = await Document.create(
|
|
{
|
|
title: faker.lorem.words(4),
|
|
content: overrides.content ?? parser.parse(text)?.toJSON(),
|
|
text,
|
|
publishedAt: isNull(overrides.collectionId) ? null : new Date(),
|
|
lastModifiedById: overrides.userId,
|
|
createdById: overrides.userId,
|
|
editorVersion: "12.0.0",
|
|
...overrides,
|
|
},
|
|
{
|
|
silent: overrides.createdAt || overrides.updatedAt ? true : false,
|
|
}
|
|
);
|
|
|
|
if (overrides.collectionId && overrides.publishedAt !== null) {
|
|
collection = collection
|
|
? await Collection.findByPk(overrides.collectionId, {
|
|
includeDocumentStructure: true,
|
|
})
|
|
: undefined;
|
|
|
|
await collection?.addDocumentToStructure(document, 0);
|
|
}
|
|
|
|
return document;
|
|
}
|
|
|
|
export async function buildTemplate(
|
|
overrides: Omit<Partial<Template>, "collectionId"> & {
|
|
userId?: string;
|
|
text?: string;
|
|
collectionId?: string | null;
|
|
} = {}
|
|
) {
|
|
if (!overrides.teamId) {
|
|
const team = await buildTeam();
|
|
overrides.teamId = team.id;
|
|
}
|
|
|
|
if (!overrides.userId) {
|
|
const user = await buildUser({
|
|
teamId: overrides.teamId,
|
|
});
|
|
overrides.userId = user.id;
|
|
}
|
|
|
|
let collection;
|
|
if (overrides.collectionId === undefined) {
|
|
collection = await buildCollection({
|
|
teamId: overrides.teamId,
|
|
userId: overrides.userId,
|
|
});
|
|
overrides.collectionId = collection.id;
|
|
}
|
|
|
|
const text = overrides.text ?? "This is the text in an example template";
|
|
const template = await Template.create(
|
|
{
|
|
title: faker.lorem.words(4),
|
|
content: overrides.content ?? parser.parse(text)?.toJSON(),
|
|
lastModifiedById: overrides.userId,
|
|
createdById: overrides.userId,
|
|
editorVersion: "12.0.0",
|
|
...overrides,
|
|
},
|
|
{
|
|
silent: overrides.createdAt || overrides.updatedAt ? true : false,
|
|
}
|
|
);
|
|
|
|
return template;
|
|
}
|
|
|
|
export async function buildComment(overrides: {
|
|
userId: string;
|
|
documentId: string;
|
|
parentCommentId?: string;
|
|
resolvedById?: string;
|
|
reactions?: ReactionSummary[];
|
|
createdAt?: Date;
|
|
}) {
|
|
const comment = await Comment.create(
|
|
{
|
|
resolvedById: overrides.resolvedById,
|
|
parentCommentId: overrides.parentCommentId,
|
|
documentId: overrides.documentId,
|
|
data: {
|
|
type: "doc",
|
|
content: [
|
|
{
|
|
type: "paragraph",
|
|
content: [
|
|
{
|
|
content: [],
|
|
type: "text",
|
|
text: "test",
|
|
},
|
|
],
|
|
},
|
|
],
|
|
},
|
|
createdById: overrides.userId,
|
|
reactions: overrides.reactions,
|
|
createdAt: overrides.createdAt,
|
|
updatedAt: overrides.createdAt,
|
|
},
|
|
{ silent: overrides.createdAt ? true : false }
|
|
);
|
|
|
|
return comment;
|
|
}
|
|
|
|
export async function buildResolvedComment(
|
|
user: User,
|
|
overrides: Parameters<typeof buildComment>[0]
|
|
) {
|
|
const comment = await buildComment(overrides);
|
|
comment.resolve(user);
|
|
await comment.save();
|
|
return comment;
|
|
}
|
|
|
|
export async function buildFileOperation(
|
|
overrides: Partial<FileOperation> = {}
|
|
) {
|
|
if (!overrides.teamId) {
|
|
const team = await buildTeam();
|
|
overrides.teamId = team.id;
|
|
}
|
|
|
|
if (!overrides.userId) {
|
|
const user = await buildAdmin({
|
|
teamId: overrides.teamId,
|
|
});
|
|
overrides.userId = user.id;
|
|
}
|
|
|
|
return FileOperation.create({
|
|
state: FileOperationState.Creating,
|
|
type: FileOperationType.Export,
|
|
size: 0,
|
|
key: "uploads/key/to/file.zip",
|
|
collectionId: null,
|
|
url: "https://www.urltos3file.com/file.zip",
|
|
...overrides,
|
|
});
|
|
}
|
|
|
|
// oxlint-disable-next-line @typescript-eslint/no-explicit-any
|
|
export async function buildImport(overrides: Partial<Import<any>> = {}) {
|
|
if (!overrides.teamId) {
|
|
const team = await buildTeam();
|
|
overrides.teamId = team.id;
|
|
}
|
|
|
|
if (!overrides.createdById) {
|
|
const user = await buildAdmin({
|
|
teamId: overrides.teamId,
|
|
});
|
|
overrides.createdById = user.id;
|
|
}
|
|
|
|
if (!overrides.integrationId) {
|
|
const integration = await buildIntegration({
|
|
service: IntegrationService.Notion,
|
|
userId: overrides.createdById,
|
|
teamId: overrides.teamId,
|
|
});
|
|
overrides.integrationId = integration.id;
|
|
}
|
|
|
|
// Skip BeforeCreate hooks so tests can seed multiple imports per team. The
|
|
// production "one in-progress import per team" rule is enforced by the
|
|
// Import.checkInProgress hook; tests don't need to abide by it.
|
|
// oxlint-disable-next-line @typescript-eslint/no-explicit-any
|
|
return Import.create<Import<any>>(
|
|
{
|
|
name: "testImport",
|
|
service: IntegrationService.Notion,
|
|
state: ImportState.Created,
|
|
input: [
|
|
{
|
|
permission: CollectionPermission.Read,
|
|
},
|
|
],
|
|
...overrides,
|
|
// oxlint-disable-next-line @typescript-eslint/no-explicit-any
|
|
} as any,
|
|
{ hooks: false }
|
|
);
|
|
}
|
|
|
|
export async function buildAttachment(
|
|
overrides: Partial<Attachment> = {},
|
|
fileName?: string
|
|
) {
|
|
if (!overrides.teamId) {
|
|
const team = await buildTeam();
|
|
overrides.teamId = team.id;
|
|
}
|
|
|
|
if (!overrides.userId) {
|
|
const user = await buildUser({
|
|
teamId: overrides.teamId,
|
|
});
|
|
overrides.userId = user.id;
|
|
}
|
|
|
|
if (!overrides.documentId) {
|
|
const document = await buildDocument({
|
|
teamId: overrides.teamId,
|
|
userId: overrides.userId,
|
|
});
|
|
overrides.documentId = document.id;
|
|
}
|
|
|
|
const id = randomUUID();
|
|
const acl = overrides.acl || "public-read";
|
|
const name = fileName || faker.system.fileName();
|
|
return Attachment.create({
|
|
id,
|
|
key: AttachmentHelper.getKey({ id, name, userId: overrides.userId }),
|
|
contentType: "image/png",
|
|
size: 1_000_000,
|
|
acl,
|
|
name,
|
|
createdAt: new Date("2018-01-02T00:00:00.000Z"),
|
|
updatedAt: new Date("2018-01-02T00:00:00.000Z"),
|
|
...overrides,
|
|
});
|
|
}
|
|
|
|
export async function buildEmoji(
|
|
overrides: Partial<Emoji> = {}
|
|
): Promise<Emoji> {
|
|
if (!overrides.teamId) {
|
|
const team = await buildTeam();
|
|
overrides.teamId = team.id;
|
|
}
|
|
|
|
if (!overrides.createdById) {
|
|
const user = await buildUser({
|
|
teamId: overrides.teamId,
|
|
});
|
|
overrides.createdById = user.id;
|
|
}
|
|
|
|
if (!overrides.attachmentId) {
|
|
const attachment = await buildAttachment({
|
|
teamId: overrides.teamId,
|
|
userId: overrides.createdById,
|
|
contentType: "image/png",
|
|
});
|
|
overrides.attachmentId = attachment.id;
|
|
}
|
|
|
|
return Emoji.create({
|
|
name: faker.word
|
|
.adjective()
|
|
.toLowerCase()
|
|
.replace(/[^a-z0-9_]/g, "_"),
|
|
...overrides,
|
|
});
|
|
}
|
|
|
|
export async function buildWebhookSubscription(
|
|
overrides: Partial<WebhookSubscription> = {}
|
|
): Promise<WebhookSubscription> {
|
|
if (!overrides.teamId) {
|
|
const team = await buildTeam();
|
|
overrides.teamId = team.id;
|
|
}
|
|
if (!overrides.createdById) {
|
|
const user = await buildUser({
|
|
teamId: overrides.teamId,
|
|
});
|
|
overrides.createdById = user.id;
|
|
}
|
|
if (!overrides.name) {
|
|
overrides.name = "Test Webhook Subscription";
|
|
}
|
|
if (!overrides.url) {
|
|
overrides.url = "https://www.example.com/webhook";
|
|
}
|
|
if (!overrides.events) {
|
|
overrides.events = ["*"];
|
|
}
|
|
if (!overrides.enabled) {
|
|
overrides.enabled = true;
|
|
}
|
|
|
|
return WebhookSubscription.create(overrides);
|
|
}
|
|
|
|
export async function buildWebhookDelivery(
|
|
overrides: Partial<WebhookDelivery> = {}
|
|
): Promise<WebhookDelivery> {
|
|
if (!overrides.status) {
|
|
overrides.status = "success";
|
|
}
|
|
if (!overrides.statusCode) {
|
|
overrides.statusCode = 200;
|
|
}
|
|
if (!overrides.requestBody) {
|
|
overrides.requestBody = "{}";
|
|
}
|
|
if (!overrides.requestHeaders) {
|
|
overrides.requestHeaders = {};
|
|
}
|
|
if (!overrides.webhookSubscriptionId) {
|
|
const webhookSubscription = await buildWebhookSubscription();
|
|
overrides.webhookSubscriptionId = webhookSubscription.id;
|
|
}
|
|
if (!overrides.createdAt) {
|
|
overrides.createdAt = new Date();
|
|
}
|
|
|
|
return WebhookDelivery.create(overrides);
|
|
}
|
|
|
|
export async function buildNotification(
|
|
overrides: Partial<Notification> = {}
|
|
): Promise<Notification> {
|
|
if (!overrides.event) {
|
|
overrides.event = NotificationEventType.UpdateDocument;
|
|
}
|
|
|
|
if (!overrides.teamId) {
|
|
const team = await buildTeam();
|
|
overrides.teamId = team.id;
|
|
}
|
|
|
|
if (!overrides.userId) {
|
|
const user = await buildUser({
|
|
teamId: overrides.teamId,
|
|
});
|
|
overrides.userId = user.id;
|
|
}
|
|
|
|
return Notification.create(overrides);
|
|
}
|
|
|
|
export async function buildSearchQuery(
|
|
overrides: Partial<SearchQuery> = {}
|
|
): Promise<SearchQuery> {
|
|
if (!overrides.teamId) {
|
|
const team = await buildTeam();
|
|
overrides.teamId = team.id;
|
|
}
|
|
|
|
if (!overrides.userId) {
|
|
const user = await buildUser({
|
|
teamId: overrides.teamId,
|
|
});
|
|
overrides.userId = user.id;
|
|
}
|
|
|
|
if (!overrides.source) {
|
|
overrides.source = "app";
|
|
}
|
|
|
|
if (isNil(overrides.query)) {
|
|
overrides.query = "query";
|
|
}
|
|
|
|
if (isNil(overrides.results)) {
|
|
overrides.results = 1;
|
|
}
|
|
|
|
return SearchQuery.create(overrides);
|
|
}
|
|
|
|
export async function buildPin(overrides: Partial<Pin> = {}): Promise<Pin> {
|
|
if (!overrides.teamId) {
|
|
const team = await buildTeam();
|
|
overrides.teamId = team.id;
|
|
}
|
|
|
|
if (!overrides.createdById) {
|
|
const user = await buildUser({
|
|
teamId: overrides.teamId,
|
|
});
|
|
overrides.createdById = user.id;
|
|
}
|
|
|
|
if (!overrides.documentId) {
|
|
const document = await buildDocument({
|
|
teamId: overrides.teamId,
|
|
});
|
|
overrides.documentId = document.id;
|
|
}
|
|
|
|
return Pin.create(overrides);
|
|
}
|
|
|
|
export async function buildOAuthClient(overrides: Partial<OAuthClient> = {}) {
|
|
if (!overrides.teamId) {
|
|
const team = await buildTeam();
|
|
overrides.teamId = team.id;
|
|
}
|
|
|
|
if (!overrides.createdById && overrides.createdById !== null) {
|
|
const user = await buildUser({
|
|
teamId: overrides.teamId,
|
|
});
|
|
overrides.createdById = user.id;
|
|
}
|
|
|
|
return OAuthClient.create(
|
|
{
|
|
name: faker.company.name(),
|
|
description: faker.lorem.sentence(),
|
|
redirectUris: ["https://example.com/oauth/callback"],
|
|
published: true,
|
|
...(overrides.createdAt && !overrides.updatedAt
|
|
? { updatedAt: overrides.createdAt }
|
|
: {}),
|
|
...overrides,
|
|
},
|
|
{
|
|
silent: overrides.createdAt || overrides.updatedAt ? true : false,
|
|
}
|
|
);
|
|
}
|
|
|
|
export async function buildOAuthAuthorizationCode(
|
|
overrides: Partial<OAuthAuthorizationCode> = {}
|
|
) {
|
|
if (!overrides.userId) {
|
|
const user = await buildUser();
|
|
overrides.userId = user.id;
|
|
}
|
|
|
|
if (!overrides.expiresAt) {
|
|
overrides.expiresAt = new Date();
|
|
}
|
|
|
|
const code = randomString(32);
|
|
|
|
let client;
|
|
if (overrides.oauthClientId) {
|
|
client = await OAuthClient.findByPk(overrides.oauthClientId, {
|
|
rejectOnEmpty: true,
|
|
});
|
|
} else {
|
|
client = await buildOAuthClient();
|
|
overrides.oauthClientId = client.id;
|
|
}
|
|
|
|
return OAuthAuthorizationCode.create({
|
|
authorizationCodeHash: hash(code),
|
|
scope: ["read"],
|
|
redirectUri: client.redirectUris[0],
|
|
...overrides,
|
|
});
|
|
}
|
|
|
|
export async function buildOAuthAuthentication({
|
|
oauthClientId,
|
|
user,
|
|
scope,
|
|
grantId,
|
|
}: {
|
|
oauthClientId?: string;
|
|
user: User;
|
|
scope: string[];
|
|
grantId?: string;
|
|
}) {
|
|
const oauthClient = oauthClientId
|
|
? await OAuthClient.findByPk(oauthClientId, { rejectOnEmpty: true })
|
|
: await buildOAuthClient({
|
|
teamId: user.teamId,
|
|
});
|
|
const oauthInterfaceClient = {
|
|
id: oauthClient.clientId,
|
|
grants: ["authorization_code"],
|
|
redirectUris: ["https://example.com/oauth/callback"],
|
|
};
|
|
const oauthInterfaceUser = {
|
|
id: user.id,
|
|
};
|
|
|
|
const accessToken = await OAuthInterface.generateAccessToken(
|
|
oauthInterfaceClient,
|
|
oauthInterfaceUser,
|
|
scope
|
|
);
|
|
const refreshToken = await OAuthInterface.generateRefreshToken(
|
|
oauthInterfaceClient,
|
|
oauthInterfaceUser,
|
|
scope
|
|
);
|
|
return OAuthAuthentication.create({
|
|
userId: user.id,
|
|
oauthClientId: oauthClient.id,
|
|
accessToken,
|
|
accessTokenHash: hash(accessToken),
|
|
accessTokenExpiresAt: new Date(Date.now() + 1000 * 60 * 60),
|
|
refreshToken,
|
|
refreshTokenHash: hash(refreshToken),
|
|
refreshTokenExpiresAt: new Date(Date.now() + 1000 * 60 * 60 * 24 * 30),
|
|
scope,
|
|
grantId,
|
|
});
|
|
}
|
|
|
|
export function buildProseMirrorDoc(content: DeepPartial<ProsemirrorData>[]) {
|
|
return Node.fromJSON(schema, {
|
|
type: "doc",
|
|
content,
|
|
});
|
|
}
|
|
|
|
export function buildMention(overrides: {
|
|
type?: MentionType;
|
|
modelId: string;
|
|
actorId: string;
|
|
label?: string;
|
|
id?: string;
|
|
}): DeepPartial<ProsemirrorData> {
|
|
return {
|
|
type: "mention",
|
|
attrs: {
|
|
id: overrides.id ?? randomUUID(),
|
|
type: overrides.type ?? MentionType.User,
|
|
label: overrides.label ?? faker.name.fullName(),
|
|
modelId: overrides.modelId,
|
|
actorId: overrides.actorId,
|
|
},
|
|
};
|
|
}
|
|
|
|
export function buildCommentMark(overrides: {
|
|
id?: string;
|
|
userId?: string;
|
|
draft?: boolean;
|
|
resolved?: boolean;
|
|
}) {
|
|
if (!overrides.id) {
|
|
overrides.id = randomString(10);
|
|
}
|
|
|
|
if (!overrides.userId) {
|
|
overrides.userId = randomString(10);
|
|
}
|
|
|
|
return {
|
|
type: "comment",
|
|
attrs: overrides,
|
|
};
|
|
}
|
|
|
|
export async function buildRelationship(overrides: Partial<Relationship> = {}) {
|
|
if (!overrides.userId) {
|
|
const user = await buildUser();
|
|
overrides.userId = user.id;
|
|
}
|
|
|
|
if (!overrides.documentId) {
|
|
const document = await buildDocument({
|
|
createdById: overrides.userId,
|
|
});
|
|
overrides.documentId = document.id;
|
|
}
|
|
|
|
if (!overrides.reverseDocumentId) {
|
|
const reverseDocument = await buildDocument({
|
|
createdById: overrides.userId,
|
|
});
|
|
overrides.reverseDocumentId = reverseDocument.id;
|
|
}
|
|
|
|
return Relationship.create({
|
|
type: RelationshipType.Backlink,
|
|
...overrides,
|
|
});
|
|
}
|
|
|
|
export async function buildUserPasskey(
|
|
overrides: Partial<InferCreationAttributes<UserPasskey>> = {}
|
|
) {
|
|
if (!overrides.userId) {
|
|
const user = await buildUser();
|
|
overrides.userId = user.id;
|
|
}
|
|
|
|
return UserPasskey.create({
|
|
credentialId: randomString(32),
|
|
credentialPublicKey: Buffer.from(randomString(64)),
|
|
counter: 0,
|
|
transports: ["internal"],
|
|
name: faker.word.noun(),
|
|
...overrides,
|
|
});
|
|
}
|