add userSession module

This commit is contained in:
rafiarrafif
2025-05-08 15:21:05 +07:00
parent cb7757a83b
commit a2c27b313a
11 changed files with 125 additions and 1 deletions

BIN
bun.lockb

Binary file not shown.

View File

@ -14,7 +14,8 @@
"cookie": "^1.0.2",
"elysia": "latest",
"joi": "^17.13.3",
"jsonwebtoken": "^9.0.2"
"jsonwebtoken": "^9.0.2",
"ua-parser-js": "^2.0.3"
},
"devDependencies": {
"bun-types": "latest",

View File

@ -0,0 +1,26 @@
import { Context } from "elysia";
import { UAParser } from "ua-parser-js";
import { UserHeaderInformation } from "./types";
export const getUserHeaderInformation = (
ctx: Context
): UserHeaderInformation => {
const headers = ctx.request.headers;
const userAgentHeader = headers.get("user-agent") || "desktop";
const userAgent = new UAParser(userAgentHeader);
const userIP =
headers.get("cf-connecting-ip") ||
headers.get("x-real-ip") ||
headers.get("x-forwarded-for")?.split(",")[0] ||
"Unknown IP";
const userHeaderInformation = {
ip: userIP,
deviceType: userAgent.getDevice().type || "desktop",
deviceOS: userAgent.getOS().name + " " + userAgent.getOS().version,
browser: userAgent.getBrowser().name + " " + userAgent.getBrowser().version,
};
return userHeaderInformation;
};

View File

@ -0,0 +1,6 @@
export interface UserHeaderInformation {
ip: string;
deviceType: string;
deviceOS: string;
browser: string;
}

View File

@ -0,0 +1,30 @@
import { Context } from "elysia";
import { createUserSessionService } from "../services/createUserSession.service";
import { getUserHeaderInformation } from "../../../helpers/cookies/userHeader/getUserHeaderInformation";
import { handlePrismaError } from "../../../utils/databases/prisma/error/handler";
import {
returnErrorResponse,
returnWriteResponse,
} from "../../../helpers/callback/httpResponse";
export const createUserSessionRole = async (
ctx: Context & { body: { userId: string } }
) => {
const userHeaderData = getUserHeaderInformation(ctx);
try {
const newUserSession = await createUserSessionService({
userId: ctx.body.userId,
userHeaderInformation: userHeaderData,
});
return returnWriteResponse(
ctx.set,
201,
"User session created",
newUserSession
);
} catch (error) {
const { status, message, details } = handlePrismaError(error);
return returnErrorResponse(ctx.set, status, message, details);
}
};

View File

@ -0,0 +1,7 @@
import Elysia from "elysia";
import { createUserSessionRole } from "./controller/createUserSession.controller";
export const userSessionModule = new Elysia({ prefix: "/user-sessions" }).post(
"/",
createUserSessionRole
);

View File

@ -0,0 +1,20 @@
import { createUserSessionServiceParams } from "../userSession.types";
import { createUserSessionRepo } from "../userSession.repository";
export const createUserSessionService = async (
data: createUserSessionServiceParams
) => {
const sessionLifetime = Number(process.env.SESSION_EXPIRE!);
try {
const newUserSession = await createUserSessionRepo({
userId: data.userId,
isAuthenticated: true,
deviceType: data.userHeaderInformation.deviceType,
deviceOs: data.userHeaderInformation.deviceOS,
deviceIp: data.userHeaderInformation.ip,
validUntil: new Date(new Date().getTime() + sessionLifetime * 1000),
});
} catch (error) {
throw error;
}
};

View File

@ -0,0 +1,3 @@
import { prisma } from "../../utils/databases/prisma/connection";
export const userSessionModel = prisma.userSession;

View File

@ -0,0 +1,25 @@
import { Prisma } from "@prisma/client";
import { userSessionModel } from "./userSession.model";
export const createUserSessionRepo = async (
data: Prisma.UserSessionUncheckedCreateInput
) => {
try {
const newUserRole = await userSessionModel.create({
data: data,
include: {
user: {
omit: {
password: true,
},
include: {
roles: true,
},
},
},
});
return newUserRole;
} catch (error) {
throw error;
}
};

View File

@ -0,0 +1,6 @@
import { UserHeaderInformation } from "../../helpers/cookies/userHeader/getUserHeaderInformation/types";
export interface createUserSessionServiceParams {
userId: string;
userHeaderInformation: UserHeaderInformation;
}