mirror of
https://github.com/Sendouc/sendou.ink.git
synced 2026-03-31 14:54:52 -05:00
171 lines
4.2 KiB
TypeScript
171 lines
4.2 KiB
TypeScript
import type { ActionFunctionArgs, LoaderFunctionArgs } from "@remix-run/node";
|
|
import type { Params } from "@remix-run/react";
|
|
import { expect } from "vitest";
|
|
import type { z } from "zod";
|
|
import { REGULAR_USER_TEST_ID } from "~/db/seed/constants";
|
|
import { db, sql } from "~/db/sql";
|
|
import { ADMIN_ID } from "~/features/admin/admin-constants";
|
|
import { SESSION_KEY } from "~/features/auth/core/authenticator.server";
|
|
import { authSessionStorage } from "~/features/auth/core/session.server";
|
|
|
|
export function arrayContainsSameItems<T>(arr1: T[], arr2: T[]) {
|
|
return (
|
|
arr1.length === arr2.length && arr1.every((item) => arr2.includes(item))
|
|
);
|
|
}
|
|
|
|
export function wrappedAction<T extends z.ZodTypeAny>({
|
|
action,
|
|
}: {
|
|
// TODO: strongly type this
|
|
action: (args: ActionFunctionArgs) => any;
|
|
}) {
|
|
return async (
|
|
args: z.infer<T>,
|
|
{
|
|
user,
|
|
params = {},
|
|
}: { user?: "admin" | "regular"; params?: Params<string> } = {},
|
|
) => {
|
|
const body = new URLSearchParams(args);
|
|
const request = new Request("http://app.com/path", {
|
|
method: "POST",
|
|
body,
|
|
headers: await authHeader(user),
|
|
});
|
|
|
|
try {
|
|
const response = await action({
|
|
request,
|
|
context: {},
|
|
params,
|
|
});
|
|
|
|
return response;
|
|
} catch (thrown) {
|
|
if (thrown instanceof Response) {
|
|
// it was a redirect
|
|
if (thrown.status === 302) return thrown;
|
|
|
|
throw new Error(`Response thrown with status code: ${thrown.status}`);
|
|
}
|
|
|
|
throw thrown;
|
|
}
|
|
};
|
|
}
|
|
|
|
export function wrappedLoader<T>({
|
|
loader,
|
|
}: {
|
|
// TODO: strongly type this
|
|
loader: (args: LoaderFunctionArgs) => any;
|
|
}) {
|
|
return async ({
|
|
user,
|
|
params = {},
|
|
}: { user?: "admin" | "regular"; params?: Params<string> } = {}) => {
|
|
const request = new Request("http://app.com/path", {
|
|
method: "GET",
|
|
headers: await authHeader(user),
|
|
});
|
|
|
|
try {
|
|
const data = await loader({
|
|
request,
|
|
params,
|
|
context: {},
|
|
});
|
|
|
|
return data as T;
|
|
} catch (thrown) {
|
|
if (thrown instanceof Response) {
|
|
throw new Error(`Response thrown with status code: ${thrown.status}`);
|
|
}
|
|
|
|
throw thrown;
|
|
}
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Asserts that the given response errored out (with a toast message, via `errorToastIfFalsy(cond)` call)
|
|
*
|
|
* @param response - The HTTP response object to check.
|
|
* @param message - Optional. The expected error toast message shown to the user.
|
|
*/
|
|
export function assertResponseErrored(response: Response, message?: string) {
|
|
expect(response.headers.get("Location")).toContain("?__error=");
|
|
if (message) {
|
|
expect(response.headers.get("Location")).toContain(message);
|
|
}
|
|
}
|
|
|
|
async function authHeader(user?: "admin" | "regular"): Promise<HeadersInit> {
|
|
if (!user) return [];
|
|
|
|
const session = await authSessionStorage.getSession();
|
|
|
|
session.set(SESSION_KEY, user === "admin" ? ADMIN_ID : REGULAR_USER_TEST_ID);
|
|
|
|
return [["Cookie", await authSessionStorage.commitSession(session)]];
|
|
}
|
|
|
|
/**
|
|
* Resets all data in the database by deleting all rows from every table,
|
|
* except for SQLite system tables and the 'migrations' table.
|
|
*
|
|
* @example
|
|
* describe("My integration test", () => {
|
|
* beforeEach(async () => {
|
|
* await dbInsertUsers(2);
|
|
* });
|
|
*
|
|
* afterEach(() => {
|
|
* dbReset();
|
|
* });
|
|
*
|
|
* // tests go here
|
|
* });
|
|
*/
|
|
export const dbReset = () => {
|
|
const tables = sql
|
|
.prepare(
|
|
"SELECT name FROM sqlite_master WHERE type='table' AND name NOT LIKE 'sqlite_%' AND name NOT LIKE 'migrations';",
|
|
)
|
|
.all() as { name: string }[];
|
|
|
|
sql.prepare("PRAGMA foreign_keys = OFF").run();
|
|
for (const table of tables) {
|
|
sql.prepare(`DELETE FROM "${table.name}"`).run();
|
|
}
|
|
sql.prepare("PRAGMA foreign_keys = ON").run();
|
|
};
|
|
|
|
/**
|
|
* Inserts a specified number of user records into the "User" table in the database for integration testing.
|
|
* 1) id: 1, discordName: "user1", discordId: "0"
|
|
* 2) id: 2, discordName: "user2", discordId: "1"
|
|
* 3) etc.
|
|
*
|
|
* @param count - The number of users to insert. Defaults to 2 if not provided.
|
|
*
|
|
* @example
|
|
* // Inserts 5 users into the database
|
|
* await dbInsertUsers(5);
|
|
*
|
|
* // Inserts 2 users (default)
|
|
* await dbInsertUsers();
|
|
*/
|
|
export const dbInsertUsers = (count = 2) =>
|
|
db
|
|
.insertInto("User")
|
|
.values(
|
|
Array.from({ length: count }).map((_, i) => ({
|
|
id: i + 1,
|
|
discordName: `user${i + 1}`,
|
|
discordId: String(i),
|
|
})),
|
|
)
|
|
.execute();
|