Files
SubMiner/src/main/runtime/immersion-startup.ts
T
sudacode 5feed360ca feat: add app-owned YouTube subtitle flow with absPlayer-style parsing (#31)
* 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
2026-03-24 00:01:24 -07:00

114 lines
3.5 KiB
TypeScript

type ImmersionRetentionPolicy = {
eventsDays: number;
telemetryDays: number;
sessionsDays: number;
dailyRollupsDays: number;
monthlyRollupsDays: number;
vacuumIntervalDays: number;
};
type ImmersionTrackingPolicy = {
enabled?: boolean;
batchSize: number;
flushIntervalMs: number;
queueCap: number;
payloadCapBytes: number;
maintenanceIntervalMs: number;
retention: ImmersionRetentionPolicy;
};
type ImmersionTrackingConfig = {
immersionTracking?: ImmersionTrackingPolicy;
};
type ImmersionTrackerPolicy = Omit<ImmersionTrackingPolicy, 'enabled'>;
const DISABLE_IMMERSION_TRACKING_SESSION_ENV = 'SUBMINER_DISABLE_IMMERSION_TRACKING';
type ImmersionTrackerServiceParams = {
dbPath: string;
policy: ImmersionTrackerPolicy;
};
type MpvClientLike = {
connected: boolean;
connect: () => void;
};
export type ImmersionTrackerStartupDeps = {
getResolvedConfig: () => ImmersionTrackingConfig;
getConfiguredDbPath: () => string;
createTrackerService: (params: ImmersionTrackerServiceParams) => unknown;
setTracker: (tracker: unknown | null) => void;
getMpvClient: () => MpvClientLike | null;
shouldAutoConnectMpv?: () => boolean;
seedTrackerFromCurrentMedia: () => void;
logInfo: (message: string) => void;
logDebug: (message: string) => void;
logWarn: (message: string, details: unknown) => void;
};
export function createImmersionTrackerStartupHandler(
deps: ImmersionTrackerStartupDeps,
): () => void {
const isSessionTrackingDisabled = process.env[DISABLE_IMMERSION_TRACKING_SESSION_ENV] === '1';
return () => {
if (isSessionTrackingDisabled) {
deps.logInfo(
`Immersion tracking disabled for this session by ${DISABLE_IMMERSION_TRACKING_SESSION_ENV}=1.`,
);
return;
}
const config = deps.getResolvedConfig();
if (config.immersionTracking?.enabled === false) {
deps.logInfo('Immersion tracking disabled in config');
return;
}
try {
deps.logDebug('Immersion tracker startup requested: creating tracker service.');
const dbPath = deps.getConfiguredDbPath();
deps.logInfo(`Creating immersion tracker with dbPath=${dbPath}`);
const policy = config.immersionTracking;
if (!policy) {
throw new Error('Immersion tracking policy missing');
}
deps.setTracker(
deps.createTrackerService({
dbPath,
policy: {
batchSize: policy.batchSize,
flushIntervalMs: policy.flushIntervalMs,
queueCap: policy.queueCap,
payloadCapBytes: policy.payloadCapBytes,
maintenanceIntervalMs: policy.maintenanceIntervalMs,
retention: {
eventsDays: policy.retention.eventsDays,
telemetryDays: policy.retention.telemetryDays,
sessionsDays: policy.retention.sessionsDays,
dailyRollupsDays: policy.retention.dailyRollupsDays,
monthlyRollupsDays: policy.retention.monthlyRollupsDays,
vacuumIntervalDays: policy.retention.vacuumIntervalDays,
},
},
}),
);
deps.logDebug('Immersion tracker initialized successfully.');
const mpvClient = deps.getMpvClient();
if ((deps.shouldAutoConnectMpv?.() ?? true) && mpvClient && !mpvClient.connected) {
deps.logInfo('Auto-connecting MPV client for immersion tracking');
mpvClient.connect();
}
deps.seedTrackerFromCurrentMedia();
} catch (error) {
deps.logWarn('Immersion tracker startup failed; disabling tracking.', error);
deps.setTracker(null);
}
};
}