POST /api/custom-domains now accepts a third body variant — { pattern }
without an explicit kind or type — which the resolver infers from the
pattern shape so the frontend can ship a single dynamic input field
instead of asking the user to choose between Seite / E-Mail in advance.
- pattern contains '@' → treat as kind='mail', strip the
local-part, store as mail_domain
after the same TLD / DOMAIN_RE
validation as the explicit-kind path
- pattern contains '.' (no '@') → treat as kind='web'
- neither → 400 INVALID_PATTERN with a clear
message ("Bitte eine Domain oder
Mail-Adresse eingeben")
Variant A ({ pattern, kind }) and Variant B ({ domain, type }) stay
fully supported, plus a `kind: 'auto'` keyword if a client prefers an
explicit opt-in to the auto-detect path. The display-name path is still
locked off in v1.0 — pure tokens without dots route into the same
INVALID_PATTERN response, which keeps the v1.0 guarantee intact.
Plan-limits.test.ts grew the matching test cases — auto-detect for a
domain, auto-detect for a full address (local-part stripped to mail_-
domain), auto-detect rejection for a bare token. All existing tests
keep their pass status.
419 lines
18 KiB
TypeScript
419 lines
18 KiB
TypeScript
/**
|
|
* Tests: Custom-Domain Plan-Limits (separate web/mail buckets)
|
|
*
|
|
* Testet:
|
|
* - getPlanLimits returnt strukturiertes { web, mail } Objekt
|
|
* - countActiveCustomDomainsSplit returnt korrekte Split-Counts (Unit ohne DB)
|
|
* - POST-Body-Compat: { pattern, kind: 'mail' } mit Domain-Pattern → mail_domain
|
|
* - POST-Body-Compat: { pattern, kind: 'mail' } mit Display-Name-Pattern → 400 INVALID_MAIL_DOMAIN (v1.0)
|
|
* - POST-Body-Compat: { domain, type: 'mail_display_name' } → 400 DISPLAY_NAME_NOT_SUPPORTED (v1.0)
|
|
* - POST-Body-Compat: volle Adresse (local@domain.tld) → local-part gestripped, gespeichert als mail_domain
|
|
* - Submit eines mail_domain → erlaubt (gleich wie web)
|
|
*
|
|
* DSGVO: keine PII. Synthetic Brand-Namen und Test-Domains.
|
|
*/
|
|
import { describe, it, expect } from "vitest";
|
|
import { getPlanLimits, PLAN_LIMITS } from "../../server/utils/plan-features";
|
|
|
|
// ─── Plan-Limits Shape ────────────────────────────────────────────────────────
|
|
|
|
describe("getPlanLimits — customDomains ist strukturiertes Objekt", () => {
|
|
it("Free: customDomains = { web: 5, mail: 5 }", () => {
|
|
const limits = getPlanLimits("free");
|
|
expect(limits.customDomains).toEqual({ web: 5, mail: 5 });
|
|
});
|
|
|
|
it("Pro: customDomains = { web: 5, mail: 5 }", () => {
|
|
const limits = getPlanLimits("pro");
|
|
expect(limits.customDomains).toEqual({ web: 5, mail: 5 });
|
|
});
|
|
|
|
it("Legend: customDomains = { web: 10, mail: 10 }", () => {
|
|
const limits = getPlanLimits("legend");
|
|
expect(limits.customDomains).toEqual({ web: 10, mail: 10 });
|
|
});
|
|
|
|
it("Legacy 'premium' → Legend limits", () => {
|
|
const limits = getPlanLimits("premium");
|
|
expect(limits.customDomains).toEqual({ web: 10, mail: 10 });
|
|
});
|
|
|
|
it("Legacy 'standard' → Pro limits", () => {
|
|
const limits = getPlanLimits("standard");
|
|
expect(limits.customDomains).toEqual({ web: 5, mail: 5 });
|
|
});
|
|
|
|
it("PLAN_LIMITS.free.customDomains hat keine 'number'-Shape mehr", () => {
|
|
const val = PLAN_LIMITS.free.customDomains;
|
|
expect(typeof val).not.toBe("number");
|
|
expect(typeof val).toBe("object");
|
|
expect(typeof val.web).toBe("number");
|
|
expect(typeof val.mail).toBe("number");
|
|
});
|
|
});
|
|
|
|
// ─── Body-Compat-Mapping ──────────────────────────────────────────────────────
|
|
|
|
/**
|
|
* Extrahiert die resolveTypeAndValue-Logik als reiner Unit-Test ohne Server-Overhead.
|
|
* Spiegelt exakt die Implementierung in index.post.ts wider (v1.0 — kein Display-Name).
|
|
*
|
|
* Returns { ok: true, type, value } oder { ok: false, error } — analog zum Endpoint.
|
|
*/
|
|
type TestResolveResult =
|
|
| { ok: true; type: string; value: string }
|
|
| { ok: false; error: "INVALID_MAIL_DOMAIN" | "DISPLAY_NAME_NOT_SUPPORTED" | "INVALID_PATTERN" };
|
|
|
|
function resolveTypeAndValueForTest(body: any): TestResolveResult {
|
|
const DOMAIN_RE = /^[a-z0-9]([a-z0-9-]*[a-z0-9])?(\.[a-z0-9]([a-z0-9-]*[a-z0-9])?)+$/;
|
|
|
|
if (body?.kind !== undefined || body?.pattern !== undefined) {
|
|
const rawKind = (body?.kind as string)?.trim();
|
|
const pattern = (body?.pattern as string)?.trim() ?? "";
|
|
|
|
// Variante C: kind fehlt oder kind='auto' → Auto-Detect
|
|
if (rawKind === undefined || rawKind === null || rawKind === "" || rawKind === "auto") {
|
|
if (pattern.includes("@")) {
|
|
const atIdx = pattern.lastIndexOf("@");
|
|
const normalized = pattern.slice(atIdx + 1).toLowerCase().replace(/^https?:\/\//, "").trim();
|
|
if (normalized.includes(".") && DOMAIN_RE.test(normalized)) {
|
|
return { ok: true, type: "mail_domain", value: normalized };
|
|
}
|
|
return { ok: false, error: "INVALID_MAIL_DOMAIN" };
|
|
}
|
|
|
|
if (pattern.includes(".")) {
|
|
return { ok: true, type: "web", value: pattern };
|
|
}
|
|
|
|
return { ok: false, error: "INVALID_PATTERN" };
|
|
}
|
|
|
|
const kind = rawKind;
|
|
|
|
if (kind === "web") {
|
|
return { ok: true, type: "web", value: pattern };
|
|
}
|
|
|
|
if (kind === "mail") {
|
|
// Defensiv: volle Adresse → local-part strippen
|
|
let normalized = pattern;
|
|
if (normalized.includes("@")) {
|
|
const atIdx = normalized.lastIndexOf("@");
|
|
normalized = normalized.slice(atIdx + 1);
|
|
}
|
|
normalized = normalized.toLowerCase().replace(/^https?:\/\//, "").trim();
|
|
|
|
if (normalized.includes(".") && DOMAIN_RE.test(normalized)) {
|
|
return { ok: true, type: "mail_domain", value: normalized };
|
|
}
|
|
|
|
// Display-Name-Input: nicht unterstützt in v1.0
|
|
return { ok: false, error: "INVALID_MAIL_DOMAIN" };
|
|
}
|
|
|
|
return { ok: true, type: kind, value: pattern };
|
|
}
|
|
|
|
const rawType = (body?.type as string)?.trim() ?? "web";
|
|
const value = (body?.domain as string)?.trim() ?? "";
|
|
|
|
// v1.0: mail_display_name wird nicht akzeptiert
|
|
if (rawType === "mail_display_name") {
|
|
return { ok: false, error: "DISPLAY_NAME_NOT_SUPPORTED" };
|
|
}
|
|
|
|
return { ok: true, type: rawType, value };
|
|
}
|
|
|
|
describe("Body-Compat-Mapping — kind='mail' Pattern-Analyse (v1.0: nur Domain-Input)", () => {
|
|
it("{ kind: 'web', pattern: 'casino.de' } → type='web'", () => {
|
|
const r = resolveTypeAndValueForTest({ kind: "web", pattern: "casino.de" });
|
|
expect(r.ok).toBe(true);
|
|
if (r.ok) {
|
|
expect(r.type).toBe("web");
|
|
expect(r.value).toBe("casino.de");
|
|
}
|
|
});
|
|
|
|
it("{ kind: 'mail', pattern: 'only4-subscribers.com' } → mail_domain (enthält Punkt + TLD)", () => {
|
|
const r = resolveTypeAndValueForTest({ kind: "mail", pattern: "only4-subscribers.com" });
|
|
expect(r.ok).toBe(true);
|
|
if (r.ok) {
|
|
expect(r.type).toBe("mail_domain");
|
|
expect(r.value).toBe("only4-subscribers.com");
|
|
}
|
|
});
|
|
|
|
it("{ kind: 'mail', pattern: 'casino-relay.de' } → mail_domain", () => {
|
|
const r = resolveTypeAndValueForTest({ kind: "mail", pattern: "casino-relay.de" });
|
|
expect(r.ok).toBe(true);
|
|
if (r.ok) expect(r.type).toBe("mail_domain");
|
|
});
|
|
|
|
// v1.0: Display-Name-Input → 400 INVALID_MAIL_DOMAIN (kein mail_display_name mehr)
|
|
it("{ kind: 'mail', pattern: 'EXTRASPIN' } → 400 INVALID_MAIL_DOMAIN (kein Punkt, Display-Name)", () => {
|
|
const r = resolveTypeAndValueForTest({ kind: "mail", pattern: "EXTRASPIN" });
|
|
expect(r.ok).toBe(false);
|
|
if (!r.ok) expect(r.error).toBe("INVALID_MAIL_DOMAIN");
|
|
});
|
|
|
|
it("{ kind: 'mail', pattern: 'Casino Bonus' } → 400 INVALID_MAIL_DOMAIN (Leerzeichen, kein TLD)", () => {
|
|
const r = resolveTypeAndValueForTest({ kind: "mail", pattern: "Casino Bonus" });
|
|
expect(r.ok).toBe(false);
|
|
if (!r.ok) expect(r.error).toBe("INVALID_MAIL_DOMAIN");
|
|
});
|
|
|
|
// Defensiv: volle Adresse → local-part gestripped, als mail_domain gespeichert
|
|
it("{ kind: 'mail', pattern: 'communications@only4-subscribers.com' } → mail_domain mit 'only4-subscribers.com'", () => {
|
|
const r = resolveTypeAndValueForTest({ kind: "mail", pattern: "communications@only4-subscribers.com" });
|
|
expect(r.ok).toBe(true);
|
|
if (r.ok) {
|
|
expect(r.type).toBe("mail_domain");
|
|
expect(r.value).toBe("only4-subscribers.com");
|
|
}
|
|
});
|
|
|
|
it("Variante B { domain, type: 'mail_domain' } passiert unverändert durch", () => {
|
|
const r = resolveTypeAndValueForTest({ domain: "spin.casino.com", type: "mail_domain" });
|
|
expect(r.ok).toBe(true);
|
|
if (r.ok) {
|
|
expect(r.type).toBe("mail_domain");
|
|
expect(r.value).toBe("spin.casino.com");
|
|
}
|
|
});
|
|
|
|
// v1.0: Variante B mit mail_display_name → 400 DISPLAY_NAME_NOT_SUPPORTED
|
|
it("Variante B { domain, type: 'mail_display_name' } → 400 DISPLAY_NAME_NOT_SUPPORTED (v1.0)", () => {
|
|
const r = resolveTypeAndValueForTest({ domain: "EXTRASPIN", type: "mail_display_name" });
|
|
expect(r.ok).toBe(false);
|
|
if (!r.ok) expect(r.error).toBe("DISPLAY_NAME_NOT_SUPPORTED");
|
|
});
|
|
});
|
|
|
|
// ─── Slot-Bucket-Logik ────────────────────────────────────────────────────────
|
|
|
|
describe("Slot-Bucket-Logik — welcher Bucket pro Type", () => {
|
|
it("type='web' → bucket='web'", () => {
|
|
const bucket = "web" === "web" ? "web" : "mail";
|
|
expect(bucket).toBe("web");
|
|
});
|
|
|
|
it("type='mail_domain' → bucket='mail'", () => {
|
|
const type = "mail_domain";
|
|
const bucket: "web" | "mail" = type === "web" ? "web" : "mail";
|
|
expect(bucket).toBe("mail");
|
|
});
|
|
|
|
it("type='mail_display_name' → bucket='mail'", () => {
|
|
const type = "mail_display_name";
|
|
const bucket: "web" | "mail" = type === "web" ? "web" : "mail";
|
|
expect(bucket).toBe("mail");
|
|
});
|
|
|
|
it("5 mail_domain + 0 mail_display_name = 5 mail-Slots belegt (kein Platz für 6tes bei Free)", () => {
|
|
const freeMailLimit = PLAN_LIMITS.free.customDomains.mail; // 5
|
|
const mailCount = 5; // 5 mail_domain belegt
|
|
expect(mailCount >= freeMailLimit).toBe(true);
|
|
});
|
|
|
|
it("5 mail belegt + neues web → kein MAIL_LIMIT_REACHED (web-Slot separat)", () => {
|
|
const freeWebLimit = PLAN_LIMITS.free.customDomains.web; // 5
|
|
const freeMailLimit = PLAN_LIMITS.free.customDomains.mail; // 5
|
|
const mailCount = 5;
|
|
const webCount = 0;
|
|
// Mail voll, aber web-Slot noch frei
|
|
expect(webCount >= freeWebLimit).toBe(false); // web nicht voll
|
|
expect(mailCount >= freeMailLimit).toBe(true); // mail voll
|
|
});
|
|
|
|
it("5 web + 5 mail belegt → beide Buckets voll (Free-Plan exhausted)", () => {
|
|
const limits = getPlanLimits("free");
|
|
const webCount = 5;
|
|
const mailCount = 5;
|
|
expect(webCount >= limits.customDomains.web).toBe(true);
|
|
expect(mailCount >= limits.customDomains.mail).toBe(true);
|
|
});
|
|
|
|
it("legacy mail_display_name-Rows (falls vorhanden) zählen weiterhin in mail-Bucket", () => {
|
|
// v1.0: mail_display_name kann nicht mehr über POST /api/custom-domains erstellt werden.
|
|
// Etwaige Legacy-Rows (aus alten Tests oder manuellem Insert) zählen dennoch korrekt
|
|
// im mail-Bucket via countActiveCustomDomainsSplit — da sie type='mail_display_name' haben.
|
|
// DB-Typ bleibt im Schema erhalten für triviale v1.1-Reaktivierung.
|
|
const mailDomainCount = 3;
|
|
const mailDisplayNameCount = 2; // Legacy-Rows (nicht mehr per API anlegbar in v1.0)
|
|
const totalMail = mailDomainCount + mailDisplayNameCount;
|
|
const freeMailLimit = PLAN_LIMITS.free.customDomains.mail;
|
|
expect(totalMail >= freeMailLimit).toBe(true);
|
|
});
|
|
});
|
|
|
|
// ─── countActiveCustomDomainsSplit Dokumentation ──────────────────────────────
|
|
|
|
describe("countActiveCustomDomainsSplit — Slot-Counting-Semantik (Dokumentationstest ohne DB)", () => {
|
|
it("grupiert type='web' in web-Bucket, 'mail_domain'+'mail_display_name' in mail-Bucket", () => {
|
|
// Diese Funktion nutzt Prisma groupBy — echter DB-Test läuft auf Hetzner.
|
|
// Hier dokumentieren wir die erwartete Semantik:
|
|
//
|
|
// Input: 3 rows web + 2 rows mail_domain + 1 row mail_display_name
|
|
// Expected: { web: 3, mail: 3 }
|
|
//
|
|
// Der mail-Bucket summiert BEIDE mail-Types.
|
|
const expectedWeb = 3;
|
|
const expectedMail = 3; // 2 mail_domain + 1 mail_display_name
|
|
expect(expectedWeb + expectedMail).toBe(6); // Gesamtanzahl korrekt
|
|
expect(typeof expectedWeb).toBe("number");
|
|
expect(typeof expectedMail).toBe("number");
|
|
});
|
|
});
|
|
|
|
// ─── DomainSubmission type-Kopier-Semantik ────────────────────────────────────
|
|
|
|
/**
|
|
* Dokumentiert die Regel: submitDomainForReview kopiert type aus UserCustomDomain.
|
|
* Kein DB-Test — reine Semantik-Verifikation.
|
|
*/
|
|
describe("DomainSubmission.type — Type-Kopier-Semantik beim Submit", () => {
|
|
it("type='web' aus CustomDomain → DomainSubmission.type='web'", () => {
|
|
// Spiegelt die Logik in submitDomainForReview: domain.type wird in Submission kopiert
|
|
const customDomainType = "web";
|
|
const submissionType = customDomainType; // direkte Zuweisung im DB-Layer
|
|
expect(submissionType).toBe("web");
|
|
});
|
|
|
|
it("type='mail_domain' aus CustomDomain → DomainSubmission.type='mail_domain'", () => {
|
|
const customDomainType = "mail_domain";
|
|
const submissionType = customDomainType;
|
|
expect(submissionType).toBe("mail_domain");
|
|
});
|
|
|
|
it("type='web' ist Default falls CustomDomain fehlt (Backfill-Semantik)", () => {
|
|
// Backfill-Migration: Rows ohne JOIN-Match behalten DEFAULT 'web'
|
|
const fallback = "web";
|
|
expect(fallback).toBe("web");
|
|
});
|
|
|
|
it("Community-Vote-Post-Text für mail_domain enthält Hinweis auf Mail-Absender", () => {
|
|
// Spiegelt die Logik in submit.post.ts — type-aware Post-Content
|
|
const type = "mail_domain";
|
|
const domain = "mailing.casino-affiliate.com";
|
|
const postContent =
|
|
type === "mail_domain"
|
|
? `Domain-Vorschlag (Mail-Absender): **${domain}**\n\nIch schlage vor, diese Absender-Domain zur globalen ReBreak-Sperrliste hinzuzufügen. Casino-Affiliates nutzen oft Mailing-Listen mit harmlosen Namen. Stimme ab: Sollte **${domain}** global gesperrt werden?`
|
|
: `Domain-Vorschlag: **${domain}**\n\nIch schlage vor, diese Domain zur globalen ReBreak-Sperrliste hinzuzufügen. Stimme ab: Sollte **${domain}** global gesperrt werden?`;
|
|
expect(postContent).toContain("Mail-Absender");
|
|
expect(postContent).toContain(domain);
|
|
});
|
|
|
|
it("Community-Vote-Post-Text für web enthält keinen Mail-Absender-Hinweis", () => {
|
|
const type = "web";
|
|
const domain = "casino.de";
|
|
const postContent =
|
|
type === "mail_domain"
|
|
? `Domain-Vorschlag (Mail-Absender): **${domain}**`
|
|
: `Domain-Vorschlag: **${domain}**\n\nIch schlage vor, diese Domain zur globalen ReBreak-Sperrliste hinzuzufügen.`;
|
|
expect(postContent).not.toContain("Mail-Absender");
|
|
expect(postContent).toContain(domain);
|
|
});
|
|
});
|
|
|
|
// ─── Submit-Guard mail_display_name ──────────────────────────────────────────
|
|
|
|
describe("Submit-Guard — DISPLAY_NAME_NOT_SUBMITTABLE", () => {
|
|
it("type='mail_display_name' darf nicht submitted werden — guard in submit.post.ts (v1.0)", () => {
|
|
// v1.0: mail_display_name-Rows können nicht mehr per POST /api/custom-domains angelegt werden.
|
|
// Falls Legacy-Rows in der DB existieren: submit.post.ts wirft 400 DISPLAY_NAME_NOT_SUBMITTABLE.
|
|
// Spiegelt die Guard-Logik in submit.post.ts wider:
|
|
// if (existing.type === 'mail_display_name') → 400 DISPLAY_NAME_NOT_SUBMITTABLE
|
|
const type = "mail_display_name";
|
|
const isSubmittable = type !== "mail_display_name";
|
|
expect(isSubmittable).toBe(false);
|
|
});
|
|
|
|
it("type='mail_domain' ist submittable (gleich wie web)", () => {
|
|
const type = "mail_domain";
|
|
const isSubmittable = type !== "mail_display_name";
|
|
expect(isSubmittable).toBe(true);
|
|
});
|
|
|
|
it("type='web' ist submittable", () => {
|
|
const type = "web";
|
|
const isSubmittable = type !== "mail_display_name";
|
|
expect(isSubmittable).toBe(true);
|
|
});
|
|
});
|
|
|
|
// ─── Variante C — Auto-Detect (kein kind im Body) ────────────────────────────
|
|
|
|
describe("Auto-Detect — { pattern } ohne kind (Variante C)", () => {
|
|
it("{ pattern: 'casino.com' } → type='web', value='casino.com'", () => {
|
|
const r = resolveTypeAndValueForTest({ pattern: "casino.com" });
|
|
expect(r.ok).toBe(true);
|
|
if (r.ok) {
|
|
expect(r.type).toBe("web");
|
|
expect(r.value).toBe("casino.com");
|
|
}
|
|
});
|
|
|
|
it("{ pattern: 'info@casino.com' } → type='mail_domain', value='casino.com'", () => {
|
|
const r = resolveTypeAndValueForTest({ pattern: "info@casino.com" });
|
|
expect(r.ok).toBe(true);
|
|
if (r.ok) {
|
|
expect(r.type).toBe("mail_domain");
|
|
expect(r.value).toBe("casino.com");
|
|
}
|
|
});
|
|
|
|
it("{ pattern: 'INVALID' } → 400 INVALID_PATTERN (kein '@', kein '.')", () => {
|
|
const r = resolveTypeAndValueForTest({ pattern: "INVALID" });
|
|
expect(r.ok).toBe(false);
|
|
if (!r.ok) expect(r.error).toBe("INVALID_PATTERN");
|
|
});
|
|
|
|
it("{ pattern: 'a.b@c.d' } → type='mail_domain', value='c.d' (lastIndexOf @ strip)", () => {
|
|
const r = resolveTypeAndValueForTest({ pattern: "a.b@c.d" });
|
|
expect(r.ok).toBe(true);
|
|
if (r.ok) {
|
|
expect(r.type).toBe("mail_domain");
|
|
expect(r.value).toBe("c.d");
|
|
}
|
|
});
|
|
|
|
it("{ kind: 'auto', pattern: 'casino.com' } → type='web' (kind='auto' triggert Auto-Detect)", () => {
|
|
const r = resolveTypeAndValueForTest({ kind: "auto", pattern: "casino.com" });
|
|
expect(r.ok).toBe(true);
|
|
if (r.ok) {
|
|
expect(r.type).toBe("web");
|
|
expect(r.value).toBe("casino.com");
|
|
}
|
|
});
|
|
|
|
it("{ kind: 'auto', pattern: 'no@valid.domain.here' } → type='mail_domain', value='valid.domain.here'", () => {
|
|
const r = resolveTypeAndValueForTest({ kind: "auto", pattern: "no@valid.domain.here" });
|
|
expect(r.ok).toBe(true);
|
|
if (r.ok) {
|
|
expect(r.type).toBe("mail_domain");
|
|
expect(r.value).toBe("valid.domain.here");
|
|
}
|
|
});
|
|
|
|
// Bestehende Variante A bleibt unberührt (Backwards-Compat)
|
|
it("Backwards-Compat: { kind: 'web', pattern: 'casino.de' } → type='web' (weiterhin)", () => {
|
|
const r = resolveTypeAndValueForTest({ kind: "web", pattern: "casino.de" });
|
|
expect(r.ok).toBe(true);
|
|
if (r.ok) {
|
|
expect(r.type).toBe("web");
|
|
expect(r.value).toBe("casino.de");
|
|
}
|
|
});
|
|
|
|
it("Backwards-Compat: { kind: 'mail', pattern: 'news@bet365.de' } → type='mail_domain', value='bet365.de'", () => {
|
|
const r = resolveTypeAndValueForTest({ kind: "mail", pattern: "news@bet365.de" });
|
|
expect(r.ok).toBe(true);
|
|
if (r.ok) {
|
|
expect(r.type).toBe("mail_domain");
|
|
expect(r.value).toBe("bet365.de");
|
|
}
|
|
});
|
|
});
|