Skip to content

Commit

Permalink
cache segment requests
Browse files Browse the repository at this point in the history
  • Loading branch information
hanydd committed Jan 16, 2025
1 parent 9f2523c commit f1c03de
Show file tree
Hide file tree
Showing 4 changed files with 101 additions and 36 deletions.
39 changes: 7 additions & 32 deletions src/content.ts
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@ import SubmissionNotice from "./render/SubmissionNotice";
import { FetchResponse } from "./requests/background-request-proxy";
import { getPortVideoByHash, postPortVideo, postPortVideoVote, updatePortedSegments } from "./requests/portVideo";
import { asyncRequestToServer } from "./requests/requests";
import { getSegmentsByHash } from "./requests/segments";
import { getSegmentsByVideoID } from "./requests/segments";
import { getVideoLabel } from "./requests/videoLabels";
import { checkPageForNewThumbnails, setupThumbnailListener } from "./thumbnail-utils/thumbnailManagement";
import {
Expand All @@ -33,7 +33,6 @@ import {
SponsorHideType,
SponsorSourceType,
SponsorTime,
SponsorTimeHashedID,
ToggleSkippable,
VideoID,
VideoInfo,
Expand Down Expand Up @@ -1342,31 +1341,19 @@ async function sponsorsLookup(keepOldSubmissions = true, ignoreServerCache = fal
return;
}

const categories: string[] = Config.config.categorySelections.map((category) => category.name);

const extraRequestData: Record<string, unknown> = {};
const hashParams = getHashParams();
if (hashParams.requiredSegment) extraRequestData.requiredSegment = hashParams.requiredSegment;

const hashPrefix = (await getVideoIDHash(getVideoID())).slice(0, 4) as VideoID & HashedValue;
const response = await getSegmentsByHash(hashPrefix, extraRequestData, ignoreServerCache);
const segmentResponse = await getSegmentsByVideoID(getVideoID(), extraRequestData, ignoreServerCache);

// store last response status
lastResponseStatus = response?.status;

if (response?.ok) {
const receivedSegments: SponsorTime[] = JSON.parse(response.responseText)
?.filter((video: SponsorTimeHashedID) => video.videoID === getVideoID())
?.map((video: SponsorTimeHashedID) => video.segments)?.[0]
?.filter(
(segment: SponsorTime) =>
getEnabledActionTypes().includes(segment.actionType) && categories.includes(segment.category)
)
?.map((segment: SponsorTime) => ({
...segment,
source: SponsorSourceType.Server,
}))
?.sort((a: SponsorTime, b: SponsorTime) => a.segment[0] - b.segment[0]);
lastResponseStatus = segmentResponse?.status;

if (segmentResponse.status === 200) {
const receivedSegments: SponsorTime[] = segmentResponse.segments;

if (receivedSegments && receivedSegments.length) {
sponsorDataFound = true;

Expand Down Expand Up @@ -1451,18 +1438,6 @@ async function sponsorsLookup(keepOldSubmissions = true, ignoreServerCache = fal
}
}

function getEnabledActionTypes(forceFullVideo = false): ActionType[] {
const actionTypes = [ActionType.Skip, ActionType.Poi];
if (Config.config.muteSegments) {
actionTypes.push(ActionType.Mute);
}
if (Config.config.fullVideoSegments || forceFullVideo) {
actionTypes.push(ActionType.Full);
}

return actionTypes;
}

async function lockedCategoriesLookup(): Promise<void> {
const hashPrefix = (await getHash(getVideoID(), 1)).slice(0, 4);
const response = await asyncRequestToServer("GET", "/api/lockCategories/" + hashPrefix);
Expand Down
74 changes: 72 additions & 2 deletions src/requests/segments.ts
Original file line number Diff line number Diff line change
@@ -1,16 +1,86 @@
import Config from "../config";
import { ActionType, SponsorSourceType, SponsorTime, SponsorTimeHashedID, VideoID } from "../types";
import { DataCache } from "../utils/cache";
import { getVideoIDHash, HashedValue } from "../utils/hash";
import { asyncRequestToServer } from "./requests";

const segmentCache = new DataCache<VideoID, SegmentResponse>(() => {
return {
segments: null,
status: 200,
};
}, 100);

export interface SegmentResponse {
segments: SponsorTime[] | null;
status: number;
}

export async function getSegmentsByHash(
hashPrefix: string,
extraRequestData: Record<string, unknown>,
ignoreServerCache: boolean
ignoreCache: boolean
) {
const response = await asyncRequestToServer(
"GET",
"/api/skipSegments/" + hashPrefix,
extraRequestData,
ignoreServerCache
ignoreCache
);

return response;
}

export async function getSegmentsByVideoID(
videoID: string,
extraRequestData: Record<string, unknown> = {},
ignoreCache: boolean = false
): Promise<SegmentResponse> {
if (ignoreCache) {
segmentCache.delete(videoID);
}
const cachedData = segmentCache.getFromCache(videoID);
if (cachedData) {
return cachedData;
}

const categories: string[] = Config.config.categorySelections.map((category) => category.name);
const hashPrefix = (await getVideoIDHash(videoID)).slice(0, 4) as VideoID & HashedValue;
const response = await getSegmentsByHash(hashPrefix, extraRequestData, ignoreCache);

const responseSegments: SegmentResponse = { segments: null, status: response.status };
if (!response?.ok) {
return responseSegments;
}
const allSegments: SponsorTimeHashedID[] = JSON.parse(response?.responseText);

for (const segmentResponse of allSegments) {
const segment = segmentResponse.segments
?.filter(
(segment: SponsorTime) =>
getEnabledActionTypes().includes(segment.actionType) && categories.includes(segment.category)
)
?.map((segment: SponsorTime) => ({
...segment,
source: SponsorSourceType.Server,
}))
?.sort((a: SponsorTime, b: SponsorTime) => a.segment[0] - b.segment[0]);
segmentCache.set(segmentResponse.videoID, { segments: segment, status: response.status });
if (videoID == segmentResponse.videoID) {
responseSegments.segments = segment;
}
}
return responseSegments;
}

function getEnabledActionTypes(forceFullVideo = false): ActionType[] {
const actionTypes = [ActionType.Skip, ActionType.Poi];
if (Config.config.muteSegments) {
actionTypes.push(ActionType.Mute);
}
if (Config.config.fullVideoSegments || forceFullVideo) {
actionTypes.push(ActionType.Full);
}

return actionTypes;
}
19 changes: 17 additions & 2 deletions src/thumbnail-utils/thumbnails.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,7 @@
import Config from "../config";
import { getSegmentsByVideoID } from "../requests/segments";
import { getVideoLabel } from "../requests/videoLabels";
import { VideoID } from "../types";
import { waitFor } from "../utils/";
import { getBvIDFromURL } from "../utils/parseVideoID";
import { getLabelAnchorSelector, getLinkAttribute, getLinkSelectors } from "./thumbnail-selectors";
Expand Down Expand Up @@ -52,7 +54,7 @@ export async function labelThumbnailProcess(
const [videoID] = videoIDs;

// 获取或创建缩略图标签
const { overlay, text } = await createOrGetThumbnail(thumbnail, containerType);
const { overlay, text } = await createOrGetThumbnail(thumbnail, containerType, videoID);

const category = await getVideoLabel(videoID);
if (!category) {
Expand Down Expand Up @@ -90,10 +92,23 @@ async function hideThumbnailLabel(thumbnail: HTMLElement): Promise<void> {
}
}

const preloadSegments = (e: MouseEvent) => {
const videoID = (e.target as HTMLElement).getAttribute("data-bsb-bvid");
getSegmentsByVideoID(videoID);
};

async function createOrGetThumbnail(
thumbnail: HTMLElement,
containerType: string
containerType: string,
videoID: VideoID
): Promise<{ overlay: HTMLElement; text: HTMLElement }> {
// only add evnet listener once, add preloadSegments to thumbnail when pointerenter
if (thumbnail.getAttribute("data-bsb-bvid") != videoID) {
thumbnail.setAttribute("data-bsb-bvid", videoID);
thumbnail.removeEventListener("mouseenter", preloadSegments);
thumbnail.addEventListener("mouseenter", preloadSegments);
}

const oldLabelElement = await getOldThumbnailLabel(thumbnail);
if (oldLabelElement) {
return {
Expand Down
5 changes: 5 additions & 0 deletions src/utils/cache.ts
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@ export class DataCache<T extends string, V> {
}

public getFromCache(key: T): (V & CacheRecord) | undefined {
this.cacheUsed(key);
return this.cache[key];
}

Expand All @@ -25,6 +26,10 @@ export class DataCache<T extends string, V> {
this.gc();
}

public delete(key: T): void {
delete this.cache[key];
}

public setupCache(key: T): V & CacheRecord {
if (!this.cache[key] && this.init) {
this.set(key, this.init());
Expand Down

0 comments on commit f1c03de

Please sign in to comment.