diff --git a/src/clients/radarr-client.ts b/src/clients/radarr-client.ts index e0a3492..e76e68f 100644 --- a/src/clients/radarr-client.ts +++ b/src/clients/radarr-client.ts @@ -1,11 +1,18 @@ import { KyHttpClient } from "../__generated__/ky-client"; -import { MergedCustomFormatResource } from "../__generated__/mergedTypes"; import { Api } from "../__generated__/radarr/Api"; +import { + CustomFormatResource, + LanguageResource, + QualityDefinitionResource, + QualityProfileResource, +} from "../__generated__/radarr/data-contracts"; import { logger } from "../logger"; +import { cloneWithJSON } from "../util"; import { IArrClient, validateClientParams } from "./unified-client"; -export class RadarrClient implements IArrClient { +export class RadarrClient implements IArrClient { private api!: Api; + private languageMap: Map = new Map(); constructor(baseUrl: string, apiKey: string) { this.initialize(baseUrl, apiKey); @@ -24,25 +31,42 @@ export class RadarrClient implements IArrClient { this.api = new Api(httpClient); } + async getLanguages() { + return this.api.v3LanguageList(); + } + // Quality Management getQualityDefinitions() { return this.api.v3QualitydefinitionList(); } - updateQualityDefinitions(definitions: any) { - return this.api.v3QualitydefinitionUpdateUpdate(definitions); + async updateQualityDefinitions(definitions: QualityDefinitionResource[]) { + await this.api.v3QualitydefinitionUpdateUpdate(definitions); + this.api.v3LanguageList(); + return this.getQualityDefinitions(); } // Quality Profiles - getQualityProfiles() { + getQualityProfiles(): Promise { return this.api.v3QualityprofileList(); } - createQualityProfile(profile: any) { - return this.api.v3QualityprofileCreate(profile); + async createQualityProfile(profile: QualityProfileResource): Promise { + const cloned = cloneWithJSON(profile); + + if (this.languageMap.size <= 0) { + const languages = await this.getLanguages(); + this.languageMap = new Map(languages.map((i) => [i.name!, i])); + } + + if (profile.language == null) { + cloned.language = this.languageMap.get("Any"); + } + + return this.api.v3QualityprofileCreate(cloned); } - updateQualityProfile(id: string, profile: any) { + updateQualityProfile(id: string, profile: QualityProfileResource): Promise { return this.api.v3QualityprofileUpdate(id, profile); } @@ -51,11 +75,11 @@ export class RadarrClient implements IArrClient { return this.api.v3CustomformatList(); } - createCustomFormat(format: MergedCustomFormatResource) { + createCustomFormat(format: CustomFormatResource) { return this.api.v3CustomformatCreate(format); } - updateCustomFormat(id: string, format: MergedCustomFormatResource) { + updateCustomFormat(id: string, format: CustomFormatResource) { return this.api.v3CustomformatUpdate(id, format); } @@ -63,19 +87,6 @@ export class RadarrClient implements IArrClient { return this.api.v3CustomformatDelete(+id); } - // Metadata Profiles - async getMetadataProfiles() { - throw new Error("Metadata profiles are not supported in Radarr"); - } - - async createMetadataProfile(profile: any) { - throw new Error("Metadata profiles are not supported in Radarr"); - } - - async updateMetadataProfile(id: number, profile: any) { - throw new Error("Metadata profiles are not supported in Radarr"); - } - // System/Health Check getSystemStatus() { return this.api.v3SystemStatusList(); diff --git a/src/clients/readarr-client.ts b/src/clients/readarr-client.ts index e88aa3f..63c58cd 100644 --- a/src/clients/readarr-client.ts +++ b/src/clients/readarr-client.ts @@ -1,10 +1,18 @@ import { KyHttpClient } from "../__generated__/ky-client"; -import { MergedCustomFormatResource } from "../__generated__/mergedTypes"; import { Api } from "../__generated__/readarr/Api"; +import { + CustomFormatResource, + LanguageResource, + MetadataProfileResource, + QualityDefinitionResource, + QualityProfileResource, +} from "../__generated__/readarr/data-contracts"; import { logger } from "../logger"; import { IArrClient, validateClientParams } from "./unified-client"; -export class ReadarrClient implements IArrClient { +export class ReadarrClient + implements IArrClient +{ private api!: Api; constructor(baseUrl: string, apiKey: string) { @@ -24,13 +32,18 @@ export class ReadarrClient implements IArrClient { this.api = new Api(httpClient); } + async getLanguages() { + return this.api.v1LanguageList(); + } + // Quality Management getQualityDefinitions() { return this.api.v1QualitydefinitionList(); } - updateQualityDefinitions(definitions: any) { - return this.api.v1QualitydefinitionUpdateUpdate(definitions); + async updateQualityDefinitions(definitions: QualityDefinitionResource[]) { + await this.api.v1QualitydefinitionUpdateUpdate(definitions); + return this.api.v1QualitydefinitionList(); } // Quality Profiles @@ -38,11 +51,11 @@ export class ReadarrClient implements IArrClient { return this.api.v1QualityprofileList(); } - createQualityProfile(profile: any) { + createQualityProfile(profile: QualityProfileResource) { return this.api.v1QualityprofileCreate(profile); } - updateQualityProfile(id: string, profile: any) { + updateQualityProfile(id: string, profile: QualityProfileResource) { return this.api.v1QualityprofileUpdate(id, profile); } @@ -51,11 +64,11 @@ export class ReadarrClient implements IArrClient { return this.api.v1CustomformatList(); } - createCustomFormat(format: MergedCustomFormatResource) { + createCustomFormat(format: CustomFormatResource) { return this.api.v1CustomformatCreate(format); } - updateCustomFormat(id: string, format: MergedCustomFormatResource) { + updateCustomFormat(id: string, format: CustomFormatResource) { return this.api.v1CustomformatUpdate(id, format); } @@ -68,11 +81,11 @@ export class ReadarrClient implements IArrClient { return this.api.v1MetadataprofileList(); } - async createMetadataProfile(profile: any) { + async createMetadataProfile(profile: MetadataProfileResource) { return this.api.v1MetadataprofileCreate(profile); } - async updateMetadataProfile(id: number, profile: any) { + async updateMetadataProfile(id: number, profile: MetadataProfileResource) { return this.api.v1MetadataprofileUpdate(id.toString(), profile); } diff --git a/src/clients/sonarr-client.ts b/src/clients/sonarr-client.ts index b901708..1509193 100644 --- a/src/clients/sonarr-client.ts +++ b/src/clients/sonarr-client.ts @@ -1,10 +1,21 @@ import { KyHttpClient } from "../__generated__/ky-client"; -import { MergedCustomFormatResource } from "../__generated__/mergedTypes"; import { Api } from "../__generated__/sonarr/Api"; +import { + CustomFormatResource, + LanguageResource, + QualityDefinitionResource, + QualityProfileResource, +} from "../__generated__/sonarr/data-contracts"; import { logger } from "../logger"; import { IArrClient, validateClientParams } from "./unified-client"; -export class SonarrClient implements IArrClient { +export type SonarrQualityProfileResource = { + id?: number; + name?: string; + // Add other common properties that all quality profiles share +}; + +export class SonarrClient implements IArrClient { private api!: Api; constructor(baseUrl: string, apiKey: string) { @@ -24,13 +35,18 @@ export class SonarrClient implements IArrClient { this.api = new Api(httpClient); } + async getLanguages() { + return this.api.v3LanguageList(); + } + // Quality Management getQualityDefinitions() { return this.api.v3QualitydefinitionList(); } - updateQualityDefinitions(definitions: any) { - return this.api.v3QualitydefinitionUpdateUpdate(definitions); + async updateQualityDefinitions(definitions: QualityDefinitionResource[]) { + this.api.v3QualitydefinitionUpdateUpdate(definitions); + return this.api.v3QualitydefinitionList(); } // Quality Profiles @@ -38,11 +54,11 @@ export class SonarrClient implements IArrClient { return this.api.v3QualityprofileList(); } - createQualityProfile(profile: any) { + createQualityProfile(profile: SonarrQualityProfileResource) { return this.api.v3QualityprofileCreate(profile); } - updateQualityProfile(id: string, profile: any) { + updateQualityProfile(id: string, profile: SonarrQualityProfileResource) { return this.api.v3QualityprofileUpdate(id, profile); } @@ -51,11 +67,11 @@ export class SonarrClient implements IArrClient { return this.api.v3CustomformatList(); } - createCustomFormat(format: MergedCustomFormatResource) { + createCustomFormat(format: CustomFormatResource) { return this.api.v3CustomformatCreate(format); } - updateCustomFormat(id: string, format: MergedCustomFormatResource) { + updateCustomFormat(id: string, format: CustomFormatResource) { return this.api.v3CustomformatUpdate(id, format); } @@ -63,19 +79,6 @@ export class SonarrClient implements IArrClient { return this.api.v3CustomformatDelete(+id); } - // Metadata Profiles - async getMetadataProfiles() { - throw new Error("Metadata profiles are not supported in Sonarr"); - } - - async createMetadataProfile(profile: any) { - throw new Error("Metadata profiles are not supported in Sonarr"); - } - - async updateMetadataProfile(id: number, profile: any) { - throw new Error("Metadata profiles are not supported in Sonarr"); - } - // System/Health Check getSystemStatus() { return this.api.v3SystemStatusList(); diff --git a/src/clients/unified-client.ts b/src/clients/unified-client.ts index 263e9ce..ee3dea3 100644 --- a/src/clients/unified-client.ts +++ b/src/clients/unified-client.ts @@ -1,4 +1,4 @@ -import { MergedCustomFormatResource } from "../__generated__/mergedTypes"; +import { MergedCustomFormatResource, MergedQualityDefinitionResource, MergedQualityProfileResource } from "../__generated__/mergedTypes"; import { logger } from "../logger"; import { ArrType } from "../types/common.types"; import { RadarrClient } from "./radarr-client"; @@ -70,26 +70,48 @@ export const configureApi = async (type: ArrType, baseUrl: string, apiKey: strin return unifiedClient; }; -export interface IArrClient { +export type ArrClientCustomFormat = { + id?: number; +}; + +export type ArrClientQualityDefinition = { + id?: number; +}; + +export type ArrClientQualityProfile = { + id?: number; + name?: string | null; + // Add other common properties that all quality profiles share +}; + +export type ArrClientLanguageResource = { + id?: number; + name?: string | null; + nameLower?: string | null; +}; + +export interface IArrClient< + QP extends ArrClientQualityProfile = MergedQualityProfileResource, + QD extends ArrClientQualityDefinition = MergedQualityDefinitionResource, + CF extends ArrClientCustomFormat = MergedCustomFormatResource, + L extends ArrClientLanguageResource = ArrClientLanguageResource, +> { // Quality Management - getQualityDefinitions(): Promise; - updateQualityDefinitions(definitions: any): Promise; + getQualityDefinitions(): Promise; + updateQualityDefinitions(definitions: QD[]): Promise; // Quality Profiles - getQualityProfiles(): Promise; - createQualityProfile(profile: any): Promise; - updateQualityProfile(id: string, profile: any): Promise; + getQualityProfiles(): Promise; + createQualityProfile(profile: QP): Promise; + updateQualityProfile(id: string, profile: QP): Promise; // Custom Formats - getCustomFormats(): Promise; - createCustomFormat(format: MergedCustomFormatResource): Promise; - updateCustomFormat(id: string, format: MergedCustomFormatResource): Promise; + getCustomFormats(): Promise; + createCustomFormat(format: CF): Promise; + updateCustomFormat(id: string, format: CF): Promise; deleteCustomFormat(id: string): Promise; - // Metadata Profiles (Readarr-specific) - getMetadataProfiles(): Promise; - createMetadataProfile(profile: any): Promise; - updateMetadataProfile(id: number, profile: any): Promise; + getLanguages(): Promise; // System/Health Check getSystemStatus(): Promise; @@ -121,15 +143,23 @@ export class UnifiedClient implements IArrClient { } } + getSpecificClient(): T { + return this.api as T; + } + + async getLanguages() { + return this.api.getLanguages(); + } + async getQualityDefinitions() { return await this.api.getQualityDefinitions(); } - async updateQualityDefinitions(definitions: any) { + async updateQualityDefinitions(definitions: MergedQualityDefinitionResource[]) { return await this.api.updateQualityDefinitions(definitions); } - async createQualityProfile(profile: any) { + async createQualityProfile(profile: MergedQualityProfileResource) { return await this.api.createQualityProfile(profile); } @@ -137,23 +167,10 @@ export class UnifiedClient implements IArrClient { return await this.api.getQualityProfiles(); } - async updateQualityProfile(id: string, profile: any) { + async updateQualityProfile(id: string, profile: MergedQualityProfileResource) { return await this.api.updateQualityProfile(id, profile); } - // Readarr-specific methods - async createMetadataProfile(profile: any) { - return await this.api.createMetadataProfile(profile); - } - - async getMetadataProfiles() { - return await this.api.getMetadataProfiles(); - } - - async updateMetadataProfile(id: number, profile: any) { - return await this.api.updateMetadataProfile(id, profile); - } - async getCustomFormats() { return await this.api.getCustomFormats(); } diff --git a/src/clients/whisparr-client.ts b/src/clients/whisparr-client.ts index 4e7b90c..1e08187 100644 --- a/src/clients/whisparr-client.ts +++ b/src/clients/whisparr-client.ts @@ -1,11 +1,29 @@ import { KyHttpClient } from "../__generated__/ky-client"; -import { MergedCustomFormatResource } from "../__generated__/mergedTypes"; import { Api } from "../__generated__/whisparr/Api"; +import { + CustomFormatResource, + LanguageResource, + QualityDefinitionResource, + QualityProfileResource, +} from "../__generated__/whisparr/data-contracts"; import { logger } from "../logger"; +import { cloneWithJSON } from "../util"; import { IArrClient, validateClientParams } from "./unified-client"; -export class WhisparrClient implements IArrClient { +/** + * Overwrite wrong types for now + */ +declare module "../__generated__/whisparr/data-contracts" { + export interface QualityProfileResource { + language?: Language; + } +} + +export class WhisparrClient + implements IArrClient +{ private api!: Api; + private languageMap: Map = new Map(); constructor(baseUrl: string, apiKey: string) { this.initialize(baseUrl, apiKey); @@ -24,13 +42,18 @@ export class WhisparrClient implements IArrClient { this.api = new Api(httpClient); } + async getLanguages() { + return this.api.v3LanguageList(); + } + // Quality Management getQualityDefinitions() { return this.api.v3QualitydefinitionList(); } - updateQualityDefinitions(definitions: any) { - return this.api.v3QualitydefinitionUpdateUpdate(definitions); + async updateQualityDefinitions(definitions: QualityDefinitionResource[]) { + await this.api.v3QualitydefinitionUpdateUpdate(definitions); + return this.api.v3QualitydefinitionList(); } // Quality Profiles @@ -38,11 +61,22 @@ export class WhisparrClient implements IArrClient { return this.api.v3QualityprofileList(); } - createQualityProfile(profile: any) { - return this.api.v3QualityprofileCreate(profile); + async createQualityProfile(profile: QualityProfileResource): Promise { + const cloned = cloneWithJSON(profile); + + if (this.languageMap.size <= 0) { + const languages = await this.getLanguages(); + this.languageMap = new Map(languages.map((i) => [i.name!, i])); + } + + if (profile.language == null) { + cloned.language = this.languageMap.get("Any"); + } + + return this.api.v3QualityprofileCreate(cloned); } - updateQualityProfile(id: string, profile: any) { + updateQualityProfile(id: string, profile: QualityProfileResource) { return this.api.v3QualityprofileUpdate(id, profile); } @@ -51,11 +85,11 @@ export class WhisparrClient implements IArrClient { return this.api.v3CustomformatList(); } - createCustomFormat(format: MergedCustomFormatResource) { + createCustomFormat(format: CustomFormatResource) { return this.api.v3CustomformatCreate(format); } - updateCustomFormat(id: string, format: MergedCustomFormatResource) { + updateCustomFormat(id: string, format: CustomFormatResource) { return this.api.v3CustomformatUpdate(id, format); } @@ -63,19 +97,6 @@ export class WhisparrClient implements IArrClient { return this.api.v3CustomformatDelete(+id); } - // Metadata Profiles - async getMetadataProfiles() { - throw new Error("Metadata profiles are not supported in Whisparr"); - } - - async createMetadataProfile(profile: any) { - throw new Error("Metadata profiles are not supported in Whisparr"); - } - - async updateMetadataProfile(id: number, profile: any) { - throw new Error("Metadata profiles are not supported in Whisparr"); - } - // System/Health Check getSystemStatus() { return this.api.v3SystemStatusList();