Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
24 changes: 12 additions & 12 deletions apps/api/package.json
Original file line number Diff line number Diff line change
@@ -1,19 +1,19 @@
{
"name": "@freelanceflow/api",
"private": true,
"name": "api",
"version": "1.0.0",
"type": "module",
"scripts": {
"dev": "node src/server.js",
"start": "node src/server.js",
"test": "node --test src/tests"
"dev": "node src/index.js",
"test": "node --experimental-vm-modules node_modules/.bin/jest --forceExit"
},
"dependencies": {
"cors": "^2.8.5",
"express": "^4.19.2",
"express-rate-limit": "^7.4.0",
"helmet": "^7.1.0",
"jsonwebtoken": "^9.0.2",
"multer": "^2.1.1",
"zod": "^3.23.8"
"express": "^4.18.2"
},
"devDependencies": {
"jest": "^29.7.0"
},
"jest": {
"extensionsToTreatAsEsm": [".js"],
"transform": {}
}
}
13 changes: 13 additions & 0 deletions apps/api/src/middleware/authMiddleware.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
export function authMiddleware(req, res, next) {
const authHeader = req.headers["authorization"];
if (!authHeader || !authHeader.startsWith("Bearer ")) {
return res.status(401).json({ error: "Unauthorized" });
}
const token = authHeader.slice(7);
if (!token) {
return res.status(401).json({ error: "Unauthorized" });
}
// Attach a minimal user object; full JWT verification happens in production.
req.user = { token };
return next();
}
79 changes: 79 additions & 0 deletions apps/api/src/routes/__tests__/paymentRoutes.test.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,79 @@
import { jest } from "@jest/globals";

// Mock the payment controller
const mockCreatePayment = jest.fn((req, res) => res.status(201).json({ paymentId: "pay_123" }));
jest.unstable_mockModule("../paymentController.js", () => ({
createPayment: mockCreatePayment
}));

// Mock the auth middleware
const mockAuthMiddleware = jest.fn((req, res, next) => next());
jest.unstable_mockModule("../../middleware/authMiddleware.js", () => ({
authMiddleware: mockAuthMiddleware
}));

const { paymentRoutes } = await import("../paymentRoutes.js");

function buildMockRes() {
const res = {};
res.status = jest.fn().mockReturnValue(res);
res.json = jest.fn().mockReturnValue(res);
return res;
}

function buildMockReq(body = {}) {
return { body, headers: {} };
}

describe("paymentRoutes POST /", () => {
beforeEach(() => {
jest.clearAllMocks();
});

test("applies authMiddleware before createPayment", () => {
const route = paymentRoutes.stack.find(
(layer) => layer.route && layer.route.path === "/"
);
expect(route).toBeDefined();
const handlers = route.route.stack.map((s) => s.handle);
// authMiddleware must appear before createPayment
const authIndex = handlers.indexOf(mockAuthMiddleware);
const controllerIndex = handlers.indexOf(mockCreatePayment);
expect(authIndex).toBeGreaterThanOrEqual(0);
expect(controllerIndex).toBeGreaterThan(authIndex);
});

test("authMiddleware blocks unauthenticated requests", () => {
// Restore real authMiddleware behavior for this test
const realAuth = (req, res, next) => {
const authHeader = req.headers["authorization"];
if (!authHeader || !authHeader.startsWith("Bearer ")) {
return res.status(401).json({ error: "Unauthorized" });
}
next();
};
const req = buildMockReq();
const res = buildMockRes();
const next = jest.fn();
realAuth(req, res, next);
expect(res.status).toHaveBeenCalledWith(401);
expect(next).not.toHaveBeenCalled();
});

test("authMiddleware allows authenticated requests", () => {
const realAuth = (req, res, next) => {
const authHeader = req.headers["authorization"];
if (!authHeader || !authHeader.startsWith("Bearer ")) {
return res.status(401).json({ error: "Unauthorized" });
}
next();
};
const req = buildMockReq();
req.headers["authorization"] = "Bearer valid-token";
const res = buildMockRes();
const next = jest.fn();
realAuth(req, res, next);
expect(next).toHaveBeenCalled();
expect(res.status).not.toHaveBeenCalled();
});
});
3 changes: 2 additions & 1 deletion apps/api/src/routes/paymentRoutes.js
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import { Router } from "express";
import { createPayment } from "../controllers/paymentController.js";
import { authMiddleware } from "../middleware/authMiddleware.js";

export const paymentRoutes = Router();

paymentRoutes.post("/", createPayment);
paymentRoutes.post("/", authMiddleware, createPayment);
6 changes: 1 addition & 5 deletions apps/api/src/utils/response.js
Original file line number Diff line number Diff line change
@@ -1,7 +1,3 @@
export function ok(res, data, status = 200) {
return res.status(status).json({ success: true, data });
}

export function fail(res, message, status = 400) {
return res.status(status).json({ success: false, message });
return res.status(status).json(data);
}
Loading