mirror of
https://github.com/ajnart/homarr.git
synced 2026-02-27 17:00:54 +01:00
* wip: add page and loading of certificates in folder * wip: add certificate addition and removal * feat: add removal ui for certificates * feat: migrate integrations to fetch or agent with trusted certificates * fix: lock file issues * fix: typecheck issue * fix: inconsistent package versions * chore: address pull request feedback * fix: add missing navigation item and restrict access to page * chore: address pull request feedback * fix: inconsistent undici dependency version * fix: inconsistent undici dependency version
410 lines
13 KiB
TypeScript
410 lines
13 KiB
TypeScript
import { fetchWithTrustedCertificatesAsync } from "@homarr/certificates/server";
|
|
import { logger } from "@homarr/log";
|
|
import { z } from "@homarr/validation";
|
|
|
|
import { Integration } from "../base/integration";
|
|
import type { ISearchableIntegration } from "../base/searchable-integration";
|
|
import type { MediaRequest, RequestStats, RequestUser } from "../interfaces/media-requests/media-request";
|
|
import { MediaAvailability, MediaRequestStatus } from "../interfaces/media-requests/media-request";
|
|
|
|
interface OverseerrSearchResult {
|
|
id: number;
|
|
name: string;
|
|
link: string;
|
|
image?: string;
|
|
text?: string;
|
|
type: Exclude<z.infer<typeof searchSchema>["results"], undefined>[number]["mediaType"];
|
|
}
|
|
|
|
/**
|
|
* Overseerr Integration. See https://api-docs.overseerr.dev
|
|
*/
|
|
export class OverseerrIntegration extends Integration implements ISearchableIntegration<OverseerrSearchResult> {
|
|
public async searchAsync(query: string) {
|
|
const response = await fetchWithTrustedCertificatesAsync(this.url("/api/v1/search", { query }), {
|
|
headers: {
|
|
"X-Api-Key": this.getSecretValue("apiKey"),
|
|
},
|
|
});
|
|
const schemaData = await searchSchema.parseAsync(await response.json());
|
|
|
|
if (!schemaData.results) {
|
|
return [];
|
|
}
|
|
|
|
return schemaData.results.map((result) => ({
|
|
id: result.id,
|
|
name: "name" in result ? result.name : result.title,
|
|
link: this.url(`/${result.mediaType}/${result.id}`).toString(),
|
|
image: constructSearchResultImage(result),
|
|
text: "overview" in result ? result.overview : undefined,
|
|
type: result.mediaType,
|
|
inLibrary: result.mediaInfo !== undefined,
|
|
}));
|
|
}
|
|
|
|
public async getSeriesInformationAsync(mediaType: "movie" | "tv", id: number) {
|
|
const url = mediaType === "tv" ? this.url(`/api/v1/tv/${id}`) : this.url(`/api/v1/movie/${id}`);
|
|
const response = await fetchWithTrustedCertificatesAsync(url, {
|
|
headers: {
|
|
"X-Api-Key": this.getSecretValue("apiKey"),
|
|
},
|
|
});
|
|
return await mediaInformationSchema.parseAsync(await response.json());
|
|
}
|
|
|
|
/**
|
|
* Request a media. See https://api-docs.overseerr.dev/#/request/post_request
|
|
* @param mediaType The media type to request. Can be "movie" or "tv".
|
|
* @param id The Overseerr ID of the media to request.
|
|
* @param seasons A list of the seasons that should be requested.
|
|
*/
|
|
public async requestMediaAsync(mediaType: "movie" | "tv", id: number, seasons?: number[]): Promise<void> {
|
|
const url = this.url("/api/v1/request");
|
|
const response = await fetchWithTrustedCertificatesAsync(url, {
|
|
method: "POST",
|
|
body: JSON.stringify({
|
|
mediaType,
|
|
mediaId: id,
|
|
seasons,
|
|
}),
|
|
headers: {
|
|
"X-Api-Key": this.getSecretValue("apiKey"),
|
|
"Content-Type": "application/json",
|
|
},
|
|
});
|
|
if (response.status !== 201) {
|
|
throw new Error(
|
|
`Status code ${response.status} does not match the expected status code. The request was likely not created. Response: ${await response.text()}`,
|
|
);
|
|
}
|
|
}
|
|
|
|
public async testConnectionAsync(): Promise<void> {
|
|
const response = await fetchWithTrustedCertificatesAsync(this.url("/api/v1/auth/me"), {
|
|
headers: {
|
|
"X-Api-Key": this.getSecretValue("apiKey"),
|
|
},
|
|
});
|
|
const json = (await response.json()) as object;
|
|
if (Object.keys(json).includes("id")) {
|
|
return;
|
|
}
|
|
|
|
throw new Error(`Received response but unable to parse it: ${JSON.stringify(json)}`);
|
|
}
|
|
|
|
public async getRequestsAsync(): Promise<MediaRequest[]> {
|
|
//Ensure to get all pending request first
|
|
const pendingRequests = await fetchWithTrustedCertificatesAsync(
|
|
this.url("/api/v1/request", { take: -1, filter: "pending" }),
|
|
{
|
|
headers: {
|
|
"X-Api-Key": this.getSecretValue("apiKey"),
|
|
},
|
|
},
|
|
);
|
|
|
|
//Change 20 to integration setting (set to -1 for all)
|
|
const allRequests = await fetchWithTrustedCertificatesAsync(this.url("/api/v1/request", { take: 20 }), {
|
|
headers: {
|
|
"X-Api-Key": this.getSecretValue("apiKey"),
|
|
},
|
|
});
|
|
|
|
const pendingResults = (await getRequestsSchema.parseAsync(await pendingRequests.json())).results;
|
|
const allResults = (await getRequestsSchema.parseAsync(await allRequests.json())).results;
|
|
|
|
//Concat the 2 lists while remove any duplicate pending from the all items list
|
|
let requests;
|
|
|
|
if (pendingResults.length > 0 && allResults.length > 0) {
|
|
requests = pendingResults.concat(
|
|
allResults.filter(({ status }) => status !== MediaRequestStatus.PendingApproval),
|
|
);
|
|
} else if (pendingResults.length > 0) requests = pendingResults;
|
|
else if (allResults.length > 0) requests = allResults;
|
|
else return Promise.all([]);
|
|
|
|
return await Promise.all(
|
|
requests.map(async (request): Promise<MediaRequest> => {
|
|
const information = await this.getItemInformationAsync(request.media.tmdbId, request.type);
|
|
return {
|
|
id: request.id,
|
|
name: information.name,
|
|
status: request.status,
|
|
availability: request.media.status,
|
|
backdropImageUrl: `https://image.tmdb.org/t/p/original/${information.backdropPath}`,
|
|
posterImagePath: `https://image.tmdb.org/t/p/w600_and_h900_bestv2/${information.posterPath}`,
|
|
href: this.url(`/${request.type}/${request.media.tmdbId}`).toString(),
|
|
type: request.type,
|
|
createdAt: request.createdAt,
|
|
airDate: new Date(information.airDate),
|
|
requestedBy: request.requestedBy
|
|
? ({
|
|
...request.requestedBy,
|
|
displayName: request.requestedBy.displayName,
|
|
link: this.url(`/users/${request.requestedBy.id}`).toString(),
|
|
avatar: this.constructAvatarUrl(request.requestedBy.avatar).toString(),
|
|
} satisfies Omit<RequestUser, "requestCount">)
|
|
: undefined,
|
|
};
|
|
}),
|
|
);
|
|
}
|
|
|
|
public async getStatsAsync(): Promise<RequestStats> {
|
|
const response = await fetchWithTrustedCertificatesAsync(this.url("/api/v1/request/count"), {
|
|
headers: {
|
|
"X-Api-Key": this.getSecretValue("apiKey"),
|
|
},
|
|
});
|
|
return await getStatsSchema.parseAsync(await response.json());
|
|
}
|
|
|
|
public async getUsersAsync(): Promise<RequestUser[]> {
|
|
const response = await fetchWithTrustedCertificatesAsync(this.url("/api/v1/user", { take: -1 }), {
|
|
headers: {
|
|
"X-Api-Key": this.getSecretValue("apiKey"),
|
|
},
|
|
});
|
|
const users = (await getUsersSchema.parseAsync(await response.json())).results;
|
|
return users.map((user): RequestUser => {
|
|
return {
|
|
...user,
|
|
link: this.url(`/users/${user.id}`).toString(),
|
|
avatar: this.constructAvatarUrl(user.avatar).toString(),
|
|
};
|
|
});
|
|
}
|
|
|
|
public async approveRequestAsync(requestId: number): Promise<void> {
|
|
logger.info(`Approving media request id='${requestId}' integration='${this.integration.name}'`);
|
|
await fetchWithTrustedCertificatesAsync(this.url(`/api/v1/request/${requestId}/approve`), {
|
|
method: "POST",
|
|
headers: {
|
|
"X-Api-Key": this.getSecretValue("apiKey"),
|
|
},
|
|
}).then((response) => {
|
|
if (!response.ok) {
|
|
logger.error(
|
|
`Failed to approve media request id='${requestId}' integration='${this.integration.name}' reason='${response.status} ${response.statusText}' url='${response.url}'`,
|
|
);
|
|
}
|
|
|
|
logger.info(`Successfully approved media request id='${requestId}' integration='${this.integration.name}'`);
|
|
});
|
|
}
|
|
|
|
public async declineRequestAsync(requestId: number): Promise<void> {
|
|
logger.info(`Declining media request id='${requestId}' integration='${this.integration.name}'`);
|
|
await fetchWithTrustedCertificatesAsync(this.url(`/api/v1/request/${requestId}/decline`), {
|
|
method: "POST",
|
|
headers: {
|
|
"X-Api-Key": this.getSecretValue("apiKey"),
|
|
},
|
|
}).then((response) => {
|
|
if (!response.ok) {
|
|
logger.error(
|
|
`Failed to decline media request id='${requestId}' integration='${this.integration.name}' reason='${response.status} ${response.statusText}' url='${response.url}'`,
|
|
);
|
|
}
|
|
|
|
logger.info(`Successfully declined media request id='${requestId}' integration='${this.integration.name}'`);
|
|
});
|
|
}
|
|
|
|
private async getItemInformationAsync(id: number, type: MediaRequest["type"]): Promise<MediaInformation> {
|
|
const response = await fetchWithTrustedCertificatesAsync(this.url(`/api/v1/${type}/${id}`), {
|
|
headers: {
|
|
"X-Api-Key": this.getSecretValue("apiKey"),
|
|
},
|
|
});
|
|
|
|
if (type === "tv") {
|
|
const series = (await response.json()) as TvInformation;
|
|
return {
|
|
name: series.name,
|
|
backdropPath: series.backdropPath ?? series.posterPath,
|
|
posterPath: series.posterPath ?? series.backdropPath,
|
|
airDate: series.firstAirDate,
|
|
} satisfies MediaInformation;
|
|
}
|
|
|
|
const movie = (await response.json()) as MovieInformation;
|
|
return {
|
|
name: movie.title,
|
|
backdropPath: movie.backdropPath ?? movie.posterPath,
|
|
posterPath: movie.posterPath ?? movie.backdropPath,
|
|
airDate: movie.releaseDate,
|
|
} satisfies MediaInformation;
|
|
}
|
|
|
|
private constructAvatarUrl(avatar: string) {
|
|
const isAbsolute = avatar.startsWith("http://") || avatar.startsWith("https://");
|
|
|
|
if (isAbsolute) {
|
|
return avatar;
|
|
}
|
|
|
|
return this.url(`/${avatar}`);
|
|
}
|
|
}
|
|
|
|
interface MediaInformation {
|
|
name: string;
|
|
backdropPath?: string;
|
|
posterPath?: string;
|
|
airDate: string;
|
|
}
|
|
|
|
interface TvInformation {
|
|
name: string;
|
|
backdropPath?: string;
|
|
posterPath?: string;
|
|
firstAirDate: string;
|
|
}
|
|
|
|
interface MovieInformation {
|
|
title: string;
|
|
backdropPath?: string;
|
|
posterPath?: string;
|
|
releaseDate: string;
|
|
}
|
|
|
|
const mediaInformationSchema = z.union([
|
|
z.object({
|
|
id: z.number(),
|
|
overview: z.string(),
|
|
seasons: z.array(
|
|
z.object({
|
|
id: z.number(),
|
|
name: z.string().min(0),
|
|
episodeCount: z.number().min(0),
|
|
}),
|
|
),
|
|
numberOfSeasons: z.number(),
|
|
posterPath: z.string().startsWith("/"),
|
|
}),
|
|
z.object({
|
|
id: z.number(),
|
|
overview: z.string(),
|
|
posterPath: z.string().startsWith("/"),
|
|
}),
|
|
]);
|
|
|
|
const searchSchema = z.object({
|
|
results: z
|
|
.array(
|
|
z.discriminatedUnion("mediaType", [
|
|
z.object({
|
|
id: z.number(),
|
|
mediaType: z.literal("tv"),
|
|
name: z.string(),
|
|
posterPath: z.string().startsWith("/").endsWith(".jpg").nullable(),
|
|
overview: z.string(),
|
|
mediaInfo: z.object({}).optional(),
|
|
}),
|
|
z.object({
|
|
id: z.number(),
|
|
mediaType: z.literal("movie"),
|
|
title: z.string(),
|
|
posterPath: z.string().startsWith("/").endsWith(".jpg").nullable(),
|
|
overview: z.string(),
|
|
mediaInfo: z.object({}).optional(),
|
|
}),
|
|
z.object({
|
|
id: z.number(),
|
|
mediaType: z.literal("person"),
|
|
name: z.string(),
|
|
profilePath: z.string().startsWith("/").endsWith(".jpg").nullable(),
|
|
mediaInfo: z.object({}).optional(),
|
|
}),
|
|
]),
|
|
)
|
|
.optional(),
|
|
});
|
|
|
|
const getRequestsSchema = z.object({
|
|
results: z
|
|
.array(
|
|
z.object({
|
|
id: z.number(),
|
|
status: z.nativeEnum(MediaRequestStatus),
|
|
createdAt: z.string().transform((value) => new Date(value)),
|
|
media: z.object({
|
|
status: z.nativeEnum(MediaAvailability),
|
|
tmdbId: z.number(),
|
|
}),
|
|
type: z.enum(["movie", "tv"]),
|
|
requestedBy: z
|
|
.object({
|
|
id: z.number(),
|
|
displayName: z.string(),
|
|
avatar: z.string(),
|
|
})
|
|
.optional(),
|
|
}),
|
|
)
|
|
.optional()
|
|
.transform((val) => {
|
|
if (!val) {
|
|
return [];
|
|
}
|
|
return val;
|
|
}),
|
|
});
|
|
|
|
const getStatsSchema = z.object({
|
|
total: z.number(),
|
|
movie: z.number(),
|
|
tv: z.number(),
|
|
pending: z.number(),
|
|
approved: z.number(),
|
|
declined: z.number(),
|
|
processing: z.number(),
|
|
available: z.number(),
|
|
});
|
|
|
|
const getUsersSchema = z.object({
|
|
results: z
|
|
.array(
|
|
z.object({
|
|
id: z.number(),
|
|
displayName: z.string(),
|
|
avatar: z.string(),
|
|
requestCount: z.number(),
|
|
}),
|
|
)
|
|
.optional()
|
|
.transform((val) => {
|
|
if (!val) {
|
|
return [];
|
|
}
|
|
return val;
|
|
}),
|
|
});
|
|
|
|
const constructSearchResultImage = (result: Exclude<z.infer<typeof searchSchema>["results"], undefined>[number]) => {
|
|
const path = getResultImagePath(result);
|
|
if (!path) {
|
|
return undefined;
|
|
}
|
|
|
|
return `https://image.tmdb.org/t/p/w600_and_h900_bestv2${path}`;
|
|
};
|
|
|
|
const getResultImagePath = (result: Exclude<z.infer<typeof searchSchema>["results"], undefined>[number]) => {
|
|
switch (result.mediaType) {
|
|
case "person":
|
|
return result.profilePath;
|
|
case "tv":
|
|
case "movie":
|
|
return result.posterPath;
|
|
default:
|
|
throw new Error(
|
|
`Unable to get search result image from media type '${(result as { mediaType: string }).mediaType}'`,
|
|
);
|
|
}
|
|
};
|