Release v1.3.0: Multi-source downloads, audio analyzer resilience, mobile improvements

Major Features:
- Multi-source download system (Soulseek/Lidarr with fallback)
- Configurable enrichment speed control (1-5x)
- Mobile touch drag support for seek sliders
- iOS PWA media controls (Control Center, Lock Screen)
- Artist name alias resolution via Last.fm
- Circuit breaker pattern for audio analysis

Critical Fixes:
- Audio analyzer stability (non-ASCII, BrokenProcessPool, OOM)
- Discovery system race conditions and import failures
- Radio decade categorization using originalYear
- LastFM API response normalization
- Mood bucket infinite loop prevention

Security:
- Bull Board admin authentication
- Lidarr webhook signature verification
- JWT token expiration and refresh
- Encryption key validation on startup

Closes #2, #6, #9, #13, #21, #26, #31, #34, #35, #37, #40, #43
This commit is contained in:
Your Name
2026-01-06 20:07:33 -06:00
parent 8fe151a0d1
commit cc8d0f6969
242 changed files with 20562 additions and 7725 deletions

View File

@@ -0,0 +1,146 @@
#!/usr/bin/env ts-node
/**
* Backfill Script: Populate originalYear for existing albums
*
* This script populates the new originalYear field for albums that don't have it yet.
*
* Strategy:
* 1. For albums already enriched with MusicBrainz data, copy year to originalYear
* (since enrichment overwrites year with the original release date)
* 2. Skip temporary albums (temp-* MBIDs)
*
* Usage:
* npx ts-node scripts/backfill-original-year.ts [--dry-run]
*
* Options:
* --dry-run Show what would be updated without making changes
*/
import { PrismaClient } from "@prisma/client";
const prisma = new PrismaClient();
async function backfillOriginalYear(dryRun: boolean = false) {
console.log("=== Backfill originalYear Script ===\n");
console.log(
`Mode: ${
dryRun ? "DRY RUN (no changes)" : "LIVE (will update database)"
}\n`
);
try {
// Find albums that need backfilling
const albumsToBackfill = await prisma.album.findMany({
where: {
originalYear: null,
year: { not: null }, // Only albums that have a year value
rgMbid: { not: { startsWith: "temp-" } }, // Skip temporary albums
},
select: {
id: true,
rgMbid: true,
title: true,
year: true,
originalYear: true,
artist: {
select: {
name: true,
},
},
},
});
console.log(`Found ${albumsToBackfill.length} albums to backfill\n`);
if (albumsToBackfill.length === 0) {
console.log("✓ No albums need backfilling. All done!");
return;
}
// Show sample of albums to be updated
console.log("Sample of albums to be updated:");
albumsToBackfill.slice(0, 5).forEach((album, idx) => {
console.log(
` ${idx + 1}. "${album.title}" by ${album.artist.name}`
);
console.log(
` Current: year=${album.year}, originalYear=${album.originalYear}`
);
console.log(` Will set: originalYear=${album.year}\n`);
});
if (albumsToBackfill.length > 5) {
console.log(
` ... and ${albumsToBackfill.length - 5} more albums\n`
);
}
if (dryRun) {
console.log(
"DRY RUN: No changes made. Remove --dry-run to apply updates."
);
return;
}
// Confirm before proceeding in live mode
console.log(
`Proceeding with backfill of ${albumsToBackfill.length} albums...\n`
);
// Process in batches to avoid overwhelming the database
const BATCH_SIZE = 100;
let processed = 0;
let updated = 0;
for (let i = 0; i < albumsToBackfill.length; i += BATCH_SIZE) {
const batch = albumsToBackfill.slice(i, i + BATCH_SIZE);
// Update each album in the batch
const updatePromises = batch.map((album) =>
prisma.album.update({
where: { id: album.id },
data: { originalYear: album.year },
})
);
await Promise.all(updatePromises);
processed += batch.length;
updated += batch.length;
const progress = (
(processed / albumsToBackfill.length) *
100
).toFixed(1);
console.log(
`Progress: ${processed}/${albumsToBackfill.length} (${progress}%) albums updated`
);
}
console.log(`\n✓ Backfill complete!`);
console.log(` - Total albums updated: ${updated}`);
console.log(` - Field populated: originalYear`);
console.log(
`\nNote: Future albums will have originalYear populated automatically during enrichment.`
);
} catch (error) {
console.error("\n✗ Error during backfill:", error);
throw error;
} finally {
await prisma.$disconnect();
}
}
// Parse command line arguments
const args = process.argv.slice(2);
const dryRun = args.includes("--dry-run");
// Run the backfill
backfillOriginalYear(dryRun)
.then(() => {
process.exit(0);
})
.catch((error) => {
console.error(error);
process.exit(1);
});