mirror of
https://github.com/ksyasuda/SubMiner.git
synced 2026-03-21 00:11:27 -07:00
109 lines
3.3 KiB
TypeScript
109 lines
3.3 KiB
TypeScript
type MpvClientLike = {
|
|
connected: boolean;
|
|
connect: () => void;
|
|
};
|
|
|
|
type SpawnedProcessLike = {
|
|
on: (event: 'error', listener: (error: unknown) => void) => void;
|
|
unref: () => void;
|
|
};
|
|
|
|
export type WaitForMpvConnectedDeps = {
|
|
getMpvClient: () => MpvClientLike | null;
|
|
now: () => number;
|
|
sleep: (delayMs: number) => Promise<void>;
|
|
};
|
|
|
|
export function createWaitForMpvConnectedHandler(deps: WaitForMpvConnectedDeps) {
|
|
return async (timeoutMs = 7000): Promise<boolean> => {
|
|
const client = deps.getMpvClient();
|
|
if (!client) return false;
|
|
if (client.connected) return true;
|
|
try {
|
|
client.connect();
|
|
} catch {}
|
|
|
|
const startedAt = deps.now();
|
|
while (deps.now() - startedAt < timeoutMs) {
|
|
if (deps.getMpvClient()?.connected) return true;
|
|
await deps.sleep(100);
|
|
}
|
|
return Boolean(deps.getMpvClient()?.connected);
|
|
};
|
|
}
|
|
|
|
export type LaunchMpvForJellyfinDeps = {
|
|
getSocketPath: () => string;
|
|
platform: NodeJS.Platform;
|
|
execPath: string;
|
|
defaultMpvLogPath: string;
|
|
defaultMpvArgs: readonly string[];
|
|
removeSocketPath: (socketPath: string) => void;
|
|
spawnMpv: (args: string[]) => SpawnedProcessLike;
|
|
logWarn: (message: string, error: unknown) => void;
|
|
logInfo: (message: string) => void;
|
|
};
|
|
|
|
export function createLaunchMpvIdleForJellyfinPlaybackHandler(deps: LaunchMpvForJellyfinDeps) {
|
|
return (): void => {
|
|
const socketPath = deps.getSocketPath();
|
|
if (deps.platform !== 'win32') {
|
|
try {
|
|
deps.removeSocketPath(socketPath);
|
|
} catch {
|
|
// ignore stale socket cleanup errors
|
|
}
|
|
}
|
|
|
|
const scriptOpts = `--script-opts=subminer-binary_path=${deps.execPath},subminer-socket_path=${socketPath}`;
|
|
const mpvArgs = [
|
|
...deps.defaultMpvArgs,
|
|
'--idle=yes',
|
|
scriptOpts,
|
|
`--log-file=${deps.defaultMpvLogPath}`,
|
|
`--input-ipc-server=${socketPath}`,
|
|
];
|
|
const proc = deps.spawnMpv(mpvArgs);
|
|
proc.on('error', (error) => {
|
|
deps.logWarn('Failed to launch mpv for Jellyfin remote playback', error);
|
|
});
|
|
proc.unref();
|
|
deps.logInfo(`Launched mpv for Jellyfin playback on socket: ${socketPath}`);
|
|
};
|
|
}
|
|
|
|
export type EnsureMpvConnectedDeps = {
|
|
getMpvClient: () => MpvClientLike | null;
|
|
setMpvClient: (client: MpvClientLike | null) => void;
|
|
createMpvClient: () => MpvClientLike;
|
|
waitForMpvConnected: (timeoutMs: number) => Promise<boolean>;
|
|
launchMpvIdleForJellyfinPlayback: () => void;
|
|
getAutoLaunchInFlight: () => Promise<boolean> | null;
|
|
setAutoLaunchInFlight: (promise: Promise<boolean> | null) => void;
|
|
connectTimeoutMs: number;
|
|
autoLaunchTimeoutMs: number;
|
|
};
|
|
|
|
export function createEnsureMpvConnectedForJellyfinPlaybackHandler(deps: EnsureMpvConnectedDeps) {
|
|
return async (): Promise<boolean> => {
|
|
if (!deps.getMpvClient()) {
|
|
deps.setMpvClient(deps.createMpvClient());
|
|
}
|
|
|
|
const connected = await deps.waitForMpvConnected(deps.connectTimeoutMs);
|
|
if (connected) return true;
|
|
|
|
if (!deps.getAutoLaunchInFlight()) {
|
|
const inFlight = (async () => {
|
|
deps.launchMpvIdleForJellyfinPlayback();
|
|
return deps.waitForMpvConnected(deps.autoLaunchTimeoutMs);
|
|
})().finally(() => {
|
|
deps.setAutoLaunchInFlight(null);
|
|
});
|
|
deps.setAutoLaunchInFlight(inFlight);
|
|
}
|
|
|
|
return deps.getAutoLaunchInFlight() as Promise<boolean>;
|
|
};
|
|
}
|