The "kein Filecode — Server gab leeren Link zurueck" error was treated as a generic upload failure → after retries exhausted, the manager called mark-failed and added the account to _failedAccounts → next batch re-primed with primedFailed=1 → pre-job-swap-blocked because no fallback override exists for a single-account hoster. One server-side flake permanently poisoned the session. It's not an account problem — same account + same file works on a later try. This is a doodstream-backend processing flake (empty CDN form, no fn / no st), the same class as a transient network error: don't blacklist, just fail this file cleanly. - doodstream-upload.js: tag the empty-form throw with err.hosterTransient=true (explicit flag, primary signal — matches the err.accountError / err.fileRejected pattern already used elsewhere). - upload-manager.js: new _isHosterTransientError classifier (flag first, message regex as defensive fallback). In the retry loop: break on first hit (server flake won't clear in 3 s, re-uploading the file 4× is pure bandwidth waste). Post-loop: dedicated branch that emits the final error WITHOUT blacklisting the account — same shape as the existing transient-network branch. - Tests: classifier unit tests (flag path, regex path, negatives) + regression test that proves the account is NOT added to _failedAccounts and mark-failed does NOT fire. Drops the hoster-transient test from ~19 s to ~1.5 ms, confirming the in-loop fast-break works. We now fail fast on this error class instead of retrying — the next-batch manual retry is the recovery path, and the account stays usable for it. Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
975 lines
41 KiB
JavaScript
975 lines
41 KiB
JavaScript
const { describe, it, mock, beforeEach } = require('node:test');
|
||
const assert = require('node:assert/strict');
|
||
const path = require('path');
|
||
const { EventEmitter } = require('events');
|
||
|
||
// We need to mock fs.statSync and the hoster upload functions before requiring upload-manager
|
||
// Use node:test mock.module (available in Node 22+)
|
||
|
||
describe('UploadManager', () => {
|
||
let UploadManager;
|
||
let mockUploadFile;
|
||
let fakeFileSize;
|
||
|
||
beforeEach(() => {
|
||
fakeFileSize = 1024 * 1024; // 1 MB default
|
||
|
||
// Clear module cache for fresh mocks each test
|
||
delete require.cache[require.resolve('../lib/upload-manager')];
|
||
|
||
// Mock the hosters module
|
||
mockUploadFile = mock.fn(async (hoster, filePath, apiKey, onProgress, signal, throttle) => {
|
||
// Simulate upload progress
|
||
if (onProgress) {
|
||
onProgress(fakeFileSize / 2, fakeFileSize);
|
||
onProgress(fakeFileSize, fakeFileSize);
|
||
}
|
||
return { download_url: `https://${hoster}/test123`, embed_url: null, file_code: 'test123' };
|
||
});
|
||
|
||
// Override require for hosters
|
||
const origRequire = module.constructor.prototype.require;
|
||
const hosters = require('../lib/hosters');
|
||
hosters.uploadFile = mockUploadFile;
|
||
|
||
// Mock fs.statSync for test file paths
|
||
const fs = require('fs');
|
||
const origStatSync = fs.statSync;
|
||
fs.statSync = function(p) {
|
||
if (typeof p === 'string' && p.startsWith('/test/')) {
|
||
return { size: fakeFileSize };
|
||
}
|
||
return origStatSync.call(this, p);
|
||
};
|
||
|
||
UploadManager = require('../lib/upload-manager');
|
||
});
|
||
|
||
it('emits progress events for each task', async () => {
|
||
const mgr = new UploadManager({});
|
||
const events = [];
|
||
mgr.on('progress', (data) => events.push(data));
|
||
|
||
await mgr.startBatch([
|
||
{ file: '/test/video1.mp4', hoster: 'doodstream.com', apiKey: 'key1' }
|
||
]);
|
||
|
||
const statuses = events.map(e => e.status);
|
||
assert.ok(statuses.includes('queued'), 'should have queued status');
|
||
assert.ok(statuses.includes('done'), 'should have done status');
|
||
});
|
||
|
||
it('emits batch-done with correct summary', async () => {
|
||
const mgr = new UploadManager({});
|
||
let summary = null;
|
||
mgr.on('batch-done', (s) => { summary = s; });
|
||
|
||
await mgr.startBatch([
|
||
{ file: '/test/video1.mp4', hoster: 'doodstream.com', apiKey: 'key1' },
|
||
{ file: '/test/video2.mp4', hoster: 'doodstream.com', apiKey: 'key1' }
|
||
]);
|
||
|
||
assert.ok(summary);
|
||
assert.equal(summary.total, 2);
|
||
assert.equal(summary.succeeded, 2);
|
||
assert.equal(summary.failed, 0);
|
||
assert.equal(summary.files.length, 2);
|
||
});
|
||
|
||
it('retries on failure then succeeds', async () => {
|
||
let callCount = 0;
|
||
mockUploadFile.mock.mockImplementation(async (hoster, filePath, apiKey, onProgress) => {
|
||
callCount++;
|
||
if (callCount <= 2) throw new Error('network error');
|
||
if (onProgress) onProgress(fakeFileSize, fakeFileSize);
|
||
return { download_url: 'https://test/ok', embed_url: null, file_code: 'ok' };
|
||
});
|
||
|
||
const mgr = new UploadManager({ 'doodstream.com': { retries: 3, parallelCount: 1, maxSpeedKbs: 0, restartBelowKbs: 0, timeIntervalSec: 0, maxSizeMb: 0 } });
|
||
const statuses = [];
|
||
mgr.on('progress', (d) => statuses.push(d.status));
|
||
|
||
await mgr.startBatch([
|
||
{ file: '/test/video.mp4', hoster: 'doodstream.com', apiKey: 'key1' }
|
||
]);
|
||
|
||
assert.ok(statuses.includes('retrying'), 'should show retrying status');
|
||
assert.ok(statuses.includes('done'), 'should eventually succeed');
|
||
assert.equal(callCount, 3);
|
||
});
|
||
|
||
it('exhausted retries result in error', async () => {
|
||
mockUploadFile.mock.mockImplementation(async () => {
|
||
throw new Error('permanent failure');
|
||
});
|
||
|
||
const mgr = new UploadManager({ 'doodstream.com': { retries: 1, parallelCount: 1, maxSpeedKbs: 0, restartBelowKbs: 0, timeIntervalSec: 0, maxSizeMb: 0 } });
|
||
let summary = null;
|
||
mgr.on('batch-done', (s) => { summary = s; });
|
||
|
||
await mgr.startBatch([
|
||
{ file: '/test/video.mp4', hoster: 'doodstream.com', apiKey: 'key1' }
|
||
]);
|
||
|
||
assert.ok(summary);
|
||
assert.equal(summary.failed, 1);
|
||
assert.equal(summary.succeeded, 0);
|
||
});
|
||
|
||
it('cancel aborts running uploads', async () => {
|
||
mockUploadFile.mock.mockImplementation(async (hoster, filePath, apiKey, onProgress, signal) => {
|
||
// Simulate a slow upload
|
||
await new Promise((resolve, reject) => {
|
||
const timer = setTimeout(() => resolve({ download_url: 'x', embed_url: null, file_code: 'x' }), 10000);
|
||
if (signal) signal.addEventListener('abort', () => { clearTimeout(timer); reject(new Error('Aborted')); });
|
||
});
|
||
});
|
||
|
||
const mgr = new UploadManager({});
|
||
let batchDone = false;
|
||
mgr.on('batch-done', () => { batchDone = true; });
|
||
|
||
const batchPromise = mgr.startBatch([
|
||
{ file: '/test/video.mp4', hoster: 'doodstream.com', apiKey: 'key1' }
|
||
]);
|
||
|
||
// Wait a bit then cancel
|
||
await new Promise(r => setTimeout(r, 100));
|
||
mgr.cancel();
|
||
|
||
await batchPromise;
|
||
assert.equal(mgr.running, false);
|
||
assert.ok(batchDone, 'batch-done should be emitted even after cancel');
|
||
});
|
||
|
||
it('maxSizeMb filter skips oversized files', async () => {
|
||
fakeFileSize = 5 * 1024 * 1024; // 5 MB
|
||
|
||
const mgr = new UploadManager({ 'doodstream.com': { retries: 0, parallelCount: 1, maxSpeedKbs: 0, restartBelowKbs: 0, timeIntervalSec: 0, maxSizeMb: 1 } });
|
||
const statuses = [];
|
||
mgr.on('progress', (d) => statuses.push(d.status));
|
||
|
||
await mgr.startBatch([
|
||
{ file: '/test/big.mp4', hoster: 'doodstream.com', apiKey: 'key1' }
|
||
]);
|
||
|
||
assert.ok(statuses.includes('skipped'), 'oversized file should be skipped');
|
||
assert.ok(!statuses.includes('uploading'), 'should not attempt upload');
|
||
});
|
||
|
||
it('per-hoster semaphore limits concurrency', async () => {
|
||
let concurrent = 0;
|
||
let maxConcurrent = 0;
|
||
|
||
mockUploadFile.mock.mockImplementation(async (hoster, filePath, apiKey, onProgress, signal) => {
|
||
concurrent++;
|
||
maxConcurrent = Math.max(maxConcurrent, concurrent);
|
||
await new Promise(r => setTimeout(r, 50));
|
||
concurrent--;
|
||
if (onProgress) onProgress(fakeFileSize, fakeFileSize);
|
||
return { download_url: 'https://test/ok', embed_url: null, file_code: 'ok' };
|
||
});
|
||
|
||
const mgr = new UploadManager({ 'doodstream.com': { retries: 0, parallelCount: 1, maxSpeedKbs: 0, restartBelowKbs: 0, timeIntervalSec: 0, maxSizeMb: 0 } });
|
||
|
||
await mgr.startBatch([
|
||
{ file: '/test/a.mp4', hoster: 'doodstream.com', apiKey: 'k' },
|
||
{ file: '/test/b.mp4', hoster: 'doodstream.com', apiKey: 'k' },
|
||
{ file: '/test/c.mp4', hoster: 'doodstream.com', apiKey: 'k' }
|
||
]);
|
||
|
||
assert.equal(maxConcurrent, 1, 'should only run 1 upload at a time');
|
||
});
|
||
|
||
it('global parallel limit caps concurrency across hosters', async () => {
|
||
let concurrent = 0;
|
||
let maxConcurrent = 0;
|
||
|
||
mockUploadFile.mock.mockImplementation(async (hoster, filePath, apiKey, onProgress) => {
|
||
concurrent++;
|
||
maxConcurrent = Math.max(maxConcurrent, concurrent);
|
||
await new Promise((resolve) => setTimeout(resolve, 40));
|
||
concurrent--;
|
||
if (onProgress) onProgress(fakeFileSize, fakeFileSize);
|
||
return { download_url: `https://${hoster}/ok`, embed_url: null, file_code: 'ok' };
|
||
});
|
||
|
||
const mgr = new UploadManager({
|
||
'doodstream.com': { retries: 0, parallelCount: 5, maxSpeedKbs: 0, restartBelowKbs: 0, timeIntervalSec: 0, maxSizeMb: 0 },
|
||
'voe.sx': { retries: 0, parallelCount: 5, maxSpeedKbs: 0, restartBelowKbs: 0, timeIntervalSec: 0, maxSizeMb: 0 },
|
||
'vidmoly.me': { retries: 0, parallelCount: 5, maxSpeedKbs: 0, restartBelowKbs: 0, timeIntervalSec: 0, maxSizeMb: 0 }
|
||
}, {
|
||
parallelUploadCount: 2,
|
||
scaleParallelUploads: true
|
||
});
|
||
|
||
await mgr.startBatch([
|
||
{ jobId: 'job-1', file: '/test/a.mp4', hoster: 'doodstream.com', apiKey: 'k' },
|
||
{ jobId: 'job-2', file: '/test/b.mp4', hoster: 'voe.sx', apiKey: 'k' },
|
||
{ jobId: 'job-3', file: '/test/c.mp4', hoster: 'vidmoly.me', apiKey: 'k' }
|
||
]);
|
||
|
||
assert.equal(maxConcurrent, 2, 'should only run 2 uploads globally at once');
|
||
});
|
||
|
||
it('cancelJobs aborts a selected running upload', async () => {
|
||
mockUploadFile.mock.mockImplementation(async (hoster, filePath, apiKey, onProgress, signal) => {
|
||
await new Promise((resolve, reject) => {
|
||
const timer = setTimeout(() => resolve(), 250);
|
||
if (signal) {
|
||
signal.addEventListener('abort', () => {
|
||
clearTimeout(timer);
|
||
reject(new Error('Aborted'));
|
||
}, { once: true });
|
||
}
|
||
});
|
||
if (onProgress) onProgress(fakeFileSize, fakeFileSize);
|
||
return { download_url: 'https://test/ok', embed_url: null, file_code: 'ok' };
|
||
});
|
||
|
||
const mgr = new UploadManager({});
|
||
const statuses = [];
|
||
mgr.on('progress', (data) => statuses.push({ jobId: data.jobId, status: data.status }));
|
||
|
||
const batchPromise = mgr.startBatch([
|
||
{ jobId: 'selected-job', file: '/test/video.mp4', hoster: 'doodstream.com', apiKey: 'key1' }
|
||
]);
|
||
|
||
await new Promise((resolve) => setTimeout(resolve, 50));
|
||
mgr.cancelJobs(['selected-job']);
|
||
|
||
await batchPromise;
|
||
assert.ok(statuses.some((entry) => entry.jobId === 'selected-job' && entry.status === 'aborted'));
|
||
});
|
||
|
||
it('addJobs returns duplicate info and still runs newly queued jobs', async () => {
|
||
let releaseFirst = null;
|
||
|
||
mockUploadFile.mock.mockImplementation(async (hoster, filePath, apiKey, onProgress, signal) => {
|
||
if (filePath.endsWith('/first.mp4')) {
|
||
await new Promise((resolve, reject) => {
|
||
releaseFirst = resolve;
|
||
if (signal) {
|
||
signal.addEventListener('abort', () => reject(new Error('Aborted')), { once: true });
|
||
}
|
||
});
|
||
} else {
|
||
await new Promise((resolve) => setTimeout(resolve, 20));
|
||
}
|
||
if (onProgress) onProgress(fakeFileSize, fakeFileSize);
|
||
return { download_url: 'https://test/ok', embed_url: null, file_code: 'ok' };
|
||
});
|
||
|
||
const mgr = new UploadManager({
|
||
'doodstream.com': { retries: 0, parallelCount: 1, maxSpeedKbs: 0, restartBelowKbs: 0, timeIntervalSec: 0, maxSizeMb: 0 }
|
||
});
|
||
const statuses = [];
|
||
mgr.on('progress', (data) => statuses.push({ jobId: data.jobId, status: data.status }));
|
||
|
||
const batchPromise = mgr.startBatch([
|
||
{ jobId: 'job-first', file: '/test/first.mp4', hoster: 'doodstream.com', apiKey: 'key1' }
|
||
]);
|
||
|
||
for (let i = 0; i < 50 && !releaseFirst; i++) {
|
||
await new Promise((resolve) => setTimeout(resolve, 10));
|
||
}
|
||
assert.equal(typeof releaseFirst, 'function', 'first job should be running before addJobs');
|
||
|
||
const addResult = mgr.addJobs([
|
||
{ jobId: 'job-first', file: '/test/first.mp4', hoster: 'doodstream.com', apiKey: 'key1' },
|
||
{ jobId: 'job-second', file: '/test/second.mp4', hoster: 'doodstream.com', apiKey: 'key1' },
|
||
{ jobId: 'job-third', file: '/test/third.mp4', hoster: 'doodstream.com', apiKey: 'key1' }
|
||
]);
|
||
|
||
assert.equal(addResult.added, 2);
|
||
assert.deepEqual(addResult.alreadyInBatchJobIds, ['job-first']);
|
||
|
||
releaseFirst();
|
||
await batchPromise;
|
||
|
||
assert.ok(statuses.some((entry) => entry.jobId === 'job-second' && entry.status === 'done'));
|
||
assert.ok(statuses.some((entry) => entry.jobId === 'job-third' && entry.status === 'done'));
|
||
});
|
||
|
||
it('_combineSignals propagates abort from either source', () => {
|
||
const mgr = new UploadManager({});
|
||
const ac1 = new AbortController();
|
||
const ac2 = new AbortController();
|
||
|
||
const { signal, cleanup } = mgr._combineSignals(ac1.signal, ac2.signal);
|
||
assert.equal(signal.aborted, false);
|
||
|
||
ac2.abort();
|
||
assert.equal(signal.aborted, true);
|
||
cleanup();
|
||
});
|
||
|
||
it('_combineSignals returns aborted signal if input already aborted', () => {
|
||
const mgr = new UploadManager({});
|
||
const ac1 = new AbortController();
|
||
ac1.abort();
|
||
const ac2 = new AbortController();
|
||
|
||
const { signal, cleanup } = mgr._combineSignals(ac1.signal, ac2.signal);
|
||
assert.equal(signal.aborted, true);
|
||
cleanup();
|
||
});
|
||
|
||
it('_sleep resolves after delay', async () => {
|
||
const mgr = new UploadManager({});
|
||
const start = Date.now();
|
||
await mgr._sleep(100);
|
||
assert.ok(Date.now() - start >= 90);
|
||
});
|
||
|
||
it('_sleep rejects on abort', async () => {
|
||
const mgr = new UploadManager({});
|
||
const ac = new AbortController();
|
||
setTimeout(() => ac.abort(), 10);
|
||
await assert.rejects(mgr._sleep(5000, ac.signal), /Aborted/);
|
||
});
|
||
|
||
it('file not found produces descriptive error', async () => {
|
||
// Override fs.statSync to throw ENOENT for a specific path
|
||
const fs = require('fs');
|
||
const origStat = fs.statSync;
|
||
fs.statSync = function(p) {
|
||
if (p === '/test/deleted.mp4') throw Object.assign(new Error('ENOENT'), { code: 'ENOENT' });
|
||
return origStat.call(this, p);
|
||
};
|
||
|
||
const mgr = new UploadManager({});
|
||
const errors = [];
|
||
mgr.on('progress', (d) => { if (d.error) errors.push(d.error); });
|
||
|
||
await mgr.startBatch([
|
||
{ file: '/test/deleted.mp4', hoster: 'doodstream.com', apiKey: 'key1' }
|
||
]);
|
||
|
||
fs.statSync = origStat;
|
||
assert.ok(errors.some(e => e.includes('nicht gefunden')), `expected "nicht gefunden" error, got: ${errors.join(', ')}`);
|
||
});
|
||
|
||
it('zero-byte file produces descriptive error', async () => {
|
||
fakeFileSize = 0;
|
||
|
||
const mgr = new UploadManager({});
|
||
const errors = [];
|
||
mgr.on('progress', (d) => { if (d.error) errors.push(d.error); });
|
||
|
||
await mgr.startBatch([
|
||
{ file: '/test/empty.mp4', hoster: 'doodstream.com', apiKey: 'key1' }
|
||
]);
|
||
|
||
assert.ok(errors.some(e => e.includes('0 Bytes')), `expected "0 Bytes" error, got: ${errors.join(', ')}`);
|
||
});
|
||
|
||
it('empty batch completes immediately with zero counts', async () => {
|
||
const mgr = new UploadManager({});
|
||
let summary = null;
|
||
mgr.on('batch-done', (s) => { summary = s; });
|
||
|
||
const start = Date.now();
|
||
await mgr.startBatch([]);
|
||
const elapsed = Date.now() - start;
|
||
|
||
assert.ok(summary, 'batch-done should be emitted');
|
||
assert.equal(summary.total, 0);
|
||
assert.equal(summary.succeeded, 0);
|
||
assert.equal(summary.failed, 0);
|
||
assert.ok(elapsed < 200, `empty batch should complete fast, took ${elapsed}ms`);
|
||
});
|
||
|
||
it('scaleParallelUploads limits per-hoster count to global limit', async () => {
|
||
let concurrent = 0;
|
||
let maxConcurrent = 0;
|
||
|
||
mockUploadFile.mock.mockImplementation(async (hoster, filePath, apiKey, onProgress) => {
|
||
concurrent++;
|
||
maxConcurrent = Math.max(maxConcurrent, concurrent);
|
||
await new Promise(r => setTimeout(r, 40));
|
||
concurrent--;
|
||
if (onProgress) onProgress(fakeFileSize, fakeFileSize);
|
||
return { download_url: 'ok', embed_url: null, file_code: 'ok' };
|
||
});
|
||
|
||
const mgr = new UploadManager(
|
||
{ 'doodstream.com': { retries: 0, parallelCount: 10, maxSpeedKbs: 0, restartBelowKbs: 0, timeIntervalSec: 0, maxSizeMb: 0 } },
|
||
{ parallelUploadCount: 2, scaleParallelUploads: true }
|
||
);
|
||
|
||
await mgr.startBatch([
|
||
{ file: '/test/a.mp4', hoster: 'doodstream.com', apiKey: 'k' },
|
||
{ file: '/test/b.mp4', hoster: 'doodstream.com', apiKey: 'k' },
|
||
{ file: '/test/c.mp4', hoster: 'doodstream.com', apiKey: 'k' },
|
||
{ file: '/test/d.mp4', hoster: 'doodstream.com', apiKey: 'k' },
|
||
{ file: '/test/e.mp4', hoster: 'doodstream.com', apiKey: 'k' }
|
||
]);
|
||
|
||
assert.ok(maxConcurrent <= 2, `scaleParallelUploads should cap at 2, was ${maxConcurrent}`);
|
||
});
|
||
|
||
it('addJobs injects new tasks into running batch', async () => {
|
||
let started = 0;
|
||
mockUploadFile.mock.mockImplementation(async (hoster, filePath, apiKey, onProgress) => {
|
||
started++;
|
||
await new Promise(r => setTimeout(r, 100));
|
||
if (onProgress) onProgress(fakeFileSize, fakeFileSize);
|
||
return { download_url: 'ok', embed_url: null, file_code: 'ok' };
|
||
});
|
||
|
||
const mgr = new UploadManager({});
|
||
let summary = null;
|
||
mgr.on('batch-done', (s) => { summary = s; });
|
||
|
||
// Start batch with 2 tasks
|
||
const batchPromise = mgr.startBatch([
|
||
{ jobId: 'job-1', file: '/test/a.mp4', hoster: 'doodstream.com', apiKey: 'k' },
|
||
{ jobId: 'job-2', file: '/test/b.mp4', hoster: 'doodstream.com', apiKey: 'k' }
|
||
]);
|
||
|
||
// After 30ms (during upload), inject 2 more tasks
|
||
await new Promise(r => setTimeout(r, 30));
|
||
const result = mgr.addJobs([
|
||
{ jobId: 'job-3', file: '/test/c.mp4', hoster: 'doodstream.com', apiKey: 'k' },
|
||
{ jobId: 'job-4', file: '/test/d.mp4', hoster: 'doodstream.com', apiKey: 'k' }
|
||
]);
|
||
assert.equal(result.added, 2, 'should add 2 new jobs');
|
||
assert.equal(result.alreadyInBatchJobIds.length, 0);
|
||
|
||
await batchPromise;
|
||
assert.ok(summary);
|
||
assert.equal(started, 4, 'all 4 jobs should have run');
|
||
});
|
||
|
||
it('addJobs rejects duplicates already in running batch', async () => {
|
||
mockUploadFile.mock.mockImplementation(async (hoster, filePath, apiKey, onProgress, signal) => {
|
||
// Slow upload so we can add jobs while it's running
|
||
await new Promise((resolve, reject) => {
|
||
const timer = setTimeout(resolve, 200);
|
||
if (signal) signal.addEventListener('abort', () => { clearTimeout(timer); reject(new Error('Aborted')); });
|
||
});
|
||
if (onProgress) onProgress(fakeFileSize, fakeFileSize);
|
||
return { download_url: 'ok', embed_url: null, file_code: 'ok' };
|
||
});
|
||
|
||
const mgr = new UploadManager({});
|
||
const batchPromise = mgr.startBatch([
|
||
{ jobId: 'job-A', file: '/test/x.mp4', hoster: 'doodstream.com', apiKey: 'k' }
|
||
]);
|
||
|
||
// Try to add the SAME jobId while it's running
|
||
await new Promise(r => setTimeout(r, 50));
|
||
const result = mgr.addJobs([
|
||
{ jobId: 'job-A', file: '/test/x.mp4', hoster: 'doodstream.com', apiKey: 'k' },
|
||
{ jobId: 'job-B', file: '/test/y.mp4', hoster: 'doodstream.com', apiKey: 'k' }
|
||
]);
|
||
assert.equal(result.added, 1, 'should skip duplicate jobId, add only the new one');
|
||
assert.deepEqual(result.alreadyInBatchJobIds, ['job-A']);
|
||
|
||
await batchPromise;
|
||
});
|
||
|
||
it('addJobs returns added=0 when not running', () => {
|
||
const mgr = new UploadManager({});
|
||
const result = mgr.addJobs([
|
||
{ jobId: 'job-1', file: '/test/a.mp4', hoster: 'doodstream.com', apiKey: 'k' }
|
||
]);
|
||
assert.equal(result.added, 0);
|
||
});
|
||
|
||
it('stats event contains expected fields', async () => {
|
||
// Make upload take long enough for stats interval to fire
|
||
mockUploadFile.mock.mockImplementation(async (hoster, filePath, apiKey, onProgress, signal) => {
|
||
await new Promise(r => setTimeout(r, 1500));
|
||
if (onProgress) onProgress(fakeFileSize, fakeFileSize);
|
||
return { download_url: 'https://test/ok', embed_url: null, file_code: 'ok' };
|
||
});
|
||
|
||
const mgr = new UploadManager({});
|
||
const statsEvents = [];
|
||
mgr.on('stats', (d) => statsEvents.push(d));
|
||
|
||
await mgr.startBatch([
|
||
{ file: '/test/video.mp4', hoster: 'doodstream.com', apiKey: 'key1' }
|
||
]);
|
||
|
||
assert.ok(statsEvents.length > 0, 'should have received stats events');
|
||
const stat = statsEvents[0];
|
||
assert.ok('globalSpeedKbs' in stat);
|
||
assert.ok('totalBytes' in stat);
|
||
assert.ok('elapsed' in stat);
|
||
assert.ok('activeJobs' in stat);
|
||
});
|
||
|
||
describe('error classification', () => {
|
||
it('treats "not enough disk space" as account-level, not file-rejected', () => {
|
||
const mgr = new UploadManager({});
|
||
// Shape matches what lib/hosters.js attaches for byse account-storage-full
|
||
const err = new Error('Byse lehnte Datei ab: 0:0:0:not enough disk space on your account');
|
||
err.accountError = true;
|
||
assert.equal(mgr._isFileRejectedError(err), false,
|
||
'account-level error must NOT be classified as file-rejected');
|
||
assert.equal(mgr._shouldSkipRetryOnAccountError(err), true,
|
||
'account-storage-full must trigger account rotation');
|
||
});
|
||
|
||
it('classifies disk-space errors by message alone (safety net)', () => {
|
||
const mgr = new UploadManager({});
|
||
const err = new Error('Byse lehnte Datei ab: not enough disk space');
|
||
// No flag set — regex alone must catch it.
|
||
assert.equal(mgr._shouldSkipRetryOnAccountError(err), true);
|
||
assert.equal(mgr._isFileRejectedError(err), false,
|
||
'must not match generic "lehnte Datei ab" as file-rejected');
|
||
});
|
||
|
||
it('keeps true file rejections as file-rejected', () => {
|
||
const mgr = new UploadManager({});
|
||
const err = new Error('Byse lehnte Datei ab: Duplicate');
|
||
err.fileRejected = true;
|
||
assert.equal(mgr._isFileRejectedError(err), true);
|
||
assert.equal(mgr._shouldSkipRetryOnAccountError(err), false);
|
||
});
|
||
|
||
it('file-rejected regex still matches known phrases without flag', () => {
|
||
const mgr = new UploadManager({});
|
||
for (const msg of [
|
||
'Not video file format',
|
||
'Duplicate',
|
||
'Datei zu klein',
|
||
'File too large',
|
||
'Invalid file'
|
||
]) {
|
||
assert.equal(mgr._isFileRejectedError(new Error(msg)), true, `should match: ${msg}`);
|
||
}
|
||
});
|
||
|
||
it('accountError flag beats fileRejected if both set (defensive)', () => {
|
||
const mgr = new UploadManager({});
|
||
const err = new Error('weird');
|
||
err.fileRejected = true;
|
||
err.accountError = true;
|
||
assert.equal(mgr._isFileRejectedError(err), false,
|
||
'account-level always wins — rotation must happen');
|
||
assert.equal(mgr._shouldSkipRetryOnAccountError(err), true);
|
||
});
|
||
});
|
||
|
||
describe('session-level account memory', () => {
|
||
// Scenario: user has 2 byse accounts. Account 1 is full ("not enough
|
||
// disk space"). First job fails on acc1 → rotation to acc2. Second job
|
||
// must NOT re-probe acc1; pre-job-swap has to kick in.
|
||
it('after account is marked failed, next job swaps straight to override without retrying acc1', async () => {
|
||
// Only acc1 throws disk-space; acc2 succeeds. Mock decides by apiKey.
|
||
mockUploadFile.mock.mockImplementation(async (hoster, filePath, apiKey, onProgress) => {
|
||
if (apiKey === 'acc1-key') {
|
||
const err = new Error('Byse lehnte Datei ab: 0:0:0:not enough disk space on your account');
|
||
err.accountError = true;
|
||
throw err;
|
||
}
|
||
if (onProgress) onProgress(fakeFileSize, fakeFileSize);
|
||
return { download_url: 'https://byse.sx/ok', embed_url: null, file_code: 'ok' };
|
||
});
|
||
|
||
const mgr = new UploadManager(
|
||
{ 'byse.sx': { retries: 3, parallelCount: 1, maxSpeedKbs: 0, restartBelowKbs: 0, timeIntervalSec: 0, maxSizeMb: 0 } }
|
||
);
|
||
|
||
// Simulate main.js: on account-failed, resolve fallback → switchAccount
|
||
mgr.on('account-failed', ({ hoster, accountId }) => {
|
||
mgr.switchAccount(hoster, { id: 'acc2', username: 'u2', password: 'p2', apiKey: 'acc2-key' });
|
||
});
|
||
|
||
const rotEvents = [];
|
||
mgr.on('rot-log', (e) => rotEvents.push(e));
|
||
const progress = [];
|
||
mgr.on('progress', (d) => progress.push({ fileName: d.fileName, status: d.status, error: d.error }));
|
||
|
||
await mgr.startBatch([
|
||
{ file: '/test/a.mp4', hoster: 'byse.sx', apiKey: 'acc1-key', accountId: 'acc1', username: 'u1', password: 'p1' },
|
||
{ file: '/test/b.mp4', hoster: 'byse.sx', apiKey: 'acc1-key', accountId: 'acc1', username: 'u1', password: 'p1' }
|
||
]);
|
||
|
||
// Event sequence we expect:
|
||
// - job A: fast-fail on acc1 → mark-failed → switchAccount → rotate → upload with acc2 → done
|
||
// - job B: pre-job-swap from acc1 → acc2 (no attempts on acc1!) → done
|
||
const events = rotEvents.map(e => e.event);
|
||
assert.ok(events.includes('fast-fail'), `expected fast-fail, got: ${events.join(',')}`);
|
||
assert.ok(events.includes('mark-failed'), `expected mark-failed, got: ${events.join(',')}`);
|
||
assert.ok(events.includes('switchAccount'), `expected switchAccount, got: ${events.join(',')}`);
|
||
assert.ok(events.includes('pre-job-swap'), `expected pre-job-swap for 2nd job, got: ${events.join(',')}`);
|
||
|
||
// job B's pre-job-swap MUST predate any upload attempt for /test/b.mp4.
|
||
// If acc1 was probed for B, the mock would have thrown and we'd see
|
||
// another fast-fail or retrying event for b.mp4.
|
||
const bProgressErrors = progress
|
||
.filter(p => p.fileName && p.fileName.includes('b.mp4') && p.error)
|
||
.map(p => p.error);
|
||
assert.equal(bProgressErrors.length, 0,
|
||
`job B should never have touched acc1; got errors: ${bProgressErrors.join(' | ')}`);
|
||
|
||
// Both jobs should be done at the end.
|
||
const doneFiles = progress.filter(p => p.status === 'done').map(p => p.fileName);
|
||
assert.ok(doneFiles.some(f => f && f.includes('a.mp4')), 'a.mp4 should finish via rotation');
|
||
assert.ok(doneFiles.some(f => f && f.includes('b.mp4')), 'b.mp4 should finish via pre-job-swap');
|
||
|
||
// Sanity: mock was called with acc2-key more often than acc1-key.
|
||
const byKey = { acc1: 0, acc2: 0 };
|
||
for (const call of mockUploadFile.mock.calls) {
|
||
if (call.arguments[2] === 'acc1-key') byKey.acc1++;
|
||
else if (call.arguments[2] === 'acc2-key') byKey.acc2++;
|
||
}
|
||
assert.ok(byKey.acc1 <= 1, `acc1 should only be tried once (for job A); got ${byKey.acc1}`);
|
||
assert.ok(byKey.acc2 >= 2, `acc2 should handle both jobs after rotation; got ${byKey.acc2}`);
|
||
});
|
||
|
||
it('on fresh UploadManager (simulates app restart), failed-account memory is gone', () => {
|
||
const mgr1 = new UploadManager({});
|
||
mgr1._failedAccounts.set('byse.sx:acc1', true);
|
||
mgr1.switchAccount('byse.sx', { id: 'acc2' });
|
||
assert.equal(mgr1._failedAccounts.size, 1);
|
||
assert.equal(mgr1._accountOverrides.size, 1);
|
||
|
||
const mgr2 = new UploadManager({});
|
||
assert.equal(mgr2._failedAccounts.size, 0, 'new manager must start clean');
|
||
assert.equal(mgr2._accountOverrides.size, 0, 'override map must be empty on fresh manager');
|
||
});
|
||
|
||
it('exposes failed-account introspection (for main.js mid-batch re-resolve)', () => {
|
||
const mgr = new UploadManager({});
|
||
assert.deepEqual(mgr.getFailedAccountKeys(), []);
|
||
assert.equal(mgr.getOverride('byse.sx'), null);
|
||
|
||
mgr._failedAccounts.set('byse.sx:acc1', true);
|
||
mgr._failedAccounts.set('voe.sx:other', true);
|
||
assert.deepEqual(mgr.getFailedAccountKeys().sort(), ['byse.sx:acc1', 'voe.sx:other']);
|
||
assert.equal(mgr.getOverride('byse.sx'), null, 'no override yet');
|
||
|
||
mgr.switchAccount('byse.sx', { id: 'acc2', apiKey: 'k2' });
|
||
assert.equal(mgr.getOverride('byse.sx').id, 'acc2');
|
||
assert.equal(mgr.getOverride('voe.sx'), null, 'unrelated hoster still has no override');
|
||
});
|
||
|
||
it('startBatch primes failed-accounts + overrides — retry after batch-done skips dead account', async () => {
|
||
// acc1 fails with disk-space; acc2 succeeds.
|
||
mockUploadFile.mock.mockImplementation(async (hoster, filePath, apiKey, onProgress) => {
|
||
if (apiKey === 'acc1-key') {
|
||
const err = new Error('Byse lehnte Datei ab: not enough disk space');
|
||
err.accountError = true;
|
||
throw err;
|
||
}
|
||
if (onProgress) onProgress(fakeFileSize, fakeFileSize);
|
||
return { download_url: 'ok', embed_url: null, file_code: 'ok' };
|
||
});
|
||
|
||
const mgr = new UploadManager(
|
||
{ 'byse.sx': { retries: 3, parallelCount: 1, maxSpeedKbs: 0, restartBelowKbs: 0, timeIntervalSec: 0, maxSizeMb: 0 } }
|
||
);
|
||
const rotEvents = [];
|
||
mgr.on('rot-log', (e) => rotEvents.push(e));
|
||
|
||
// Simulate a retry-after-batch-done: main.js would pass the
|
||
// session-cached failed-accounts + overrides from the previous batch.
|
||
await mgr.startBatch([
|
||
{ file: '/test/a.mp4', hoster: 'byse.sx', apiKey: 'acc1-key', accountId: 'acc1', username: 'u1', password: 'p1' }
|
||
], {
|
||
primeFailedAccounts: ['byse.sx:acc1'],
|
||
primeOverrides: [['byse.sx', { id: 'acc2', username: 'u2', password: 'p2', apiKey: 'acc2-key' }]]
|
||
});
|
||
|
||
const events = rotEvents.map(e => e.event);
|
||
// pre-job-swap should fire on the very first attempt — no fast-fail
|
||
// because acc1 was never touched.
|
||
assert.ok(events.includes('pre-job-swap'),
|
||
`expected pre-job-swap from primed state; got: ${events.join(',')}`);
|
||
assert.ok(!events.includes('fast-fail'),
|
||
`must NOT burn a fast-fail on primed-dead acc1; got: ${events.join(',')}`);
|
||
assert.ok(!events.includes('mark-failed'),
|
||
`acc1 was already marked failed (primed); must not emit mark-failed again; got: ${events.join(',')}`);
|
||
|
||
// acc1 must not be touched at all.
|
||
const acc1Calls = mockUploadFile.mock.calls.filter(c => c.arguments[2] === 'acc1-key').length;
|
||
assert.equal(acc1Calls, 0, 'primed-dead acc1 must not receive any upload attempts');
|
||
});
|
||
|
||
it('generic error + pre-resolved override: rotates after 1 attempt (no more 5x on primary)', async () => {
|
||
// acc1 throws a generic non-transient, non-account-specific error.
|
||
// acc2 succeeds. With a pre-resolved override (from main.js at batch
|
||
// start), the retry loop must break after 1 attempt on acc1 and rotate.
|
||
let acc1Calls = 0;
|
||
let acc2Calls = 0;
|
||
mockUploadFile.mock.mockImplementation(async (hoster, filePath, apiKey, onProgress) => {
|
||
if (apiKey === 'acc1-key') {
|
||
acc1Calls++;
|
||
throw new Error('VOE Upload: irgendein generischer Fehler');
|
||
}
|
||
acc2Calls++;
|
||
if (onProgress) onProgress(fakeFileSize, fakeFileSize);
|
||
return { download_url: 'ok', embed_url: null, file_code: 'ok' };
|
||
});
|
||
|
||
const mgr = new UploadManager(
|
||
{ 'voe.sx': { retries: 5, parallelCount: 1, maxSpeedKbs: 0, restartBelowKbs: 0, timeIntervalSec: 0, maxSizeMb: 0 } }
|
||
);
|
||
const events = [];
|
||
mgr.on('rot-log', (e) => events.push(e.event));
|
||
|
||
await mgr.startBatch([
|
||
{ file: '/test/a.mp4', hoster: 'voe.sx', apiKey: 'acc1-key', accountId: 'acc1' }
|
||
], {
|
||
// Main.js pre-resolves the fallback at batch start.
|
||
primeOverrides: [['voe.sx', { id: 'acc2', apiKey: 'acc2-key' }]]
|
||
});
|
||
|
||
assert.equal(acc1Calls, 1, 'acc1 must get exactly 1 attempt before rotation kicks in');
|
||
assert.ok(acc2Calls >= 1, 'acc2 must take over');
|
||
assert.ok(events.includes('try-alternate-after-fail'),
|
||
`expected try-alternate-after-fail; got: ${events.join(',')}`);
|
||
});
|
||
|
||
it('transient error + pre-resolved override: retries SAME acc (network, not acc, is the issue)', async () => {
|
||
let acc1Calls = 0;
|
||
let acc2Calls = 0;
|
||
mockUploadFile.mock.mockImplementation(async (hoster, filePath, apiKey, onProgress) => {
|
||
if (apiKey === 'acc1-key') {
|
||
acc1Calls++;
|
||
if (acc1Calls <= 2) throw new Error('connect ECONNRESET 1.2.3.4:443');
|
||
if (onProgress) onProgress(fakeFileSize, fakeFileSize);
|
||
return { download_url: 'ok-on-acc1-retry', embed_url: null, file_code: 'ok' };
|
||
}
|
||
acc2Calls++;
|
||
if (onProgress) onProgress(fakeFileSize, fakeFileSize);
|
||
return { download_url: 'ok', embed_url: null, file_code: 'ok' };
|
||
});
|
||
|
||
const mgr = new UploadManager(
|
||
{ 'voe.sx': { retries: 5, parallelCount: 1, maxSpeedKbs: 0, restartBelowKbs: 0, timeIntervalSec: 0, maxSizeMb: 0 } }
|
||
);
|
||
const events = [];
|
||
mgr.on('rot-log', (e) => events.push(e.event));
|
||
|
||
await mgr.startBatch([
|
||
{ file: '/test/a.mp4', hoster: 'voe.sx', apiKey: 'acc1-key', accountId: 'acc1' }
|
||
], {
|
||
primeOverrides: [['voe.sx', { id: 'acc2', apiKey: 'acc2-key' }]]
|
||
});
|
||
|
||
assert.equal(acc1Calls, 3, 'transient must retry same acc until success');
|
||
assert.equal(acc2Calls, 0, 'must NOT rotate away on transient network errors');
|
||
assert.ok(!events.includes('try-alternate-after-fail'),
|
||
`transient should NOT trigger try-alternate-after-fail; got: ${events.join(',')}`);
|
||
});
|
||
|
||
it('generic error + NO override: falls back to classic retry on same account', async () => {
|
||
let acc1Calls = 0;
|
||
mockUploadFile.mock.mockImplementation(async (hoster, filePath, apiKey, onProgress) => {
|
||
acc1Calls++;
|
||
throw new Error('something generic');
|
||
});
|
||
|
||
const mgr = new UploadManager(
|
||
{ 'voe.sx': { retries: 3, parallelCount: 1, maxSpeedKbs: 0, restartBelowKbs: 0, timeIntervalSec: 0, maxSizeMb: 0 } }
|
||
);
|
||
|
||
await mgr.startBatch([
|
||
{ file: '/test/a.mp4', hoster: 'voe.sx', apiKey: 'acc1-key', accountId: 'acc1' }
|
||
]);
|
||
|
||
// retries=3 → maxAttempts=4 (retries + 1). Without an override to rotate
|
||
// to, must exhaust all 4 attempts on acc1.
|
||
assert.equal(acc1Calls, 4, 'single-account hoster must retry N+1 times on same account');
|
||
});
|
||
|
||
it('startBatch without prime opts still clears state (back-compat)', async () => {
|
||
const mgr = new UploadManager({});
|
||
mgr._failedAccounts.set('byse.sx:acc1', true);
|
||
mgr._accountOverrides.set('byse.sx', { id: 'leftover' });
|
||
|
||
mockUploadFile.mock.mockImplementation(async (hoster, filePath, apiKey, onProgress) => {
|
||
if (onProgress) onProgress(fakeFileSize, fakeFileSize);
|
||
return { download_url: 'ok', embed_url: null, file_code: 'ok' };
|
||
});
|
||
|
||
await mgr.startBatch([
|
||
{ file: '/test/a.mp4', hoster: 'doodstream.com', apiKey: 'k' }
|
||
]);
|
||
|
||
assert.equal(mgr._failedAccounts.size, 0, 'legacy callers still get a clean slate');
|
||
assert.equal(mgr._accountOverrides.size, 0, 'legacy callers still get a clean slate for overrides');
|
||
});
|
||
|
||
it('transient network errors skip rotation (account stays fine)', () => {
|
||
const mgr = new UploadManager({});
|
||
const cases = [
|
||
'getaddrinfo ENOTFOUND api.byse.sx',
|
||
'connect ECONNRESET 104.18.10.10:443',
|
||
'connect ETIMEDOUT 1.2.3.4:443',
|
||
'socket hang up',
|
||
'request to https://voe.sx failed, reason: getaddrinfo EAI_AGAIN',
|
||
'fetch failed',
|
||
'connect ECONNREFUSED 127.0.0.1:443',
|
||
'network error'
|
||
];
|
||
for (const msg of cases) {
|
||
const err = new Error(msg);
|
||
assert.equal(mgr._isTransientNetworkError(err), true, `should mark transient: ${msg}`);
|
||
assert.equal(mgr._isFileRejectedError(err), false, `transient must NOT be file-rejected: ${msg}`);
|
||
assert.equal(mgr._shouldSkipRetryOnAccountError(err), false, `transient must NOT be account-specific: ${msg}`);
|
||
}
|
||
});
|
||
|
||
it('transient classification does not swallow real account failures', () => {
|
||
const mgr = new UploadManager({});
|
||
const notTransient = [
|
||
'HTTP 429 Too Many Requests',
|
||
'quota exceeded',
|
||
'account suspended',
|
||
'Byse lehnte Datei ab: Duplicate',
|
||
'Falscher Passwort',
|
||
'Session expired'
|
||
];
|
||
for (const msg of notTransient) {
|
||
const err = new Error(msg);
|
||
assert.equal(mgr._isTransientNetworkError(err), false,
|
||
`must NOT be transient: "${msg}"`);
|
||
}
|
||
});
|
||
|
||
it('hoster-transient flag is recognised (primary path)', () => {
|
||
const mgr = new UploadManager({});
|
||
const err = new Error('whatever');
|
||
err.hosterTransient = true;
|
||
assert.equal(mgr._isHosterTransientError(err), true);
|
||
// Must not be confused with other classes.
|
||
assert.equal(mgr._isFileRejectedError(err), false);
|
||
assert.equal(mgr._isTransientNetworkError(err), false);
|
||
assert.equal(mgr._shouldSkipRetryOnAccountError(err), false);
|
||
});
|
||
|
||
it('hoster-transient regex fallback catches wrapped doodstream empty-form errors', () => {
|
||
const mgr = new UploadManager({});
|
||
const cases = [
|
||
'Doodstream Upload: kein Filecode — Server gab leeren Link zurueck (st=?, fn=fehlt/leer ...)',
|
||
'wrapper: Server gab leeren Link zurueck while parsing'
|
||
];
|
||
for (const msg of cases) {
|
||
assert.equal(mgr._isHosterTransientError(new Error(msg)), true, `should match: ${msg}`);
|
||
}
|
||
// Plain network and account errors must NOT match the hoster-transient class.
|
||
const negatives = [
|
||
'fetch failed',
|
||
'getaddrinfo ENOTFOUND',
|
||
'HTTP 429',
|
||
'quota exceeded',
|
||
'Byse lehnte Datei ab: Duplicate'
|
||
];
|
||
for (const msg of negatives) {
|
||
assert.equal(mgr._isHosterTransientError(new Error(msg)), false, `must NOT match: ${msg}`);
|
||
}
|
||
});
|
||
|
||
it('regression: hoster-transient does NOT blacklist the account (account stays usable across batches)', async () => {
|
||
// Simulate doodstream-upload throwing the tagged empty-form error.
|
||
mockUploadFile.mock.mockImplementation(async () => {
|
||
const err = new Error('Doodstream Upload: kein Filecode — Server gab leeren Link zurueck (st=?, fn=fehlt/leer ...)');
|
||
err.hosterTransient = true;
|
||
throw err;
|
||
});
|
||
|
||
const mgr = new UploadManager(
|
||
{ 'doodstream.com': { retries: 3, parallelCount: 1, maxSpeedKbs: 0, restartBelowKbs: 0, timeIntervalSec: 0, maxSizeMb: 0 } }
|
||
);
|
||
const rotEvents = [];
|
||
mgr.on('rot-log', (e) => rotEvents.push(e));
|
||
|
||
// No username/password so the manager routes through the mocked
|
||
// hosters.uploadFile (instead of DoodstreamUploader directly).
|
||
await mgr.startBatch([
|
||
{ file: '/test/Arrested.Development.mkv', hoster: 'doodstream.com', apiKey: 'acc1-key', accountId: 'acc1' }
|
||
]);
|
||
|
||
const events = rotEvents.map(e => e.event);
|
||
// Must NOT poison the account — that's the entire point of this fix.
|
||
assert.equal(mgr._failedAccounts.size, 0, `account must NOT be blacklisted; _failedAccounts=${JSON.stringify(mgr.getFailedAccountKeys())}`);
|
||
assert.ok(!events.includes('mark-failed'), `must NOT mark-failed for hoster-transient; got: ${events.join(',')}`);
|
||
// The in-loop fast-break and the post-loop classification must both fire.
|
||
assert.ok(events.includes('hoster-transient'),
|
||
`expected hoster-transient (in-loop break, no wasted retries); got: ${events.join(',')}`);
|
||
assert.ok(events.includes('skip-rotation-hoster-transient'),
|
||
`expected skip-rotation-hoster-transient (post-loop branch); got: ${events.join(',')}`);
|
||
// And the retry loop must NOT burn the full retries=3 -> only 1 attempt on this account.
|
||
assert.equal(mockUploadFile.mock.calls.length, 1,
|
||
`must fail fast on hoster-transient, not re-upload the file 4× wasting bandwidth; got ${mockUploadFile.mock.calls.length} calls`);
|
||
});
|
||
|
||
it('late-resolved override is honored by subsequent jobs (simulates mid-batch config add)', async () => {
|
||
// Only acc1 throws; acc2 succeeds.
|
||
mockUploadFile.mock.mockImplementation(async (hoster, filePath, apiKey, onProgress) => {
|
||
if (apiKey === 'acc1-key') {
|
||
const err = new Error('Byse lehnte Datei ab: not enough disk space');
|
||
err.accountError = true;
|
||
throw err;
|
||
}
|
||
if (onProgress) onProgress(fakeFileSize, fakeFileSize);
|
||
return { download_url: 'ok', embed_url: null, file_code: 'ok' };
|
||
});
|
||
|
||
const mgr = new UploadManager(
|
||
{ 'byse.sx': { retries: 1, parallelCount: 1, maxSpeedKbs: 0, restartBelowKbs: 0, timeIntervalSec: 0, maxSizeMb: 0 } }
|
||
);
|
||
|
||
// Scenario: initially config has ONLY acc1. No account-failed listener
|
||
// resolves a fallback (because none exists in config yet). Job A fails
|
||
// with rotation-end.
|
||
const rotEvents = [];
|
||
mgr.on('rot-log', (e) => rotEvents.push(e));
|
||
|
||
await mgr.startBatch([
|
||
{ file: '/test/a.mp4', hoster: 'byse.sx', apiKey: 'acc1-key', accountId: 'acc1', username: 'u1', password: 'p1' }
|
||
]);
|
||
|
||
// Job A should have ended with rotation-end (no fallback available).
|
||
const eventsA = rotEvents.map(e => e.event);
|
||
assert.ok(eventsA.includes('mark-failed'), 'acc1 must be marked failed');
|
||
assert.ok(eventsA.includes('rotation-end'), 'expected rotation-end without a fallback');
|
||
assert.equal(mgr.getFailedAccountKeys().length, 1);
|
||
assert.equal(mgr.getOverride('byse.sx'), null, 'no override set during first batch');
|
||
|
||
// --- Simulate: user adds acc2 in Settings → save-config handler finds
|
||
// that byse.sx:acc1 is failed without an override → resolves + switches.
|
||
mgr.switchAccount('byse.sx', { id: 'acc2', username: 'u2', password: 'p2', apiKey: 'acc2-key' });
|
||
|
||
// Now a follow-up batch (same running session — in production, addJobs
|
||
// or a new startBatch without clearing maps would reach this state).
|
||
// We need to ALSO clear _failedAccounts manually here because startBatch
|
||
// resets it — so we poke the inner state to emulate "still mid-batch
|
||
// with late config". The switchAccount-after-fail path is what matters.
|
||
rotEvents.length = 0;
|
||
// Re-run just the _runJob path by manually setting up state and using
|
||
// addJobs — simulates mid-batch job add after config change.
|
||
mgr.running = true;
|
||
mgr._batchResults = new Map();
|
||
mgr._batchResults.set('/test/b.mp4', { name: 'b.mp4', size: fakeFileSize, results: [] });
|
||
mgr._failedAccounts.set('byse.sx:acc1', true); // re-establish failed state
|
||
mgr._additionalPromises = [];
|
||
|
||
// Spawn a new job through addJobs() path (uses _runJob internally)
|
||
const addResult = await mgr.addJobs([
|
||
{ file: '/test/b.mp4', hoster: 'byse.sx', apiKey: 'acc1-key', accountId: 'acc1', username: 'u1', password: 'p1', jobId: 'jb' }
|
||
]);
|
||
assert.ok(addResult.added >= 1 || addResult.alreadyInBatch === 0,
|
||
`addJobs should accept new job: ${JSON.stringify(addResult)}`);
|
||
await Promise.allSettled(mgr._additionalPromises);
|
||
|
||
const eventsB = rotEvents.map(e => e.event);
|
||
assert.ok(eventsB.includes('pre-job-swap'),
|
||
`job B should have pre-job-swap after late override was set; got: ${eventsB.join(',')}`);
|
||
|
||
// Mock must have been called with acc2-key for the new job (not acc1-key again)
|
||
const acc1ForB = mockUploadFile.mock.calls.filter(c =>
|
||
c.arguments[1] === '/test/b.mp4' && c.arguments[2] === 'acc1-key').length;
|
||
assert.equal(acc1ForB, 0, 'job B must never touch acc1-key after late fallback was set');
|
||
});
|
||
});
|
||
});
|