mirror of
https://github.com/ksyasuda/SubMiner.git
synced 2026-04-12 04:19:25 -07:00
* fix: harden preload argv parsing for popup windows * fix: align youtube playback with shared overlay startup * fix: unwrap mpv youtube streams for anki media mining * docs: update docs for youtube subtitle and mining flow * refactor: unify cli and runtime wiring for startup and youtube flow * feat: update subtitle sidebar overlay behavior * chore: add shared log-file source for diagnostics * fix(ci): add changelog fragment for immersion changes * fix: address CodeRabbit review feedback * fix: persist canonical title from youtube metadata * style: format stats library tab * fix: address latest review feedback * style: format stats library files * test: stub launcher youtube deps in CI * test: isolate launcher youtube flow deps * test: stub launcher youtube deps in failing case * test: force x11 backend in launcher ci harness * test: address latest review feedback * fix(launcher): preserve user YouTube ytdl raw options * docs(backlog): update task tracking notes * fix(immersion): special-case youtube media paths in runtime and tracking * feat(stats): improve YouTube media metadata and picker key handling * fix(ci): format stats media library hook * fix: address latest CodeRabbit review items * docs: update youtube release notes and docs * feat: auto-load youtube subtitles before manual picker * fix: restore app-owned youtube subtitle flow * docs: update youtube playback docs and config copy * refactor: remove legacy youtube launcher mode plumbing * fix: refine youtube subtitle startup binding * docs: clarify youtube subtitle startup behavior * fix: address PR #31 latest review follow-ups * fix: address PR #31 follow-up review comments * test: harden youtube picker test harness * udpate backlog * fix: add timeout to youtube metadata probe * docs: refresh youtube and stats docs * update backlog * update backlog * chore: release v0.9.0
198 lines
6.3 KiB
TypeScript
198 lines
6.3 KiB
TypeScript
import assert from 'node:assert/strict';
|
|
import test from 'node:test';
|
|
import {
|
|
createYoutubePrimarySubtitleNotificationRuntime,
|
|
type YoutubePrimarySubtitleNotificationTimer,
|
|
} from './youtube-primary-subtitle-notification';
|
|
|
|
function createTimerHarness() {
|
|
let nextId = 1;
|
|
const timers = new Map<number, () => void>();
|
|
return {
|
|
schedule: (fn: () => void): YoutubePrimarySubtitleNotificationTimer => {
|
|
const id = nextId++;
|
|
timers.set(id, fn);
|
|
return { id };
|
|
},
|
|
clear: (timer: YoutubePrimarySubtitleNotificationTimer | null) => {
|
|
if (!timer) {
|
|
return;
|
|
}
|
|
if (typeof timer === 'object' && 'id' in timer) {
|
|
timers.delete(timer.id);
|
|
}
|
|
},
|
|
runAll: () => {
|
|
const pending = [...timers.values()];
|
|
timers.clear();
|
|
for (const fn of pending) {
|
|
fn();
|
|
}
|
|
},
|
|
size: () => timers.size,
|
|
};
|
|
}
|
|
|
|
test('notifier reports missing preferred primary subtitle once for youtube media', () => {
|
|
const notifications: string[] = [];
|
|
const timers = createTimerHarness();
|
|
const runtime = createYoutubePrimarySubtitleNotificationRuntime({
|
|
getPrimarySubtitleLanguages: () => ['ja', 'jpn'],
|
|
notifyFailure: (message) => {
|
|
notifications.push(message);
|
|
},
|
|
schedule: (fn) => timers.schedule(fn),
|
|
clearSchedule: (timer) => timers.clear(timer),
|
|
});
|
|
|
|
runtime.handleMediaPathChange('https://www.youtube.com/watch?v=abc');
|
|
runtime.handleSubtitleTrackChange(null);
|
|
runtime.handleSubtitleTrackListChange([
|
|
{ type: 'sub', id: 2, lang: 'en', title: 'English', external: true },
|
|
]);
|
|
|
|
assert.equal(timers.size(), 1);
|
|
timers.runAll();
|
|
timers.runAll();
|
|
|
|
assert.deepEqual(notifications, [
|
|
'Primary subtitle failed to download or load. Try again from the subtitle modal.',
|
|
]);
|
|
});
|
|
|
|
test('notifier suppresses failure when preferred primary subtitle is selected', () => {
|
|
const notifications: string[] = [];
|
|
const timers = createTimerHarness();
|
|
const runtime = createYoutubePrimarySubtitleNotificationRuntime({
|
|
getPrimarySubtitleLanguages: () => ['ja', 'jpn'],
|
|
notifyFailure: (message) => {
|
|
notifications.push(message);
|
|
},
|
|
schedule: (fn) => timers.schedule(fn),
|
|
clearSchedule: (timer) => timers.clear(timer),
|
|
});
|
|
|
|
runtime.handleMediaPathChange('https://www.youtube.com/watch?v=abc');
|
|
runtime.handleSubtitleTrackListChange([
|
|
{ type: 'sub', id: 5, lang: 'ja', title: 'Japanese', external: true },
|
|
]);
|
|
runtime.handleSubtitleTrackChange(5);
|
|
timers.runAll();
|
|
|
|
assert.deepEqual(notifications, []);
|
|
});
|
|
|
|
test('notifier suppresses failure when selected track is marked active before sid arrives', () => {
|
|
const notifications: string[] = [];
|
|
const timers = createTimerHarness();
|
|
const runtime = createYoutubePrimarySubtitleNotificationRuntime({
|
|
getPrimarySubtitleLanguages: () => ['ja', 'jpn'],
|
|
notifyFailure: (message) => {
|
|
notifications.push(message);
|
|
},
|
|
schedule: (fn) => timers.schedule(fn),
|
|
clearSchedule: (timer) => timers.clear(timer),
|
|
});
|
|
|
|
runtime.handleMediaPathChange('https://www.youtube.com/watch?v=abc');
|
|
runtime.handleSubtitleTrackChange(null);
|
|
runtime.handleSubtitleTrackListChange([
|
|
{ type: 'sub', id: 5, lang: 'ja', title: 'Japanese', external: false, selected: true },
|
|
]);
|
|
timers.runAll();
|
|
|
|
assert.deepEqual(notifications, []);
|
|
});
|
|
|
|
test('notifier suppresses failure when any external subtitle track is selected', () => {
|
|
const notifications: string[] = [];
|
|
const timers = createTimerHarness();
|
|
const runtime = createYoutubePrimarySubtitleNotificationRuntime({
|
|
getPrimarySubtitleLanguages: () => ['ja', 'jpn'],
|
|
notifyFailure: (message) => {
|
|
notifications.push(message);
|
|
},
|
|
schedule: (fn) => timers.schedule(fn),
|
|
clearSchedule: (timer) => timers.clear(timer),
|
|
});
|
|
|
|
runtime.handleMediaPathChange('https://www.youtube.com/watch?v=abc');
|
|
runtime.handleSubtitleTrackListChange([
|
|
{ type: 'sub', id: 5, lang: '', title: 'auto-ja-orig.ja-orig.vtt', external: true },
|
|
]);
|
|
runtime.handleSubtitleTrackChange(5);
|
|
timers.runAll();
|
|
|
|
assert.deepEqual(notifications, []);
|
|
});
|
|
|
|
test('notifier resets when media changes away from youtube', () => {
|
|
const notifications: string[] = [];
|
|
const timers = createTimerHarness();
|
|
const runtime = createYoutubePrimarySubtitleNotificationRuntime({
|
|
getPrimarySubtitleLanguages: () => ['ja'],
|
|
notifyFailure: (message) => {
|
|
notifications.push(message);
|
|
},
|
|
schedule: (fn) => timers.schedule(fn),
|
|
clearSchedule: (timer) => timers.clear(timer),
|
|
});
|
|
|
|
runtime.handleMediaPathChange('https://www.youtube.com/watch?v=abc');
|
|
runtime.handleMediaPathChange('/tmp/video.mkv');
|
|
timers.runAll();
|
|
|
|
assert.deepEqual(notifications, []);
|
|
});
|
|
|
|
test('notifier ignores empty and null media paths and waits for track list before reporting', () => {
|
|
const notifications: string[] = [];
|
|
const timers = createTimerHarness();
|
|
const runtime = createYoutubePrimarySubtitleNotificationRuntime({
|
|
getPrimarySubtitleLanguages: () => ['ja'],
|
|
notifyFailure: (message) => {
|
|
notifications.push(message);
|
|
},
|
|
schedule: (fn) => timers.schedule(fn),
|
|
clearSchedule: (timer) => timers.clear(timer),
|
|
});
|
|
|
|
runtime.handleMediaPathChange(null);
|
|
runtime.handleMediaPathChange('');
|
|
assert.equal(timers.size(), 0);
|
|
|
|
runtime.handleMediaPathChange('https://www.youtube.com/watch?v=abc');
|
|
runtime.handleSubtitleTrackChange(7);
|
|
runtime.handleSubtitleTrackListChange([
|
|
{ type: 'sub', id: 7, lang: 'ja', title: 'Japanese', external: true },
|
|
]);
|
|
timers.runAll();
|
|
assert.deepEqual(notifications, []);
|
|
});
|
|
|
|
test('notifier suppresses timer while app-owned youtube flow is still settling', () => {
|
|
const notifications: string[] = [];
|
|
const timers = createTimerHarness();
|
|
const runtime = createYoutubePrimarySubtitleNotificationRuntime({
|
|
getPrimarySubtitleLanguages: () => ['ja'],
|
|
notifyFailure: (message) => {
|
|
notifications.push(message);
|
|
},
|
|
schedule: (fn) => timers.schedule(fn),
|
|
clearSchedule: (timer) => timers.clear(timer),
|
|
});
|
|
|
|
runtime.setAppOwnedFlowInFlight(true);
|
|
runtime.handleMediaPathChange('https://www.youtube.com/watch?v=abc');
|
|
|
|
assert.equal(timers.size(), 0);
|
|
|
|
runtime.setAppOwnedFlowInFlight(false);
|
|
assert.equal(timers.size(), 1);
|
|
|
|
timers.runAll();
|
|
assert.deepEqual(notifications, [
|
|
'Primary subtitle failed to download or load. Try again from the subtitle modal.',
|
|
]);
|
|
});
|