Compare commits
22 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| b40307a421 | |||
| 97e2f256e6 | |||
| d29b6e9223 | |||
| 7f7a91f056 | |||
| 879aee8eea | |||
| f8f693e32a | |||
| c18f5bd26e | |||
| e726fbe33f | |||
| 4107bdf611 | |||
| 52806d00dc | |||
| dcb77dd27f | |||
| 6c3ffb9b77 | |||
| 7226404221 | |||
| 0eaa54cf4a | |||
| fd0dc7528d | |||
| c4e584cc3b | |||
| 610632c4f0 | |||
| 1535a5725a | |||
| 8fb00843ef | |||
| 6f6aa5b967 | |||
| fb68bddedb | |||
| 7d7304637c |
+1
-15
@@ -2,21 +2,7 @@
|
|||||||
|
|
||||||
All notable changes to this project will be documented in this file.
|
All notable changes to this project will be documented in this file.
|
||||||
Format follows [Keep a Changelog](https://keepachangelog.com/en/1.0.0/).
|
Format follows [Keep a Changelog](https://keepachangelog.com/en/1.0.0/).
|
||||||
This project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
This project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).## [1.7.36] - 2026-05-29
|
||||||
|
|
||||||
## [1.7.38] - 2026-05-29
|
|
||||||
|
|
||||||
### Fixed
|
|
||||||
|
|
||||||
- **SABnzbd History Legacy Slot Name Compatibility (Issue #74)** — Hardened SABnzbd active-download and history slot title matching in `DownloadMatcher.js` to support all slot name property variations (`filename`, `nzbname`, `name`, `nzb_name`). This ensures history matching succeeds against cached/legacy data schemas where the name is stored solely under the `filename` property, preventing completed downloads awaiting import from incorrectly displaying as `"Unknown"` client cards. Added unit tests for legacy `filename` slot matching compatibility.
|
|
||||||
|
|
||||||
## [1.7.37] - 2026-05-29
|
|
||||||
|
|
||||||
### Fixed
|
|
||||||
|
|
||||||
- **SABnzbd History Matching Symmetry (Issue #74)** — Consolidated SABnzbd active-download matching algorithms in `DownloadMatcher.js` by introducing a unified, type-safe internal helper `findSabMatch(sabDownloadId, nzbName, context, caller)`. Refactored `matchSabSlots` and `matchSabHistory` to route entirely through `findSabMatch`. This resolves a bug where completed SABnzbd downloads awaiting manual import in Sonarr or Radarr queues were incorrectly flagged as "unknown" client/"Orphaned (unconfigured client)". Added detailed unit tests to safeguard this behavior.
|
|
||||||
|
|
||||||
## [1.7.36] - 2026-05-29
|
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
|
||||||
|
|||||||
Generated
+2
-2
@@ -1,12 +1,12 @@
|
|||||||
{
|
{
|
||||||
"name": "sofarr",
|
"name": "sofarr",
|
||||||
"version": "1.7.38",
|
"version": "1.7.36",
|
||||||
"lockfileVersion": 3,
|
"lockfileVersion": 3,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"packages": {
|
"packages": {
|
||||||
"": {
|
"": {
|
||||||
"name": "sofarr",
|
"name": "sofarr",
|
||||||
"version": "1.7.38",
|
"version": "1.7.36",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"axios": "^1.6.0",
|
"axios": "^1.6.0",
|
||||||
|
|||||||
+1
-1
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "sofarr",
|
"name": "sofarr",
|
||||||
"version": "1.7.38",
|
"version": "1.7.36",
|
||||||
"description": "A personal media download dashboard that shows your downloads 'so far' while you relax on the sofa waiting for your *arr services to finish",
|
"description": "A personal media download dashboard that shows your downloads 'so far' while you relax on the sofa waiting for your *arr services to finish",
|
||||||
"main": "server/index.js",
|
"main": "server/index.js",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
|
|||||||
+1
-1
@@ -133,7 +133,7 @@ function createApp({ skipRateLimits = false } = {}) {
|
|||||||
* version:
|
* version:
|
||||||
* type: string
|
* type: string
|
||||||
* description: sofarr version
|
* description: sofarr version
|
||||||
* example: "1.7.38"
|
* example: "1.7.36"
|
||||||
* x-code-samples:
|
* x-code-samples:
|
||||||
* - lang: curl
|
* - lang: curl
|
||||||
* label: cURL
|
* label: cURL
|
||||||
|
|||||||
+1
-1
@@ -22,7 +22,7 @@ info:
|
|||||||
|
|
||||||
## SSE Streaming
|
## SSE Streaming
|
||||||
Real-time updates are available via Server-Sent Events at GET /api/dashboard/stream.
|
Real-time updates are available via Server-Sent Events at GET /api/dashboard/stream.
|
||||||
version: 1.7.38
|
version: 1.7.36
|
||||||
contact:
|
contact:
|
||||||
name: sofarr
|
name: sofarr
|
||||||
license:
|
license:
|
||||||
|
|||||||
@@ -53,65 +53,6 @@ function titleMatches(clientName, arrTitle, { isFallback = true, caller = 'Downl
|
|||||||
return matched;
|
return matched;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Internal helper: Finds the best matching Sonarr or Radarr record for a SABnzbd slot.
|
|
||||||
* Performs robust case-insensitive downloadId matching (queue → history),
|
|
||||||
* then bidirectional title fallback (queue → history).
|
|
||||||
* This eliminates all duplication and asymmetry between matchSabSlots and matchSabHistory.
|
|
||||||
*
|
|
||||||
* @param {string|null} sabDownloadId
|
|
||||||
* @param {string} nzbName
|
|
||||||
* @param {Object} context
|
|
||||||
* @param {string} caller - e.g. 'matchSabHistory' or 'matchSabSlots'
|
|
||||||
* @returns {{ sonarrMatch: Object|null, radarrMatch: Object|null }}
|
|
||||||
*/
|
|
||||||
function findSabMatch(sabDownloadId, nzbName, context, caller = 'DownloadMatcher') {
|
|
||||||
const {
|
|
||||||
sonarrQueueRecords = [],
|
|
||||||
sonarrHistoryRecords = [],
|
|
||||||
radarrQueueRecords = [],
|
|
||||||
radarrHistoryRecords = []
|
|
||||||
} = context;
|
|
||||||
|
|
||||||
const findBest = (queueRecords, historyRecords) => {
|
|
||||||
// 1. Robust ID match (queue first)
|
|
||||||
let match = sabDownloadId
|
|
||||||
? queueRecords.find(r => {
|
|
||||||
const dl = r && r.downloadId;
|
|
||||||
return dl && String(dl).toLowerCase() === String(sabDownloadId).toLowerCase();
|
|
||||||
})
|
|
||||||
: null;
|
|
||||||
|
|
||||||
if (!match && sabDownloadId) {
|
|
||||||
match = historyRecords.find(r => {
|
|
||||||
const dl = r && r.downloadId;
|
|
||||||
return dl && String(dl).toLowerCase() === String(sabDownloadId).toLowerCase();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
// 2. Title fallback (queue first, then history)
|
|
||||||
if (!match && nzbName) {
|
|
||||||
match = queueRecords.find(r => {
|
|
||||||
const rTitle = r && (r.title || r.sourceTitle);
|
|
||||||
return rTitle && titleMatches(nzbName, rTitle, { isFallback: true, caller });
|
|
||||||
});
|
|
||||||
}
|
|
||||||
if (!match && nzbName) {
|
|
||||||
match = historyRecords.find(r => {
|
|
||||||
const rTitle = r && (r.title || r.sourceTitle);
|
|
||||||
return rTitle && titleMatches(nzbName, rTitle, { isFallback: true, caller });
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
return match || null;
|
|
||||||
};
|
|
||||||
|
|
||||||
return {
|
|
||||||
sonarrMatch: findBest(sonarrQueueRecords, sonarrHistoryRecords),
|
|
||||||
radarrMatch: findBest(radarrQueueRecords, radarrHistoryRecords)
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* All callers (matchers + orphan path) must supply client, instanceId, and instanceName via options.
|
* All callers (matchers + orphan path) must supply client, instanceId, and instanceName via options.
|
||||||
* Defaults exist only as a last-resort safety net.
|
* Defaults exist only as a last-resort safety net.
|
||||||
@@ -290,14 +231,52 @@ async function matchSabSlots(slots, context) {
|
|||||||
|
|
||||||
const matched = [];
|
const matched = [];
|
||||||
for (const slot of slots) {
|
for (const slot of slots) {
|
||||||
const nzbName = slot.filename || slot.nzbname || slot.name || slot.nzb_name;
|
const nzbName = slot.filename || slot.nzbname;
|
||||||
if (!nzbName) continue;
|
if (!nzbName) continue;
|
||||||
|
|
||||||
const slotState = getSlotStatusAndSpeed(slot, queueStatus, queueSpeed, queueKbpersec);
|
const slotState = getSlotStatusAndSpeed(slot, queueStatus, queueSpeed, queueKbpersec);
|
||||||
const nzbNameLower = nzbName.toLowerCase();
|
const nzbNameLower = nzbName.toLowerCase();
|
||||||
|
|
||||||
|
// Try to match by downloadId first (most reliable)
|
||||||
const sabDownloadId = slot.nzo_id || slot.id;
|
const sabDownloadId = slot.nzo_id || slot.id;
|
||||||
const { sonarrMatch, radarrMatch } = findSabMatch(sabDownloadId, nzbName, context, 'matchSabSlots');
|
let sonarrMatch = sabDownloadId ? sonarrQueueRecords.find(r => r.downloadId === sabDownloadId) : null;
|
||||||
|
let radarrMatch = sabDownloadId ? radarrQueueRecords.find(r => r.downloadId === sabDownloadId) : null;
|
||||||
|
|
||||||
|
// Also check HISTORY by downloadId
|
||||||
|
if (!sonarrMatch && sabDownloadId) {
|
||||||
|
sonarrMatch = sonarrHistoryRecords.find(r => r.downloadId === sabDownloadId);
|
||||||
|
}
|
||||||
|
if (!radarrMatch && sabDownloadId) {
|
||||||
|
radarrMatch = radarrHistoryRecords.find(r => r.downloadId === sabDownloadId);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fallback: Check by title matching
|
||||||
|
if (!sonarrMatch) {
|
||||||
|
sonarrMatch = sonarrQueueRecords.find(r => {
|
||||||
|
const rTitle = r.title || r.sourceTitle;
|
||||||
|
return rTitle && titleMatches(nzbName, rTitle, { isFallback: true, caller: 'matchSabSlots' });
|
||||||
|
});
|
||||||
|
}
|
||||||
|
if (!radarrMatch) {
|
||||||
|
radarrMatch = radarrQueueRecords.find(r => {
|
||||||
|
const rTitle = r.title || r.sourceTitle;
|
||||||
|
return rTitle && titleMatches(nzbName, rTitle, { isFallback: true, caller: 'matchSabSlots' });
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Also check HISTORY (completed downloads) if no queue match
|
||||||
|
if (!sonarrMatch) {
|
||||||
|
sonarrMatch = sonarrHistoryRecords.find(r => {
|
||||||
|
const rTitle = r.title || r.sourceTitle;
|
||||||
|
return rTitle && titleMatches(nzbName, rTitle, { isFallback: true, caller: 'matchSabSlots' });
|
||||||
|
});
|
||||||
|
}
|
||||||
|
if (!radarrMatch) {
|
||||||
|
radarrMatch = radarrHistoryRecords.find(r => {
|
||||||
|
const rTitle = r.title || r.sourceTitle;
|
||||||
|
return rTitle && titleMatches(nzbName, rTitle, { isFallback: true, caller: 'matchSabSlots' });
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
// Progress calculation
|
// Progress calculation
|
||||||
const mbValue = slot.mb !== undefined && slot.mb !== null ? parseFloat(slot.mb) : 0;
|
const mbValue = slot.mb !== undefined && slot.mb !== null ? parseFloat(slot.mb) : 0;
|
||||||
@@ -343,20 +322,63 @@ async function matchSabSlots(slots, context) {
|
|||||||
return matched;
|
return matched;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Matches SABnzbd history slots to Sonarr/Radarr activity using title matching.
|
||||||
|
* @param {Array} slots - SABnzbd history slots
|
||||||
|
* @param {Object} context - Matching context with records, maps, and user info
|
||||||
|
* @returns {Array} Array of matched download objects
|
||||||
|
*/
|
||||||
async function matchSabHistory(slots, context) {
|
async function matchSabHistory(slots, context) {
|
||||||
|
const {
|
||||||
|
sonarrQueueRecords,
|
||||||
|
sonarrHistoryRecords,
|
||||||
|
radarrQueueRecords,
|
||||||
|
radarrHistoryRecords
|
||||||
|
} = context;
|
||||||
|
|
||||||
const matched = [];
|
const matched = [];
|
||||||
|
|
||||||
for (const slot of slots) {
|
for (const slot of slots) {
|
||||||
const nzbName = slot.filename || slot.nzbname || slot.name || slot.nzb_name;
|
const nzbName = slot.name || slot.nzb_name || slot.nzbname;
|
||||||
if (!nzbName) continue;
|
if (!nzbName) continue;
|
||||||
|
const nzbNameLower = nzbName.toLowerCase();
|
||||||
|
|
||||||
|
// Try to match by downloadId (nzo_id or slot ID) first (most reliable)
|
||||||
const sabDownloadId = slot.nzo_id || slot.id;
|
const sabDownloadId = slot.nzo_id || slot.id;
|
||||||
const { sonarrMatch, radarrMatch } = findSabMatch(sabDownloadId, nzbName, context, 'matchSabHistory');
|
const matchesSabId = (r) => {
|
||||||
|
const dl = r && r.downloadId;
|
||||||
|
if (!dl || !sabDownloadId) return false;
|
||||||
|
return String(dl).toLowerCase() === String(sabDownloadId).toLowerCase();
|
||||||
|
};
|
||||||
|
|
||||||
|
let sonarrMatch = sabDownloadId ? sonarrHistoryRecords.find(matchesSabId) : null;
|
||||||
|
let radarrMatch = sabDownloadId ? radarrHistoryRecords.find(matchesSabId) : null;
|
||||||
|
|
||||||
|
// Dual-lookup: also try against active queue records (history slot may still be in *arr queue)
|
||||||
|
if (!sonarrMatch && sabDownloadId) {
|
||||||
|
sonarrMatch = sonarrQueueRecords.find(matchesSabId);
|
||||||
|
}
|
||||||
|
if (!radarrMatch && sabDownloadId) {
|
||||||
|
radarrMatch = radarrQueueRecords.find(matchesSabId);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fallback: Check by title matching
|
||||||
|
if (!sonarrMatch) {
|
||||||
|
sonarrMatch = sonarrHistoryRecords.find(r => {
|
||||||
|
const rTitle = r.title || r.sourceTitle;
|
||||||
|
return rTitle && titleMatches(nzbName, rTitle, { isFallback: true, caller: 'matchSabHistory' });
|
||||||
|
});
|
||||||
|
}
|
||||||
|
if (!radarrMatch) {
|
||||||
|
radarrMatch = radarrHistoryRecords.find(r => {
|
||||||
|
const rTitle = r.title || r.sourceTitle;
|
||||||
|
return rTitle && titleMatches(nzbName, rTitle, { isFallback: true, caller: 'matchSabHistory' });
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
const commonOptions = {
|
const commonOptions = {
|
||||||
title: nzbName,
|
title: nzbName,
|
||||||
status: slot.status || 'Completed',
|
status: slot.status || 'Completed',
|
||||||
progress: 100,
|
progress: 100, // History items are completed
|
||||||
mb: slot.mb,
|
mb: slot.mb,
|
||||||
size: Math.round((slot.mb || 0) * 1024 * 1024),
|
size: Math.round((slot.mb || 0) * 1024 * 1024),
|
||||||
completedAt: slot.completed_time,
|
completedAt: slot.completed_time,
|
||||||
@@ -370,7 +392,7 @@ async function matchSabHistory(slots, context) {
|
|||||||
const dlObj = buildArrDownload(sonarrMatch, context, {
|
const dlObj = buildArrDownload(sonarrMatch, context, {
|
||||||
...commonOptions,
|
...commonOptions,
|
||||||
arrType: 'sonarr',
|
arrType: 'sonarr',
|
||||||
episodes: DownloadAssembler.gatherEpisodes(nzbName.toLowerCase(), context.sonarrHistoryRecords || [])
|
episodes: DownloadAssembler.gatherEpisodes(nzbNameLower, sonarrHistoryRecords)
|
||||||
});
|
});
|
||||||
if (dlObj) matched.push(dlObj);
|
if (dlObj) matched.push(dlObj);
|
||||||
}
|
}
|
||||||
@@ -383,7 +405,6 @@ async function matchSabHistory(slots, context) {
|
|||||||
if (dlObj) matched.push(dlObj);
|
if (dlObj) matched.push(dlObj);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return matched;
|
return matched;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -226,61 +226,6 @@ describe('DownloadMatcher', () => {
|
|||||||
expect(result).toHaveLength(1);
|
expect(result).toHaveLength(1);
|
||||||
expect(result[0].arrQueueId).toBe(101);
|
expect(result[0].arrQueueId).toBe(101);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('falls back to title matching against Sonarr queue records when downloadId is absent/unmatched', async () => {
|
|
||||||
const testContext = {
|
|
||||||
...context,
|
|
||||||
sonarrHistoryRecords: [],
|
|
||||||
sonarrQueueRecords: [
|
|
||||||
{ id: 201, seriesId: 1, title: 'My.Cool.Show.S01E01.720p-Group' }
|
|
||||||
],
|
|
||||||
seriesMap: new Map([[1, { id: 1, title: 'My Cool Show', tags: [1] }]])
|
|
||||||
};
|
|
||||||
|
|
||||||
const slots = [{ id: null, name: 'My_Cool_Show_S01E01_720p-Group.nzb', status: 'Completed', mb: 1000 }];
|
|
||||||
const result = await DownloadMatcher.matchSabHistory(slots, testContext);
|
|
||||||
|
|
||||||
expect(result).toHaveLength(1);
|
|
||||||
expect(result[0].arrQueueId).toBe(201);
|
|
||||||
expect(result[0].arrType).toBe('sonarr');
|
|
||||||
});
|
|
||||||
|
|
||||||
it('falls back to title matching against Radarr queue records when downloadId is absent/unmatched', async () => {
|
|
||||||
const testContext = {
|
|
||||||
...context,
|
|
||||||
radarrHistoryRecords: [],
|
|
||||||
radarrQueueRecords: [
|
|
||||||
{ id: 301, movieId: 2, title: 'Awesome Movie 2026 1080p' }
|
|
||||||
],
|
|
||||||
moviesMap: new Map([[2, { id: 2, title: 'Awesome Movie', tags: [1] }]]),
|
|
||||||
radarrTagMap: new Map([[1, 'alice']])
|
|
||||||
};
|
|
||||||
|
|
||||||
const slots = [{ id: null, name: 'Awesome.Movie.2026.1080p.nzb', status: 'Completed', mb: 1000 }];
|
|
||||||
const result = await DownloadMatcher.matchSabHistory(slots, testContext);
|
|
||||||
|
|
||||||
expect(result).toHaveLength(1);
|
|
||||||
expect(result[0].arrQueueId).toBe(301);
|
|
||||||
expect(result[0].arrType).toBe('radarr');
|
|
||||||
});
|
|
||||||
|
|
||||||
it('matches when history slots only have the filename field (cached legacy format)', async () => {
|
|
||||||
const testContext = {
|
|
||||||
...context,
|
|
||||||
sonarrHistoryRecords: [],
|
|
||||||
sonarrQueueRecords: [
|
|
||||||
{ id: 201, seriesId: 1, title: 'My.Cool.Show.S01E01.720p-Group' }
|
|
||||||
],
|
|
||||||
seriesMap: new Map([[1, { id: 1, title: 'My Cool Show', tags: [1] }]])
|
|
||||||
};
|
|
||||||
|
|
||||||
const slots = [{ id: null, filename: 'My_Cool_Show_S01E01_720p-Group.nzb', status: 'Completed', mb: 1000 }];
|
|
||||||
const result = await DownloadMatcher.matchSabHistory(slots, testContext);
|
|
||||||
|
|
||||||
expect(result).toHaveLength(1);
|
|
||||||
expect(result[0].arrQueueId).toBe(201);
|
|
||||||
expect(result[0].arrType).toBe('sonarr');
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('titleMatches helper', () => {
|
describe('titleMatches helper', () => {
|
||||||
|
|||||||
Reference in New Issue
Block a user