Files
fotospiel-app/resources/js/admin/mobile/__tests__/billingUsage.test.ts
Codex Agent ad829ae509
Some checks failed
linter / quality (push) Has been cancelled
tests / ci (push) Has been cancelled
tests / ui (push) Has been cancelled
Update partner packages, copy, and demo switcher
2026-01-15 17:33:36 +01:00

95 lines
2.8 KiB
TypeScript

import { describe, expect, it } from 'vitest';
import type { TenantPackageSummary } from '../../api';
import { buildPackageUsageMetrics, getUsageState, usagePercent } from '../billingUsage';
const basePackage: TenantPackageSummary = {
id: 1,
package_id: 1,
package_name: 'Pro',
package_type: 'reseller',
included_package_slug: 'pro',
active: true,
used_events: 2,
remaining_events: 3,
price: 120,
currency: 'EUR',
purchased_at: null,
expires_at: null,
package_limits: {
max_events_per_year: 5,
max_guests: 150,
max_photos: 1000,
gallery_days: 90,
remaining_gallery_days: 30,
},
branding_allowed: true,
watermark_allowed: true,
features: null,
};
describe('buildPackageUsageMetrics', () => {
it('builds usage metrics for event, guest, and photo limits', () => {
const metrics = buildPackageUsageMetrics(basePackage);
const keys = metrics.map((metric) => metric.key);
expect(keys).toEqual(['events', 'guests', 'photos', 'gallery']);
const eventMetric = metrics.find((metric) => metric.key === 'events');
expect(eventMetric?.used).toBe(2);
expect(eventMetric?.limit).toBe(5);
expect(eventMetric?.remaining).toBe(3);
const galleryMetric = metrics.find((metric) => metric.key === 'gallery');
expect(galleryMetric?.remaining).toBe(30);
});
it('derives remaining when only usage is known', () => {
const metrics = buildPackageUsageMetrics({
...basePackage,
remaining_events: null,
package_limits: { max_events_per_year: 10, gallery_days: 60 },
used_events: 4,
});
expect(metrics[0].remaining).toBe(6);
});
it('filters metrics without limits', () => {
const metrics = buildPackageUsageMetrics({
...basePackage,
package_limits: { max_events_per_year: 0 },
});
expect(metrics).toHaveLength(0);
});
});
describe('usagePercent', () => {
it('calculates usage percent when usage is known', () => {
const metrics = buildPackageUsageMetrics(basePackage);
const eventMetric = metrics.find((metric) => metric.key === 'events');
expect(eventMetric).toBeTruthy();
expect(usagePercent(eventMetric!)).toBe(40);
});
it('defaults to full bar when usage is unknown', () => {
const metrics = buildPackageUsageMetrics({
...basePackage,
used_events: NaN,
remaining_events: null,
package_limits: { max_events_per_year: 10 },
});
expect(usagePercent(metrics[0])).toBe(100);
});
});
describe('getUsageState', () => {
it('returns warning and danger based on thresholds', () => {
const metrics = buildPackageUsageMetrics({
...basePackage,
used_events: 8,
remaining_events: null,
package_limits: { max_events_per_year: 10 },
});
expect(getUsageState(metrics[0])).toBe('warning');
expect(getUsageState({ ...metrics[0], used: 10, remaining: 0 })).toBe('danger');
});
});