ircu2/tools/iauthd-ts/tests/auth/manager.test.ts

192 lines
6.3 KiB
TypeScript

/**
* Tests for AuthManager
*/
import { describe, it, expect, beforeEach, afterEach } from 'vitest';
import { writeFileSync, unlinkSync, mkdtempSync } from 'node:fs';
import { join } from 'node:path';
import { tmpdir } from 'node:os';
import { AuthManager } from '../../src/auth/manager.js';
import type { AuthProviderConfig } from '../../src/auth/types.js';
describe('AuthManager', () => {
let tempDir: string;
let usersFile1: string;
let usersFile2: string;
beforeEach(() => {
tempDir = mkdtempSync(join(tmpdir(), 'iauthd-test-'));
usersFile1 = join(tempDir, 'users1');
usersFile2 = join(tempDir, 'users2');
});
afterEach(() => {
try {
unlinkSync(usersFile1);
unlinkSync(usersFile2);
} catch {
// Ignore if files don't exist
}
});
describe('initialization', () => {
it('should initialize with no providers', async () => {
const manager = new AuthManager([]);
await manager.initialize();
expect(manager.hasProviders()).toBe(false);
expect(manager.getProviderCount()).toBe(0);
});
it('should initialize single file provider', async () => {
writeFileSync(usersFile1, 'testuser:testpass\n');
const configs: AuthProviderConfig[] = [
{ provider: 'file', path: usersFile1 },
];
const manager = new AuthManager(configs);
await manager.initialize();
expect(manager.hasProviders()).toBe(true);
expect(manager.getProviderCount()).toBe(1);
});
it('should initialize multiple providers', async () => {
writeFileSync(usersFile1, 'user1:pass1\n');
writeFileSync(usersFile2, 'user2:pass2\n');
const configs: AuthProviderConfig[] = [
{ provider: 'file', path: usersFile1, priority: 50 },
{ provider: 'file', path: usersFile2, priority: 100 },
];
const manager = new AuthManager(configs);
await manager.initialize();
expect(manager.getProviderCount()).toBe(2);
});
it('should sort providers by priority', async () => {
writeFileSync(usersFile1, 'user1:pass1\n');
writeFileSync(usersFile2, 'user2:pass2\n');
const configs: AuthProviderConfig[] = [
{ provider: 'file', path: usersFile1, priority: 100 },
{ provider: 'file', path: usersFile2, priority: 50 },
];
const manager = new AuthManager(configs);
await manager.initialize();
const info = manager.getProviderInfo();
expect(info[0].priority).toBe(50);
expect(info[1].priority).toBe(100);
});
});
describe('authentication with fallback chain', () => {
it('should authenticate against first matching provider', async () => {
writeFileSync(usersFile1, 'user1:pass1\n');
writeFileSync(usersFile2, 'user2:pass2\n');
const configs: AuthProviderConfig[] = [
{ provider: 'file', path: usersFile1, priority: 50 },
{ provider: 'file', path: usersFile2, priority: 100 },
];
const manager = new AuthManager(configs);
await manager.initialize();
const result = await manager.authenticate('user1', 'pass1');
expect(result.success).toBe(true);
expect(result.account).toBe('user1');
});
it('should fallback to second provider if first fails', async () => {
writeFileSync(usersFile1, 'user1:pass1\n');
writeFileSync(usersFile2, 'user2:pass2\n');
const configs: AuthProviderConfig[] = [
{ provider: 'file', path: usersFile1, priority: 50 },
{ provider: 'file', path: usersFile2, priority: 100 },
];
const manager = new AuthManager(configs);
await manager.initialize();
// user2 is only in the second file
const result = await manager.authenticate('user2', 'pass2');
expect(result.success).toBe(true);
expect(result.account).toBe('user2');
});
it('should fail if no provider matches', async () => {
writeFileSync(usersFile1, 'user1:pass1\n');
writeFileSync(usersFile2, 'user2:pass2\n');
const configs: AuthProviderConfig[] = [
{ provider: 'file', path: usersFile1 },
{ provider: 'file', path: usersFile2 },
];
const manager = new AuthManager(configs);
await manager.initialize();
const result = await manager.authenticate('unknownuser', 'anypass');
expect(result.success).toBe(false);
});
it('should return error if not initialized', async () => {
const manager = new AuthManager([]);
// Don't call initialize
const result = await manager.authenticate('user', 'pass');
expect(result.success).toBe(false);
expect(result.error).toBe('AuthManager not initialized');
});
it('should return error if no providers configured', async () => {
const manager = new AuthManager([]);
await manager.initialize();
const result = await manager.authenticate('user', 'pass');
expect(result.success).toBe(false);
expect(result.error).toBe('No auth providers configured');
});
});
describe('provider health', () => {
it('should report healthy providers', async () => {
writeFileSync(usersFile1, 'user1:pass1\n');
const configs: AuthProviderConfig[] = [
{ provider: 'file', path: usersFile1 },
];
const manager = new AuthManager(configs);
await manager.initialize();
expect(manager.hasHealthyProviders()).toBe(true);
const info = manager.getProviderInfo();
expect(info[0].healthy).toBe(true);
});
});
describe('reload', () => {
it('should reload all providers', async () => {
writeFileSync(usersFile1, 'user1:pass1\n');
const configs: AuthProviderConfig[] = [
{ provider: 'file', path: usersFile1 },
];
const manager = new AuthManager(configs);
await manager.initialize();
// Modify file
writeFileSync(usersFile1, 'user1:newpass\n');
await manager.reload();
// Should now use new password
const result = await manager.authenticate('user1', 'newpass');
expect(result.success).toBe(true);
});
});
describe('shutdown', () => {
it('should shutdown all providers', async () => {
writeFileSync(usersFile1, 'user1:pass1\n');
const configs: AuthProviderConfig[] = [
{ provider: 'file', path: usersFile1 },
];
const manager = new AuthManager(configs);
await manager.initialize();
await manager.shutdown();
expect(manager.hasProviders()).toBe(false);
});
});
});