mirror of
https://github.com/ksyasuda/SubMiner.git
synced 2026-03-24 00:11:27 -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
65 lines
2.2 KiB
TypeScript
65 lines
2.2 KiB
TypeScript
import assert from 'node:assert/strict';
|
|
import test from 'node:test';
|
|
|
|
import { resolveMediaGenerationInputPath } from './media-source';
|
|
|
|
test('resolveMediaGenerationInputPath keeps local file paths', async () => {
|
|
const result = await resolveMediaGenerationInputPath({
|
|
currentVideoPath: '/tmp/video.mkv',
|
|
});
|
|
|
|
assert.equal(result, '/tmp/video.mkv');
|
|
});
|
|
|
|
test('resolveMediaGenerationInputPath prefers stream-open-filename for remote media', async () => {
|
|
const requests: string[] = [];
|
|
|
|
const result = await resolveMediaGenerationInputPath({
|
|
currentVideoPath: 'https://www.youtube.com/watch?v=abc123',
|
|
requestProperty: async (name: string) => {
|
|
requests.push(name);
|
|
return 'https://rr1---sn.example.googlevideo.com/videoplayback?id=123';
|
|
},
|
|
});
|
|
|
|
assert.equal(result, 'https://rr1---sn.example.googlevideo.com/videoplayback?id=123');
|
|
assert.deepEqual(requests, ['stream-open-filename']);
|
|
});
|
|
|
|
test('resolveMediaGenerationInputPath unwraps mpv edl source for audio and video', async () => {
|
|
const edlSource = [
|
|
'edl://!new_stream;!no_clip;!no_chapters;%70%https://audio.example/videoplayback?mime=audio%2Fwebm',
|
|
'!new_stream;!no_clip;!no_chapters;%69%https://video.example/videoplayback?mime=video%2Fmp4',
|
|
'!global_tags,title=test',
|
|
].join(';');
|
|
|
|
const audioResult = await resolveMediaGenerationInputPath(
|
|
{
|
|
currentVideoPath: 'https://www.youtube.com/watch?v=abc123',
|
|
requestProperty: async () => edlSource,
|
|
},
|
|
'audio',
|
|
);
|
|
const videoResult = await resolveMediaGenerationInputPath(
|
|
{
|
|
currentVideoPath: 'https://www.youtube.com/watch?v=abc123',
|
|
requestProperty: async () => edlSource,
|
|
},
|
|
'video',
|
|
);
|
|
|
|
assert.equal(audioResult, 'https://audio.example/videoplayback?mime=audio%2Fwebm');
|
|
assert.equal(videoResult, 'https://video.example/videoplayback?mime=video%2Fmp4');
|
|
});
|
|
|
|
test('resolveMediaGenerationInputPath falls back to currentVideoPath when stream-open-filename fails', async () => {
|
|
const result = await resolveMediaGenerationInputPath({
|
|
currentVideoPath: 'https://www.youtube.com/watch?v=abc123',
|
|
requestProperty: async () => {
|
|
throw new Error('property unavailable');
|
|
},
|
|
});
|
|
|
|
assert.equal(result, 'https://www.youtube.com/watch?v=abc123');
|
|
});
|