brunner56's picture
implement app
0bfe2e3
import { AddonDetail, ParseResult, StreamRequest } from '@aiostreams/types';
import { ParsedStream, Stream, Config } from '@aiostreams/types';
import { BaseWrapper } from './base';
import { addonDetails, createLogger } from '@aiostreams/utils';
import { Settings } from '@aiostreams/utils';
const logger = createLogger('wrappers');
export class Debridio extends BaseWrapper {
constructor(
configString: string | null,
overrideUrl: string | null,
addonName: string = 'Debridio',
addonId: string,
userConfig: Config,
indexerTimeout?: number
) {
let url = overrideUrl
? overrideUrl
: Settings.DEBRIDIO_URL + (configString ? configString + '/' : '');
super(
addonName,
url,
addonId,
userConfig,
indexerTimeout || Settings.DEFAULT_DEBRIDIO_TIMEOUT,
Settings.DEFAULT_DEBRIDIO_USER_AGENT
? { 'User-Agent': Settings.DEFAULT_DEBRIDIO_USER_AGENT }
: undefined
);
}
protected parseStream(stream: Stream): ParseResult {
const parsedStream = super.parseStream(stream);
if (stream.url && parsedStream.type === 'stream') {
parsedStream.result.filename = stream.description?.split('\n')[0];
if (
Settings.FORCE_DEBRIDIO_HOSTNAME !== undefined ||
Settings.FORCE_DEBRIDIO_PORT !== undefined ||
Settings.FORCE_DEBRIDIO_PROTOCOL !== undefined
) {
// modify the URL according to settings, needed when using a local URL for requests but a public stream URL is needed.
const url = new URL(stream.url);
if (Settings.FORCE_DEBRIDIO_PROTOCOL !== undefined) {
url.protocol = Settings.FORCE_DEBRIDIO_PROTOCOL;
}
if (Settings.FORCE_DEBRIDIO_PORT !== undefined) {
url.port = Settings.FORCE_DEBRIDIO_PORT.toString();
}
if (Settings.FORCE_DEBRIDIO_HOSTNAME !== undefined) {
url.hostname = Settings.FORCE_DEBRIDIO_HOSTNAME;
}
parsedStream.result.url = url.toString();
}
}
return parsedStream;
}
}
const getDebridioConfigString = (provider: string, apiKey: string) => {
const config = {
provider,
apiKey,
disableUncached: false,
qualityOrder: [],
excludeSize: '',
maxReturnPerQuality: '',
};
return Buffer.from(JSON.stringify(config)).toString('base64');
};
export async function getDebridioStreams(
config: Config,
debridioOptions: {
prioritiseDebrid?: string;
overrideName?: string;
overrideUrl?: string;
indexerTimeout?: string;
},
streamRequest: StreamRequest,
addonId: string
): Promise<{
addonStreams: ParsedStream[];
addonErrors: string[];
}> {
const supportedServices: string[] =
addonDetails.find((addon: AddonDetail) => addon.id === 'debridio')
?.supportedServices || [];
const addonStreams: ParsedStream[] = [];
const indexerTimeout = debridioOptions.indexerTimeout
? parseInt(debridioOptions.indexerTimeout)
: undefined;
// If overrideUrl is provided, use it to get streams and skip all other steps
if (debridioOptions.overrideUrl) {
const debridio = new Debridio(
null,
debridioOptions.overrideUrl as string,
debridioOptions.overrideName,
addonId,
config,
indexerTimeout
);
return debridio.getParsedStreams(streamRequest);
}
// find all usable and enabled services
const usableServices = config.services.filter(
(service) => supportedServices.includes(service.id) && service.enabled
);
// if no usable services found, throw an error
if (usableServices.length < 1) {
throw new Error('No supported service(s) enabled');
}
// otherwise, depending on the configuration, create multiple instances of comet or use a single instance with the prioritised service
if (
debridioOptions.prioritiseDebrid &&
!supportedServices.includes(debridioOptions.prioritiseDebrid)
) {
throw new Error('Invalid debrid service');
}
if (debridioOptions.prioritiseDebrid) {
const debridService = usableServices.find(
(service) => service.id === debridioOptions.prioritiseDebrid
);
if (!debridService) {
throw new Error(
'Debrid service not found for ' + debridioOptions.prioritiseDebrid
);
}
if (!debridService.credentials.apiKey) {
throw new Error(
'Debrid service API key not found for ' +
debridioOptions.prioritiseDebrid
);
}
// get the comet config and b64 encode it
const debridioConfigString = getDebridioConfigString(
debridioOptions.prioritiseDebrid,
debridService.credentials.apiKey
);
const debridio = new Debridio(
debridioConfigString,
null,
debridioOptions.overrideName,
addonId,
config,
indexerTimeout
);
return debridio.getParsedStreams(streamRequest);
}
// if no prioritised service is provided, create a debridio instance for each service
const servicesToUse = usableServices.filter((service) => service.enabled);
if (servicesToUse.length < 1) {
throw new Error('No supported service(s) enabled');
}
const addonErrors: string[] = [];
const streamPromises = servicesToUse.map(async (service) => {
logger.info(`Getting Debridio streams for ${service.name}`, {
func: 'debridio',
});
const debridioConfigString = getDebridioConfigString(
service.id,
service.credentials.apiKey
);
const debridio = new Debridio(
debridioConfigString,
null,
debridioOptions.overrideName,
addonId,
config,
indexerTimeout
);
return debridio.getParsedStreams(streamRequest);
});
const streamsArray = await Promise.allSettled(streamPromises);
streamsArray.forEach((result) => {
if (result.status === 'fulfilled') {
addonStreams.push(...result.value.addonStreams);
addonErrors.push(...result.value.addonErrors);
} else {
addonErrors.push(result.reason.message);
}
});
return { addonStreams, addonErrors };
}