Adds ConsentFactoryProtocol / ConsentGrantMeta / ConsentProtocol to core-shared/di/bind-protocols so feature binders can wire per-user consent without a hard dep on the optional @repo/core-consent package. BindContext gains an optional consentFactory? field following the same pattern as bus?, auditLog?, etc. signUpUseCase gains a 4th optional dep (consentFactory). When present and the input includes a cookieHeader containing cc_consent=<categories>, the use case calls consent.grant for each category with method:"signup-migration" and returns a clearCookie payload (Max-Age:0) so the anonymous cookie is cleared on the HTTP response. Tests use RecordingConsent from @repo/core-testing to assert migration call shape and cookie-clear; no-cookie and no-factory branches are also covered. All coverage bands hold at 100% for use-cases. Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
72 lines
2.3 KiB
TypeScript
72 lines
2.3 KiB
TypeScript
import { describe, it, expect } from "vitest";
|
|
import { RecordingEventBus } from "@repo/core-testing/instrumentation";
|
|
import { signUpController } from "@/interface-adapters/controllers/sign-up.controller";
|
|
import { MockUsersRepository } from "@/infrastructure/repositories/users.repository.mock";
|
|
import { MockAuthenticationService } from "@/infrastructure/services/authentication.service.mock";
|
|
import { signUpUseCase } from "@/application/use-cases/sign-up.use-case";
|
|
import { InputParseError } from "@/entities/errors/common";
|
|
import { userFactory } from "@/__factories__/user.factory";
|
|
|
|
describe("signUpController", () => {
|
|
it("returns a cookie on successful sign-up", async () => {
|
|
const users = new MockUsersRepository([]);
|
|
const auth = new MockAuthenticationService(users);
|
|
const useCase = signUpUseCase(
|
|
users,
|
|
auth,
|
|
new RecordingEventBus(),
|
|
undefined,
|
|
);
|
|
const controller = signUpController(useCase);
|
|
|
|
const result = await controller({
|
|
username: "carol",
|
|
password: "secret_password",
|
|
confirmPassword: "secret_password",
|
|
});
|
|
expect(result.name).toBe("session");
|
|
expect(result.value).toBeTruthy();
|
|
});
|
|
|
|
it("throws InputParseError when passwords do not match", async () => {
|
|
const users = new MockUsersRepository([]);
|
|
const auth = new MockAuthenticationService(users);
|
|
const useCase = signUpUseCase(
|
|
users,
|
|
auth,
|
|
new RecordingEventBus(),
|
|
undefined,
|
|
);
|
|
const controller = signUpController(useCase);
|
|
|
|
await expect(
|
|
controller({
|
|
username: "dave",
|
|
password: "secret_password",
|
|
confirmPassword: "different_password",
|
|
}),
|
|
).rejects.toBeInstanceOf(InputParseError);
|
|
});
|
|
|
|
it("throws InputParseError when username is too short", async () => {
|
|
const users = new MockUsersRepository([]);
|
|
const auth = new MockAuthenticationService(users);
|
|
await users.createUser(userFactory.build({ username: "alice" }));
|
|
const useCase = signUpUseCase(
|
|
users,
|
|
auth,
|
|
new RecordingEventBus(),
|
|
undefined,
|
|
);
|
|
const controller = signUpController(useCase);
|
|
|
|
await expect(
|
|
controller({
|
|
username: "ab",
|
|
password: "secret_password",
|
|
confirmPassword: "secret_password",
|
|
}),
|
|
).rejects.toBeInstanceOf(InputParseError);
|
|
});
|
|
});
|