Files
outline/server/test/factories.ts
T
Hemachandar 04c3d81b1f chore: Setup missing oxlint configs (#9862)
* shared

* server

* app

* remove vestigial eslintrc files

* update comment directives
2025-08-06 19:54:22 -04:00

832 lines
19 KiB
TypeScript

import { faker } from "@faker-js/faker";
import isNil from "lodash/isNil";
import isNull from "lodash/isNull";
import { Node } from "prosemirror-model";
import { InferCreationAttributes } from "sequelize";
import { DeepPartial } from "utility-types";
import { v4 as uuidv4 } from "uuid";
import { randomString } from "@shared/random";
import {
CollectionPermission,
FileOperationState,
FileOperationType,
ImportState,
IntegrationService,
IntegrationType,
NotificationEventType,
ProsemirrorData,
ReactionSummary,
SubscriptionType,
UserRole,
} from "@shared/types";
import { parser, schema } from "@server/editor";
import {
Share,
Team,
User,
Event,
Document,
Star,
Collection,
Group,
GroupUser,
Attachment,
IntegrationAuthentication,
Integration,
FileOperation,
WebhookSubscription,
WebhookDelivery,
ApiKey,
Subscription,
Notification,
SearchQuery,
Pin,
Comment,
Import,
OAuthAuthorizationCode,
OAuthClient,
AuthenticationProvider,
OAuthAuthentication,
} from "@server/models";
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(),
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: uuidv4(),
},
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 buildComment(overrides: {
userId: string;
documentId: string;
parentCommentId?: string;
resolvedById?: string;
reactions?: ReactionSummary[];
}) {
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,
});
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;
}
// 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,
});
}
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 = uuidv4();
const acl = overrides.acl || "public-read";
const name = fileName || faker.system.fileName();
return Attachment.create({
id,
key: AttachmentHelper.getKey({ acl, id, name, userId: overrides.userId }),
contentType: "image/png",
size: 100,
acl,
name,
createdAt: new Date("2018-01-02T00:00:00.000Z"),
updatedAt: new Date("2018-01-02T00:00:00.000Z"),
...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) {
const user = await buildUser({
teamId: overrides.teamId,
});
overrides.createdById = user.id;
}
return OAuthClient.create({
name: faker.company.name(),
description: faker.lorem.paragraph(),
redirectUris: ["https://example.com/oauth/callback"],
published: true,
...overrides,
});
}
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,
}: {
oauthClientId?: string;
user: User;
scope: 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,
});
}
export function buildProseMirrorDoc(content: DeepPartial<ProsemirrorData>[]) {
return Node.fromJSON(schema, {
type: "doc",
content,
});
}
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,
};
}