mirror of
https://github.com/Drop-OSS/drop.git
synced 2026-06-22 04:11:32 +10:00
refactor: session handler
This commit is contained in:
+1
-1
@@ -45,7 +45,7 @@ export default defineNuxtConfig({
|
|||||||
},
|
},
|
||||||
|
|
||||||
scheduledTasks: {
|
scheduledTasks: {
|
||||||
"0 * * * *": ["cleanup:invitations"],
|
"0 * * * *": ["cleanup:invitations", "cleanup:sessions"],
|
||||||
},
|
},
|
||||||
|
|
||||||
compressPublicAssets: true,
|
compressPublicAssets: true,
|
||||||
|
|||||||
@@ -53,6 +53,11 @@ model Certificate {
|
|||||||
}
|
}
|
||||||
|
|
||||||
model Session {
|
model Session {
|
||||||
token String @id
|
token String @id
|
||||||
data Json
|
expiresAt DateTime
|
||||||
|
|
||||||
|
userId String
|
||||||
|
user User? @relation(fields: [userId], references: [id])
|
||||||
|
|
||||||
|
data Json // misc extra data
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -14,7 +14,8 @@ model User {
|
|||||||
collections Collection[]
|
collections Collection[]
|
||||||
articles Article[]
|
articles Article[]
|
||||||
|
|
||||||
tokens APIToken[]
|
tokens APIToken[]
|
||||||
|
sessions Session[]
|
||||||
}
|
}
|
||||||
|
|
||||||
model Notification {
|
model Notification {
|
||||||
|
|||||||
@@ -8,24 +8,30 @@ import {
|
|||||||
} from "~/server/internal/security/simple";
|
} from "~/server/internal/security/simple";
|
||||||
import sessionHandler from "~/server/internal/session";
|
import sessionHandler from "~/server/internal/session";
|
||||||
|
|
||||||
export default defineEventHandler(async (h3) => {
|
const signinValidator = type({
|
||||||
const body = await readBody(h3);
|
username: "string",
|
||||||
|
password: "string",
|
||||||
|
"rememberMe?": "boolean | undefined",
|
||||||
|
});
|
||||||
|
|
||||||
|
export default defineEventHandler(async (h3) => {
|
||||||
|
const body = signinValidator(await readBody(h3));
|
||||||
|
if (body instanceof type.errors) {
|
||||||
|
// hover out.summary to see validation errors
|
||||||
|
console.error(body.summary);
|
||||||
|
|
||||||
const username = body.username;
|
|
||||||
const password = body.password;
|
|
||||||
const rememberMe = body.rememberMe ?? false;
|
|
||||||
if (username === undefined || password === undefined)
|
|
||||||
throw createError({
|
throw createError({
|
||||||
statusCode: 403,
|
statusCode: 400,
|
||||||
statusMessage: "Username or password missing from request.",
|
statusMessage: body.summary,
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
|
||||||
const authMek = await prisma.linkedAuthMec.findFirst({
|
const authMek = await prisma.linkedAuthMec.findFirst({
|
||||||
where: {
|
where: {
|
||||||
mec: AuthMec.Simple,
|
mec: AuthMec.Simple,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
user: {
|
user: {
|
||||||
username,
|
username: body.username,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
include: {
|
include: {
|
||||||
@@ -62,14 +68,14 @@ export default defineEventHandler(async (h3) => {
|
|||||||
"Invalid password state. Please contact the server administrator.",
|
"Invalid password state. Please contact the server administrator.",
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!(await checkHashBcrypt(password, hash)))
|
if (!(await checkHashBcrypt(body.password, hash)))
|
||||||
throw createError({
|
throw createError({
|
||||||
statusCode: 401,
|
statusCode: 401,
|
||||||
statusMessage: "Invalid username or password.",
|
statusMessage: "Invalid username or password.",
|
||||||
});
|
});
|
||||||
|
|
||||||
// TODO: send user to forgot password screen or something to force them to change their password to new system
|
// TODO: send user to forgot password screen or something to force them to change their password to new system
|
||||||
await sessionHandler.setUserId(h3, authMek.userId, rememberMe);
|
await sessionHandler.signin(h3, authMek.userId, body.rememberMe);
|
||||||
return { result: true, userId: authMek.userId };
|
return { result: true, userId: authMek.userId };
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -82,13 +88,12 @@ export default defineEventHandler(async (h3) => {
|
|||||||
"Invalid password state. Please contact the server administrator.",
|
"Invalid password state. Please contact the server administrator.",
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!(await checkHashArgon2(password, hash)))
|
if (!(await checkHashArgon2(body.password, hash)))
|
||||||
throw createError({
|
throw createError({
|
||||||
statusCode: 401,
|
statusCode: 401,
|
||||||
statusMessage: "Invalid username or password.",
|
statusMessage: "Invalid username or password.",
|
||||||
});
|
});
|
||||||
|
|
||||||
await sessionHandler.setUserId(h3, authMek.userId, rememberMe);
|
await sessionHandler.signin(h3, authMek.userId, body.rememberMe);
|
||||||
|
|
||||||
return { result: true, userId: authMek.userId };
|
return { result: true, userId: authMek.userId };
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -2,8 +2,8 @@ import clientHandler from "~/server/internal/clients/handler";
|
|||||||
import sessionHandler from "~/server/internal/session";
|
import sessionHandler from "~/server/internal/session";
|
||||||
|
|
||||||
export default defineEventHandler(async (h3) => {
|
export default defineEventHandler(async (h3) => {
|
||||||
const userId = await sessionHandler.getUserId(h3);
|
const user = await sessionHandler.getSession(h3);
|
||||||
if (!userId) throw createError({ statusCode: 403 });
|
if (!user) throw createError({ statusCode: 403 });
|
||||||
|
|
||||||
const query = getQuery(h3);
|
const query = getQuery(h3);
|
||||||
const providedClientId = query.id?.toString();
|
const providedClientId = query.id?.toString();
|
||||||
@@ -13,16 +13,14 @@ export default defineEventHandler(async (h3) => {
|
|||||||
statusMessage: "Provide client ID in request params as 'id'",
|
statusMessage: "Provide client ID in request params as 'id'",
|
||||||
});
|
});
|
||||||
|
|
||||||
const data = await clientHandler.fetchClientMetadata(
|
const data = await clientHandler.fetchClientMetadata(providedClientId);
|
||||||
providedClientId
|
|
||||||
);
|
|
||||||
if (!data)
|
if (!data)
|
||||||
throw createError({
|
throw createError({
|
||||||
statusCode: 404,
|
statusCode: 404,
|
||||||
statusMessage: "Request not found.",
|
statusMessage: "Request not found.",
|
||||||
});
|
});
|
||||||
|
|
||||||
await clientHandler.attachUserId(providedClientId, userId);
|
await clientHandler.attachUserId(providedClientId, user.userId);
|
||||||
|
|
||||||
return data;
|
return data;
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -2,8 +2,8 @@ import clientHandler from "~/server/internal/clients/handler";
|
|||||||
import sessionHandler from "~/server/internal/session";
|
import sessionHandler from "~/server/internal/session";
|
||||||
|
|
||||||
export default defineEventHandler(async (h3) => {
|
export default defineEventHandler(async (h3) => {
|
||||||
const userId = await sessionHandler.getUserId(h3);
|
const user = await sessionHandler.getSession(h3);
|
||||||
if (!userId) throw createError({ statusCode: 403 });
|
if (!user) throw createError({ statusCode: 403 });
|
||||||
|
|
||||||
const body = await readBody(h3);
|
const body = await readBody(h3);
|
||||||
const clientId = await body.id;
|
const clientId = await body.id;
|
||||||
|
|||||||
@@ -81,8 +81,8 @@ class ACLManager {
|
|||||||
if (!request)
|
if (!request)
|
||||||
throw new Error("Native web requests not available - weird deployment?");
|
throw new Error("Native web requests not available - weird deployment?");
|
||||||
// Sessions automatically have all ACLs
|
// Sessions automatically have all ACLs
|
||||||
const userId = await sessionHandler.getUserId(request);
|
const user = await sessionHandler.getSession(request);
|
||||||
if (userId) return userId;
|
if (user) return user.userId;
|
||||||
|
|
||||||
const authorizationToken = this.getAuthorizationToken(request);
|
const authorizationToken = this.getAuthorizationToken(request);
|
||||||
if (!authorizationToken) return undefined;
|
if (!authorizationToken) return undefined;
|
||||||
@@ -116,9 +116,11 @@ class ACLManager {
|
|||||||
) {
|
) {
|
||||||
if (!request)
|
if (!request)
|
||||||
throw new Error("Native web requests not available - weird deployment?");
|
throw new Error("Native web requests not available - weird deployment?");
|
||||||
const userId = await sessionHandler.getUserId(request);
|
const userSession = await sessionHandler.getSession(request);
|
||||||
if (userId) {
|
if (userSession) {
|
||||||
const user = await prisma.user.findUnique({ where: { id: userId } });
|
const user = await prisma.user.findUnique({
|
||||||
|
where: { id: userSession.userId },
|
||||||
|
});
|
||||||
if (!user) return false;
|
if (!user) return false;
|
||||||
if (user.admin) return true;
|
if (user.admin) return true;
|
||||||
return false;
|
return false;
|
||||||
|
|||||||
@@ -10,8 +10,8 @@ export default function createDBSessionHandler(): SessionProvider {
|
|||||||
});
|
});
|
||||||
|
|
||||||
return {
|
return {
|
||||||
async setSession(token, data) {
|
async setSession(token, session) {
|
||||||
cache.set(token, data);
|
cache.set(token, session);
|
||||||
|
|
||||||
// const strData = JSON.stringify(data);
|
// const strData = JSON.stringify(data);
|
||||||
await prisma.session.upsert({
|
await prisma.session.upsert({
|
||||||
@@ -20,54 +20,16 @@ export default function createDBSessionHandler(): SessionProvider {
|
|||||||
},
|
},
|
||||||
create: {
|
create: {
|
||||||
token,
|
token,
|
||||||
data,
|
...session,
|
||||||
},
|
|
||||||
update: {
|
|
||||||
data,
|
|
||||||
},
|
},
|
||||||
|
update: session,
|
||||||
});
|
});
|
||||||
return true;
|
return true;
|
||||||
},
|
},
|
||||||
async updateSession(token, key, data) {
|
async updateSession(token, data) {
|
||||||
const newObj: { [key: string]: any } = {};
|
return await this.setSession(token, data);
|
||||||
newObj[key] = data;
|
|
||||||
cache.set(token, newObj);
|
|
||||||
|
|
||||||
const session = await prisma.session.upsert({
|
|
||||||
where: {
|
|
||||||
token,
|
|
||||||
},
|
|
||||||
create: {
|
|
||||||
token,
|
|
||||||
data: newObj,
|
|
||||||
},
|
|
||||||
update: {},
|
|
||||||
});
|
|
||||||
|
|
||||||
// if json object and not arrary, update session
|
|
||||||
if (
|
|
||||||
typeof session.data === "object" &&
|
|
||||||
!Array.isArray(session.data) &&
|
|
||||||
session.data !== null
|
|
||||||
) {
|
|
||||||
// means we set it above
|
|
||||||
if (session.data[key] === data) return true;
|
|
||||||
|
|
||||||
// else we need to set it ourselves
|
|
||||||
(session.data as Prisma.JsonObject)[key] = data;
|
|
||||||
await prisma.session.update({
|
|
||||||
where: {
|
|
||||||
token,
|
|
||||||
},
|
|
||||||
data: {
|
|
||||||
data: session.data,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
},
|
},
|
||||||
async getSession<T>(token: string) {
|
async getSession<T extends Session>(token: string) {
|
||||||
const cached = cache.get(token);
|
const cached = cache.get(token);
|
||||||
if (cached !== undefined) return cached as T;
|
if (cached !== undefined) return cached as T;
|
||||||
|
|
||||||
@@ -77,15 +39,31 @@ export default function createDBSessionHandler(): SessionProvider {
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
if (result === null) return undefined;
|
if (result === null) return undefined;
|
||||||
return result.data as T;
|
|
||||||
|
// i hate casting
|
||||||
|
// need to cast to unknown since result.data can be an N deep json object technically
|
||||||
|
// ts doesn't like that be cast down to the more constraining session type
|
||||||
|
return result as unknown as T;
|
||||||
},
|
},
|
||||||
async clearSession(token) {
|
async removeSession(token) {
|
||||||
cache.delete(token);
|
cache.delete(token);
|
||||||
await prisma.session.delete({
|
await prisma.session.delete({
|
||||||
where: {
|
where: {
|
||||||
token,
|
token,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
return true;
|
||||||
|
},
|
||||||
|
async cleanupSessions() {
|
||||||
|
const now = new Date();
|
||||||
|
|
||||||
|
await prisma.session.deleteMany({
|
||||||
|
where: {
|
||||||
|
expiresAt: {
|
||||||
|
lt: now,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,8 +1,7 @@
|
|||||||
import { H3Event, Session } from "h3";
|
import { H3Event } from "h3";
|
||||||
import createMemorySessionProvider from "./memory";
|
import createMemorySessionProvider from "./memory";
|
||||||
import { SessionProvider } from "./types";
|
import { Session, SessionProvider } from "./types";
|
||||||
import prisma from "../db/database";
|
import { randomUUID } from "node:crypto";
|
||||||
import { v4 as uuidv4 } from "uuid";
|
|
||||||
import moment from "moment";
|
import moment from "moment";
|
||||||
import { parse as parseCookies } from "cookie-es";
|
import { parse as parseCookies } from "cookie-es";
|
||||||
import { MinimumRequestObject } from "~/server/h3";
|
import { MinimumRequestObject } from "~/server/h3";
|
||||||
@@ -14,9 +13,7 @@ This implementation may need work.
|
|||||||
It exposes an API that should stay static, but there are plenty of opportunities for optimisation/organisation under the hood
|
It exposes an API that should stay static, but there are plenty of opportunities for optimisation/organisation under the hood
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const userSessionKey = "_userSession";
|
const dropTokenCookieName = "drop-token";
|
||||||
const userIdKey = "_userId";
|
|
||||||
const dropTokenCookie = "drop-token";
|
|
||||||
const normalSessionLength = [31, "days"];
|
const normalSessionLength = [31, "days"];
|
||||||
const extendedSessionLength = [1, "year"];
|
const extendedSessionLength = [1, "year"];
|
||||||
|
|
||||||
@@ -29,86 +26,94 @@ export class SessionHandler {
|
|||||||
// this.sessionProvider = createMemorySessionProvider();
|
// this.sessionProvider = createMemorySessionProvider();
|
||||||
}
|
}
|
||||||
|
|
||||||
private getSessionToken(request: MinimumRequestObject | undefined) {
|
async signin(h3: H3Event, userId: string, rememberMe: boolean = false) {
|
||||||
|
const expiresAt = this.createExipreAt(rememberMe);
|
||||||
|
const token = this.createSessionCookie(h3, expiresAt);
|
||||||
|
return await this.sessionProvider.setSession(token, {
|
||||||
|
userId,
|
||||||
|
expiresAt,
|
||||||
|
data: {},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get a session associated with a request
|
||||||
|
* @returns session
|
||||||
|
*/
|
||||||
|
async getSession<T extends Session>(request: MinimumRequestObject) {
|
||||||
|
const token = this.getSessionToken(request);
|
||||||
|
if (!token) return undefined;
|
||||||
|
// TODO: should validate if session is expired or not here, not in application code
|
||||||
|
|
||||||
|
const data = await this.sessionProvider.getSession<T>(token);
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Signout session associated with request and deauthenticates it
|
||||||
|
* @param request
|
||||||
|
* @returns
|
||||||
|
*/
|
||||||
|
async signout(h3: H3Event) {
|
||||||
|
const token = this.getSessionToken(h3);
|
||||||
|
if (!token) return false;
|
||||||
|
const res = await this.sessionProvider.removeSession(token);
|
||||||
|
if (!res) return false;
|
||||||
|
deleteCookie(h3, dropTokenCookieName);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
async cleanupSessions() {
|
||||||
|
await this.sessionProvider.cleanupSessions();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Update session info
|
||||||
|
* @param token session token
|
||||||
|
* @param data new session data
|
||||||
|
* @returns success or not
|
||||||
|
*/
|
||||||
|
private async updateSession(token: string, data: Session) {
|
||||||
|
return await this.sessionProvider.updateSession(token, data);
|
||||||
|
}
|
||||||
|
|
||||||
|
// ---------------------- Private API Below ------------------------
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get session token on a request
|
||||||
|
* @param request
|
||||||
|
* @returns session token
|
||||||
|
*/
|
||||||
|
private getSessionToken(
|
||||||
|
request: MinimumRequestObject | undefined
|
||||||
|
): string | undefined {
|
||||||
if (!request) throw new Error("Native web request not available");
|
if (!request) throw new Error("Native web request not available");
|
||||||
const cookieHeader = request.headers.get("Cookie");
|
const cookieHeader = request.headers.get("Cookie");
|
||||||
if (!cookieHeader) return undefined;
|
if (!cookieHeader) return undefined;
|
||||||
const cookies = parseCookies(cookieHeader);
|
const cookies = parseCookies(cookieHeader);
|
||||||
const cookie = cookies[dropTokenCookie];
|
const cookie = cookies[dropTokenCookieName];
|
||||||
return cookie;
|
return cookie;
|
||||||
}
|
}
|
||||||
|
|
||||||
private async createSession(h3: H3Event, extend = false) {
|
private createExipreAt(rememberMe: boolean) {
|
||||||
const token = uuidv4();
|
return moment()
|
||||||
const expiry = moment().add(
|
.add(...(rememberMe ? extendedSessionLength : normalSessionLength))
|
||||||
...(extend ? extendedSessionLength : normalSessionLength)
|
.toDate();
|
||||||
);
|
}
|
||||||
|
|
||||||
setCookie(h3, dropTokenCookie, token, { expires: expiry.toDate() });
|
|
||||||
|
|
||||||
this.sessionProvider.setSession(dropTokenCookie, {});
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates cookie that represents user session
|
||||||
|
* @param h3
|
||||||
|
* @param extend
|
||||||
|
* @returns
|
||||||
|
*/
|
||||||
|
private createSessionCookie(h3: H3Event, expiresAt: Date) {
|
||||||
|
const token = randomUUID();
|
||||||
|
// TODO: we should probably switch to jwts to minimize possibility of someone
|
||||||
|
// trying to guess a session id (jwts let us sign + encrypt stuff in a std way)
|
||||||
|
setCookie(h3, dropTokenCookieName, token, { expires: expiresAt });
|
||||||
return token;
|
return token;
|
||||||
}
|
}
|
||||||
|
|
||||||
getDropTokenCookie() {
|
|
||||||
return dropTokenCookie;
|
|
||||||
}
|
|
||||||
|
|
||||||
async getSession<T extends Session>(request: MinimumRequestObject) {
|
|
||||||
const token = this.getSessionToken(request);
|
|
||||||
if (!token) return undefined;
|
|
||||||
const data = await this.sessionProvider.getSession<{ [userSessionKey]: T }>(
|
|
||||||
token
|
|
||||||
);
|
|
||||||
if (!data) return undefined;
|
|
||||||
|
|
||||||
return data[userSessionKey];
|
|
||||||
}
|
|
||||||
async setSession(h3: H3Event, data: any, extend = false) {
|
|
||||||
const token =
|
|
||||||
this.getSessionToken(h3) ?? (await this.createSession(h3, extend));
|
|
||||||
const result = await this.sessionProvider.updateSession(
|
|
||||||
token,
|
|
||||||
userSessionKey,
|
|
||||||
data
|
|
||||||
);
|
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
async clearSession(request: MinimumRequestObject) {
|
|
||||||
const token = this.getSessionToken(request);
|
|
||||||
if (!token) return false;
|
|
||||||
await this.sessionProvider.clearSession(token);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
async getUserId(h3: MinimumRequestObject) {
|
|
||||||
const token = this.getSessionToken(h3);
|
|
||||||
if (!token) return undefined;
|
|
||||||
|
|
||||||
return await this.getUserIdRaw(token);
|
|
||||||
}
|
|
||||||
async getUserIdRaw(token: string) {
|
|
||||||
const session = await this.sessionProvider.getSession<{
|
|
||||||
[userIdKey]: string | undefined;
|
|
||||||
}>(token);
|
|
||||||
|
|
||||||
if (!session) return undefined;
|
|
||||||
|
|
||||||
return session[userIdKey];
|
|
||||||
}
|
|
||||||
|
|
||||||
async setUserId(h3: H3Event, userId: string, extend = false) {
|
|
||||||
const token =
|
|
||||||
this.getSessionToken(h3) ?? (await this.createSession(h3, extend));
|
|
||||||
|
|
||||||
const result = await this.sessionProvider.updateSession(
|
|
||||||
token,
|
|
||||||
userIdKey,
|
|
||||||
userId
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export const sessionHandler = new SessionHandler();
|
export const sessionHandler = new SessionHandler();
|
||||||
|
|||||||
@@ -8,15 +8,23 @@ export default function createMemorySessionHandler() {
|
|||||||
sessions[token] = data;
|
sessions[token] = data;
|
||||||
return true;
|
return true;
|
||||||
},
|
},
|
||||||
async updateSession(token, key, data) {
|
async getSession<T extends Session>(token: string): Promise<T | undefined> {
|
||||||
sessions[token] = Object.assign({}, sessions[token], { [key]: data });
|
const session = sessions[token];
|
||||||
|
return session ? (session as T) : undefined; // Ensure undefined is returned if session is not found
|
||||||
|
},
|
||||||
|
async updateSession(token, data) {
|
||||||
|
return this.setSession(token, data);
|
||||||
|
},
|
||||||
|
async removeSession(token) {
|
||||||
|
delete sessions[token];
|
||||||
return true;
|
return true;
|
||||||
},
|
},
|
||||||
async getSession(token) {
|
async cleanupSessions() {
|
||||||
return sessions[token] as any; // Wild type cast because we let the user specify types if they want
|
const now = new Date();
|
||||||
},
|
for (let token in sessions) {
|
||||||
async clearSession(token) {
|
// if expires at time is before now, the session is expired
|
||||||
delete sessions[token];
|
if (sessions[token].expiresAt < now) await this.removeSession(token);
|
||||||
|
}
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
Vendored
+11
-4
@@ -1,10 +1,17 @@
|
|||||||
import { H3Event } from "h3";
|
import { H3Event } from "h3";
|
||||||
|
|
||||||
export type Session = { [key: string]: any };
|
export type Session = {
|
||||||
|
userId: string;
|
||||||
|
expiresAt: Date;
|
||||||
|
data: {
|
||||||
|
[key: string]: any;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
export interface SessionProvider {
|
export interface SessionProvider {
|
||||||
setSession: (token: string, data: Session) => Promise<boolean>;
|
|
||||||
updateSession: (token: string, key: string, data: any) => Promise<boolean>;
|
|
||||||
getSession: <T extends Session>(token: string) => Promise<T | undefined>;
|
getSession: <T extends Session>(token: string) => Promise<T | undefined>;
|
||||||
clearSession: (token: string) => Promise<void>;
|
setSession: (token: string, data: Session) => Promise<boolean>;
|
||||||
|
updateSession: (token: string, data: Session) => Promise<boolean>;
|
||||||
|
removeSession: (token: string) => Promise<boolean>;
|
||||||
|
cleanupSessions: () => Promise<void>;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -14,8 +14,8 @@ export default defineNitroPlugin((nitro) => {
|
|||||||
switch (error.statusCode) {
|
switch (error.statusCode) {
|
||||||
case 401:
|
case 401:
|
||||||
case 403:
|
case 403:
|
||||||
const userId = await sessionHandler.getUserId(event);
|
const user = await sessionHandler.getSession(event);
|
||||||
if (userId) break;
|
if (user) break;
|
||||||
return sendRedirect(
|
return sendRedirect(
|
||||||
event,
|
event,
|
||||||
`/auth/signin?redirect=${encodeURIComponent(event.path)}`
|
`/auth/signin?redirect=${encodeURIComponent(event.path)}`
|
||||||
|
|||||||
@@ -0,0 +1,12 @@
|
|||||||
|
import sessionHandler from "~/server/internal/session";
|
||||||
|
|
||||||
|
export default defineTask({
|
||||||
|
meta: {
|
||||||
|
name: "cleanup:invitations",
|
||||||
|
},
|
||||||
|
async run({}) {
|
||||||
|
await sessionHandler.cleanupSessions();
|
||||||
|
|
||||||
|
return { result: true };
|
||||||
|
},
|
||||||
|
});
|
||||||
Reference in New Issue
Block a user