Files
SubMiner/src/main/ipc-runtime.test.ts
sudacode ec64eebb80 fix: wire playlist-browser IPC deps through bootstrap surfaces
- Thread `openPlaylistBrowser` action into `IpcRuntimeBootstrapInput`
- Pass `playlistBrowserMainDeps` through bootstrap into `createIpcRuntime`
- Add playlist-browser mock deps to ipc-runtime tests
- Bump subminer-yomitan submodule
2026-04-01 00:16:39 -07:00

197 lines
6.7 KiB
TypeScript

import assert from 'node:assert/strict';
import test from 'node:test';
import { createIpcRuntime, createIpcRuntimeFromMainState } from './ipc-runtime';
function createBaseRuntimeInput(capturedRegistration: { value: unknown | null }) {
const manualResult = { ok: true, summary: 'done' };
const main = {
window: {
getMainWindow: () => null,
getVisibleOverlayVisibility: () => false,
focusMainWindow: () => {},
onOverlayModalClosed: () => {},
onOverlayModalOpened: () => {},
onYoutubePickerResolve: async () => ({ ok: true }) as never,
openYomitanSettings: () => {},
quitApp: () => {},
toggleVisibleOverlay: () => {},
},
playlistBrowser: {
getPlaylistBrowserSnapshot: async () => ({
directoryPath: null,
directoryAvailable: false,
directoryStatus: '',
directoryItems: [],
playlistItems: [],
playingIndex: null,
currentFilePath: null,
}),
appendPlaylistBrowserFile: async () => ({ ok: true, message: 'ok' }),
playPlaylistBrowserIndex: async () => ({ ok: true, message: 'ok' }),
removePlaylistBrowserIndex: async () => ({ ok: true, message: 'ok' }),
movePlaylistBrowserIndex: async () => ({ ok: true, message: 'ok' }),
},
subtitle: {
tokenizeCurrentSubtitle: async () => null,
getCurrentSubtitleRaw: () => '',
getCurrentSubtitleAss: () => '',
getSubtitleSidebarSnapshot: async () => null as never,
getPlaybackPaused: () => false,
getSubtitlePosition: () => null,
getSubtitleStyle: () => null,
saveSubtitlePosition: () => {},
getMecabTokenizer: () => null,
getKeybindings: () => [],
getConfiguredShortcuts: () => null,
getStatsToggleKey: () => '',
getMarkWatchedKey: () => '',
getSecondarySubMode: () => 'hover',
},
controller: {
getControllerConfig: () => ({}) as never,
saveControllerConfig: () => {},
saveControllerPreference: () => {},
},
runtime: {
getMpvClient: () => null,
getAnkiConnectStatus: () => false,
getRuntimeOptions: () => [],
reportOverlayContentBounds: () => {},
getImmersionTracker: () => null,
},
anilist: {
getStatus: () => null,
clearToken: () => {},
openSetup: () => {},
getQueueStatus: () => null,
retryQueueNow: async () => ({ ok: true, message: 'ok' }) as never,
},
mining: {
appendClipboardVideoToQueue: () => ({ ok: true, message: 'ok' }),
},
};
const ankiJimaku = {
patchAnkiConnectEnabled: () => {},
getResolvedConfig: () => ({}),
getRuntimeOptionsManager: () => null,
getSubtitleTimingTracker: () => null,
getMpvClient: () => null,
getAnkiIntegration: () => null,
setAnkiIntegration: () => {},
getKnownWordCacheStatePath: () => '/tmp/known-words.json',
showDesktopNotification: () => {},
createFieldGroupingCallback: () => async () => ({}) as never,
broadcastRuntimeOptionsChanged: () => {},
getFieldGroupingResolver: () => null,
setFieldGroupingResolver: () => {},
parseMediaInfo: () => ({}) as never,
getCurrentMediaPath: () => null,
jimakuFetchJson: async () => ({ data: null, error: null }) as never,
getJimakuMaxEntryResults: () => 5,
getJimakuLanguagePreference: () => 'ja' as const,
resolveJimakuApiKey: async () => null,
isRemoteMediaPath: () => false,
downloadToFile: async () => ({ ok: true, path: '/tmp/file' }) as never,
};
return {
mpv: {
mainDeps: {
triggerSubsyncFromConfig: () => {},
openRuntimeOptionsPalette: () => {},
openYoutubeTrackPicker: () => {},
openPlaylistBrowser: () => {},
cycleRuntimeOption: () => ({ ok: true }),
showMpvOsd: () => {},
replayCurrentSubtitle: () => {},
playNextSubtitle: () => {},
shiftSubDelayToAdjacentSubtitle: async () => {},
sendMpvCommand: () => {},
getMpvClient: () => null,
isMpvConnected: () => true,
hasRuntimeOptionsManager: () => true,
},
handleMpvCommandFromIpcRuntime: () => {},
runSubsyncManualFromIpc: async () => manualResult as never,
},
main,
ankiJimaku,
registration: {
runtimeOptions: {
getRuntimeOptionsManager: () => null,
showMpvOsd: () => {},
},
main,
ankiJimaku,
registerIpcRuntimeServices: (params: unknown) => {
capturedRegistration.value = params;
},
},
registerIpcRuntimeServices: (params: unknown) => {
capturedRegistration.value = params;
},
manualResult,
};
}
test('ipc runtime registers composed IPC handlers from explicit registration input', async () => {
const capturedRegistration = { value: null as unknown | null };
const input = createBaseRuntimeInput(capturedRegistration);
const runtime = createIpcRuntime({
mpv: input.mpv,
registration: input.registration,
});
runtime.registerIpcRuntimeHandlers();
assert.ok(capturedRegistration.value);
const registration = capturedRegistration.value as {
runtimeOptions: { showMpvOsd: unknown };
mainDeps: {
getPlaylistBrowserSnapshot: unknown;
handleMpvCommand: unknown;
runSubsyncManual: (payload: unknown) => Promise<unknown>;
};
};
assert.equal(registration.runtimeOptions.showMpvOsd !== undefined, true);
assert.equal(registration.mainDeps.getPlaylistBrowserSnapshot instanceof Function, true);
assert.equal(registration.mainDeps.handleMpvCommand instanceof Function, true);
assert.deepEqual(
await registration.mainDeps.runSubsyncManual({ payload: null } as never),
input.manualResult,
);
});
test('ipc runtime builds grouped registration input from main state', async () => {
const capturedRegistration = { value: null as unknown | null };
const input = createBaseRuntimeInput(capturedRegistration);
const runtime = createIpcRuntimeFromMainState({
mpv: input.mpv,
runtimeOptions: input.registration.runtimeOptions,
main: input.main,
ankiJimaku: input.ankiJimaku,
});
runtime.registerIpcRuntimeHandlers();
assert.ok(capturedRegistration.value);
const registration = capturedRegistration.value as {
runtimeOptions: { showMpvOsd: unknown };
mainDeps: {
getPlaylistBrowserSnapshot: unknown;
handleMpvCommand: unknown;
runSubsyncManual: (payload: unknown) => Promise<unknown>;
};
};
assert.equal(registration.runtimeOptions.showMpvOsd !== undefined, true);
assert.equal(registration.mainDeps.getPlaylistBrowserSnapshot instanceof Function, true);
assert.equal(registration.mainDeps.handleMpvCommand instanceof Function, true);
assert.deepEqual(
await registration.mainDeps.runSubsyncManual({ payload: null } as never),
input.manualResult,
);
});