Add rootless Podman fixes, and others

improve container startup for rootless Podman, plus related refactors and tests. Key changes:

- Add/modify Audiobookshelf-related code and wiring (src/lib/services/audiobookshelf/api.ts, library service refs) and update documentation TABLEOFCONTENTS to reference ABS implementation.
- Detect user namespace in docker/unified app-start.sh and redis-start.sh and skip gosu when running in rootless Podman to preserve UID mapping; improve startup logging and verification.
- Add utility/service files (auth-token-cache.service.ts, credential-migration.service.ts, cleanup-helpers.ts) and corresponding tests; update chapter-merger and metadata-tagger utilities/tests.
- Update many admin/auth API routes and tests to reflect changes in settings and integrations.
- Remove large AI agent and Audiobookshelf implementation guide docs (AGENTS.md and the implementation guide) and add README note about AI-assisted workflow.

These changes enable Audiobookshelf backend mode, improve compatibility with rootless container runtimes, and include cleanup/refactor work and unit tests.
This commit is contained in:
kikootwo
2026-02-04 14:05:28 -05:00
parent 2ef9ac7be1
commit a0f2ba680d
42 changed files with 1843 additions and 3820 deletions
+7 -3
View File
@@ -45,7 +45,7 @@ describe('SelectProfilePage', () => {
});
it('selects an unprotected profile and stores auth data', async () => {
sessionStorage.setItem('plex_main_token', 'main-token');
// Token is now stored server-side, only pinId needed in URL
setMockSearchParams('pinId=123');
const setAuthDataMock = vi.fn();
@@ -71,6 +71,9 @@ describe('SelectProfilePage', () => {
const fetchMock = vi.fn(async (input: RequestInfo, init?: RequestInit) => {
const url = typeof input === 'string' ? input : input.url;
if (url === '/api/auth/plex/home-users') {
// Verify pinId header is sent instead of token
const headers = (init as RequestInit)?.headers as Record<string, string>;
expect(headers?.['X-Plex-Pin-Id']).toBe('123');
return makeJsonResponse({ users: profiles });
}
if (url === '/api/auth/plex/switch-profile') {
@@ -107,7 +110,7 @@ describe('SelectProfilePage', () => {
});
it('prompts for a PIN and handles invalid submissions', async () => {
sessionStorage.setItem('plex_main_token', 'main-token');
// Token is now stored server-side, only pinId needed in URL
setMockSearchParams('pinId=555');
const setAuthDataMock = vi.fn();
@@ -136,7 +139,8 @@ describe('SelectProfilePage', () => {
return makeJsonResponse({ users: profiles });
}
if (url === '/api/auth/plex/switch-profile') {
return makeJsonResponse({ message: 'Invalid PIN' }, false, 401);
// Return InvalidPIN error type to trigger PIN error message
return makeJsonResponse({ error: 'InvalidPIN', message: 'Invalid PIN' }, false, 401);
}
throw new Error(`Unexpected fetch: ${url}`);
});