mirror of
https://github.com/wkulhanek/bonob.git
synced 2025-12-21 17:33:29 +01:00
Compare commits
22 Commits
v0.6.0
...
feature/nd
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
add87e5df9 | ||
|
|
38f53168fa | ||
|
|
166a4b5ec2 | ||
|
|
eb66393fe6 | ||
|
|
730524d7a1 | ||
|
|
1b14b88fb4 | ||
|
|
d2f13416f6 | ||
|
|
2997e5ac3b | ||
|
|
d1ff224e89 | ||
|
|
ac266a3c46 | ||
|
|
25857d7e5a | ||
|
|
50cb5b2550 | ||
|
|
e37a09c266 | ||
|
|
88661d7c26 | ||
|
|
6ad39ce044 | ||
|
|
1c94a6d565 | ||
|
|
00944a7a25 | ||
|
|
c7352aefa3 | ||
|
|
192f65a56b | ||
|
|
9b3df4ce1a | ||
|
|
df9a6d4663 | ||
|
|
d0c80b2f20 |
2
.github/workflows/ci.yml
vendored
2
.github/workflows/ci.yml
vendored
@@ -62,7 +62,7 @@ jobs:
|
|||||||
uses: docker/build-push-action@v2
|
uses: docker/build-push-action@v2
|
||||||
with:
|
with:
|
||||||
context: .
|
context: .
|
||||||
platforms: linux/amd64,linux/arm/v7
|
platforms: linux/amd64,linux/arm/v7,linux/arm64
|
||||||
push: ${{ github.event_name != 'pull_request' }}
|
push: ${{ github.event_name != 'pull_request' }}
|
||||||
tags: ${{ steps.meta.outputs.tags }}
|
tags: ${{ steps.meta.outputs.tags }}
|
||||||
labels: ${{ steps.meta.outputs.labels }}
|
labels: ${{ steps.meta.outputs.labels }}
|
||||||
10
README.md
10
README.md
@@ -209,10 +209,16 @@ In this case you could set;
|
|||||||
BNB_SUBSONIC_CUSTOM_CLIENTS="audio/flac"
|
BNB_SUBSONIC_CUSTOM_CLIENTS="audio/flac"
|
||||||
```
|
```
|
||||||
|
|
||||||
This would result in 2 players in Navidrome, one called 'bonob', the other called 'bonob+audio/flac'. You could then configure a custom flac transcoder in Navidrome that re-samples the flacs to a sonos supported format, ie [Using something like this](https://stackoverflow.com/questions/41420391/ffmpeg-flac-24-bit-96khz-to-16-bit-48khz);
|
This would result in 2 players in Navidrome, one called 'bonob', the other called 'bonob+audio/flac'. You could then configure a custom flac transcoder in Navidrome that re-samples the flacs to a sonos supported format, ie [Using something like this](https://stackoverflow.com/questions/41420391/ffmpeg-flac-24-bit-96khz-to-16-bit-48khz) or [this](https://stackoverflow.com/questions/52119489/ffmpeg-limit-audio-sample-rate):
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
ffmpeg -i %s -af aresample=resampler=soxr:out_sample_fmt=s16:out_sample_rate=48000 -f flac -
|
ffmpeg -i %s -af aformat=sample_fmts=s16|s32:sample_rates=8000|11025|16000|22050|24000|32000|44100|48000 -f flac -
|
||||||
|
```
|
||||||
|
|
||||||
|
**Note for Sonos S1:** [24-bit depth is only supported by Sonos S2](https://support.sonos.com/s/article/79?language=en_US), so if your system is still on Sonos S1, transcoding should convert all FLACs to 16-bit:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
ffmpeg -i %s -af aformat=sample_fmts=s16:sample_rates=8000|11025|16000|22050|24000|32000|44100|48000 -f flac -
|
||||||
```
|
```
|
||||||
|
|
||||||
### Changing Icon colors
|
### Changing Icon colors
|
||||||
|
|||||||
@@ -27,8 +27,8 @@ services:
|
|||||||
BNB_URL: http://192.168.1.111:4534
|
BNB_URL: http://192.168.1.111:4534
|
||||||
BNB_SECRET: changeme
|
BNB_SECRET: changeme
|
||||||
BNB_SONOS_SERVICE_ID: 246
|
BNB_SONOS_SERVICE_ID: 246
|
||||||
BNB_SONOS_AUTO_REGISTER: "true"
|
BNB_SONOS_AUTO_REGISTER: true
|
||||||
BNB_SONOS_DEVICE_DISCOVERY: "true"
|
BNB_SONOS_DEVICE_DISCOVERY: true
|
||||||
# ip address of one of your sonos devices
|
# ip address of one of your sonos devices
|
||||||
BNB_SONOS_SEED_HOST: 192.168.1.121
|
BNB_SONOS_SEED_HOST: 192.168.1.121
|
||||||
BNB_SUBSONIC_URL: http://navidrome:4533
|
BNB_SUBSONIC_URL: http://navidrome:4533
|
||||||
|
|||||||
@@ -60,8 +60,8 @@
|
|||||||
"scripts": {
|
"scripts": {
|
||||||
"clean": "rm -Rf build node_modules",
|
"clean": "rm -Rf build node_modules",
|
||||||
"build": "tsc",
|
"build": "tsc",
|
||||||
"dev": "BNB_DEBUG=true BNB_SCROBBLE_TRACKS=false BNB_REPORT_NOW_PLAYING=false BNB_ICON_FOREGROUND_COLOR=white BNB_ICON_BACKGROUND_COLOR=darkgrey BNB_SONOS_SERVICE_NAME=bonobDev BNB_SONOS_DEVICE_DISCOVERY=true nodemon -V ./src/app.ts",
|
"dev": "BNB_DEBUG=true BNB_SCROBBLE_TRACKS=false BNB_REPORT_NOW_PLAYING=false BNB_ICON_FOREGROUND_COLOR=white BNB_ICON_BACKGROUND_COLOR=darkgrey BNB_SONOS_SERVICE_NAME=z_bonobDev BNB_SONOS_DEVICE_DISCOVERY=true nodemon -V ./src/app.ts",
|
||||||
"devr": "BNB_DEBUG=true BNB_SCROBBLE_TRACKS=false BNB_REPORT_NOW_PLAYING=false BNB_ICON_FOREGROUND_COLOR=white BNB_ICON_BACKGROUND_COLOR=darkgrey BNB_SONOS_SERVICE_NAME=bonobDev BNB_SONOS_DEVICE_DISCOVERY=true BNB_SONOS_AUTO_REGISTER=true nodemon -V ./src/app.ts",
|
"devr": "BNB_DEBUG=true BNB_SCROBBLE_TRACKS=false BNB_REPORT_NOW_PLAYING=false BNB_ICON_FOREGROUND_COLOR=white BNB_ICON_BACKGROUND_COLOR=darkgrey BNB_SONOS_SERVICE_NAME=z_bonobDev BNB_SONOS_DEVICE_DISCOVERY=true BNB_SONOS_AUTO_REGISTER=true nodemon -V ./src/app.ts",
|
||||||
"register-dev": "ts-node ./src/register.ts http://$(hostname):4534",
|
"register-dev": "ts-node ./src/register.ts http://$(hostname):4534",
|
||||||
"test": "jest",
|
"test": "jest",
|
||||||
"gitinfo": "git describe --tags > .gitinfo"
|
"gitinfo": "git describe --tags > .gitinfo"
|
||||||
|
|||||||
@@ -5,8 +5,6 @@ import logger from "./logger";
|
|||||||
|
|
||||||
import {
|
import {
|
||||||
appendMimeTypeToClientFor,
|
appendMimeTypeToClientFor,
|
||||||
axiosImageFetcher,
|
|
||||||
cachingImageFetcher,
|
|
||||||
DEFAULT,
|
DEFAULT,
|
||||||
Subsonic,
|
Subsonic,
|
||||||
} from "./subsonic";
|
} from "./subsonic";
|
||||||
@@ -17,6 +15,7 @@ import sonos, { bonobService } from "./sonos";
|
|||||||
import { MusicService } from "./music_service";
|
import { MusicService } from "./music_service";
|
||||||
import { SystemClock } from "./clock";
|
import { SystemClock } from "./clock";
|
||||||
import { JWTSmapiLoginTokens } from "./smapi_auth";
|
import { JWTSmapiLoginTokens } from "./smapi_auth";
|
||||||
|
import { axiosImageFetcher, cachingImageFetcher } from "./images";
|
||||||
|
|
||||||
const config = readConfig();
|
const config = readConfig();
|
||||||
const clock = SystemClock;
|
const clock = SystemClock;
|
||||||
@@ -32,6 +31,7 @@ const bonob = bonobService(
|
|||||||
|
|
||||||
const sonosSystem = sonos(config.sonos.discovery);
|
const sonosSystem = sonos(config.sonos.discovery);
|
||||||
|
|
||||||
|
// todo: just pass in the customClientsForStringArray into subsonic and make it sort it out.
|
||||||
const streamUserAgent = config.subsonic.customClientsFor
|
const streamUserAgent = config.subsonic.customClientsFor
|
||||||
? appendMimeTypeToClientFor(config.subsonic.customClientsFor.split(","))
|
? appendMimeTypeToClientFor(config.subsonic.customClientsFor.split(","))
|
||||||
: DEFAULT;
|
: DEFAULT;
|
||||||
|
|||||||
44
src/clock.ts
44
src/clock.ts
@@ -1,13 +1,38 @@
|
|||||||
import dayjs, { Dayjs } from "dayjs";
|
import dayjs, { Dayjs } from "dayjs";
|
||||||
|
|
||||||
export const isChristmas = (clock: Clock = SystemClock) => clock.now().month() == 11 && clock.now().date() == 25;
|
function fixedDateMonthEvent(dateMonth: string) {
|
||||||
export const isMay4 = (clock: Clock = SystemClock) => clock.now().month() == 4 && clock.now().date() == 4;
|
const date = Number.parseInt(dateMonth.split("/")[0]!);
|
||||||
export const isHalloween = (clock: Clock = SystemClock) => clock.now().month() == 9 && clock.now().date() == 31
|
const month = Number.parseInt(dateMonth.split("/")[1]!);
|
||||||
export const isHoli = (clock: Clock = SystemClock) => ["2022/03/18", "2023/03/07", "2024/03/25", "2025/03/14"].map(dayjs).find(it => it.isSame(clock.now())) != undefined
|
return (clock: Clock = SystemClock) => {
|
||||||
export const isCNY = (clock: Clock = SystemClock) => ["2022/02/01", "2023/01/22", "2024/02/10", "2025/02/29"].map(dayjs).find(it => it.isSame(clock.now())) != undefined
|
return clock.now().date() == date && clock.now().month() == month - 1;
|
||||||
export const isCNY_2022 = (clock: Clock = SystemClock) => clock.now().isSame(dayjs("2022/02/01"))
|
};
|
||||||
export const isCNY_2023 = (clock: Clock = SystemClock) => clock.now().isSame(dayjs("2023/01/22"))
|
}
|
||||||
export const isCNY_2024 = (clock: Clock = SystemClock) => clock.now().isSame(dayjs("2024/02/10"))
|
|
||||||
|
function fixedDateEvent(date: string) {
|
||||||
|
const dayjsDate = dayjs(date);
|
||||||
|
return (clock: Clock = SystemClock) => {
|
||||||
|
return clock.now().isSame(dayjsDate, "day");
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function anyOf(rules: ((clock: Clock) => boolean)[]) {
|
||||||
|
return (clock: Clock = SystemClock) => {
|
||||||
|
return rules.find((rule) => rule(clock)) != undefined;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export const isChristmas = fixedDateMonthEvent("25/12");
|
||||||
|
export const isMay4 = fixedDateMonthEvent("04/05");
|
||||||
|
export const isHalloween = fixedDateMonthEvent("31/10");
|
||||||
|
export const isHoli = anyOf(
|
||||||
|
["2022/03/18", "2023/03/07", "2024/03/25", "2025/03/14"].map(fixedDateEvent)
|
||||||
|
)
|
||||||
|
|
||||||
|
export const isCNY_2022 = fixedDateEvent("2022/02/01");
|
||||||
|
export const isCNY_2023 = fixedDateEvent("2023/01/22");
|
||||||
|
export const isCNY_2024 = fixedDateEvent("2024/02/10");
|
||||||
|
export const isCNY_2025 = fixedDateEvent("2025/02/29");
|
||||||
|
export const isCNY = anyOf([isCNY_2022, isCNY_2023, isCNY_2024, isCNY_2025]);
|
||||||
|
|
||||||
export interface Clock {
|
export interface Clock {
|
||||||
now(): Dayjs;
|
now(): Dayjs;
|
||||||
@@ -22,7 +47,8 @@ export class FixedClock implements Clock {
|
|||||||
this.time = time;
|
this.time = time;
|
||||||
}
|
}
|
||||||
|
|
||||||
add = (t: number, unit: dayjs.UnitTypeShort) => this.time = this.time.add(t, unit)
|
add = (t: number, unit: dayjs.UnitTypeShort) =>
|
||||||
|
(this.time = this.time.add(t, unit));
|
||||||
|
|
||||||
now = () => this.time;
|
now = () => this.time;
|
||||||
}
|
}
|
||||||
@@ -5,20 +5,22 @@ import url from "./url_builder";
|
|||||||
export const WORD = /^\w+$/;
|
export const WORD = /^\w+$/;
|
||||||
export const COLOR = /^#?\w+$/;
|
export const COLOR = /^#?\w+$/;
|
||||||
|
|
||||||
type EnvVarOpts = {
|
type EnvVarOpts<T> = {
|
||||||
default: string | undefined;
|
default: T | undefined;
|
||||||
legacy: string[] | undefined;
|
legacy: string[] | undefined;
|
||||||
validationPattern: RegExp | undefined;
|
validationPattern: RegExp | undefined;
|
||||||
|
parser: ((value: string) => T) | undefined
|
||||||
};
|
};
|
||||||
|
|
||||||
export function envVar(
|
export function envVar<T>(
|
||||||
name: string,
|
name: string,
|
||||||
opts: Partial<EnvVarOpts> = {
|
opts: Partial<EnvVarOpts<T>> = {
|
||||||
default: undefined,
|
default: undefined,
|
||||||
legacy: undefined,
|
legacy: undefined,
|
||||||
validationPattern: undefined,
|
validationPattern: undefined,
|
||||||
|
parser: undefined
|
||||||
}
|
}
|
||||||
) {
|
): T {
|
||||||
const result = [name, ...(opts.legacy || [])]
|
const result = [name, ...(opts.legacy || [])]
|
||||||
.map((it) => ({ key: it, value: process.env[it] }))
|
.map((it) => ({ key: it, value: process.env[it] }))
|
||||||
.find((it) => it.value);
|
.find((it) => it.value);
|
||||||
@@ -36,17 +38,28 @@ export function envVar(
|
|||||||
logger.warn(`Configuration key '${result.key}' is deprecated, replace with '${name}'`)
|
logger.warn(`Configuration key '${result.key}' is deprecated, replace with '${name}'`)
|
||||||
}
|
}
|
||||||
|
|
||||||
return result?.value || opts.default;
|
let value: T | undefined = undefined;
|
||||||
|
|
||||||
|
if(result?.value && opts.parser) {
|
||||||
|
value = opts.parser(result?.value)
|
||||||
|
} else if(result?.value)
|
||||||
|
value = result?.value as any as T
|
||||||
|
|
||||||
|
return value == undefined ? opts.default as T : value;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const bnbEnvVar = (key: string, opts: Partial<EnvVarOpts> = {}) =>
|
export const bnbEnvVar = <T>(key: string, opts: Partial<EnvVarOpts<T>> = {}) =>
|
||||||
envVar(`BNB_${key}`, {
|
envVar(`BNB_${key}`, {
|
||||||
...opts,
|
...opts,
|
||||||
legacy: [`BONOB_${key}`, ...(opts.legacy || [])],
|
legacy: [`BONOB_${key}`, ...(opts.legacy || [])],
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const asBoolean = (value: string) => value == "true";
|
||||||
|
|
||||||
|
const asInt = (value: string) => Number.parseInt(value);
|
||||||
|
|
||||||
export default function () {
|
export default function () {
|
||||||
const port = +bnbEnvVar("PORT", { default: "4534" })!;
|
const port = bnbEnvVar<number>("PORT", { default: 4534, parser: asInt })!;
|
||||||
const bonobUrl = bnbEnvVar("URL", {
|
const bonobUrl = bnbEnvVar("URL", {
|
||||||
legacy: ["BONOB_WEB_ADDRESS"],
|
legacy: ["BONOB_WEB_ADDRESS"],
|
||||||
default: `http://${hostname()}:${port}`,
|
default: `http://${hostname()}:${port}`,
|
||||||
@@ -62,34 +75,34 @@ export default function () {
|
|||||||
return {
|
return {
|
||||||
port,
|
port,
|
||||||
bonobUrl: url(bonobUrl),
|
bonobUrl: url(bonobUrl),
|
||||||
secret: bnbEnvVar("SECRET", { default: "bonob" })!,
|
secret: bnbEnvVar<string>("SECRET", { default: "bonob" })!,
|
||||||
authTimeout: bnbEnvVar("AUTH_TIMEOUT", { default: "1h" })!,
|
authTimeout: bnbEnvVar<string>("AUTH_TIMEOUT", { default: "1h" })!,
|
||||||
icons: {
|
icons: {
|
||||||
foregroundColor: bnbEnvVar("ICON_FOREGROUND_COLOR", {
|
foregroundColor: bnbEnvVar<string>("ICON_FOREGROUND_COLOR", {
|
||||||
validationPattern: COLOR,
|
validationPattern: COLOR,
|
||||||
}),
|
}),
|
||||||
backgroundColor: bnbEnvVar("ICON_BACKGROUND_COLOR", {
|
backgroundColor: bnbEnvVar<string>("ICON_BACKGROUND_COLOR", {
|
||||||
validationPattern: COLOR,
|
validationPattern: COLOR,
|
||||||
}),
|
}),
|
||||||
},
|
},
|
||||||
sonos: {
|
sonos: {
|
||||||
serviceName: bnbEnvVar("SONOS_SERVICE_NAME", { default: "bonob" })!,
|
serviceName: bnbEnvVar<string>("SONOS_SERVICE_NAME", { default: "bonob" })!,
|
||||||
discovery: {
|
discovery: {
|
||||||
enabled:
|
enabled:
|
||||||
bnbEnvVar("SONOS_DEVICE_DISCOVERY", { default: "true" }) == "true",
|
bnbEnvVar<boolean>("SONOS_DEVICE_DISCOVERY", { default: true, parser: asBoolean }),
|
||||||
seedHost: bnbEnvVar("SONOS_SEED_HOST"),
|
seedHost: bnbEnvVar<string>("SONOS_SEED_HOST"),
|
||||||
},
|
},
|
||||||
autoRegister:
|
autoRegister:
|
||||||
bnbEnvVar("SONOS_AUTO_REGISTER", { default: "false" }) == "true",
|
bnbEnvVar<boolean>("SONOS_AUTO_REGISTER", { default: false, parser: asBoolean }),
|
||||||
sid: Number(bnbEnvVar("SONOS_SERVICE_ID", { default: "246" })),
|
sid: bnbEnvVar<number>("SONOS_SERVICE_ID", { default: 246, parser: asInt }),
|
||||||
},
|
},
|
||||||
subsonic: {
|
subsonic: {
|
||||||
url: bnbEnvVar("SUBSONIC_URL", { legacy: ["BONOB_NAVIDROME_URL"], default: `http://${hostname()}:4533` })!,
|
url: bnbEnvVar("SUBSONIC_URL", { legacy: ["BONOB_NAVIDROME_URL"], default: `http://${hostname()}:4533` })!,
|
||||||
customClientsFor: bnbEnvVar("SUBSONIC_CUSTOM_CLIENTS", { legacy: ["BONOB_NAVIDROME_CUSTOM_CLIENTS"] }),
|
customClientsFor: bnbEnvVar<string>("SUBSONIC_CUSTOM_CLIENTS", { legacy: ["BONOB_NAVIDROME_CUSTOM_CLIENTS"] }),
|
||||||
artistImageCache: bnbEnvVar("SUBSONIC_ARTIST_IMAGE_CACHE"),
|
artistImageCache: bnbEnvVar<string>("SUBSONIC_ARTIST_IMAGE_CACHE"),
|
||||||
},
|
},
|
||||||
scrobbleTracks: bnbEnvVar("SCROBBLE_TRACKS", { default: "true" }) == "true",
|
scrobbleTracks: bnbEnvVar<boolean>("SCROBBLE_TRACKS", { default: true, parser: asBoolean }),
|
||||||
reportNowPlaying:
|
reportNowPlaying:
|
||||||
bnbEnvVar("REPORT_NOW_PLAYING", { default: "true" }) == "true",
|
bnbEnvVar<boolean>("REPORT_NOW_PLAYING", { default: true, parser: asBoolean }),
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
67
src/http.ts
Normal file
67
src/http.ts
Normal file
@@ -0,0 +1,67 @@
|
|||||||
|
import {
|
||||||
|
AxiosPromise,
|
||||||
|
AxiosRequestConfig,
|
||||||
|
Method,
|
||||||
|
ResponseType,
|
||||||
|
} from "axios";
|
||||||
|
|
||||||
|
// todo: do i need this anymore?
|
||||||
|
export interface Http {
|
||||||
|
(config: AxiosRequestConfig): AxiosPromise<any>;
|
||||||
|
}
|
||||||
|
export interface Http2 extends Http {
|
||||||
|
with: (params: Partial<RequestParams>) => Http2;
|
||||||
|
}
|
||||||
|
|
||||||
|
export type RequestParams = {
|
||||||
|
baseURL: string;
|
||||||
|
url: string;
|
||||||
|
params: any;
|
||||||
|
headers: any;
|
||||||
|
responseType: ResponseType;
|
||||||
|
method: Method;
|
||||||
|
};
|
||||||
|
|
||||||
|
const wrap = (http2: Http2, params: Partial<RequestParams>): Http2 => {
|
||||||
|
const f = ((config: AxiosRequestConfig) => http2(merge(params, config))) as Http2;
|
||||||
|
f.with = (params: Partial<RequestParams>) => wrap(f, params);
|
||||||
|
return f;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const http2From = (http: Http): Http2 => {
|
||||||
|
const f = ((config: AxiosRequestConfig) => http(config)) as Http2;
|
||||||
|
f.with = (defaults: Partial<RequestParams>) => wrap(f, defaults);
|
||||||
|
return f;
|
||||||
|
}
|
||||||
|
|
||||||
|
const merge = (
|
||||||
|
defaults: Partial<RequestParams>,
|
||||||
|
config: AxiosRequestConfig
|
||||||
|
) => {
|
||||||
|
let toApply = {
|
||||||
|
...defaults,
|
||||||
|
...config,
|
||||||
|
};
|
||||||
|
if (defaults.params) {
|
||||||
|
toApply = {
|
||||||
|
...toApply,
|
||||||
|
params: {
|
||||||
|
...defaults.params,
|
||||||
|
...config.params,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
if (defaults.headers) {
|
||||||
|
toApply = {
|
||||||
|
...toApply,
|
||||||
|
headers: {
|
||||||
|
...defaults.headers,
|
||||||
|
...config.headers,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
return toApply;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const http =
|
||||||
|
(base: Http, defaults: Partial<RequestParams>): Http => (config: AxiosRequestConfig) => base(merge(defaults, config));
|
||||||
48
src/images.ts
Normal file
48
src/images.ts
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
|
||||||
|
import sharp from "sharp";
|
||||||
|
import fse from "fs-extra";
|
||||||
|
import path from "path";
|
||||||
|
import { Md5 } from "ts-md5/dist/md5";
|
||||||
|
import axios from "axios";
|
||||||
|
|
||||||
|
import { CoverArt } from "./music_service";
|
||||||
|
import { BROWSER_HEADERS } from "./utils";
|
||||||
|
|
||||||
|
export type ImageFetcher = (url: string) => Promise<CoverArt | undefined>;
|
||||||
|
|
||||||
|
export const cachingImageFetcher =
|
||||||
|
(cacheDir: string, delegate: ImageFetcher) =>
|
||||||
|
async (url: string): Promise<CoverArt | undefined> => {
|
||||||
|
const filename = path.join(cacheDir, `${Md5.hashStr(url)}.png`);
|
||||||
|
return fse
|
||||||
|
.readFile(filename)
|
||||||
|
.then((data) => ({ contentType: "image/png", data }))
|
||||||
|
.catch(() =>
|
||||||
|
delegate(url).then((image) => {
|
||||||
|
if (image) {
|
||||||
|
return sharp(image.data)
|
||||||
|
.png()
|
||||||
|
.toBuffer()
|
||||||
|
.then((png) => {
|
||||||
|
return fse
|
||||||
|
.writeFile(filename, png)
|
||||||
|
.then(() => ({ contentType: "image/png", data: png }));
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
})
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export const axiosImageFetcher = (url: string): Promise<CoverArt | undefined> =>
|
||||||
|
axios
|
||||||
|
.get(url, {
|
||||||
|
headers: BROWSER_HEADERS,
|
||||||
|
responseType: "arraybuffer",
|
||||||
|
})
|
||||||
|
.then((res) => ({
|
||||||
|
contentType: res.headers["content-type"],
|
||||||
|
data: Buffer.from(res.data, "binary"),
|
||||||
|
}))
|
||||||
|
.catch(() => undefined);
|
||||||
@@ -15,7 +15,13 @@ export class AuthFailure extends Error {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export type IdName = {
|
||||||
|
id: string;
|
||||||
|
name: string;
|
||||||
|
};
|
||||||
|
|
||||||
export type ArtistSummary = {
|
export type ArtistSummary = {
|
||||||
|
// todo: why can this be undefined?
|
||||||
id: string | undefined;
|
id: string | undefined;
|
||||||
name: string;
|
name: string;
|
||||||
image: BUrn | undefined;
|
image: BUrn | undefined;
|
||||||
@@ -65,8 +71,8 @@ export type Track = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
export type Paging = {
|
export type Paging = {
|
||||||
_index: number;
|
_index: number | undefined;
|
||||||
_count: number;
|
_count: number | undefined;
|
||||||
};
|
};
|
||||||
|
|
||||||
export type Result<T> = {
|
export type Result<T> = {
|
||||||
@@ -74,9 +80,10 @@ export type Result<T> = {
|
|||||||
total: number;
|
total: number;
|
||||||
};
|
};
|
||||||
|
|
||||||
export function slice2<T>({ _index, _count }: Paging) {
|
export function slice2<T>({ _index, _count }: Partial<Paging> = {}) {
|
||||||
|
const i = _index || 0;
|
||||||
return (things: T[]): [T[], number] => [
|
return (things: T[]): [T[], number] => [
|
||||||
things.slice(_index, _index + _count),
|
_count ? things.slice(i, i + _count) : things.slice(i),
|
||||||
things.length,
|
things.length,
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
@@ -138,6 +145,10 @@ export type Playlist = PlaylistSummary & {
|
|||||||
entries: Track[]
|
entries: Track[]
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export type Sortable = {
|
||||||
|
sortName: string
|
||||||
|
}
|
||||||
|
|
||||||
export const range = (size: number) => [...Array(size).keys()];
|
export const range = (size: number) => [...Array(size).keys()];
|
||||||
|
|
||||||
export const asArtistAlbumPairs = (artists: Artist[]): [Artist, Album][] =>
|
export const asArtistAlbumPairs = (artists: Artist[]): [Artist, Album][] =>
|
||||||
@@ -152,7 +163,7 @@ export interface MusicService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export interface MusicLibrary {
|
export interface MusicLibrary {
|
||||||
artists(q: ArtistQuery): Promise<Result<ArtistSummary>>;
|
artists(q: ArtistQuery): Promise<Result<ArtistSummary & Sortable>>;
|
||||||
artist(id: string): Promise<Artist>;
|
artist(id: string): Promise<Artist>;
|
||||||
albums(q: AlbumQuery): Promise<Result<AlbumSummary>>;
|
albums(q: AlbumQuery): Promise<Result<AlbumSummary>>;
|
||||||
album(id: string): Promise<Album>;
|
album(id: string): Promise<Album>;
|
||||||
|
|||||||
@@ -33,13 +33,10 @@ import makeI8N, { asLANGs, KEY, keys as i8nKeys, LANG } from "./i8n";
|
|||||||
import { Icon, ICONS, festivals, features } from "./icon";
|
import { Icon, ICONS, festivals, features } from "./icon";
|
||||||
import _, { shuffle } from "underscore";
|
import _, { shuffle } from "underscore";
|
||||||
import morgan from "morgan";
|
import morgan from "morgan";
|
||||||
import { takeWithRepeats } from "./utils";
|
import { mask, takeWithRepeats } from "./utils";
|
||||||
import { parse } from "./burn";
|
import { parse } from "./burn";
|
||||||
import { axiosImageFetcher, ImageFetcher } from "./subsonic";
|
import { axiosImageFetcher, ImageFetcher } from "./images";
|
||||||
import {
|
import { JWTSmapiLoginTokens, SmapiAuthTokens } from "./smapi_auth";
|
||||||
JWTSmapiLoginTokens,
|
|
||||||
SmapiAuthTokens,
|
|
||||||
} from "./smapi_auth";
|
|
||||||
|
|
||||||
export const BONOB_ACCESS_TOKEN_HEADER = "bat";
|
export const BONOB_ACCESS_TOKEN_HEADER = "bat";
|
||||||
|
|
||||||
@@ -377,23 +374,28 @@ function server(
|
|||||||
logger.info(
|
logger.info(
|
||||||
`${trace} bnb<- ${req.method} ${req.path}?${JSON.stringify(
|
`${trace} bnb<- ${req.method} ${req.path}?${JSON.stringify(
|
||||||
req.query
|
req.query
|
||||||
)}, headers=${JSON.stringify({ ...req.headers, "bnbt": "*****", "bnbk": "*****" })}`
|
)}, headers=${JSON.stringify(mask(req.headers, ["bnbt", "bnbk"]))}`
|
||||||
);
|
);
|
||||||
|
|
||||||
const serviceToken = pipe(
|
const serviceToken = pipe(
|
||||||
E.fromNullable("Missing bnbt header")(req.headers["bnbt"] as string),
|
E.fromNullable("Missing bnbt header")(req.headers["bnbt"] as string),
|
||||||
E.chain(token => pipe(
|
E.chain((token) =>
|
||||||
|
pipe(
|
||||||
E.fromNullable("Missing bnbk header")(req.headers["bnbk"] as string),
|
E.fromNullable("Missing bnbk header")(req.headers["bnbk"] as string),
|
||||||
E.map(key => ({ token, key }))
|
E.map((key) => ({ token, key }))
|
||||||
)),
|
)
|
||||||
|
),
|
||||||
E.chain((auth) =>
|
E.chain((auth) =>
|
||||||
pipe(
|
pipe(
|
||||||
smapiAuthTokens.verify(auth),
|
smapiAuthTokens.verify(auth),
|
||||||
E.mapLeft((_) => "Auth token failed to verify")
|
E.mapLeft((_) => "Auth token failed to verify")
|
||||||
)
|
)
|
||||||
),
|
),
|
||||||
E.getOrElseW(() => undefined)
|
E.getOrElseW((e: string) => {
|
||||||
)
|
logger.error(`Failed to get serviceToken for stream: ${e}`);
|
||||||
|
return undefined;
|
||||||
|
})
|
||||||
|
);
|
||||||
|
|
||||||
if (!serviceToken) {
|
if (!serviceToken) {
|
||||||
return res.status(401).send();
|
return res.status(401).send();
|
||||||
|
|||||||
67
src/smapi.ts
67
src/smapi.ts
@@ -19,6 +19,7 @@ import {
|
|||||||
Playlist,
|
Playlist,
|
||||||
Rating,
|
Rating,
|
||||||
slice2,
|
slice2,
|
||||||
|
Sortable,
|
||||||
Track,
|
Track,
|
||||||
} from "./music_service";
|
} from "./music_service";
|
||||||
import { APITokens } from "./api_tokens";
|
import { APITokens } from "./api_tokens";
|
||||||
@@ -366,6 +367,54 @@ export const artist = (bonobUrl: URLBuilder, artist: ArtistSummary) => ({
|
|||||||
albumArtURI: defaultArtistArtURI(bonobUrl, artist).href(),
|
albumArtURI: defaultArtistArtURI(bonobUrl, artist).href(),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
export const scrollIndicesFrom = (things: Sortable[]) => {
|
||||||
|
const indicies: Record<string, number | undefined> = {
|
||||||
|
"A":undefined,
|
||||||
|
"B":undefined,
|
||||||
|
"C":undefined,
|
||||||
|
"D":undefined,
|
||||||
|
"E":undefined,
|
||||||
|
"F":undefined,
|
||||||
|
"G":undefined,
|
||||||
|
"H":undefined,
|
||||||
|
"I":undefined,
|
||||||
|
"J":undefined,
|
||||||
|
"K":undefined,
|
||||||
|
"L":undefined,
|
||||||
|
"M":undefined,
|
||||||
|
"N":undefined,
|
||||||
|
"O":undefined,
|
||||||
|
"P":undefined,
|
||||||
|
"Q":undefined,
|
||||||
|
"R":undefined,
|
||||||
|
"S":undefined,
|
||||||
|
"T":undefined,
|
||||||
|
"U":undefined,
|
||||||
|
"V":undefined,
|
||||||
|
"W":undefined,
|
||||||
|
"X":undefined,
|
||||||
|
"Y":undefined,
|
||||||
|
"Z":undefined,
|
||||||
|
}
|
||||||
|
const upperNames = things.map(thing => thing.sortName.toUpperCase());
|
||||||
|
for(var i = 0; i < upperNames.length; i++) {
|
||||||
|
const char = upperNames[i]![0]!;
|
||||||
|
if(Object.keys(indicies).includes(char) && indicies[char] == undefined) {
|
||||||
|
indicies[char] = i;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
var lastIndex = 0;
|
||||||
|
const result: string[] = [];
|
||||||
|
Object.entries(indicies).forEach(([letter, index]) => {
|
||||||
|
result.push(letter);
|
||||||
|
if(index) {
|
||||||
|
lastIndex = index;
|
||||||
|
}
|
||||||
|
result.push(`${lastIndex}`);
|
||||||
|
})
|
||||||
|
return result.join(",")
|
||||||
|
}
|
||||||
|
|
||||||
function splitId<T>(id: string) {
|
function splitId<T>(id: string) {
|
||||||
const [type, typeId] = id.split(":");
|
const [type, typeId] = id.split(":");
|
||||||
return (t: T) => ({
|
return (t: T) => ({
|
||||||
@@ -707,6 +756,7 @@ function bindSmapiSoapServiceToExpress(
|
|||||||
title: lang("artists"),
|
title: lang("artists"),
|
||||||
albumArtURI: iconArtURI(bonobUrl, "artists").href(),
|
albumArtURI: iconArtURI(bonobUrl, "artists").href(),
|
||||||
itemType: "container",
|
itemType: "container",
|
||||||
|
canScroll: true,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: "albums",
|
id: "albums",
|
||||||
@@ -945,6 +995,23 @@ function bindSmapiSoapServiceToExpress(
|
|||||||
throw `Unsupported getMetadata id=${id}`;
|
throw `Unsupported getMetadata id=${id}`;
|
||||||
}
|
}
|
||||||
}),
|
}),
|
||||||
|
getScrollIndices: async (
|
||||||
|
{ id }: { id: string },
|
||||||
|
_,
|
||||||
|
soapyHeaders: SoapyHeaders
|
||||||
|
) => {
|
||||||
|
switch(id) {
|
||||||
|
case "artists": {
|
||||||
|
return login(soapyHeaders?.credentials)
|
||||||
|
.then(({ musicLibrary }) => musicLibrary.artists({ _index: 0, _count: undefined }))
|
||||||
|
.then((artists) => ({
|
||||||
|
getScrollIndicesResult: scrollIndicesFrom(artists.results)
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
throw `Unsupported getScrollIndices id=${id}`;
|
||||||
|
}
|
||||||
|
},
|
||||||
createContainer: async (
|
createContainer: async (
|
||||||
{ title, seedId }: { title: string; seedId: string | undefined },
|
{ title, seedId }: { title: string; seedId: string | undefined },
|
||||||
_,
|
_,
|
||||||
|
|||||||
978
src/subsonic.ts
978
src/subsonic.ts
@@ -1,978 +0,0 @@
|
|||||||
import { option as O, taskEither as TE } from "fp-ts";
|
|
||||||
import * as A from "fp-ts/Array";
|
|
||||||
import { ordString } from "fp-ts/lib/Ord";
|
|
||||||
import { pipe } from "fp-ts/lib/function";
|
|
||||||
import { Md5 } from "ts-md5/dist/md5";
|
|
||||||
import {
|
|
||||||
Credentials,
|
|
||||||
MusicService,
|
|
||||||
Album,
|
|
||||||
Result,
|
|
||||||
slice2,
|
|
||||||
AlbumQuery,
|
|
||||||
ArtistQuery,
|
|
||||||
MusicLibrary,
|
|
||||||
AlbumSummary,
|
|
||||||
Genre,
|
|
||||||
Track,
|
|
||||||
CoverArt,
|
|
||||||
Rating,
|
|
||||||
AlbumQueryType,
|
|
||||||
Artist,
|
|
||||||
AuthFailure,
|
|
||||||
} from "./music_service";
|
|
||||||
import sharp from "sharp";
|
|
||||||
import _ from "underscore";
|
|
||||||
import fse from "fs-extra";
|
|
||||||
import path from "path";
|
|
||||||
|
|
||||||
import axios, { AxiosRequestConfig } from "axios";
|
|
||||||
import randomstring from "randomstring";
|
|
||||||
import { b64Encode, b64Decode } from "./b64";
|
|
||||||
import logger from "./logger";
|
|
||||||
import { assertSystem, BUrn } from "./burn";
|
|
||||||
import { artist } from "./smapi";
|
|
||||||
|
|
||||||
export const BROWSER_HEADERS = {
|
|
||||||
accept:
|
|
||||||
"text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8",
|
|
||||||
"accept-encoding": "gzip, deflate, br",
|
|
||||||
"accept-language": "en-GB,en;q=0.5",
|
|
||||||
"upgrade-insecure-requests": "1",
|
|
||||||
"user-agent":
|
|
||||||
"Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:86.0) Gecko/20100101 Firefox/86.0",
|
|
||||||
};
|
|
||||||
|
|
||||||
export const t = (password: string, s: string) =>
|
|
||||||
Md5.hashStr(`${password}${s}`);
|
|
||||||
|
|
||||||
export const t_and_s = (password: string) => {
|
|
||||||
const s = randomstring.generate();
|
|
||||||
return {
|
|
||||||
t: t(password, s),
|
|
||||||
s,
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
export const DODGY_IMAGE_NAME = "2a96cbd8b46e442fc41c2b86b821562f.png";
|
|
||||||
|
|
||||||
export const isValidImage = (url: string | undefined) =>
|
|
||||||
url != undefined && !url.endsWith(DODGY_IMAGE_NAME);
|
|
||||||
|
|
||||||
type SubsonicEnvelope = {
|
|
||||||
"subsonic-response": SubsonicResponse;
|
|
||||||
};
|
|
||||||
|
|
||||||
type SubsonicResponse = {
|
|
||||||
status: string;
|
|
||||||
};
|
|
||||||
|
|
||||||
type album = {
|
|
||||||
id: string;
|
|
||||||
name: string;
|
|
||||||
artist: string | undefined;
|
|
||||||
artistId: string | undefined;
|
|
||||||
coverArt: string | undefined;
|
|
||||||
genre: string | undefined;
|
|
||||||
year: string | undefined;
|
|
||||||
};
|
|
||||||
|
|
||||||
type artist = {
|
|
||||||
id: string;
|
|
||||||
name: string;
|
|
||||||
albumCount: number;
|
|
||||||
artistImageUrl: string | undefined;
|
|
||||||
};
|
|
||||||
|
|
||||||
type GetArtistsResponse = SubsonicResponse & {
|
|
||||||
artists: {
|
|
||||||
index: {
|
|
||||||
artist: artist[];
|
|
||||||
name: string;
|
|
||||||
}[];
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
type GetAlbumListResponse = SubsonicResponse & {
|
|
||||||
albumList2: {
|
|
||||||
album: album[];
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
type genre = {
|
|
||||||
songCount: number;
|
|
||||||
albumCount: number;
|
|
||||||
value: string;
|
|
||||||
};
|
|
||||||
|
|
||||||
type GetGenresResponse = SubsonicResponse & {
|
|
||||||
genres: {
|
|
||||||
genre: genre[];
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
type SubsonicError = SubsonicResponse & {
|
|
||||||
error: {
|
|
||||||
code: string;
|
|
||||||
message: string;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
export type images = {
|
|
||||||
smallImageUrl: string | undefined;
|
|
||||||
mediumImageUrl: string | undefined;
|
|
||||||
largeImageUrl: string | undefined;
|
|
||||||
};
|
|
||||||
|
|
||||||
type artistInfo = images & {
|
|
||||||
biography: string | undefined;
|
|
||||||
musicBrainzId: string | undefined;
|
|
||||||
lastFmUrl: string | undefined;
|
|
||||||
similarArtist: artist[];
|
|
||||||
};
|
|
||||||
|
|
||||||
type ArtistSummary = IdName & {
|
|
||||||
image: BUrn | undefined;
|
|
||||||
};
|
|
||||||
|
|
||||||
type GetArtistInfoResponse = SubsonicResponse & {
|
|
||||||
artistInfo2: artistInfo;
|
|
||||||
};
|
|
||||||
|
|
||||||
type GetArtistResponse = SubsonicResponse & {
|
|
||||||
artist: artist & {
|
|
||||||
album: album[];
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
export type song = {
|
|
||||||
id: string;
|
|
||||||
parent: string | undefined;
|
|
||||||
title: string;
|
|
||||||
album: string | undefined;
|
|
||||||
albumId: string | undefined;
|
|
||||||
artist: string | undefined;
|
|
||||||
artistId: string | undefined;
|
|
||||||
track: number | undefined;
|
|
||||||
year: string | undefined;
|
|
||||||
genre: string | undefined;
|
|
||||||
coverArt: string | undefined;
|
|
||||||
created: string | undefined;
|
|
||||||
duration: number | undefined;
|
|
||||||
bitRate: number | undefined;
|
|
||||||
suffix: string | undefined;
|
|
||||||
contentType: string | undefined;
|
|
||||||
type: string | undefined;
|
|
||||||
userRating: number | undefined;
|
|
||||||
starred: string | undefined;
|
|
||||||
};
|
|
||||||
|
|
||||||
type GetAlbumResponse = {
|
|
||||||
album: album & {
|
|
||||||
song: song[];
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
type playlist = {
|
|
||||||
id: string;
|
|
||||||
name: string;
|
|
||||||
};
|
|
||||||
|
|
||||||
type GetPlaylistResponse = {
|
|
||||||
playlist: {
|
|
||||||
id: string;
|
|
||||||
name: string;
|
|
||||||
entry: song[];
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
type GetPlaylistsResponse = {
|
|
||||||
playlists: { playlist: playlist[] };
|
|
||||||
};
|
|
||||||
|
|
||||||
type GetSimilarSongsResponse = {
|
|
||||||
similarSongs2: { song: song[] };
|
|
||||||
};
|
|
||||||
|
|
||||||
type GetTopSongsResponse = {
|
|
||||||
topSongs: { song: song[] };
|
|
||||||
};
|
|
||||||
|
|
||||||
type GetSongResponse = {
|
|
||||||
song: song;
|
|
||||||
};
|
|
||||||
|
|
||||||
type GetStarredResponse = {
|
|
||||||
starred2: {
|
|
||||||
song: song[];
|
|
||||||
album: album[];
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
export type PingResponse = {
|
|
||||||
status: string;
|
|
||||||
version: string;
|
|
||||||
type: string;
|
|
||||||
serverVersion: string;
|
|
||||||
};
|
|
||||||
|
|
||||||
type Search3Response = SubsonicResponse & {
|
|
||||||
searchResult3: {
|
|
||||||
artist: artist[];
|
|
||||||
album: album[];
|
|
||||||
song: song[];
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
export function isError(
|
|
||||||
subsonicResponse: SubsonicResponse
|
|
||||||
): subsonicResponse is SubsonicError {
|
|
||||||
return (subsonicResponse as SubsonicError).error !== undefined;
|
|
||||||
}
|
|
||||||
|
|
||||||
type IdName = {
|
|
||||||
id: string;
|
|
||||||
name: string;
|
|
||||||
};
|
|
||||||
|
|
||||||
const coverArtURN = (coverArt: string | undefined): BUrn | undefined =>
|
|
||||||
pipe(
|
|
||||||
coverArt,
|
|
||||||
O.fromNullable,
|
|
||||||
O.map((it: string) => ({ system: "subsonic", resource: `art:${it}` })),
|
|
||||||
O.getOrElseW(() => undefined)
|
|
||||||
);
|
|
||||||
|
|
||||||
export const artistImageURN = (
|
|
||||||
spec: Partial<{
|
|
||||||
artistId: string | undefined;
|
|
||||||
artistImageURL: string | undefined;
|
|
||||||
}>
|
|
||||||
): BUrn | undefined => {
|
|
||||||
const deets = {
|
|
||||||
artistId: undefined,
|
|
||||||
artistImageURL: undefined,
|
|
||||||
...spec,
|
|
||||||
};
|
|
||||||
if (deets.artistImageURL && isValidImage(deets.artistImageURL)) {
|
|
||||||
return {
|
|
||||||
system: "external",
|
|
||||||
resource: deets.artistImageURL,
|
|
||||||
};
|
|
||||||
} else if (artistIsInLibrary(deets.artistId)) {
|
|
||||||
return {
|
|
||||||
system: "subsonic",
|
|
||||||
resource: `art:${deets.artistId!}`,
|
|
||||||
};
|
|
||||||
} else {
|
|
||||||
return undefined;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
export const asTrack = (album: Album, song: song): Track => ({
|
|
||||||
id: song.id,
|
|
||||||
name: song.title,
|
|
||||||
mimeType: song.contentType!,
|
|
||||||
duration: song.duration || 0,
|
|
||||||
number: song.track || 0,
|
|
||||||
genre: maybeAsGenre(song.genre),
|
|
||||||
coverArt: coverArtURN(song.coverArt),
|
|
||||||
album,
|
|
||||||
artist: {
|
|
||||||
id: song.artistId,
|
|
||||||
name: song.artist ? song.artist : "?",
|
|
||||||
image: song.artistId
|
|
||||||
? artistImageURN({ artistId: song.artistId })
|
|
||||||
: undefined,
|
|
||||||
},
|
|
||||||
rating: {
|
|
||||||
love: song.starred != undefined,
|
|
||||||
stars:
|
|
||||||
song.userRating && song.userRating <= 5 && song.userRating >= 0
|
|
||||||
? song.userRating
|
|
||||||
: 0,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
const asAlbum = (album: album): Album => ({
|
|
||||||
id: album.id,
|
|
||||||
name: album.name,
|
|
||||||
year: album.year,
|
|
||||||
genre: maybeAsGenre(album.genre),
|
|
||||||
artistId: album.artistId,
|
|
||||||
artistName: album.artist,
|
|
||||||
coverArt: coverArtURN(album.coverArt),
|
|
||||||
});
|
|
||||||
|
|
||||||
export const asGenre = (genreName: string) => ({
|
|
||||||
id: b64Encode(genreName),
|
|
||||||
name: genreName,
|
|
||||||
});
|
|
||||||
|
|
||||||
const maybeAsGenre = (genreName: string | undefined): Genre | undefined =>
|
|
||||||
pipe(
|
|
||||||
genreName,
|
|
||||||
O.fromNullable,
|
|
||||||
O.map(asGenre),
|
|
||||||
O.getOrElseW(() => undefined)
|
|
||||||
);
|
|
||||||
|
|
||||||
export type StreamClientApplication = (track: Track) => string;
|
|
||||||
|
|
||||||
const DEFAULT_CLIENT_APPLICATION = "bonob";
|
|
||||||
const USER_AGENT = "bonob";
|
|
||||||
|
|
||||||
export const DEFAULT: StreamClientApplication = (_: Track) =>
|
|
||||||
DEFAULT_CLIENT_APPLICATION;
|
|
||||||
|
|
||||||
export function appendMimeTypeToClientFor(mimeTypes: string[]) {
|
|
||||||
return (track: Track) =>
|
|
||||||
mimeTypes.includes(track.mimeType) ? `bonob+${track.mimeType}` : "bonob";
|
|
||||||
}
|
|
||||||
|
|
||||||
export const asURLSearchParams = (q: any) => {
|
|
||||||
const urlSearchParams = new URLSearchParams();
|
|
||||||
Object.keys(q).forEach((k) => {
|
|
||||||
_.flatten([q[k]]).forEach((v) => {
|
|
||||||
urlSearchParams.append(k, `${v}`);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
return urlSearchParams;
|
|
||||||
};
|
|
||||||
|
|
||||||
export type ImageFetcher = (url: string) => Promise<CoverArt | undefined>;
|
|
||||||
|
|
||||||
export const cachingImageFetcher =
|
|
||||||
(cacheDir: string, delegate: ImageFetcher) =>
|
|
||||||
async (url: string): Promise<CoverArt | undefined> => {
|
|
||||||
const filename = path.join(cacheDir, `${Md5.hashStr(url)}.png`);
|
|
||||||
return fse
|
|
||||||
.readFile(filename)
|
|
||||||
.then((data) => ({ contentType: "image/png", data }))
|
|
||||||
.catch(() =>
|
|
||||||
delegate(url).then((image) => {
|
|
||||||
if (image) {
|
|
||||||
return sharp(image.data)
|
|
||||||
.png()
|
|
||||||
.toBuffer()
|
|
||||||
.then((png) => {
|
|
||||||
return fse
|
|
||||||
.writeFile(filename, png)
|
|
||||||
.then(() => ({ contentType: "image/png", data: png }));
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
return undefined;
|
|
||||||
}
|
|
||||||
})
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
export const axiosImageFetcher = (url: string): Promise<CoverArt | undefined> =>
|
|
||||||
axios
|
|
||||||
.get(url, {
|
|
||||||
headers: BROWSER_HEADERS,
|
|
||||||
responseType: "arraybuffer",
|
|
||||||
})
|
|
||||||
.then((res) => ({
|
|
||||||
contentType: res.headers["content-type"],
|
|
||||||
data: Buffer.from(res.data, "binary"),
|
|
||||||
}))
|
|
||||||
.catch(() => undefined);
|
|
||||||
|
|
||||||
const AlbumQueryTypeToSubsonicType: Record<AlbumQueryType, string> = {
|
|
||||||
alphabeticalByArtist: "alphabeticalByArtist",
|
|
||||||
alphabeticalByName: "alphabeticalByName",
|
|
||||||
byGenre: "byGenre",
|
|
||||||
random: "random",
|
|
||||||
recentlyPlayed: "recent",
|
|
||||||
mostPlayed: "frequent",
|
|
||||||
recentlyAdded: "newest",
|
|
||||||
favourited: "starred",
|
|
||||||
starred: "highest",
|
|
||||||
};
|
|
||||||
|
|
||||||
const artistIsInLibrary = (artistId: string | undefined) =>
|
|
||||||
artistId != undefined && artistId != "-1";
|
|
||||||
|
|
||||||
type SubsonicCredentials = Credentials & {
|
|
||||||
type: string;
|
|
||||||
bearer: string | undefined;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const asToken = (credentials: SubsonicCredentials) =>
|
|
||||||
b64Encode(JSON.stringify(credentials));
|
|
||||||
export const parseToken = (token: string): SubsonicCredentials =>
|
|
||||||
JSON.parse(b64Decode(token));
|
|
||||||
|
|
||||||
interface SubsonicMusicLibrary extends MusicLibrary {
|
|
||||||
flavour(): string;
|
|
||||||
bearerToken(
|
|
||||||
credentials: Credentials
|
|
||||||
): TE.TaskEither<Error, string | undefined>;
|
|
||||||
}
|
|
||||||
|
|
||||||
export class Subsonic implements MusicService {
|
|
||||||
url: string;
|
|
||||||
streamClientApplication: StreamClientApplication;
|
|
||||||
externalImageFetcher: ImageFetcher;
|
|
||||||
|
|
||||||
constructor(
|
|
||||||
url: string,
|
|
||||||
streamClientApplication: StreamClientApplication = DEFAULT,
|
|
||||||
externalImageFetcher: ImageFetcher = axiosImageFetcher
|
|
||||||
) {
|
|
||||||
this.url = url;
|
|
||||||
this.streamClientApplication = streamClientApplication;
|
|
||||||
this.externalImageFetcher = externalImageFetcher;
|
|
||||||
}
|
|
||||||
|
|
||||||
get = async (
|
|
||||||
{ username, password }: Credentials,
|
|
||||||
path: string,
|
|
||||||
q: {} = {},
|
|
||||||
config: AxiosRequestConfig | undefined = {}
|
|
||||||
) =>
|
|
||||||
axios
|
|
||||||
.get(`${this.url}${path}`, {
|
|
||||||
params: asURLSearchParams({
|
|
||||||
u: username,
|
|
||||||
v: "1.16.1",
|
|
||||||
c: DEFAULT_CLIENT_APPLICATION,
|
|
||||||
...t_and_s(password),
|
|
||||||
...q,
|
|
||||||
}),
|
|
||||||
headers: {
|
|
||||||
"User-Agent": USER_AGENT,
|
|
||||||
},
|
|
||||||
...config,
|
|
||||||
})
|
|
||||||
.catch((e) => {
|
|
||||||
throw `Subsonic failed with: ${e}`;
|
|
||||||
})
|
|
||||||
.then((response) => {
|
|
||||||
if (response.status != 200 && response.status != 206) {
|
|
||||||
throw `Subsonic failed with a ${response.status || "no!"} status`;
|
|
||||||
} else return response;
|
|
||||||
});
|
|
||||||
|
|
||||||
getJSON = async <T>(
|
|
||||||
{ username, password }: Credentials,
|
|
||||||
path: string,
|
|
||||||
q: {} = {}
|
|
||||||
): Promise<T> =>
|
|
||||||
this.get({ username, password }, path, { f: "json", ...q })
|
|
||||||
.then((response) => response.data as SubsonicEnvelope)
|
|
||||||
.then((json) => json["subsonic-response"])
|
|
||||||
.then((json) => {
|
|
||||||
if (isError(json)) throw `Subsonic error:${json.error.message}`;
|
|
||||||
else return json as unknown as T;
|
|
||||||
});
|
|
||||||
|
|
||||||
generateToken = (credentials: Credentials) =>
|
|
||||||
pipe(
|
|
||||||
TE.tryCatch(
|
|
||||||
() =>
|
|
||||||
this.getJSON<PingResponse>(
|
|
||||||
_.pick(credentials, "username", "password"),
|
|
||||||
"/rest/ping.view"
|
|
||||||
),
|
|
||||||
(e) => new AuthFailure(e as string)
|
|
||||||
),
|
|
||||||
TE.chain(({ type }) =>
|
|
||||||
pipe(
|
|
||||||
TE.tryCatch(
|
|
||||||
() => this.libraryFor({ ...credentials, type }),
|
|
||||||
() => new AuthFailure("Failed to get library")
|
|
||||||
),
|
|
||||||
TE.map((library) => ({ type, library }))
|
|
||||||
)
|
|
||||||
),
|
|
||||||
TE.chain(({ library, type }) =>
|
|
||||||
pipe(
|
|
||||||
library.bearerToken(credentials),
|
|
||||||
TE.map((bearer) => ({ bearer, type }))
|
|
||||||
)
|
|
||||||
),
|
|
||||||
TE.map(({ bearer, type }) => ({
|
|
||||||
serviceToken: asToken({ ...credentials, bearer, type }),
|
|
||||||
userId: credentials.username,
|
|
||||||
nickname: credentials.username,
|
|
||||||
}))
|
|
||||||
);
|
|
||||||
|
|
||||||
refreshToken = (serviceToken: string) =>
|
|
||||||
this.generateToken(parseToken(serviceToken));
|
|
||||||
|
|
||||||
getArtists = (
|
|
||||||
credentials: Credentials
|
|
||||||
): Promise<(IdName & { albumCount: number; image: BUrn | undefined })[]> =>
|
|
||||||
this.getJSON<GetArtistsResponse>(credentials, "/rest/getArtists")
|
|
||||||
.then((it) => (it.artists.index || []).flatMap((it) => it.artist || []))
|
|
||||||
.then((artists) =>
|
|
||||||
artists.map((artist) => ({
|
|
||||||
id: `${artist.id}`,
|
|
||||||
name: artist.name,
|
|
||||||
albumCount: artist.albumCount,
|
|
||||||
image: artistImageURN({
|
|
||||||
artistId: artist.id,
|
|
||||||
artistImageURL: artist.artistImageUrl,
|
|
||||||
}),
|
|
||||||
}))
|
|
||||||
);
|
|
||||||
|
|
||||||
getArtistInfo = (
|
|
||||||
credentials: Credentials,
|
|
||||||
id: string
|
|
||||||
): Promise<{
|
|
||||||
similarArtist: (ArtistSummary & { inLibrary: boolean })[];
|
|
||||||
images: {
|
|
||||||
s: string | undefined;
|
|
||||||
m: string | undefined;
|
|
||||||
l: string | undefined;
|
|
||||||
};
|
|
||||||
}> =>
|
|
||||||
this.getJSON<GetArtistInfoResponse>(credentials, "/rest/getArtistInfo2", {
|
|
||||||
id,
|
|
||||||
count: 50,
|
|
||||||
includeNotPresent: true,
|
|
||||||
})
|
|
||||||
.then((it) => it.artistInfo2)
|
|
||||||
.then((it) => ({
|
|
||||||
images: {
|
|
||||||
s: it.smallImageUrl,
|
|
||||||
m: it.mediumImageUrl,
|
|
||||||
l: it.largeImageUrl,
|
|
||||||
},
|
|
||||||
similarArtist: (it.similarArtist || []).map((artist) => ({
|
|
||||||
id: `${artist.id}`,
|
|
||||||
name: artist.name,
|
|
||||||
inLibrary: artistIsInLibrary(artist.id),
|
|
||||||
image: artistImageURN({
|
|
||||||
artistId: artist.id,
|
|
||||||
artistImageURL: artist.artistImageUrl,
|
|
||||||
}),
|
|
||||||
})),
|
|
||||||
}));
|
|
||||||
|
|
||||||
getAlbum = (credentials: Credentials, id: string): Promise<Album> =>
|
|
||||||
this.getJSON<GetAlbumResponse>(credentials, "/rest/getAlbum", { id })
|
|
||||||
.then((it) => it.album)
|
|
||||||
.then((album) => ({
|
|
||||||
id: album.id,
|
|
||||||
name: album.name,
|
|
||||||
year: album.year,
|
|
||||||
genre: maybeAsGenre(album.genre),
|
|
||||||
artistId: album.artistId,
|
|
||||||
artistName: album.artist,
|
|
||||||
coverArt: coverArtURN(album.coverArt),
|
|
||||||
}));
|
|
||||||
|
|
||||||
getArtist = (
|
|
||||||
credentials: Credentials,
|
|
||||||
id: string
|
|
||||||
): Promise<
|
|
||||||
IdName & { artistImageUrl: string | undefined; albums: AlbumSummary[] }
|
|
||||||
> =>
|
|
||||||
this.getJSON<GetArtistResponse>(credentials, "/rest/getArtist", {
|
|
||||||
id,
|
|
||||||
})
|
|
||||||
.then((it) => it.artist)
|
|
||||||
.then((it) => ({
|
|
||||||
id: it.id,
|
|
||||||
name: it.name,
|
|
||||||
artistImageUrl: it.artistImageUrl,
|
|
||||||
albums: this.toAlbumSummary(it.album || []),
|
|
||||||
}));
|
|
||||||
|
|
||||||
getArtistWithInfo = (credentials: Credentials, id: string) =>
|
|
||||||
Promise.all([
|
|
||||||
this.getArtist(credentials, id),
|
|
||||||
this.getArtistInfo(credentials, id),
|
|
||||||
]).then(([artist, artistInfo]) => ({
|
|
||||||
id: artist.id,
|
|
||||||
name: artist.name,
|
|
||||||
image: artistImageURN({
|
|
||||||
artistId: artist.id,
|
|
||||||
artistImageURL: [
|
|
||||||
artist.artistImageUrl,
|
|
||||||
artistInfo.images.l,
|
|
||||||
artistInfo.images.m,
|
|
||||||
artistInfo.images.s,
|
|
||||||
].find(isValidImage),
|
|
||||||
}),
|
|
||||||
albums: artist.albums,
|
|
||||||
similarArtists: artistInfo.similarArtist,
|
|
||||||
}));
|
|
||||||
|
|
||||||
getCoverArt = (credentials: Credentials, id: string, size?: number) =>
|
|
||||||
this.get(credentials, "/rest/getCoverArt", size ? { id, size } : { id }, {
|
|
||||||
headers: { "User-Agent": "bonob" },
|
|
||||||
responseType: "arraybuffer",
|
|
||||||
});
|
|
||||||
|
|
||||||
getTrack = (credentials: Credentials, id: string) =>
|
|
||||||
this.getJSON<GetSongResponse>(credentials, "/rest/getSong", {
|
|
||||||
id,
|
|
||||||
})
|
|
||||||
.then((it) => it.song)
|
|
||||||
.then((song) =>
|
|
||||||
this.getAlbum(credentials, song.albumId!).then((album) =>
|
|
||||||
asTrack(album, song)
|
|
||||||
)
|
|
||||||
);
|
|
||||||
|
|
||||||
getStarred = (credentials: Credentials) =>
|
|
||||||
this.getJSON<GetStarredResponse>(credentials, "/rest/getStarred2").then(
|
|
||||||
(it) => new Set(it.starred2.song.map((it) => it.id))
|
|
||||||
);
|
|
||||||
|
|
||||||
toAlbumSummary = (albumList: album[]): AlbumSummary[] =>
|
|
||||||
albumList.map((album) => ({
|
|
||||||
id: album.id,
|
|
||||||
name: album.name,
|
|
||||||
year: album.year,
|
|
||||||
genre: maybeAsGenre(album.genre),
|
|
||||||
artistId: album.artistId,
|
|
||||||
artistName: album.artist,
|
|
||||||
coverArt: coverArtURN(album.coverArt),
|
|
||||||
}));
|
|
||||||
|
|
||||||
search3 = (credentials: Credentials, q: any) =>
|
|
||||||
this.getJSON<Search3Response>(credentials, "/rest/search3", {
|
|
||||||
artistCount: 0,
|
|
||||||
albumCount: 0,
|
|
||||||
songCount: 0,
|
|
||||||
...q,
|
|
||||||
}).then((it) => ({
|
|
||||||
artists: it.searchResult3.artist || [],
|
|
||||||
albums: it.searchResult3.album || [],
|
|
||||||
songs: it.searchResult3.song || [],
|
|
||||||
}));
|
|
||||||
|
|
||||||
getAlbumList2 = (credentials: Credentials, q: AlbumQuery) =>
|
|
||||||
Promise.all([
|
|
||||||
this.getArtists(credentials).then((it) =>
|
|
||||||
_.inject(it, (total, artist) => total + artist.albumCount, 0)
|
|
||||||
),
|
|
||||||
this.getJSON<GetAlbumListResponse>(credentials, "/rest/getAlbumList2", {
|
|
||||||
type: AlbumQueryTypeToSubsonicType[q.type],
|
|
||||||
...(q.genre ? { genre: b64Decode(q.genre) } : {}),
|
|
||||||
size: 500,
|
|
||||||
offset: q._index,
|
|
||||||
})
|
|
||||||
.then((response) => response.albumList2.album || [])
|
|
||||||
.then(this.toAlbumSummary),
|
|
||||||
]).then(([total, albums]) => ({
|
|
||||||
results: albums.slice(0, q._count),
|
|
||||||
total: albums.length == 500 ? total : q._index + albums.length,
|
|
||||||
}));
|
|
||||||
|
|
||||||
// getStarred2 = (credentials: Credentials): Promise<{ albums: Album[] }> =>
|
|
||||||
// this.getJSON<GetStarredResponse>(credentials, "/rest/getStarred2")
|
|
||||||
// .then((it) => it.starred2)
|
|
||||||
// .then((it) => ({
|
|
||||||
// albums: it.album.map(asAlbum),
|
|
||||||
// }));
|
|
||||||
|
|
||||||
login = async (token: string) => this.libraryFor(parseToken(token));
|
|
||||||
|
|
||||||
private libraryFor = (
|
|
||||||
credentials: Credentials & { type: string }
|
|
||||||
): Promise<SubsonicMusicLibrary> => {
|
|
||||||
const subsonic = this;
|
|
||||||
|
|
||||||
const genericSubsonic: SubsonicMusicLibrary = {
|
|
||||||
flavour: () => "subsonic",
|
|
||||||
bearerToken: (_: Credentials) => TE.right(undefined),
|
|
||||||
artists: (q: ArtistQuery): Promise<Result<ArtistSummary>> =>
|
|
||||||
subsonic
|
|
||||||
.getArtists(credentials)
|
|
||||||
.then(slice2(q))
|
|
||||||
.then(([page, total]) => ({
|
|
||||||
total,
|
|
||||||
results: page.map((it) => ({
|
|
||||||
id: it.id,
|
|
||||||
name: it.name,
|
|
||||||
image: it.image,
|
|
||||||
})),
|
|
||||||
})),
|
|
||||||
artist: async (id: string): Promise<Artist> =>
|
|
||||||
subsonic.getArtistWithInfo(credentials, id),
|
|
||||||
albums: async (q: AlbumQuery): Promise<Result<AlbumSummary>> =>
|
|
||||||
subsonic.getAlbumList2(credentials, q),
|
|
||||||
album: (id: string): Promise<Album> => subsonic.getAlbum(credentials, id),
|
|
||||||
genres: () =>
|
|
||||||
subsonic
|
|
||||||
.getJSON<GetGenresResponse>(credentials, "/rest/getGenres")
|
|
||||||
.then((it) =>
|
|
||||||
pipe(
|
|
||||||
it.genres.genre || [],
|
|
||||||
A.filter((it) => it.albumCount > 0),
|
|
||||||
A.map((it) => it.value),
|
|
||||||
A.sort(ordString),
|
|
||||||
A.map((it) => ({ id: b64Encode(it), name: it }))
|
|
||||||
)
|
|
||||||
),
|
|
||||||
tracks: (albumId: string) =>
|
|
||||||
subsonic
|
|
||||||
.getJSON<GetAlbumResponse>(credentials, "/rest/getAlbum", {
|
|
||||||
id: albumId,
|
|
||||||
})
|
|
||||||
.then((it) => it.album)
|
|
||||||
.then((album) =>
|
|
||||||
(album.song || []).map((song) => asTrack(asAlbum(album), song))
|
|
||||||
),
|
|
||||||
track: (trackId: string) => subsonic.getTrack(credentials, trackId),
|
|
||||||
rate: (trackId: string, rating: Rating) =>
|
|
||||||
Promise.resolve(true)
|
|
||||||
.then(() => {
|
|
||||||
if (rating.stars >= 0 && rating.stars <= 5) {
|
|
||||||
return subsonic.getTrack(credentials, trackId);
|
|
||||||
} else {
|
|
||||||
throw `Invalid rating.stars value of ${rating.stars}`;
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.then((track) => {
|
|
||||||
const thingsToUpdate = [];
|
|
||||||
if (track.rating.love != rating.love) {
|
|
||||||
thingsToUpdate.push(
|
|
||||||
subsonic.getJSON(
|
|
||||||
credentials,
|
|
||||||
`/rest/${rating.love ? "star" : "unstar"}`,
|
|
||||||
{
|
|
||||||
id: trackId,
|
|
||||||
}
|
|
||||||
)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
if (track.rating.stars != rating.stars) {
|
|
||||||
thingsToUpdate.push(
|
|
||||||
subsonic.getJSON(credentials, `/rest/setRating`, {
|
|
||||||
id: trackId,
|
|
||||||
rating: rating.stars,
|
|
||||||
})
|
|
||||||
);
|
|
||||||
}
|
|
||||||
return Promise.all(thingsToUpdate);
|
|
||||||
})
|
|
||||||
.then(() => true)
|
|
||||||
.catch(() => false),
|
|
||||||
stream: async ({
|
|
||||||
trackId,
|
|
||||||
range,
|
|
||||||
}: {
|
|
||||||
trackId: string;
|
|
||||||
range: string | undefined;
|
|
||||||
}) =>
|
|
||||||
subsonic.getTrack(credentials, trackId).then((track) =>
|
|
||||||
subsonic
|
|
||||||
.get(
|
|
||||||
credentials,
|
|
||||||
`/rest/stream`,
|
|
||||||
{
|
|
||||||
id: trackId,
|
|
||||||
c: this.streamClientApplication(track),
|
|
||||||
},
|
|
||||||
{
|
|
||||||
headers: pipe(
|
|
||||||
range,
|
|
||||||
O.fromNullable,
|
|
||||||
O.map((range) => ({
|
|
||||||
"User-Agent": USER_AGENT,
|
|
||||||
Range: range,
|
|
||||||
})),
|
|
||||||
O.getOrElse(() => ({
|
|
||||||
"User-Agent": USER_AGENT,
|
|
||||||
}))
|
|
||||||
),
|
|
||||||
responseType: "stream",
|
|
||||||
}
|
|
||||||
)
|
|
||||||
.then((res) => ({
|
|
||||||
status: res.status,
|
|
||||||
headers: {
|
|
||||||
"content-type": res.headers["content-type"],
|
|
||||||
"content-length": res.headers["content-length"],
|
|
||||||
"content-range": res.headers["content-range"],
|
|
||||||
"accept-ranges": res.headers["accept-ranges"],
|
|
||||||
},
|
|
||||||
stream: res.data,
|
|
||||||
}))
|
|
||||||
),
|
|
||||||
coverArt: async (coverArtURN: BUrn, size?: number) =>
|
|
||||||
Promise.resolve(coverArtURN)
|
|
||||||
.then((it) => assertSystem(it, "subsonic"))
|
|
||||||
.then((it) => it.resource.split(":")[1]!)
|
|
||||||
.then((it) => subsonic.getCoverArt(credentials, it, size))
|
|
||||||
.then((res) => ({
|
|
||||||
contentType: res.headers["content-type"],
|
|
||||||
data: Buffer.from(res.data, "binary"),
|
|
||||||
}))
|
|
||||||
.catch((e) => {
|
|
||||||
logger.error(
|
|
||||||
`Failed getting coverArt for urn:'${coverArtURN}': ${e}`
|
|
||||||
);
|
|
||||||
return undefined;
|
|
||||||
}),
|
|
||||||
scrobble: async (id: string) =>
|
|
||||||
subsonic
|
|
||||||
.getJSON(credentials, `/rest/scrobble`, {
|
|
||||||
id,
|
|
||||||
submission: true,
|
|
||||||
})
|
|
||||||
.then((_) => true)
|
|
||||||
.catch(() => false),
|
|
||||||
nowPlaying: async (id: string) =>
|
|
||||||
subsonic
|
|
||||||
.getJSON(credentials, `/rest/scrobble`, {
|
|
||||||
id,
|
|
||||||
submission: false,
|
|
||||||
})
|
|
||||||
.then((_) => true)
|
|
||||||
.catch(() => false),
|
|
||||||
searchArtists: async (query: string) =>
|
|
||||||
subsonic
|
|
||||||
.search3(credentials, { query, artistCount: 20 })
|
|
||||||
.then(({ artists }) =>
|
|
||||||
artists.map((artist) => ({
|
|
||||||
id: artist.id,
|
|
||||||
name: artist.name,
|
|
||||||
image: artistImageURN({
|
|
||||||
artistId: artist.id,
|
|
||||||
artistImageURL: artist.artistImageUrl,
|
|
||||||
}),
|
|
||||||
}))
|
|
||||||
),
|
|
||||||
searchAlbums: async (query: string) =>
|
|
||||||
subsonic
|
|
||||||
.search3(credentials, { query, albumCount: 20 })
|
|
||||||
.then(({ albums }) => subsonic.toAlbumSummary(albums)),
|
|
||||||
searchTracks: async (query: string) =>
|
|
||||||
subsonic
|
|
||||||
.search3(credentials, { query, songCount: 20 })
|
|
||||||
.then(({ songs }) =>
|
|
||||||
Promise.all(
|
|
||||||
songs.map((it) => subsonic.getTrack(credentials, it.id))
|
|
||||||
)
|
|
||||||
),
|
|
||||||
playlists: async () =>
|
|
||||||
subsonic
|
|
||||||
.getJSON<GetPlaylistsResponse>(credentials, "/rest/getPlaylists")
|
|
||||||
.then((it) => it.playlists.playlist || [])
|
|
||||||
.then((playlists) =>
|
|
||||||
playlists.map((it) => ({ id: it.id, name: it.name }))
|
|
||||||
),
|
|
||||||
playlist: async (id: string) =>
|
|
||||||
subsonic
|
|
||||||
.getJSON<GetPlaylistResponse>(credentials, "/rest/getPlaylist", {
|
|
||||||
id,
|
|
||||||
})
|
|
||||||
.then((it) => it.playlist)
|
|
||||||
.then((playlist) => {
|
|
||||||
let trackNumber = 1;
|
|
||||||
return {
|
|
||||||
id: playlist.id,
|
|
||||||
name: playlist.name,
|
|
||||||
entries: (playlist.entry || []).map((entry) => ({
|
|
||||||
...asTrack(
|
|
||||||
{
|
|
||||||
id: entry.albumId!,
|
|
||||||
name: entry.album!,
|
|
||||||
year: entry.year,
|
|
||||||
genre: maybeAsGenre(entry.genre),
|
|
||||||
artistName: entry.artist,
|
|
||||||
artistId: entry.artistId,
|
|
||||||
coverArt: coverArtURN(entry.coverArt),
|
|
||||||
},
|
|
||||||
entry
|
|
||||||
),
|
|
||||||
number: trackNumber++,
|
|
||||||
})),
|
|
||||||
};
|
|
||||||
}),
|
|
||||||
createPlaylist: async (name: string) =>
|
|
||||||
subsonic
|
|
||||||
.getJSON<GetPlaylistResponse>(credentials, "/rest/createPlaylist", {
|
|
||||||
name,
|
|
||||||
})
|
|
||||||
.then((it) => it.playlist)
|
|
||||||
.then((it) => ({ id: it.id, name: it.name })),
|
|
||||||
deletePlaylist: async (id: string) =>
|
|
||||||
subsonic
|
|
||||||
.getJSON<GetPlaylistResponse>(credentials, "/rest/deletePlaylist", {
|
|
||||||
id,
|
|
||||||
})
|
|
||||||
.then((_) => true),
|
|
||||||
addToPlaylist: async (playlistId: string, trackId: string) =>
|
|
||||||
subsonic
|
|
||||||
.getJSON<GetPlaylistResponse>(credentials, "/rest/updatePlaylist", {
|
|
||||||
playlistId,
|
|
||||||
songIdToAdd: trackId,
|
|
||||||
})
|
|
||||||
.then((_) => true),
|
|
||||||
removeFromPlaylist: async (playlistId: string, indicies: number[]) =>
|
|
||||||
subsonic
|
|
||||||
.getJSON<GetPlaylistResponse>(credentials, "/rest/updatePlaylist", {
|
|
||||||
playlistId,
|
|
||||||
songIndexToRemove: indicies,
|
|
||||||
})
|
|
||||||
.then((_) => true),
|
|
||||||
similarSongs: async (id: string) =>
|
|
||||||
subsonic
|
|
||||||
.getJSON<GetSimilarSongsResponse>(
|
|
||||||
credentials,
|
|
||||||
"/rest/getSimilarSongs2",
|
|
||||||
{ id, count: 50 }
|
|
||||||
)
|
|
||||||
.then((it) => it.similarSongs2.song || [])
|
|
||||||
.then((songs) =>
|
|
||||||
Promise.all(
|
|
||||||
songs.map((song) =>
|
|
||||||
subsonic
|
|
||||||
.getAlbum(credentials, song.albumId!)
|
|
||||||
.then((album) => asTrack(album, song))
|
|
||||||
)
|
|
||||||
)
|
|
||||||
),
|
|
||||||
topSongs: async (artistId: string) =>
|
|
||||||
subsonic.getArtist(credentials, artistId).then(({ name }) =>
|
|
||||||
subsonic
|
|
||||||
.getJSON<GetTopSongsResponse>(credentials, "/rest/getTopSongs", {
|
|
||||||
artist: name,
|
|
||||||
count: 50,
|
|
||||||
})
|
|
||||||
.then((it) => it.topSongs.song || [])
|
|
||||||
.then((songs) =>
|
|
||||||
Promise.all(
|
|
||||||
songs.map((song) =>
|
|
||||||
subsonic
|
|
||||||
.getAlbum(credentials, song.albumId!)
|
|
||||||
.then((album) => asTrack(album, song))
|
|
||||||
)
|
|
||||||
)
|
|
||||||
)
|
|
||||||
),
|
|
||||||
};
|
|
||||||
|
|
||||||
if (credentials.type == "navidrome") {
|
|
||||||
return Promise.resolve({
|
|
||||||
...genericSubsonic,
|
|
||||||
flavour: () => "navidrome",
|
|
||||||
bearerToken: (credentials: Credentials) =>
|
|
||||||
pipe(
|
|
||||||
TE.tryCatch(
|
|
||||||
() =>
|
|
||||||
axios.post(
|
|
||||||
`${this.url}/auth/login`,
|
|
||||||
_.pick(credentials, "username", "password")
|
|
||||||
),
|
|
||||||
() => new AuthFailure("Failed to get bearerToken")
|
|
||||||
),
|
|
||||||
TE.map((it) => it.data.token as string | undefined)
|
|
||||||
),
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
return Promise.resolve(genericSubsonic);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
770
src/subsonic/generic.ts
Normal file
770
src/subsonic/generic.ts
Normal file
@@ -0,0 +1,770 @@
|
|||||||
|
import { option as O, taskEither as TE } from "fp-ts";
|
||||||
|
import * as A from "fp-ts/Array";
|
||||||
|
import { pipe } from "fp-ts/lib/function";
|
||||||
|
import { ordString } from "fp-ts/lib/Ord";
|
||||||
|
import { inject } from "underscore";
|
||||||
|
import _ from "underscore";
|
||||||
|
|
||||||
|
import logger from "../logger";
|
||||||
|
import { b64Decode, b64Encode } from "../b64";
|
||||||
|
import { assertSystem, BUrn, format } from "../burn";
|
||||||
|
|
||||||
|
import {
|
||||||
|
Album,
|
||||||
|
AlbumQuery,
|
||||||
|
AlbumQueryType,
|
||||||
|
AlbumSummary,
|
||||||
|
Artist,
|
||||||
|
ArtistQuery,
|
||||||
|
ArtistSummary,
|
||||||
|
AuthFailure,
|
||||||
|
Credentials,
|
||||||
|
Genre,
|
||||||
|
IdName,
|
||||||
|
Rating,
|
||||||
|
Result,
|
||||||
|
slice2,
|
||||||
|
Sortable,
|
||||||
|
Track,
|
||||||
|
} from "../music_service";
|
||||||
|
import {
|
||||||
|
DODGY_IMAGE_NAME,
|
||||||
|
StreamClientApplication,
|
||||||
|
SubsonicCredentials,
|
||||||
|
SubsonicMusicLibrary,
|
||||||
|
SubsonicResponse,
|
||||||
|
USER_AGENT,
|
||||||
|
} from ".";
|
||||||
|
import axios from "axios";
|
||||||
|
import { asURLSearchParams } from "../utils";
|
||||||
|
import { artistSummaryFromNDArtist, NDArtist } from "./navidrome";
|
||||||
|
import { Http2, RequestParams } from "../http";
|
||||||
|
import { client } from "./subsonic_http";
|
||||||
|
|
||||||
|
type album = {
|
||||||
|
id: string;
|
||||||
|
name: string;
|
||||||
|
artist: string | undefined;
|
||||||
|
artistId: string | undefined;
|
||||||
|
coverArt: string | undefined;
|
||||||
|
genre: string | undefined;
|
||||||
|
year: string | undefined;
|
||||||
|
};
|
||||||
|
|
||||||
|
type artist = {
|
||||||
|
id: string;
|
||||||
|
name: string;
|
||||||
|
albumCount: number;
|
||||||
|
artistImageUrl: string | undefined;
|
||||||
|
};
|
||||||
|
|
||||||
|
type GetArtistsResponse = SubsonicResponse & {
|
||||||
|
artists: {
|
||||||
|
index: {
|
||||||
|
artist: artist[];
|
||||||
|
name: string;
|
||||||
|
}[];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
type GetAlbumListResponse = SubsonicResponse & {
|
||||||
|
albumList2: {
|
||||||
|
album: album[];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
type genre = {
|
||||||
|
songCount: number;
|
||||||
|
albumCount: number;
|
||||||
|
value: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
type GetGenresResponse = SubsonicResponse & {
|
||||||
|
genres: {
|
||||||
|
genre: genre[];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
type GetArtistInfoResponse = SubsonicResponse & {
|
||||||
|
artistInfo2: artistInfo;
|
||||||
|
};
|
||||||
|
|
||||||
|
type GetArtistResponse = SubsonicResponse & {
|
||||||
|
artist: artist & {
|
||||||
|
album: album[];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
export type images = {
|
||||||
|
smallImageUrl: string | undefined;
|
||||||
|
mediumImageUrl: string | undefined;
|
||||||
|
largeImageUrl: string | undefined;
|
||||||
|
};
|
||||||
|
|
||||||
|
type artistInfo = images & {
|
||||||
|
biography: string | undefined;
|
||||||
|
musicBrainzId: string | undefined;
|
||||||
|
lastFmUrl: string | undefined;
|
||||||
|
similarArtist: artist[];
|
||||||
|
};
|
||||||
|
|
||||||
|
export type song = {
|
||||||
|
id: string;
|
||||||
|
parent: string | undefined;
|
||||||
|
title: string;
|
||||||
|
album: string | undefined;
|
||||||
|
albumId: string | undefined;
|
||||||
|
artist: string | undefined;
|
||||||
|
artistId: string | undefined;
|
||||||
|
track: number | undefined;
|
||||||
|
year: string | undefined;
|
||||||
|
genre: string | undefined;
|
||||||
|
coverArt: string | undefined;
|
||||||
|
created: string | undefined;
|
||||||
|
duration: number | undefined;
|
||||||
|
bitRate: number | undefined;
|
||||||
|
suffix: string | undefined;
|
||||||
|
contentType: string | undefined;
|
||||||
|
type: string | undefined;
|
||||||
|
userRating: number | undefined;
|
||||||
|
starred: string | undefined;
|
||||||
|
};
|
||||||
|
|
||||||
|
type GetAlbumResponse = {
|
||||||
|
album: album & {
|
||||||
|
song: song[];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
type playlist = {
|
||||||
|
id: string;
|
||||||
|
name: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
type GetPlaylistResponse = {
|
||||||
|
playlist: {
|
||||||
|
id: string;
|
||||||
|
name: string;
|
||||||
|
entry: song[];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
type GetPlaylistsResponse = {
|
||||||
|
playlists: { playlist: playlist[] };
|
||||||
|
};
|
||||||
|
|
||||||
|
type GetSimilarSongsResponse = {
|
||||||
|
similarSongs2: { song: song[] };
|
||||||
|
};
|
||||||
|
|
||||||
|
type GetTopSongsResponse = {
|
||||||
|
topSongs: { song: song[] };
|
||||||
|
};
|
||||||
|
|
||||||
|
type GetSongResponse = {
|
||||||
|
song: song;
|
||||||
|
};
|
||||||
|
|
||||||
|
type Search3Response = SubsonicResponse & {
|
||||||
|
searchResult3: {
|
||||||
|
artist: artist[];
|
||||||
|
album: album[];
|
||||||
|
song: song[];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
const AlbumQueryTypeToSubsonicType: Record<AlbumQueryType, string> = {
|
||||||
|
alphabeticalByArtist: "alphabeticalByArtist",
|
||||||
|
alphabeticalByName: "alphabeticalByName",
|
||||||
|
byGenre: "byGenre",
|
||||||
|
random: "random",
|
||||||
|
recentlyPlayed: "recent",
|
||||||
|
mostPlayed: "frequent",
|
||||||
|
recentlyAdded: "newest",
|
||||||
|
favourited: "starred",
|
||||||
|
starred: "highest",
|
||||||
|
};
|
||||||
|
|
||||||
|
export const isValidImage = (url: string | undefined) =>
|
||||||
|
url != undefined && !url.endsWith(DODGY_IMAGE_NAME);
|
||||||
|
|
||||||
|
const artistIsInLibrary = (artistId: string | undefined) =>
|
||||||
|
artistId != undefined && artistId != "-1";
|
||||||
|
|
||||||
|
const coverArtURN = (coverArt: string | undefined): BUrn | undefined =>
|
||||||
|
pipe(
|
||||||
|
coverArt,
|
||||||
|
O.fromNullable,
|
||||||
|
O.map((it: string) => ({ system: "subsonic", resource: `art:${it}` })),
|
||||||
|
O.getOrElseW(() => undefined)
|
||||||
|
);
|
||||||
|
|
||||||
|
// todo: is this the right place for this??
|
||||||
|
export const artistImageURN = (
|
||||||
|
spec: Partial<{
|
||||||
|
artistId: string | undefined;
|
||||||
|
artistImageURL: string | undefined;
|
||||||
|
}>
|
||||||
|
): BUrn | undefined => {
|
||||||
|
const deets = {
|
||||||
|
artistId: undefined,
|
||||||
|
artistImageURL: undefined,
|
||||||
|
...spec,
|
||||||
|
};
|
||||||
|
if (deets.artistImageURL && isValidImage(deets.artistImageURL)) {
|
||||||
|
return {
|
||||||
|
system: "external",
|
||||||
|
resource: deets.artistImageURL,
|
||||||
|
};
|
||||||
|
} else if (artistIsInLibrary(deets.artistId)) {
|
||||||
|
return {
|
||||||
|
system: "subsonic",
|
||||||
|
resource: `art:${deets.artistId!}`,
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const asTrack = (album: Album, song: song): Track => ({
|
||||||
|
id: song.id,
|
||||||
|
name: song.title,
|
||||||
|
mimeType: song.contentType!,
|
||||||
|
duration: song.duration || 0,
|
||||||
|
number: song.track || 0,
|
||||||
|
genre: maybeAsGenre(song.genre),
|
||||||
|
coverArt: coverArtURN(song.coverArt),
|
||||||
|
album,
|
||||||
|
artist: {
|
||||||
|
id: song.artistId,
|
||||||
|
name: song.artist ? song.artist : "?",
|
||||||
|
image: song.artistId
|
||||||
|
? artistImageURN({ artistId: song.artistId })
|
||||||
|
: undefined,
|
||||||
|
},
|
||||||
|
rating: {
|
||||||
|
love: song.starred != undefined,
|
||||||
|
stars:
|
||||||
|
song.userRating && song.userRating <= 5 && song.userRating >= 0
|
||||||
|
? song.userRating
|
||||||
|
: 0,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const asAlbum = (album: album): Album => ({
|
||||||
|
id: album.id,
|
||||||
|
name: album.name,
|
||||||
|
year: album.year,
|
||||||
|
genre: maybeAsGenre(album.genre),
|
||||||
|
artistId: album.artistId,
|
||||||
|
artistName: album.artist,
|
||||||
|
coverArt: coverArtURN(album.coverArt),
|
||||||
|
});
|
||||||
|
|
||||||
|
export const asGenre = (genreName: string) => ({
|
||||||
|
id: b64Encode(genreName),
|
||||||
|
name: genreName,
|
||||||
|
});
|
||||||
|
|
||||||
|
const maybeAsGenre = (genreName: string | undefined): Genre | undefined =>
|
||||||
|
pipe(
|
||||||
|
genreName,
|
||||||
|
O.fromNullable,
|
||||||
|
O.map(asGenre),
|
||||||
|
O.getOrElseW(() => undefined)
|
||||||
|
);
|
||||||
|
|
||||||
|
export class SubsonicGenericMusicLibrary implements SubsonicMusicLibrary {
|
||||||
|
streamClientApplication: StreamClientApplication;
|
||||||
|
subsonicHttp: Http2;
|
||||||
|
|
||||||
|
constructor(
|
||||||
|
streamClientApplication: StreamClientApplication,
|
||||||
|
subsonicHttp: Http2
|
||||||
|
) {
|
||||||
|
this.streamClientApplication = streamClientApplication;
|
||||||
|
this.subsonicHttp = subsonicHttp;
|
||||||
|
}
|
||||||
|
|
||||||
|
GET = (query: Partial<RequestParams>) => client(this.subsonicHttp)({ method: 'get', ...query });
|
||||||
|
|
||||||
|
flavour = () => "subsonic";
|
||||||
|
|
||||||
|
bearerToken = (_: Credentials): TE.TaskEither<Error, string | undefined> =>
|
||||||
|
TE.right(undefined);
|
||||||
|
|
||||||
|
artists = async (q: ArtistQuery): Promise<Result<ArtistSummary & Sortable>> =>
|
||||||
|
this.getArtists()
|
||||||
|
.then(slice2(q))
|
||||||
|
.then(([page, total]) => ({
|
||||||
|
total,
|
||||||
|
results: page.map((it) => ({
|
||||||
|
id: it.id,
|
||||||
|
name: it.name,
|
||||||
|
sortName: it.name,
|
||||||
|
image: it.image,
|
||||||
|
})),
|
||||||
|
}));
|
||||||
|
|
||||||
|
artist = async (id: string): Promise<Artist> => this.getArtistWithInfo(id);
|
||||||
|
|
||||||
|
albums = async (q: AlbumQuery): Promise<Result<AlbumSummary>> =>
|
||||||
|
this.getAlbumList2(q);
|
||||||
|
|
||||||
|
album = (id: string): Promise<Album> => this.getAlbum(id);
|
||||||
|
|
||||||
|
genres = () =>
|
||||||
|
this.GET({
|
||||||
|
url: "/rest/getGenres",
|
||||||
|
})
|
||||||
|
.asJSON<GetGenresResponse>()
|
||||||
|
.then((it) =>
|
||||||
|
pipe(
|
||||||
|
it.genres.genre || [],
|
||||||
|
A.filter((it) => it.albumCount > 0),
|
||||||
|
A.map((it) => it.value),
|
||||||
|
A.sort(ordString),
|
||||||
|
A.map((it) => ({ id: b64Encode(it), name: it }))
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
tracks = (albumId: string) =>
|
||||||
|
this.GET({
|
||||||
|
url: "/rest/getAlbum",
|
||||||
|
params: {
|
||||||
|
id: albumId,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
.asJSON<GetAlbumResponse>()
|
||||||
|
.then((it) => it.album)
|
||||||
|
.then((album) =>
|
||||||
|
(album.song || []).map((song) => asTrack(asAlbum(album), song))
|
||||||
|
);
|
||||||
|
|
||||||
|
track = (trackId: string) => this.getTrack(trackId);
|
||||||
|
|
||||||
|
rate = (trackId: string, rating: Rating) =>
|
||||||
|
Promise.resolve(true)
|
||||||
|
.then(() => {
|
||||||
|
if (rating.stars >= 0 && rating.stars <= 5) {
|
||||||
|
return this.getTrack(trackId);
|
||||||
|
} else {
|
||||||
|
throw `Invalid rating.stars value of ${rating.stars}`;
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.then((track) => {
|
||||||
|
const thingsToUpdate = [];
|
||||||
|
if (track.rating.love != rating.love) {
|
||||||
|
thingsToUpdate.push(
|
||||||
|
this.GET({
|
||||||
|
url: `/rest/${rating.love ? "star" : "unstar"}`,
|
||||||
|
params: {
|
||||||
|
id: trackId,
|
||||||
|
},
|
||||||
|
}).asJSON()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
if (track.rating.stars != rating.stars) {
|
||||||
|
thingsToUpdate.push(
|
||||||
|
this.GET({
|
||||||
|
url: `/rest/setRating`,
|
||||||
|
params: {
|
||||||
|
id: trackId,
|
||||||
|
rating: rating.stars,
|
||||||
|
},
|
||||||
|
}).asJSON()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return Promise.all(thingsToUpdate);
|
||||||
|
})
|
||||||
|
.then(() => true)
|
||||||
|
.catch(() => false);
|
||||||
|
|
||||||
|
stream = async ({
|
||||||
|
trackId,
|
||||||
|
range,
|
||||||
|
}: {
|
||||||
|
trackId: string;
|
||||||
|
range: string | undefined;
|
||||||
|
}) =>
|
||||||
|
this.getTrack(trackId).then((track) =>
|
||||||
|
this.GET({
|
||||||
|
url: "/rest/stream",
|
||||||
|
params: {
|
||||||
|
id: trackId,
|
||||||
|
c: this.streamClientApplication(track),
|
||||||
|
},
|
||||||
|
headers: range != undefined ? { Range: range } : {},
|
||||||
|
responseType: "stream",
|
||||||
|
})
|
||||||
|
.asRaw()
|
||||||
|
.then((res) => ({
|
||||||
|
status: res.status,
|
||||||
|
headers: {
|
||||||
|
"content-type": res.headers["content-type"],
|
||||||
|
"content-length": res.headers["content-length"],
|
||||||
|
"content-range": res.headers["content-range"],
|
||||||
|
"accept-ranges": res.headers["accept-ranges"],
|
||||||
|
},
|
||||||
|
stream: res.data,
|
||||||
|
}))
|
||||||
|
);
|
||||||
|
|
||||||
|
coverArt = async (coverArtURN: BUrn, size?: number) =>
|
||||||
|
Promise.resolve(coverArtURN)
|
||||||
|
.then((it) => assertSystem(it, "subsonic"))
|
||||||
|
.then((it) => it.resource.split(":")[1]!)
|
||||||
|
.then((it) => this.getCoverArt(it, size))
|
||||||
|
.then((res) => ({
|
||||||
|
contentType: res.headers["content-type"],
|
||||||
|
data: Buffer.from(res.data, "binary"),
|
||||||
|
}))
|
||||||
|
.catch((e) => {
|
||||||
|
logger.error(
|
||||||
|
`Failed getting coverArt for '${format(coverArtURN)}': ${e}`
|
||||||
|
);
|
||||||
|
return undefined;
|
||||||
|
});
|
||||||
|
|
||||||
|
scrobble = async (id: string) =>
|
||||||
|
this.GET({
|
||||||
|
url: `/rest/scrobble`,
|
||||||
|
params: {
|
||||||
|
id,
|
||||||
|
submission: true,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
.asJSON()
|
||||||
|
.then((_) => true)
|
||||||
|
.catch(() => false);
|
||||||
|
|
||||||
|
nowPlaying = async (id: string) =>
|
||||||
|
this.GET({
|
||||||
|
url: `/rest/scrobble`,
|
||||||
|
params: {
|
||||||
|
id,
|
||||||
|
submission: false,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
.asJSON()
|
||||||
|
.then((_) => true)
|
||||||
|
.catch(() => false);
|
||||||
|
|
||||||
|
searchArtists = async (query: string) =>
|
||||||
|
this.search3({ query, artistCount: 20 }).then(({ artists }) =>
|
||||||
|
artists.map((artist) => ({
|
||||||
|
id: artist.id,
|
||||||
|
name: artist.name,
|
||||||
|
image: artistImageURN({
|
||||||
|
artistId: artist.id,
|
||||||
|
artistImageURL: artist.artistImageUrl,
|
||||||
|
}),
|
||||||
|
}))
|
||||||
|
);
|
||||||
|
|
||||||
|
searchAlbums = async (query: string) =>
|
||||||
|
this.search3({ query, albumCount: 20 }).then(({ albums }) =>
|
||||||
|
this.toAlbumSummary(albums)
|
||||||
|
);
|
||||||
|
|
||||||
|
searchTracks = async (query: string) =>
|
||||||
|
this.search3({ query, songCount: 20 }).then(({ songs }) =>
|
||||||
|
Promise.all(songs.map((it) => this.getTrack(it.id)))
|
||||||
|
);
|
||||||
|
|
||||||
|
playlists = async () =>
|
||||||
|
this.GET({ url: "/rest/getPlaylists" })
|
||||||
|
.asJSON<GetPlaylistsResponse>()
|
||||||
|
.then((it) => it.playlists.playlist || [])
|
||||||
|
.then((playlists) =>
|
||||||
|
playlists.map((it) => ({ id: it.id, name: it.name }))
|
||||||
|
);
|
||||||
|
|
||||||
|
playlist = async (id: string) =>
|
||||||
|
this.GET({
|
||||||
|
url: "/rest/getPlaylist",
|
||||||
|
params: {
|
||||||
|
id,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
.asJSON<GetPlaylistResponse>()
|
||||||
|
.then((it) => it.playlist)
|
||||||
|
.then((playlist) => {
|
||||||
|
let trackNumber = 1;
|
||||||
|
return {
|
||||||
|
id: playlist.id,
|
||||||
|
name: playlist.name,
|
||||||
|
entries: (playlist.entry || []).map((entry) => ({
|
||||||
|
...asTrack(
|
||||||
|
{
|
||||||
|
id: entry.albumId!,
|
||||||
|
name: entry.album!,
|
||||||
|
year: entry.year,
|
||||||
|
genre: maybeAsGenre(entry.genre),
|
||||||
|
artistName: entry.artist,
|
||||||
|
artistId: entry.artistId,
|
||||||
|
coverArt: coverArtURN(entry.coverArt),
|
||||||
|
},
|
||||||
|
entry
|
||||||
|
),
|
||||||
|
number: trackNumber++,
|
||||||
|
})),
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
createPlaylist = async (name: string) =>
|
||||||
|
this.GET({
|
||||||
|
url: "/rest/createPlaylist",
|
||||||
|
params: {
|
||||||
|
name,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
.asJSON<GetPlaylistResponse>()
|
||||||
|
.then((it) => it.playlist)
|
||||||
|
.then((it) => ({ id: it.id, name: it.name }));
|
||||||
|
|
||||||
|
deletePlaylist = async (id: string) =>
|
||||||
|
this.GET({
|
||||||
|
url: "/rest/deletePlaylist",
|
||||||
|
params: {
|
||||||
|
id,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
.asJSON<GetPlaylistResponse>()
|
||||||
|
.then((_) => true);
|
||||||
|
|
||||||
|
addToPlaylist = async (playlistId: string, trackId: string) =>
|
||||||
|
this.GET({
|
||||||
|
url: "/rest/updatePlaylist",
|
||||||
|
params: {
|
||||||
|
playlistId,
|
||||||
|
songIdToAdd: trackId,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
.asJSON<GetPlaylistResponse>()
|
||||||
|
.then((_) => true);
|
||||||
|
|
||||||
|
removeFromPlaylist = async (playlistId: string, indicies: number[]) =>
|
||||||
|
this.GET({
|
||||||
|
url: "/rest/updatePlaylist",
|
||||||
|
params: {
|
||||||
|
playlistId,
|
||||||
|
songIndexToRemove: indicies,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
.asJSON<GetPlaylistResponse>()
|
||||||
|
.then((_) => true);
|
||||||
|
|
||||||
|
similarSongs = async (id: string) =>
|
||||||
|
this.GET({
|
||||||
|
url: "/rest/getSimilarSongs2",
|
||||||
|
params: { id, count: 50 },
|
||||||
|
})
|
||||||
|
.asJSON<GetSimilarSongsResponse>()
|
||||||
|
.then((it) => it.similarSongs2.song || [])
|
||||||
|
.then((songs) =>
|
||||||
|
Promise.all(
|
||||||
|
songs.map((song) =>
|
||||||
|
this.getAlbum(song.albumId!).then((album) => asTrack(album, song))
|
||||||
|
)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
topSongs = async (artistId: string) =>
|
||||||
|
this.getArtist(artistId).then(({ name }) =>
|
||||||
|
this.GET({
|
||||||
|
url: "/rest/getTopSongs",
|
||||||
|
params: {
|
||||||
|
artist: name,
|
||||||
|
count: 50,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
.asJSON<GetTopSongsResponse>()
|
||||||
|
.then((it) => it.topSongs.song || [])
|
||||||
|
.then((songs) =>
|
||||||
|
Promise.all(
|
||||||
|
songs.map((song) =>
|
||||||
|
this.getAlbum(song.albumId!).then((album) => asTrack(album, song))
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
private getArtists = (): Promise<
|
||||||
|
(IdName & { albumCount: number; image: BUrn | undefined })[]
|
||||||
|
> =>
|
||||||
|
this.GET({ url: "/rest/getArtists" })
|
||||||
|
.asJSON<GetArtistsResponse>()
|
||||||
|
.then((it) => (it.artists.index || []).flatMap((it) => it.artist || []))
|
||||||
|
.then((artists) =>
|
||||||
|
artists.map((artist) => ({
|
||||||
|
id: `${artist.id}`,
|
||||||
|
name: artist.name,
|
||||||
|
albumCount: artist.albumCount,
|
||||||
|
image: artistImageURN({
|
||||||
|
artistId: artist.id,
|
||||||
|
artistImageURL: artist.artistImageUrl,
|
||||||
|
}),
|
||||||
|
}))
|
||||||
|
);
|
||||||
|
|
||||||
|
private getArtistInfo = (
|
||||||
|
id: string
|
||||||
|
): Promise<{
|
||||||
|
similarArtist: (ArtistSummary & { inLibrary: boolean })[];
|
||||||
|
images: {
|
||||||
|
s: string | undefined;
|
||||||
|
m: string | undefined;
|
||||||
|
l: string | undefined;
|
||||||
|
};
|
||||||
|
}> =>
|
||||||
|
this.GET({
|
||||||
|
url: "/rest/getArtistInfo2",
|
||||||
|
params: {
|
||||||
|
id,
|
||||||
|
count: 50,
|
||||||
|
includeNotPresent: true,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
.asJSON<GetArtistInfoResponse>()
|
||||||
|
.then((it) => it.artistInfo2)
|
||||||
|
.then((it) => ({
|
||||||
|
images: {
|
||||||
|
s: it.smallImageUrl,
|
||||||
|
m: it.mediumImageUrl,
|
||||||
|
l: it.largeImageUrl,
|
||||||
|
},
|
||||||
|
similarArtist: (it.similarArtist || []).map((artist) => ({
|
||||||
|
id: `${artist.id}`,
|
||||||
|
name: artist.name,
|
||||||
|
inLibrary: artistIsInLibrary(artist.id),
|
||||||
|
image: artistImageURN({
|
||||||
|
artistId: artist.id,
|
||||||
|
artistImageURL: artist.artistImageUrl,
|
||||||
|
}),
|
||||||
|
})),
|
||||||
|
}));
|
||||||
|
|
||||||
|
private getAlbum = (id: string): Promise<Album> =>
|
||||||
|
this.GET({ url: "/rest/getAlbum", params: { id } })
|
||||||
|
.asJSON<GetAlbumResponse>()
|
||||||
|
.then((it) => it.album)
|
||||||
|
.then((album) => ({
|
||||||
|
id: album.id,
|
||||||
|
name: album.name,
|
||||||
|
year: album.year,
|
||||||
|
genre: maybeAsGenre(album.genre),
|
||||||
|
artistId: album.artistId,
|
||||||
|
artistName: album.artist,
|
||||||
|
coverArt: coverArtURN(album.coverArt),
|
||||||
|
}));
|
||||||
|
|
||||||
|
private getArtist = (
|
||||||
|
id: string
|
||||||
|
): Promise<
|
||||||
|
IdName & { artistImageUrl: string | undefined; albums: AlbumSummary[] }
|
||||||
|
> =>
|
||||||
|
this.GET({
|
||||||
|
url: "/rest/getArtist",
|
||||||
|
params: {
|
||||||
|
id,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
.asJSON<GetArtistResponse>()
|
||||||
|
.then((it) => it.artist)
|
||||||
|
.then((it) => ({
|
||||||
|
id: it.id,
|
||||||
|
name: it.name,
|
||||||
|
artistImageUrl: it.artistImageUrl,
|
||||||
|
albums: this.toAlbumSummary(it.album || []),
|
||||||
|
}));
|
||||||
|
|
||||||
|
private getArtistWithInfo = (id: string) =>
|
||||||
|
Promise.all([this.getArtist(id), this.getArtistInfo(id)]).then(
|
||||||
|
([artist, artistInfo]) => ({
|
||||||
|
id: artist.id,
|
||||||
|
name: artist.name,
|
||||||
|
image: artistImageURN({
|
||||||
|
artistId: artist.id,
|
||||||
|
artistImageURL: [
|
||||||
|
artist.artistImageUrl,
|
||||||
|
artistInfo.images.l,
|
||||||
|
artistInfo.images.m,
|
||||||
|
artistInfo.images.s,
|
||||||
|
].find(isValidImage),
|
||||||
|
}),
|
||||||
|
albums: artist.albums,
|
||||||
|
similarArtists: artistInfo.similarArtist,
|
||||||
|
})
|
||||||
|
);
|
||||||
|
|
||||||
|
private getCoverArt = (id: string, size?: number) =>
|
||||||
|
this.GET({
|
||||||
|
url: "/rest/getCoverArt",
|
||||||
|
params: { id, size },
|
||||||
|
responseType: "arraybuffer",
|
||||||
|
}).asRaw();
|
||||||
|
|
||||||
|
private getTrack = (id: string) =>
|
||||||
|
this.GET({
|
||||||
|
url: "/rest/getSong",
|
||||||
|
params: {
|
||||||
|
id,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
.asJSON<GetSongResponse>()
|
||||||
|
.then((it) => it.song)
|
||||||
|
.then((song) =>
|
||||||
|
this.getAlbum(song.albumId!).then((album) => asTrack(album, song))
|
||||||
|
);
|
||||||
|
|
||||||
|
private toAlbumSummary = (albumList: album[]): AlbumSummary[] =>
|
||||||
|
albumList.map((album) => ({
|
||||||
|
id: album.id,
|
||||||
|
name: album.name,
|
||||||
|
year: album.year,
|
||||||
|
genre: maybeAsGenre(album.genre),
|
||||||
|
artistId: album.artistId,
|
||||||
|
artistName: album.artist,
|
||||||
|
coverArt: coverArtURN(album.coverArt),
|
||||||
|
}));
|
||||||
|
|
||||||
|
private search3 = (q: any) =>
|
||||||
|
this.GET({
|
||||||
|
url: "/rest/search3",
|
||||||
|
params: {
|
||||||
|
artistCount: 0,
|
||||||
|
albumCount: 0,
|
||||||
|
songCount: 0,
|
||||||
|
...q,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
.asJSON<Search3Response>()
|
||||||
|
.then((it) => ({
|
||||||
|
artists: it.searchResult3.artist || [],
|
||||||
|
albums: it.searchResult3.album || [],
|
||||||
|
songs: it.searchResult3.song || [],
|
||||||
|
}));
|
||||||
|
|
||||||
|
private getAlbumList2 = (q: AlbumQuery) =>
|
||||||
|
Promise.all([
|
||||||
|
this.getArtists().then((it) =>
|
||||||
|
inject(it, (total, artist) => total + artist.albumCount, 0)
|
||||||
|
),
|
||||||
|
this.GET({
|
||||||
|
url: "/rest/getAlbumList2",
|
||||||
|
params: {
|
||||||
|
type: AlbumQueryTypeToSubsonicType[q.type],
|
||||||
|
...(q.genre ? { genre: b64Decode(q.genre) } : {}),
|
||||||
|
size: 500,
|
||||||
|
offset: q._index,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
.asJSON<GetAlbumListResponse>()
|
||||||
|
.then((response) => response.albumList2.album || [])
|
||||||
|
.then(this.toAlbumSummary),
|
||||||
|
]).then(([total, albums]) => ({
|
||||||
|
results: albums.slice(0, q._count),
|
||||||
|
total: albums.length == 500 ? total : (q._index || 0) + albums.length,
|
||||||
|
}));
|
||||||
|
}
|
||||||
176
src/subsonic/index.ts
Normal file
176
src/subsonic/index.ts
Normal file
@@ -0,0 +1,176 @@
|
|||||||
|
import { taskEither as TE } from "fp-ts";
|
||||||
|
import { pipe } from "fp-ts/lib/function";
|
||||||
|
import { Md5 } from "ts-md5/dist/md5";
|
||||||
|
import axios from "axios";
|
||||||
|
import randomstring from "randomstring";
|
||||||
|
import _ from "underscore";
|
||||||
|
// todo: rename http2 to http
|
||||||
|
import { Http2, http2From } from "../http";
|
||||||
|
|
||||||
|
import {
|
||||||
|
Credentials,
|
||||||
|
MusicService,
|
||||||
|
MusicLibrary,
|
||||||
|
Track,
|
||||||
|
AuthFailure,
|
||||||
|
} from "../music_service";
|
||||||
|
import { b64Encode, b64Decode } from "../b64";
|
||||||
|
import { axiosImageFetcher, ImageFetcher } from "../images";
|
||||||
|
import { navidromeMusicLibrary, SubsonicGenericMusicLibrary } from "./generic";
|
||||||
|
import { client } from "./subsonic_http";
|
||||||
|
|
||||||
|
export const t = (password: string, s: string) =>
|
||||||
|
Md5.hashStr(`${password}${s}`);
|
||||||
|
|
||||||
|
export const t_and_s = (password: string) => {
|
||||||
|
const s = randomstring.generate();
|
||||||
|
return {
|
||||||
|
t: t(password, s),
|
||||||
|
s,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
// todo: this is an ND thing
|
||||||
|
export const DODGY_IMAGE_NAME = "2a96cbd8b46e442fc41c2b86b821562f.png";
|
||||||
|
|
||||||
|
export type SubsonicEnvelope = {
|
||||||
|
"subsonic-response": SubsonicResponse;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type SubsonicResponse = {
|
||||||
|
status: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type SubsonicError = SubsonicResponse & {
|
||||||
|
error: {
|
||||||
|
code: string;
|
||||||
|
message: string;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
export type PingResponse = {
|
||||||
|
status: string;
|
||||||
|
version: string;
|
||||||
|
type: string;
|
||||||
|
serverVersion: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export function isError(
|
||||||
|
subsonicResponse: SubsonicResponse
|
||||||
|
): subsonicResponse is SubsonicError {
|
||||||
|
return (subsonicResponse as SubsonicError).error !== undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
// todo: is this a good name?
|
||||||
|
export type StreamClientApplication = (track: Track) => string;
|
||||||
|
|
||||||
|
export const DEFAULT_CLIENT_APPLICATION = "bonob";
|
||||||
|
export const USER_AGENT = "bonob";
|
||||||
|
|
||||||
|
export const DEFAULT: StreamClientApplication = (_: Track) =>
|
||||||
|
DEFAULT_CLIENT_APPLICATION;
|
||||||
|
|
||||||
|
export function appendMimeTypeToClientFor(mimeTypes: string[]) {
|
||||||
|
return (track: Track) =>
|
||||||
|
mimeTypes.includes(track.mimeType) ? `bonob+${track.mimeType}` : "bonob";
|
||||||
|
}
|
||||||
|
|
||||||
|
export type SubsonicCredentials = Credentials & {
|
||||||
|
type: string;
|
||||||
|
bearer: string | undefined;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const asToken = (credentials: SubsonicCredentials) =>
|
||||||
|
b64Encode(JSON.stringify(credentials));
|
||||||
|
|
||||||
|
export const parseToken = (token: string): SubsonicCredentials =>
|
||||||
|
JSON.parse(b64Decode(token));
|
||||||
|
|
||||||
|
export interface SubsonicMusicLibrary extends MusicLibrary {
|
||||||
|
flavour(): string;
|
||||||
|
bearerToken(
|
||||||
|
credentials: Credentials
|
||||||
|
): TE.TaskEither<Error, string | undefined>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export class Subsonic implements MusicService {
|
||||||
|
url: string;
|
||||||
|
|
||||||
|
// todo: does this need to be in here now?
|
||||||
|
streamClientApplication: StreamClientApplication;
|
||||||
|
// todo: why is this in here?
|
||||||
|
externalImageFetcher: ImageFetcher;
|
||||||
|
|
||||||
|
subsonicHttp: Http2;
|
||||||
|
|
||||||
|
constructor(
|
||||||
|
url: string,
|
||||||
|
streamClientApplication: StreamClientApplication = DEFAULT,
|
||||||
|
externalImageFetcher: ImageFetcher = axiosImageFetcher
|
||||||
|
) {
|
||||||
|
this.url = url;
|
||||||
|
this.streamClientApplication = streamClientApplication;
|
||||||
|
this.externalImageFetcher = externalImageFetcher;
|
||||||
|
this.subsonicHttp = http2From(axios).with({
|
||||||
|
baseURL: this.url,
|
||||||
|
params: { v: "1.16.1", c: DEFAULT_CLIENT_APPLICATION },
|
||||||
|
headers: { "User-Agent": "bonob" },
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
asAuthParams = (credentials: Credentials) => ({
|
||||||
|
u: credentials.username,
|
||||||
|
...t_and_s(credentials.password),
|
||||||
|
})
|
||||||
|
|
||||||
|
generateToken = (credentials: Credentials) =>
|
||||||
|
pipe(
|
||||||
|
TE.tryCatch(
|
||||||
|
() => client(this.subsonicHttp.with({ params: this.asAuthParams(credentials) } ))({ method: 'get', url: "/rest/ping.view" }).asJSON<PingResponse>(),
|
||||||
|
(e) => new AuthFailure(e as string)
|
||||||
|
),
|
||||||
|
TE.chain(({ type }) =>
|
||||||
|
pipe(
|
||||||
|
TE.tryCatch(
|
||||||
|
() => this.libraryFor({ ...credentials, type, bearer: undefined }),
|
||||||
|
() => new AuthFailure("Failed to get library")
|
||||||
|
),
|
||||||
|
TE.map((library) => ({ type, library }))
|
||||||
|
)
|
||||||
|
),
|
||||||
|
TE.chain(({ library, type }) =>
|
||||||
|
pipe(
|
||||||
|
library.bearerToken(credentials),
|
||||||
|
TE.map((bearer) => ({ bearer, type }))
|
||||||
|
)
|
||||||
|
),
|
||||||
|
TE.map(({ bearer, type }) => ({
|
||||||
|
serviceToken: asToken({ ...credentials, bearer, type }),
|
||||||
|
userId: credentials.username,
|
||||||
|
nickname: credentials.username,
|
||||||
|
}))
|
||||||
|
);
|
||||||
|
|
||||||
|
refreshToken = (serviceToken: string) =>
|
||||||
|
this.generateToken(parseToken(serviceToken));
|
||||||
|
|
||||||
|
login = async (token: string) => this.libraryFor(parseToken(token));
|
||||||
|
|
||||||
|
private libraryFor = (
|
||||||
|
credentials: SubsonicCredentials
|
||||||
|
): Promise<SubsonicMusicLibrary> => {
|
||||||
|
const subsonicGenericLibrary = new SubsonicGenericMusicLibrary(
|
||||||
|
this.streamClientApplication,
|
||||||
|
this.subsonicHttp.with({ params: this.asAuthParams(credentials) } )
|
||||||
|
);
|
||||||
|
if (credentials.type == "navidrome") {
|
||||||
|
return Promise.resolve(
|
||||||
|
navidromeMusicLibrary(this.url, subsonicGenericLibrary, credentials)
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
return Promise.resolve(subsonicGenericLibrary);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export default Subsonic;
|
||||||
95
src/subsonic/navidrome.ts
Normal file
95
src/subsonic/navidrome.ts
Normal file
@@ -0,0 +1,95 @@
|
|||||||
|
import { option as O, taskEither as TE } from "fp-ts";
|
||||||
|
import * as A from "fp-ts/Array";
|
||||||
|
import { pipe } from "fp-ts/lib/function";
|
||||||
|
import { ordString } from "fp-ts/lib/Ord";
|
||||||
|
import { inject } from "underscore";
|
||||||
|
import _ from "underscore";
|
||||||
|
import axios from "axios";
|
||||||
|
|
||||||
|
import { SubsonicCredentials, SubsonicMusicLibrary } from ".";
|
||||||
|
import { ArtistQuery, ArtistSummary, AuthFailure, Credentials, Result, Sortable } from "../music_service";
|
||||||
|
import { artistImageURN } from "./generic";
|
||||||
|
|
||||||
|
export type NDArtist = {
|
||||||
|
id: string;
|
||||||
|
name: string;
|
||||||
|
orderArtistName: string | undefined;
|
||||||
|
largeImageUrl: string | undefined;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const artistSummaryFromNDArtist = (
|
||||||
|
artist: NDArtist
|
||||||
|
): ArtistSummary & Sortable => ({
|
||||||
|
id: artist.id,
|
||||||
|
name: artist.name,
|
||||||
|
sortName: artist.orderArtistName || artist.name,
|
||||||
|
image: artistImageURN({
|
||||||
|
artistId: artist.id,
|
||||||
|
artistImageURL: artist.largeImageUrl,
|
||||||
|
}),
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
export const navidromeMusicLibrary = (
|
||||||
|
url: string,
|
||||||
|
subsonicLibrary: SubsonicMusicLibrary,
|
||||||
|
subsonicCredentials: SubsonicCredentials
|
||||||
|
): SubsonicMusicLibrary => ({
|
||||||
|
...subsonicLibrary,
|
||||||
|
flavour: () => "navidrome",
|
||||||
|
bearerToken: (
|
||||||
|
credentials: Credentials
|
||||||
|
): TE.TaskEither<Error, string | undefined> =>
|
||||||
|
pipe(
|
||||||
|
TE.tryCatch(
|
||||||
|
() =>
|
||||||
|
// todo: not hardcode axios in here
|
||||||
|
axios({
|
||||||
|
method: "post",
|
||||||
|
baseURL: url,
|
||||||
|
url: `/auth/login`,
|
||||||
|
data: _.pick(credentials, "username", "password"),
|
||||||
|
}),
|
||||||
|
() => new AuthFailure("Failed to get bearerToken")
|
||||||
|
),
|
||||||
|
TE.map((it) => it.data.token as string | undefined)
|
||||||
|
),
|
||||||
|
artists: async (
|
||||||
|
q: ArtistQuery
|
||||||
|
): Promise<Result<ArtistSummary & Sortable>> => {
|
||||||
|
let params: any = {
|
||||||
|
_sort: "name",
|
||||||
|
_order: "ASC",
|
||||||
|
_start: q._index || "0",
|
||||||
|
};
|
||||||
|
if (q._count) {
|
||||||
|
params = {
|
||||||
|
...params,
|
||||||
|
_end: (q._index || 0) + q._count,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const x: Promise<Result<ArtistSummary & Sortable>> = axios
|
||||||
|
.get(`${url}/api/artist`, {
|
||||||
|
params: asURLSearchParams(params),
|
||||||
|
headers: {
|
||||||
|
"User-Agent": USER_AGENT,
|
||||||
|
"x-nd-authorization": `Bearer ${subsonicCredentials.bearer}`,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
.catch((e) => {
|
||||||
|
throw `Navidrome failed with: ${e}`;
|
||||||
|
})
|
||||||
|
.then((response) => {
|
||||||
|
if (response.status != 200 && response.status != 206) {
|
||||||
|
throw `Navidrome failed with a ${response.status || "no!"} status`;
|
||||||
|
} else return response;
|
||||||
|
})
|
||||||
|
.then((it) => ({
|
||||||
|
results: (it.data as NDArtist[]).map(artistSummaryFromNDArtist),
|
||||||
|
total: Number.parseInt(it.headers["x-total-count"] || "0"),
|
||||||
|
}));
|
||||||
|
|
||||||
|
return x;
|
||||||
|
},
|
||||||
|
});
|
||||||
51
src/subsonic/subsonic_http.ts
Normal file
51
src/subsonic/subsonic_http.ts
Normal file
@@ -0,0 +1,51 @@
|
|||||||
|
import { AxiosResponse } from "axios";
|
||||||
|
import { isError, SubsonicEnvelope } from ".";
|
||||||
|
// todo: rename http2 to http
|
||||||
|
import { Http2, RequestParams } from "../http";
|
||||||
|
|
||||||
|
export type HttpResponse = {
|
||||||
|
data: any;
|
||||||
|
status: number;
|
||||||
|
headers: any;
|
||||||
|
};
|
||||||
|
|
||||||
|
const asJSON = <T>(response: HttpResponse): T => {
|
||||||
|
const subsonicResponse = (response.data as SubsonicEnvelope)[
|
||||||
|
"subsonic-response"
|
||||||
|
];
|
||||||
|
if (isError(subsonicResponse))
|
||||||
|
throw `Subsonic error:${subsonicResponse.error.message}`;
|
||||||
|
else return subsonicResponse as unknown as T;
|
||||||
|
};
|
||||||
|
const throwUp = (error: any) => {
|
||||||
|
throw `Subsonic failed with: ${error}`;
|
||||||
|
};
|
||||||
|
const verifyResponse = (response: AxiosResponse<any>) => {
|
||||||
|
if (response.status != 200 && response.status != 206) {
|
||||||
|
throw `Subsonic failed with a ${response.status || "no!"} status`;
|
||||||
|
} else return response;
|
||||||
|
};
|
||||||
|
|
||||||
|
export interface SubsonicHttpResponse {
|
||||||
|
asRaw(): Promise<AxiosResponse<any>>;
|
||||||
|
asJSON<T>(): Promise<T>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface SubsonicHttp {
|
||||||
|
(query: Partial<RequestParams>): SubsonicHttpResponse;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const client = (http: Http2): SubsonicHttp => {
|
||||||
|
return (query: Partial<RequestParams>): SubsonicHttpResponse => {
|
||||||
|
return {
|
||||||
|
asRaw: () => http(query).catch(throwUp).then(verifyResponse),
|
||||||
|
|
||||||
|
asJSON: <T>() =>
|
||||||
|
http
|
||||||
|
.with({ params: { f: "json" } })(query)
|
||||||
|
.catch(throwUp)
|
||||||
|
.then(verifyResponse)
|
||||||
|
.then(asJSON) as Promise<T>,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
41
src/utils.ts
41
src/utils.ts
@@ -1,7 +1,42 @@
|
|||||||
export function takeWithRepeats<T>(things:T[], count: number) {
|
import { flatten } from "underscore";
|
||||||
|
|
||||||
|
// todo: move this
|
||||||
|
export const BROWSER_HEADERS = {
|
||||||
|
accept:
|
||||||
|
"text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8",
|
||||||
|
"accept-encoding": "gzip, deflate, br",
|
||||||
|
"accept-language": "en-GB,en;q=0.5",
|
||||||
|
"upgrade-insecure-requests": "1",
|
||||||
|
"user-agent":
|
||||||
|
"Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:86.0) Gecko/20100101 Firefox/86.0",
|
||||||
|
};
|
||||||
|
|
||||||
|
// todo: move this
|
||||||
|
export const asURLSearchParams = (q: any) => {
|
||||||
|
const urlSearchParams = new URLSearchParams();
|
||||||
|
Object.keys(q).forEach((k) => {
|
||||||
|
flatten([q[k]]).forEach((v) => {
|
||||||
|
urlSearchParams.append(k, `${v}`);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
return urlSearchParams;
|
||||||
|
};
|
||||||
|
|
||||||
|
export function takeWithRepeats<T>(things: T[], count: number) {
|
||||||
const result = [];
|
const result = [];
|
||||||
for(let i = 0; i < count; i++) {
|
for (let i = 0; i < count; i++) {
|
||||||
result.push(things[i % things.length])
|
result.push(things[i % things.length]);
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export const mask = (thing: any, fields: string[]) =>
|
||||||
|
fields.reduce(
|
||||||
|
(res: any, key: string) => {
|
||||||
|
if (Object.keys(res).includes(key)) {
|
||||||
|
res[key] = "****";
|
||||||
|
}
|
||||||
|
return res;
|
||||||
|
},
|
||||||
|
{ ...thing }
|
||||||
|
);
|
||||||
|
|||||||
@@ -17,7 +17,7 @@ import {
|
|||||||
} from "../src/music_service";
|
} from "../src/music_service";
|
||||||
|
|
||||||
import { b64Encode } from "../src/b64";
|
import { b64Encode } from "../src/b64";
|
||||||
import { artistImageURN } from "../src/subsonic";
|
import { artistImageURN } from "../src/subsonic/generic";
|
||||||
|
|
||||||
const randomInt = (max: number) => Math.floor(Math.random() * Math.floor(max));
|
const randomInt = (max: number) => Math.floor(Math.random() * Math.floor(max));
|
||||||
const randomIpAddress = () => `127.0.${randomInt(255)}.${randomInt(255)}`;
|
const randomIpAddress = () => `127.0.${randomInt(255)}.${randomInt(255)}`;
|
||||||
|
|||||||
@@ -1,58 +1,85 @@
|
|||||||
import dayjs from "dayjs";
|
import { randomInt } from "crypto";
|
||||||
import { isChristmas, isCNY, isHalloween, isHoli } from "../src/clock";
|
import dayjs, { Dayjs } from "dayjs";
|
||||||
|
import timezone from "dayjs/plugin/timezone";
|
||||||
|
dayjs.extend(timezone);
|
||||||
|
|
||||||
describe("isChristmas", () => {
|
import { Clock, isChristmas, isCNY, isCNY_2022, isCNY_2023, isCNY_2024, isCNY_2025, isHalloween, isHoli, isMay4 } from "../src/clock";
|
||||||
["2000/12/25", "2022/12/25", "2030/12/25"].forEach((date) => {
|
|
||||||
it(`should return true for ${date} regardless of year`, () => {
|
|
||||||
expect(isChristmas({ now: () => dayjs(date) })).toEqual(true);
|
|
||||||
|
const randomDate = () => dayjs().subtract(randomInt(1, 1000), 'days');
|
||||||
|
const randomDates = (count: number, exclude: string[]) => {
|
||||||
|
const result: Dayjs[] = [];
|
||||||
|
while(result.length < count) {
|
||||||
|
const next = randomDate();
|
||||||
|
if(!exclude.find(it => dayjs(it).isSame(next, 'date'))) {
|
||||||
|
result.push(next)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
|
||||||
|
function describeFixedDateMonthEvent(
|
||||||
|
name: string,
|
||||||
|
dateMonth: string,
|
||||||
|
f: (clock: Clock) => boolean
|
||||||
|
) {
|
||||||
|
const randomYear = randomInt(2020, 3000);
|
||||||
|
const date = dateMonth.split("/")[0];
|
||||||
|
const month = dateMonth.split("/")[1];
|
||||||
|
|
||||||
|
describe(name, () => {
|
||||||
|
it(`should return true for ${randomYear}-${month}-${date}T00:00:00 ragardless of year`, () => {
|
||||||
|
expect(f({ now: () => dayjs(`${randomYear}-${month}-${date}T00:00:00Z`) })).toEqual(true);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it(`should return true for ${randomYear}-${month}-${date}T12:00:00 regardless of year`, () => {
|
||||||
|
expect(f({ now: () => dayjs(`${randomYear}-${month}-${date}T12:00:00Z`) })).toEqual(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it(`should return true for ${randomYear}-${month}-${date}T23:59:00 regardless of year`, () => {
|
||||||
|
expect(f({ now: () => dayjs(`${randomYear}-${month}-${date}T23:59:00`) })).toEqual(true);
|
||||||
});
|
});
|
||||||
|
|
||||||
["2000/12/24", "2000/12/26", "2021/01/01"].forEach((date) => {
|
["2000/12/24", "2000/12/26", "2021/01/01"].forEach((date) => {
|
||||||
it(`should return false for ${date} regardless of year`, () => {
|
it(`should return false for ${date}`, () => {
|
||||||
expect(isChristmas({ now: () => dayjs(date) })).toEqual(false);
|
expect(f({ now: () => dayjs(date) })).toEqual(false);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
|
||||||
describe("isHalloween", () => {
|
function describeFixedDateEvent(
|
||||||
["2000/10/31", "2022/10/31", "2030/10/31"].forEach((date) => {
|
name: string,
|
||||||
it(`should return true for ${date} regardless of year`, () => {
|
dates: string[],
|
||||||
expect(isHalloween({ now: () => dayjs(date) })).toEqual(true);
|
f: (clock: Clock) => boolean
|
||||||
|
) {
|
||||||
|
describe(name, () => {
|
||||||
|
dates.forEach((date) => {
|
||||||
|
it(`should return true for ${date}T00:00:00`, () => {
|
||||||
|
expect(f({ now: () => dayjs(`${date}T00:00:00`) })).toEqual(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it(`should return true for ${date}T23:59:59`, () => {
|
||||||
|
expect(f({ now: () => dayjs(`${date}T23:59:59`) })).toEqual(true);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
["2000/09/31", "2000/10/30", "2021/01/01"].forEach((date) => {
|
randomDates(10, dates).forEach((date) => {
|
||||||
it(`should return false for ${date} regardless of year`, () => {
|
it(`should return false for ${date}`, () => {
|
||||||
expect(isHalloween({ now: () => dayjs(date) })).toEqual(false);
|
expect(f({ now: () => dayjs(date) })).toEqual(false);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
|
||||||
describe("isHoli", () => {
|
describeFixedDateMonthEvent("christmas", "25/12", isChristmas);
|
||||||
["2022/03/18", "2023/03/07", "2024/03/25", "2025/03/14"].forEach((date) => {
|
describeFixedDateMonthEvent("halloween", "31/10", isHalloween);
|
||||||
it(`should return true for ${date} regardless of year`, () => {
|
describeFixedDateMonthEvent("may4", "04/05", isMay4);
|
||||||
expect(isHoli({ now: () => dayjs(date) })).toEqual(true);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
["2000/09/31", "2000/10/30", "2021/01/01"].forEach((date) => {
|
describeFixedDateEvent("holi", ["2022-03-18", "2023-03-07", "2024-03-25", "2025-03-14"], isHoli);
|
||||||
it(`should return false for ${date} regardless of year`, () => {
|
describeFixedDateEvent("cny", ["2022-02-01", "2023-01-22", "2024-02-10", "2025-02-29"], isCNY);
|
||||||
expect(isHoli({ now: () => dayjs(date) })).toEqual(false);
|
describeFixedDateEvent("cny 2022", ["2022-02-01"], isCNY_2022);
|
||||||
});
|
describeFixedDateEvent("cny 2023", ["2023/01/22"], isCNY_2023);
|
||||||
});
|
describeFixedDateEvent("cny 2024", ["2024/02/10"], isCNY_2024);
|
||||||
});
|
describeFixedDateEvent("cny 2025", ["2025/02/29"], isCNY_2025);
|
||||||
|
|
||||||
describe("isCNY", () => {
|
|
||||||
["2022/02/01", "2023/01/22", "2024/02/10", "2025/02/29"].forEach((date) => {
|
|
||||||
it(`should return true for ${date} regardless of year`, () => {
|
|
||||||
expect(isCNY({ now: () => dayjs(date) })).toEqual(true);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
["2000/09/31", "2000/10/30", "2021/01/01"].forEach((date) => {
|
|
||||||
it(`should return false for ${date} regardless of year`, () => {
|
|
||||||
expect(isCNY({ now: () => dayjs(date) })).toEqual(false);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|||||||
@@ -96,43 +96,36 @@ describe("config", () => {
|
|||||||
propertyGetter: (config: any) => any
|
propertyGetter: (config: any) => any
|
||||||
) {
|
) {
|
||||||
describe(name, () => {
|
describe(name, () => {
|
||||||
function expecting({
|
it.each([
|
||||||
value,
|
[expectedDefault, ""],
|
||||||
expected,
|
[expectedDefault, undefined],
|
||||||
}: {
|
[true, "true"],
|
||||||
value: string;
|
[false, "false"],
|
||||||
expected: boolean;
|
[false, "foo"],
|
||||||
}) {
|
])("should be %s when env var is '%s'", (expected, value) => {
|
||||||
describe(`when value is '${value}'`, () => {
|
|
||||||
it(`should be ${expected}`, () => {
|
|
||||||
process.env[envVar] = value;
|
process.env[envVar] = value;
|
||||||
expect(propertyGetter(config())).toEqual(expected);
|
expect(propertyGetter(config())).toEqual(expected);
|
||||||
});
|
})
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
expecting({ value: "", expected: expectedDefault });
|
|
||||||
expecting({ value: "true", expected: true });
|
|
||||||
expecting({ value: "false", expected: false });
|
|
||||||
expecting({ value: "foo", expected: false });
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
describe("bonobUrl", () => {
|
describe("bonobUrl", () => {
|
||||||
["BNB_URL", "BONOB_URL", "BONOB_WEB_ADDRESS"].forEach((key) => {
|
describe.each([
|
||||||
describe(`when ${key} is specified`, () => {
|
"BNB_URL",
|
||||||
|
"BONOB_URL",
|
||||||
|
"BONOB_WEB_ADDRESS"
|
||||||
|
])("when %s is specified", (k) => {
|
||||||
it("should be used", () => {
|
it("should be used", () => {
|
||||||
const url = "http://bonob1.example.com:8877/";
|
const url = "http://bonob1.example.com:8877/";
|
||||||
|
|
||||||
process.env["BNB_URL"] = "";
|
process.env["BNB_URL"] = "";
|
||||||
process.env["BONOB_URL"] = "";
|
process.env["BONOB_URL"] = "";
|
||||||
process.env["BONOB_WEB_ADDRESS"] = "";
|
process.env["BONOB_WEB_ADDRESS"] = "";
|
||||||
process.env[key] = url;
|
process.env[k] = url;
|
||||||
|
|
||||||
expect(config().bonobUrl.href()).toEqual(url);
|
expect(config().bonobUrl.href()).toEqual(url);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
|
||||||
|
|
||||||
describe("when none of BNB_URL, BONOB_URL, BONOB_WEB_ADDRESS are specified", () => {
|
describe("when none of BNB_URL, BONOB_URL, BONOB_WEB_ADDRESS are specified", () => {
|
||||||
describe("when BONOB_PORT is not specified", () => {
|
describe("when BONOB_PORT is not specified", () => {
|
||||||
@@ -165,8 +158,10 @@ describe("config", () => {
|
|||||||
|
|
||||||
describe("icons", () => {
|
describe("icons", () => {
|
||||||
describe("foregroundColor", () => {
|
describe("foregroundColor", () => {
|
||||||
["BNB_ICON_FOREGROUND_COLOR", "BONOB_ICON_FOREGROUND_COLOR"].forEach(
|
describe.each([
|
||||||
(k) => {
|
"BNB_ICON_FOREGROUND_COLOR",
|
||||||
|
"BONOB_ICON_FOREGROUND_COLOR",
|
||||||
|
])("%s", (k) => {
|
||||||
describe(`when ${k} is not specified`, () => {
|
describe(`when ${k} is not specified`, () => {
|
||||||
it(`should default to undefined`, () => {
|
it(`should default to undefined`, () => {
|
||||||
expect(config().icons.foregroundColor).toEqual(undefined);
|
expect(config().icons.foregroundColor).toEqual(undefined);
|
||||||
@@ -202,13 +197,14 @@ describe("config", () => {
|
|||||||
);
|
);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
});
|
||||||
);
|
|
||||||
});
|
});
|
||||||
|
|
||||||
describe("backgroundColor", () => {
|
describe("backgroundColor", () => {
|
||||||
["BNB_ICON_BACKGROUND_COLOR", "BONOB_ICON_BACKGROUND_COLOR"].forEach(
|
describe.each([
|
||||||
(k) => {
|
"BNB_ICON_BACKGROUND_COLOR",
|
||||||
|
"BONOB_ICON_BACKGROUND_COLOR",
|
||||||
|
])("%s", (k) => {
|
||||||
describe(`when ${k} is not specified`, () => {
|
describe(`when ${k} is not specified`, () => {
|
||||||
it(`should default to undefined`, () => {
|
it(`should default to undefined`, () => {
|
||||||
expect(config().icons.backgroundColor).toEqual(undefined);
|
expect(config().icons.backgroundColor).toEqual(undefined);
|
||||||
@@ -244,8 +240,7 @@ describe("config", () => {
|
|||||||
);
|
);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
});
|
||||||
);
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -254,9 +249,12 @@ describe("config", () => {
|
|||||||
expect(config().secret).toEqual("bonob");
|
expect(config().secret).toEqual("bonob");
|
||||||
});
|
});
|
||||||
|
|
||||||
["BNB_SECRET", "BONOB_SECRET"].forEach((key) => {
|
describe.each([
|
||||||
it(`should be overridable using ${key}`, () => {
|
"BNB_SECRET",
|
||||||
process.env[key] = "new secret";
|
"BONOB_SECRET"
|
||||||
|
])("%s", (k) => {
|
||||||
|
it(`should be overridable using ${k}`, () => {
|
||||||
|
process.env[k] = "new secret";
|
||||||
expect(config().secret).toEqual("new secret");
|
expect(config().secret).toEqual("new secret");
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -271,7 +269,7 @@ describe("config", () => {
|
|||||||
process.env["BNB_AUTH_TIMEOUT"] = "33s";
|
process.env["BNB_AUTH_TIMEOUT"] = "33s";
|
||||||
expect(config().authTimeout).toEqual("33s");
|
expect(config().authTimeout).toEqual("33s");
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe("sonos", () => {
|
describe("sonos", () => {
|
||||||
describe("serviceName", () => {
|
describe("serviceName", () => {
|
||||||
@@ -279,79 +277,103 @@ describe("config", () => {
|
|||||||
expect(config().sonos.serviceName).toEqual("bonob");
|
expect(config().sonos.serviceName).toEqual("bonob");
|
||||||
});
|
});
|
||||||
|
|
||||||
["BNB_SONOS_SERVICE_NAME", "BONOB_SONOS_SERVICE_NAME"].forEach((k) => {
|
describe.each([
|
||||||
|
"BNB_SONOS_SERVICE_NAME",
|
||||||
|
"BONOB_SONOS_SERVICE_NAME"
|
||||||
|
])(
|
||||||
|
"%s",
|
||||||
|
(k) => {
|
||||||
it("should be overridable", () => {
|
it("should be overridable", () => {
|
||||||
process.env[k] = "foobar1000";
|
process.env[k] = "foobar1000";
|
||||||
expect(config().sonos.serviceName).toEqual("foobar1000");
|
expect(config().sonos.serviceName).toEqual("foobar1000");
|
||||||
});
|
});
|
||||||
});
|
}
|
||||||
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
["BNB_SONOS_DEVICE_DISCOVERY", "BONOB_SONOS_DEVICE_DISCOVERY"].forEach(
|
describe.each([
|
||||||
(k) => {
|
"BNB_SONOS_DEVICE_DISCOVERY",
|
||||||
|
"BONOB_SONOS_DEVICE_DISCOVERY",
|
||||||
|
])("%s", (k) => {
|
||||||
describeBooleanConfigValue(
|
describeBooleanConfigValue(
|
||||||
"deviceDiscovery",
|
"deviceDiscovery",
|
||||||
k,
|
k,
|
||||||
true,
|
true,
|
||||||
(config) => config.sonos.discovery.enabled
|
(config) => config.sonos.discovery.enabled
|
||||||
);
|
);
|
||||||
}
|
});
|
||||||
);
|
|
||||||
|
|
||||||
describe("seedHost", () => {
|
describe("seedHost", () => {
|
||||||
it("should default to undefined", () => {
|
it("should default to undefined", () => {
|
||||||
expect(config().sonos.discovery.seedHost).toBeUndefined();
|
expect(config().sonos.discovery.seedHost).toBeUndefined();
|
||||||
});
|
});
|
||||||
|
|
||||||
["BNB_SONOS_SEED_HOST", "BONOB_SONOS_SEED_HOST"].forEach((k) => {
|
describe.each([
|
||||||
|
"BNB_SONOS_SEED_HOST",
|
||||||
|
"BONOB_SONOS_SEED_HOST"
|
||||||
|
])(
|
||||||
|
"%s",
|
||||||
|
(k) => {
|
||||||
it("should be overridable", () => {
|
it("should be overridable", () => {
|
||||||
process.env[k] = "123.456.789.0";
|
process.env[k] = "123.456.789.0";
|
||||||
expect(config().sonos.discovery.seedHost).toEqual("123.456.789.0");
|
expect(config().sonos.discovery.seedHost).toEqual("123.456.789.0");
|
||||||
});
|
});
|
||||||
});
|
}
|
||||||
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
["BNB_SONOS_AUTO_REGISTER", "BONOB_SONOS_AUTO_REGISTER"].forEach((k) => {
|
describe.each([
|
||||||
|
"BNB_SONOS_AUTO_REGISTER",
|
||||||
|
"BONOB_SONOS_AUTO_REGISTER"
|
||||||
|
])(
|
||||||
|
"%s",
|
||||||
|
(k) => {
|
||||||
describeBooleanConfigValue(
|
describeBooleanConfigValue(
|
||||||
"autoRegister",
|
"autoRegister",
|
||||||
k,
|
k,
|
||||||
false,
|
false,
|
||||||
(config) => config.sonos.autoRegister
|
(config) => config.sonos.autoRegister
|
||||||
);
|
);
|
||||||
});
|
}
|
||||||
|
);
|
||||||
|
|
||||||
describe("sid", () => {
|
describe("sid", () => {
|
||||||
it("should default to 246", () => {
|
it("should default to 246", () => {
|
||||||
expect(config().sonos.sid).toEqual(246);
|
expect(config().sonos.sid).toEqual(246);
|
||||||
});
|
});
|
||||||
|
|
||||||
["BNB_SONOS_SERVICE_ID", "BONOB_SONOS_SERVICE_ID"].forEach((k) => {
|
describe.each([
|
||||||
|
"BNB_SONOS_SERVICE_ID",
|
||||||
|
"BONOB_SONOS_SERVICE_ID"
|
||||||
|
])(
|
||||||
|
"%s",
|
||||||
|
(k) => {
|
||||||
it("should be overridable", () => {
|
it("should be overridable", () => {
|
||||||
process.env[k] = "786";
|
process.env[k] = "786";
|
||||||
expect(config().sonos.sid).toEqual(786);
|
expect(config().sonos.sid).toEqual(786);
|
||||||
});
|
});
|
||||||
});
|
}
|
||||||
|
);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe("subsonic", () => {
|
describe("subsonic", () => {
|
||||||
describe("url", () => {
|
describe("url", () => {
|
||||||
["BNB_SUBSONIC_URL", "BONOB_SUBSONIC_URL", "BONOB_NAVIDROME_URL"].forEach(
|
describe.each([
|
||||||
(k) => {
|
"BNB_SUBSONIC_URL",
|
||||||
|
"BONOB_SUBSONIC_URL",
|
||||||
|
"BONOB_NAVIDROME_URL",
|
||||||
|
])("%s", (k) => {
|
||||||
describe(`when ${k} is not specified`, () => {
|
describe(`when ${k} is not specified`, () => {
|
||||||
it(`should default to http://${hostname()}:4533`, () => {
|
it(`should default to http://${hostname()}:4533`, () => {
|
||||||
expect(config().subsonic.url).toEqual(
|
expect(config().subsonic.url).toEqual(`http://${hostname()}:4533`);
|
||||||
`http://${hostname()}:4533`
|
|
||||||
);
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe(`when ${k} is ''`, () => {
|
describe(`when ${k} is ''`, () => {
|
||||||
it(`should default to http://${hostname()}:4533`, () => {
|
it(`should default to http://${hostname()}:4533`, () => {
|
||||||
process.env[k] = "";
|
process.env[k] = "";
|
||||||
expect(config().subsonic.url).toEqual(
|
expect(config().subsonic.url).toEqual(`http://${hostname()}:4533`);
|
||||||
`http://${hostname()}:4533`
|
|
||||||
);
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -362,8 +384,7 @@ describe("config", () => {
|
|||||||
expect(config().subsonic.url).toEqual(url);
|
expect(config().subsonic.url).toEqual(url);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
});
|
||||||
);
|
|
||||||
});
|
});
|
||||||
|
|
||||||
describe("customClientsFor", () => {
|
describe("customClientsFor", () => {
|
||||||
@@ -371,11 +392,11 @@ describe("config", () => {
|
|||||||
expect(config().subsonic.customClientsFor).toBeUndefined();
|
expect(config().subsonic.customClientsFor).toBeUndefined();
|
||||||
});
|
});
|
||||||
|
|
||||||
[
|
describe.each([
|
||||||
"BNB_SUBSONIC_CUSTOM_CLIENTS",
|
"BNB_SUBSONIC_CUSTOM_CLIENTS",
|
||||||
"BONOB_SUBSONIC_CUSTOM_CLIENTS",
|
"BONOB_SUBSONIC_CUSTOM_CLIENTS",
|
||||||
"BONOB_NAVIDROME_CUSTOM_CLIENTS",
|
"BONOB_NAVIDROME_CUSTOM_CLIENTS",
|
||||||
].forEach((k) => {
|
])("%s", (k) => {
|
||||||
it(`should be overridable for ${k}`, () => {
|
it(`should be overridable for ${k}`, () => {
|
||||||
process.env[k] = "whoop/whoop";
|
process.env[k] = "whoop/whoop";
|
||||||
expect(config().subsonic.customClientsFor).toEqual("whoop/whoop");
|
expect(config().subsonic.customClientsFor).toEqual("whoop/whoop");
|
||||||
@@ -395,7 +416,10 @@ describe("config", () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
["BNB_SCROBBLE_TRACKS", "BONOB_SCROBBLE_TRACKS"].forEach((k) => {
|
describe.each([
|
||||||
|
"BNB_SCROBBLE_TRACKS",
|
||||||
|
"BONOB_SCROBBLE_TRACKS"
|
||||||
|
])("%s", (k) => {
|
||||||
describeBooleanConfigValue(
|
describeBooleanConfigValue(
|
||||||
"scrobbleTracks",
|
"scrobbleTracks",
|
||||||
k,
|
k,
|
||||||
@@ -404,12 +428,18 @@ describe("config", () => {
|
|||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
["BNB_REPORT_NOW_PLAYING", "BONOB_REPORT_NOW_PLAYING"].forEach((k) => {
|
describe.each([
|
||||||
|
"BNB_REPORT_NOW_PLAYING",
|
||||||
|
"BONOB_REPORT_NOW_PLAYING"
|
||||||
|
])(
|
||||||
|
"%s",
|
||||||
|
(k) => {
|
||||||
describeBooleanConfigValue(
|
describeBooleanConfigValue(
|
||||||
"reportNowPlaying",
|
"reportNowPlaying",
|
||||||
k,
|
k,
|
||||||
true,
|
true,
|
||||||
(config) => config.reportNowPlaying
|
(config) => config.reportNowPlaying
|
||||||
);
|
);
|
||||||
});
|
}
|
||||||
|
);
|
||||||
});
|
});
|
||||||
|
|||||||
277
tests/http.test.ts
Normal file
277
tests/http.test.ts
Normal file
@@ -0,0 +1,277 @@
|
|||||||
|
import { http, http2From, } from "../src/http";
|
||||||
|
|
||||||
|
describe("http", () => {
|
||||||
|
const mockAxios = jest.fn();
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
jest.clearAllMocks();
|
||||||
|
jest.resetAllMocks();
|
||||||
|
});
|
||||||
|
|
||||||
|
describe.each([
|
||||||
|
["baseURL"],
|
||||||
|
["url"],
|
||||||
|
["method"],
|
||||||
|
])('%s', (field) => {
|
||||||
|
const getValue = (value: string) => {
|
||||||
|
const thing = {} as any;
|
||||||
|
thing[field] = value;
|
||||||
|
return thing;
|
||||||
|
};
|
||||||
|
|
||||||
|
const base = http(mockAxios, getValue('base'));
|
||||||
|
|
||||||
|
describe("using default", () => {
|
||||||
|
it("should use the default", () => {
|
||||||
|
base({})
|
||||||
|
expect(mockAxios).toHaveBeenCalledWith(getValue('base'));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("overriding", () => {
|
||||||
|
it("should use the override", () => {
|
||||||
|
base(getValue('override'))
|
||||||
|
expect(mockAxios).toHaveBeenCalledWith(getValue('override'));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("wrapping", () => {
|
||||||
|
const firstLayer = http(base, getValue('level1'));
|
||||||
|
const secondLayer = http(firstLayer, getValue('level2'));
|
||||||
|
|
||||||
|
describe("when the outter call provides a value", () => {
|
||||||
|
it("should apply it", () => {
|
||||||
|
secondLayer(getValue('outter'))
|
||||||
|
expect(mockAxios).toHaveBeenCalledWith(getValue('outter'));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("when the outter call does not provide a value", () => {
|
||||||
|
it("should use the second layer", () => {
|
||||||
|
secondLayer({ })
|
||||||
|
expect(mockAxios).toHaveBeenCalledWith(getValue('level2'));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("requestType", () => {
|
||||||
|
const base = http(mockAxios, { responseType: 'stream' });
|
||||||
|
|
||||||
|
describe("using default", () => {
|
||||||
|
it("should use the default", () => {
|
||||||
|
base({})
|
||||||
|
expect(mockAxios).toHaveBeenCalledWith({ responseType: 'stream' });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("overriding", () => {
|
||||||
|
it("should use the override", () => {
|
||||||
|
base({ responseType: 'arraybuffer' })
|
||||||
|
expect(mockAxios).toHaveBeenCalledWith({ responseType: 'arraybuffer' });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("wrapping", () => {
|
||||||
|
const firstLayer = http(base, { responseType: 'arraybuffer' });
|
||||||
|
const secondLayer = http(firstLayer, { responseType: 'blob' });
|
||||||
|
|
||||||
|
describe("when the outter call provides a value", () => {
|
||||||
|
it("should apply it", () => {
|
||||||
|
secondLayer({ responseType: 'text' })
|
||||||
|
expect(mockAxios).toHaveBeenCalledWith({ responseType: 'text' });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("when the outter call does not provide a value", () => {
|
||||||
|
it("should use the second layer", () => {
|
||||||
|
secondLayer({ })
|
||||||
|
expect(mockAxios).toHaveBeenCalledWith({ responseType: 'blob' });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe.each([
|
||||||
|
["params"],
|
||||||
|
["headers"],
|
||||||
|
])('%s', (field) => {
|
||||||
|
const getValues = (values: any) => {
|
||||||
|
const thing = {} as any;
|
||||||
|
thing[field] = values;
|
||||||
|
return thing;
|
||||||
|
}
|
||||||
|
const base = http(mockAxios, getValues({ a: 1, b: 2, c: 3, d: 4 }));
|
||||||
|
|
||||||
|
describe("using default", () => {
|
||||||
|
it("should use the default", () => {
|
||||||
|
base({});
|
||||||
|
expect(mockAxios).toHaveBeenCalledWith(getValues({ a: 1, b: 2, c: 3, d: 4 }));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("overriding", () => {
|
||||||
|
it("should use the override", () => {
|
||||||
|
base(getValues({ b: 22, e: 5 }));
|
||||||
|
expect(mockAxios).toHaveBeenCalledWith(getValues({ a: 1, b: 22, c: 3, d: 4, e: 5 }));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("wrapping", () => {
|
||||||
|
const firstLayer = http(base, getValues({ b: 22 }));
|
||||||
|
const secondLayer = http(firstLayer, getValues({ c: 33 }));
|
||||||
|
|
||||||
|
describe("when the outter call provides a value", () => {
|
||||||
|
it("should apply it", () => {
|
||||||
|
secondLayer(getValues({ a: 11, e: 5 }));
|
||||||
|
expect(mockAxios).toHaveBeenCalledWith(getValues({ a: 11, b: 22, c: 33, d: 4, e: 5 }));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("when the outter call does not provide a value", () => {
|
||||||
|
it("should use the second layer", () => {
|
||||||
|
secondLayer({ });
|
||||||
|
expect(mockAxios).toHaveBeenCalledWith(getValues({ a: 1, b: 22, c: 33, d: 4 }));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
})
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("http2", () => {
|
||||||
|
const mockAxios = jest.fn();
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
jest.clearAllMocks();
|
||||||
|
jest.resetAllMocks();
|
||||||
|
});
|
||||||
|
|
||||||
|
describe.each([
|
||||||
|
["baseURL"],
|
||||||
|
["url"],
|
||||||
|
["method"],
|
||||||
|
])('%s', (field) => {
|
||||||
|
const fieldWithValue = (value: string) => {
|
||||||
|
const thing = {} as any;
|
||||||
|
thing[field] = value;
|
||||||
|
return thing;
|
||||||
|
};
|
||||||
|
|
||||||
|
const base = http2From(mockAxios).with(fieldWithValue('default'));
|
||||||
|
|
||||||
|
describe("using default", () => {
|
||||||
|
it("should use the default", () => {
|
||||||
|
base({})
|
||||||
|
expect(mockAxios).toHaveBeenCalledWith(fieldWithValue('default'));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("overriding", () => {
|
||||||
|
it("should use the override", () => {
|
||||||
|
base(fieldWithValue('override'))
|
||||||
|
expect(mockAxios).toHaveBeenCalledWith(fieldWithValue('override'));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("wrapping", () => {
|
||||||
|
const firstLayer = http2From(base).with(fieldWithValue('level1'));
|
||||||
|
const secondLayer = firstLayer.with(fieldWithValue('level2'));
|
||||||
|
|
||||||
|
describe("when the outter call provides a value", () => {
|
||||||
|
it("should apply it", () => {
|
||||||
|
secondLayer(fieldWithValue('outter'))
|
||||||
|
expect(mockAxios).toHaveBeenCalledWith(fieldWithValue('outter'));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("when the outter call does not provide a value", () => {
|
||||||
|
it("should use the second layer", () => {
|
||||||
|
secondLayer({ })
|
||||||
|
expect(mockAxios).toHaveBeenCalledWith(fieldWithValue('level2'));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("requestType", () => {
|
||||||
|
const base = http2From(mockAxios).with({ responseType: 'stream' });
|
||||||
|
|
||||||
|
describe("using default", () => {
|
||||||
|
it("should use the default", () => {
|
||||||
|
base({})
|
||||||
|
expect(mockAxios).toHaveBeenCalledWith({ responseType: 'stream' });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("overriding", () => {
|
||||||
|
it("should use the override", () => {
|
||||||
|
base({ responseType: 'arraybuffer' })
|
||||||
|
expect(mockAxios).toHaveBeenCalledWith({ responseType: 'arraybuffer' });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("wrapping", () => {
|
||||||
|
const firstLayer = base.with({ responseType: 'arraybuffer' });
|
||||||
|
const secondLayer = firstLayer.with({ responseType: 'blob' });
|
||||||
|
|
||||||
|
describe("when the outter call provides a value", () => {
|
||||||
|
it("should apply it", () => {
|
||||||
|
secondLayer({ responseType: 'text' })
|
||||||
|
expect(mockAxios).toHaveBeenCalledWith({ responseType: 'text' });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("when the outter call does not provide a value", () => {
|
||||||
|
it("should use the second layer", () => {
|
||||||
|
secondLayer({ })
|
||||||
|
expect(mockAxios).toHaveBeenCalledWith({ responseType: 'blob' });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe.each([
|
||||||
|
["params"],
|
||||||
|
["headers"],
|
||||||
|
])('%s', (field) => {
|
||||||
|
const fieldWithValues = (values: any) => {
|
||||||
|
const thing = {} as any;
|
||||||
|
thing[field] = values;
|
||||||
|
return thing;
|
||||||
|
}
|
||||||
|
const base = http2From(mockAxios).with(fieldWithValues({ a: 1, b: 2, c: 3, d: 4 }));
|
||||||
|
|
||||||
|
describe("using default", () => {
|
||||||
|
it("should use the default", () => {
|
||||||
|
base({});
|
||||||
|
expect(mockAxios).toHaveBeenCalledWith(fieldWithValues({ a: 1, b: 2, c: 3, d: 4 }));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("overriding", () => {
|
||||||
|
it("should use the override", () => {
|
||||||
|
base(fieldWithValues({ b: 22, e: 5 }));
|
||||||
|
expect(mockAxios).toHaveBeenCalledWith(fieldWithValues({ a: 1, b: 22, c: 3, d: 4, e: 5 }));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("wrapping", () => {
|
||||||
|
const firstLayer = base.with(fieldWithValues({ b: 22 }));
|
||||||
|
const secondLayer = firstLayer.with(fieldWithValues({ c: 33 }));
|
||||||
|
|
||||||
|
describe("when the outter call provides a value", () => {
|
||||||
|
it("should apply it", () => {
|
||||||
|
secondLayer(fieldWithValues({ a: 11, e: 5 }));
|
||||||
|
expect(mockAxios).toHaveBeenCalledWith(fieldWithValues({ a: 11, b: 22, c: 33, d: 4, e: 5 }));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("when the outter call does not provide a value", () => {
|
||||||
|
it("should use the second layer", () => {
|
||||||
|
secondLayer({ });
|
||||||
|
expect(mockAxios).toHaveBeenCalledWith(fieldWithValues({ a: 1, b: 22, c: 33, d: 4 }));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
})
|
||||||
|
});
|
||||||
78
tests/images.test.ts
Normal file
78
tests/images.test.ts
Normal file
@@ -0,0 +1,78 @@
|
|||||||
|
|
||||||
|
import tmp from "tmp";
|
||||||
|
import fse from "fs-extra";
|
||||||
|
import path from "path";
|
||||||
|
import { Md5 } from "ts-md5";
|
||||||
|
|
||||||
|
import sharp from "sharp";
|
||||||
|
jest.mock("sharp");
|
||||||
|
|
||||||
|
import { cachingImageFetcher } from "../src/images";
|
||||||
|
|
||||||
|
describe("cachingImageFetcher", () => {
|
||||||
|
const delegate = jest.fn();
|
||||||
|
const url = "http://test.example.com/someimage.jpg";
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
jest.clearAllMocks();
|
||||||
|
jest.resetAllMocks();
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("when there is no image in the cache", () => {
|
||||||
|
it("should fetch the image from the source and then cache and return it", async () => {
|
||||||
|
const dir = tmp.dirSync();
|
||||||
|
const cacheFile = path.join(dir.name, `${Md5.hashStr(url)}.png`);
|
||||||
|
const jpgImage = Buffer.from("jpg-image", "utf-8");
|
||||||
|
const pngImage = Buffer.from("png-image", "utf-8");
|
||||||
|
|
||||||
|
delegate.mockResolvedValue({ contentType: "image/jpeg", data: jpgImage });
|
||||||
|
const png = jest.fn();
|
||||||
|
(sharp as unknown as jest.Mock).mockReturnValue({ png });
|
||||||
|
png.mockReturnValue({
|
||||||
|
toBuffer: () => Promise.resolve(pngImage),
|
||||||
|
});
|
||||||
|
|
||||||
|
const result = await cachingImageFetcher(dir.name, delegate)(url);
|
||||||
|
|
||||||
|
expect(result!.contentType).toEqual("image/png");
|
||||||
|
expect(result!.data).toEqual(pngImage);
|
||||||
|
|
||||||
|
expect(delegate).toHaveBeenCalledWith(url);
|
||||||
|
expect(fse.existsSync(cacheFile)).toEqual(true);
|
||||||
|
expect(fse.readFileSync(cacheFile)).toEqual(pngImage);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("when the image is already in the cache", () => {
|
||||||
|
it("should fetch the image from the cache and return it", async () => {
|
||||||
|
const dir = tmp.dirSync();
|
||||||
|
const cacheFile = path.join(dir.name, `${Md5.hashStr(url)}.png`);
|
||||||
|
const data = Buffer.from("foobar2", "utf-8");
|
||||||
|
|
||||||
|
fse.writeFileSync(cacheFile, data);
|
||||||
|
|
||||||
|
const result = await cachingImageFetcher(dir.name, delegate)(url);
|
||||||
|
|
||||||
|
expect(result!.contentType).toEqual("image/png");
|
||||||
|
expect(result!.data).toEqual(data);
|
||||||
|
|
||||||
|
expect(delegate).not.toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("when the delegate returns undefined", () => {
|
||||||
|
it("should return undefined", async () => {
|
||||||
|
const dir = tmp.dirSync();
|
||||||
|
const cacheFile = path.join(dir.name, `${Md5.hashStr(url)}.png`);
|
||||||
|
|
||||||
|
delegate.mockResolvedValue(undefined);
|
||||||
|
|
||||||
|
const result = await cachingImageFetcher(dir.name, delegate)(url);
|
||||||
|
|
||||||
|
expect(result).toBeUndefined();
|
||||||
|
|
||||||
|
expect(delegate).toHaveBeenCalledWith(url);
|
||||||
|
expect(fse.existsSync(cacheFile)).toEqual(false);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -6,6 +6,7 @@ import {
|
|||||||
MusicLibrary,
|
MusicLibrary,
|
||||||
artistToArtistSummary,
|
artistToArtistSummary,
|
||||||
albumToAlbumSummary,
|
albumToAlbumSummary,
|
||||||
|
Artist,
|
||||||
} from "../src/music_service";
|
} from "../src/music_service";
|
||||||
import { v4 as uuid } from "uuid";
|
import { v4 as uuid } from "uuid";
|
||||||
import {
|
import {
|
||||||
@@ -78,6 +79,11 @@ describe("InMemoryMusicService", () => {
|
|||||||
musicLibrary = (await service.login(token.serviceToken)) as MusicLibrary;
|
musicLibrary = (await service.login(token.serviceToken)) as MusicLibrary;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const artistToArtistSummaryWithSortName = (artist: Artist) => ({
|
||||||
|
...artistToArtistSummary(artist),
|
||||||
|
sortName: artist.name
|
||||||
|
})
|
||||||
|
|
||||||
describe("artists", () => {
|
describe("artists", () => {
|
||||||
const artist1 = anArtist();
|
const artist1 = anArtist();
|
||||||
const artist2 = anArtist();
|
const artist2 = anArtist();
|
||||||
@@ -95,11 +101,11 @@ describe("InMemoryMusicService", () => {
|
|||||||
await musicLibrary.artists({ _index: 0, _count: 100 })
|
await musicLibrary.artists({ _index: 0, _count: 100 })
|
||||||
).toEqual({
|
).toEqual({
|
||||||
results: [
|
results: [
|
||||||
artistToArtistSummary(artist1),
|
artistToArtistSummaryWithSortName(artist1),
|
||||||
artistToArtistSummary(artist2),
|
artistToArtistSummaryWithSortName(artist2),
|
||||||
artistToArtistSummary(artist3),
|
artistToArtistSummaryWithSortName(artist3),
|
||||||
artistToArtistSummary(artist4),
|
artistToArtistSummaryWithSortName(artist4),
|
||||||
artistToArtistSummary(artist5),
|
artistToArtistSummaryWithSortName(artist5),
|
||||||
],
|
],
|
||||||
total: 5,
|
total: 5,
|
||||||
});
|
});
|
||||||
@@ -110,8 +116,8 @@ describe("InMemoryMusicService", () => {
|
|||||||
it("should provide an array of artists", async () => {
|
it("should provide an array of artists", async () => {
|
||||||
expect(await musicLibrary.artists({ _index: 2, _count: 2 })).toEqual({
|
expect(await musicLibrary.artists({ _index: 2, _count: 2 })).toEqual({
|
||||||
results: [
|
results: [
|
||||||
artistToArtistSummary(artist3),
|
artistToArtistSummaryWithSortName(artist3),
|
||||||
artistToArtistSummary(artist4),
|
artistToArtistSummaryWithSortName(artist4),
|
||||||
],
|
],
|
||||||
total: 5,
|
total: 5,
|
||||||
});
|
});
|
||||||
@@ -121,7 +127,7 @@ describe("InMemoryMusicService", () => {
|
|||||||
describe("fetching the last page", () => {
|
describe("fetching the last page", () => {
|
||||||
it("should provide an array of artists", async () => {
|
it("should provide an array of artists", async () => {
|
||||||
expect(await musicLibrary.artists({ _index: 4, _count: 2 })).toEqual({
|
expect(await musicLibrary.artists({ _index: 4, _count: 2 })).toEqual({
|
||||||
results: [artistToArtistSummary(artist5)],
|
results: [artistToArtistSummaryWithSortName(artist5)],
|
||||||
total: 5,
|
total: 5,
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -62,7 +62,7 @@ export class InMemoryMusicService implements MusicService {
|
|||||||
|
|
||||||
return Promise.resolve({
|
return Promise.resolve({
|
||||||
artists: (q: ArtistQuery) =>
|
artists: (q: ArtistQuery) =>
|
||||||
Promise.resolve(this.artists.map(artistToArtistSummary))
|
Promise.resolve(this.artists.map(artistToArtistSummary).map(it => ({ ...it, sortName: it.name })))
|
||||||
.then(slice2(q))
|
.then(slice2(q))
|
||||||
.then(asResult),
|
.then(asResult),
|
||||||
artist: (id: string) =>
|
artist: (id: string) =>
|
||||||
|
|||||||
@@ -1,7 +1,57 @@
|
|||||||
import { v4 as uuid } from "uuid";
|
import { v4 as uuid } from "uuid";
|
||||||
|
|
||||||
import { anArtist } from "./builders";
|
import { anArtist } from "./builders";
|
||||||
import { artistToArtistSummary } from "../src/music_service";
|
import { artistToArtistSummary, slice2 } from "../src/music_service";
|
||||||
|
|
||||||
|
|
||||||
|
describe("slice2", () => {
|
||||||
|
const things = ["a", "b", "c", "d", "e", "f", "g", "h", "i"];
|
||||||
|
|
||||||
|
describe("when slice is a subset of the things", () => {
|
||||||
|
it("should return the page", () => {
|
||||||
|
expect(slice2({ _index: 3, _count: 4 })(things)).toEqual([
|
||||||
|
["d", "e", "f", "g"],
|
||||||
|
things.length
|
||||||
|
])
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("when slice goes off the end of the things", () => {
|
||||||
|
it("should return the page", () => {
|
||||||
|
expect(slice2({ _index: 5, _count: 100 })(things)).toEqual([
|
||||||
|
["f", "g", "h", "i"],
|
||||||
|
things.length
|
||||||
|
])
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("when no _count is provided", () => {
|
||||||
|
it("should return from the index", () => {
|
||||||
|
expect(slice2({ _index: 5 })(things)).toEqual([
|
||||||
|
["f", "g", "h", "i"],
|
||||||
|
things.length
|
||||||
|
])
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("when no _index is provided", () => {
|
||||||
|
it("should assume from the start", () => {
|
||||||
|
expect(slice2({ _count: 3 })(things)).toEqual([
|
||||||
|
["a", "b", "c"],
|
||||||
|
things.length
|
||||||
|
])
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("when no _index or _count is provided", () => {
|
||||||
|
it("should return all the things", () => {
|
||||||
|
expect(slice2()(things)).toEqual([
|
||||||
|
things,
|
||||||
|
things.length
|
||||||
|
])
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
describe("artistToArtistSummary", () => {
|
describe("artistToArtistSummary", () => {
|
||||||
it("should map fields correctly", () => {
|
it("should map fields correctly", () => {
|
||||||
|
|||||||
@@ -167,7 +167,7 @@ describe("RangeBytesFromFilter", () => {
|
|||||||
|
|
||||||
|
|
||||||
describe("server", () => {
|
describe("server", () => {
|
||||||
jest.setTimeout(Number.parseInt(process.env["JEST_TIMEOUT"] || "2000"));
|
jest.setTimeout(Number.parseInt(process.env["JEST_TIMEOUT"] || "5000"));
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
jest.clearAllMocks();
|
jest.clearAllMocks();
|
||||||
|
|||||||
@@ -26,6 +26,7 @@ import {
|
|||||||
sonosifyMimeType,
|
sonosifyMimeType,
|
||||||
ratingAsInt,
|
ratingAsInt,
|
||||||
ratingFromInt,
|
ratingFromInt,
|
||||||
|
scrollIndicesFrom,
|
||||||
} from "../src/smapi";
|
} from "../src/smapi";
|
||||||
|
|
||||||
import { keys as i8nKeys } from "../src/i8n";
|
import { keys as i8nKeys } from "../src/i8n";
|
||||||
@@ -56,7 +57,7 @@ import dayjs from "dayjs";
|
|||||||
import url, { URLBuilder } from "../src/url_builder";
|
import url, { URLBuilder } from "../src/url_builder";
|
||||||
import { iconForGenre } from "../src/icon";
|
import { iconForGenre } from "../src/icon";
|
||||||
import { formatForURL } from "../src/burn";
|
import { formatForURL } from "../src/burn";
|
||||||
import { range } from "underscore";
|
import _, { range } from "underscore";
|
||||||
import { FixedClock } from "../src/clock";
|
import { FixedClock } from "../src/clock";
|
||||||
import { ExpiredTokenError, InvalidTokenError, SmapiAuthTokens, SmapiToken, ToSmapiFault } from "../src/smapi_auth";
|
import { ExpiredTokenError, InvalidTokenError, SmapiAuthTokens, SmapiToken, ToSmapiFault } from "../src/smapi_auth";
|
||||||
|
|
||||||
@@ -90,6 +91,8 @@ describe("rating to and from ints", () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
describe("service config", () => {
|
describe("service config", () => {
|
||||||
|
jest.setTimeout(Number.parseInt(process.env["JEST_TIMEOUT"] || "5000"));
|
||||||
|
|
||||||
const bonobWithNoContextPath = url("http://localhost:1234");
|
const bonobWithNoContextPath = url("http://localhost:1234");
|
||||||
const bonobWithContextPath = url("http://localhost:5678/some-context-path");
|
const bonobWithContextPath = url("http://localhost:5678/some-context-path");
|
||||||
|
|
||||||
@@ -858,6 +861,54 @@ describe("defaultArtistArtURI", () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe("scrollIndicesFrom", () => {
|
||||||
|
describe("artists", () => {
|
||||||
|
describe("when sortName is the same as name", () => {
|
||||||
|
it("should be scroll indicies", () => {
|
||||||
|
const artistNames = [
|
||||||
|
"10,000 Maniacs",
|
||||||
|
"99 Bacon Sandwiches",
|
||||||
|
"[something with square brackets]",
|
||||||
|
"Aerosmith",
|
||||||
|
"Bob Marley",
|
||||||
|
"beatles", // intentionally lower case
|
||||||
|
"Cans",
|
||||||
|
"egg heads", // intentionally lower case
|
||||||
|
"Moon Cakes",
|
||||||
|
"Moon Boots",
|
||||||
|
"Numpty",
|
||||||
|
"Yellow brick road"
|
||||||
|
]
|
||||||
|
const scrollIndicies = scrollIndicesFrom(artistNames.map(name => ({ name, sortName: name })))
|
||||||
|
|
||||||
|
expect(scrollIndicies).toEqual("A,3,B,4,C,6,D,6,E,7,F,7,G,7,H,7,I,7,J,7,K,7,L,7,M,8,N,10,O,10,P,10,Q,10,R,10,S,10,T,10,U,10,V,10,W,10,X,10,Y,11,Z,11")
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("when sortName is different to the name name", () => {
|
||||||
|
it("should be scroll indicies", () => {
|
||||||
|
const artistSortNames = [
|
||||||
|
"10,000 Maniacs",
|
||||||
|
"99 Bacon Sandwiches",
|
||||||
|
"[something with square brackets]",
|
||||||
|
"Aerosmith",
|
||||||
|
"Bob Marley",
|
||||||
|
"beatles", // intentionally lower case
|
||||||
|
"Cans",
|
||||||
|
"egg heads", // intentionally lower case
|
||||||
|
"Moon Cakes",
|
||||||
|
"Moon Boots",
|
||||||
|
"Numpty",
|
||||||
|
"Yellow brick road"
|
||||||
|
]
|
||||||
|
const scrollIndicies = scrollIndicesFrom(artistSortNames.map(name => ({ name: uuid(), sortName: name })))
|
||||||
|
|
||||||
|
expect(scrollIndicies).toEqual("A,3,B,4,C,6,D,6,E,7,F,7,G,7,H,7,I,7,J,7,K,7,L,7,M,8,N,10,O,10,P,10,Q,10,R,10,S,10,T,10,U,10,V,10,W,10,X,10,Y,11,Z,11")
|
||||||
|
});
|
||||||
|
})
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
describe("wsdl api", () => {
|
describe("wsdl api", () => {
|
||||||
const musicService = {
|
const musicService = {
|
||||||
generateToken: jest.fn(),
|
generateToken: jest.fn(),
|
||||||
@@ -1408,6 +1459,7 @@ describe("wsdl api", () => {
|
|||||||
title: "Artists",
|
title: "Artists",
|
||||||
albumArtURI: iconArtURI(bonobUrl, "artists").href(),
|
albumArtURI: iconArtURI(bonobUrl, "artists").href(),
|
||||||
itemType: "container",
|
itemType: "container",
|
||||||
|
canScroll: true,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: "albums",
|
id: "albums",
|
||||||
@@ -1496,6 +1548,7 @@ describe("wsdl api", () => {
|
|||||||
title: "Artiesten",
|
title: "Artiesten",
|
||||||
albumArtURI: iconArtURI(bonobUrl, "artists").href(),
|
albumArtURI: iconArtURI(bonobUrl, "artists").href(),
|
||||||
itemType: "container",
|
itemType: "container",
|
||||||
|
canScroll: true,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: "albums",
|
id: "albums",
|
||||||
@@ -3109,6 +3162,51 @@ describe("wsdl api", () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe("getScrollIndices", () => {
|
||||||
|
itShouldHandleInvalidCredentials((ws) =>
|
||||||
|
ws.getScrollIndicesAsync({ id: `artists` })
|
||||||
|
);
|
||||||
|
|
||||||
|
describe("for artists", () => {
|
||||||
|
let ws: Client;
|
||||||
|
|
||||||
|
const artist1 = anArtist({ name: "Aerosmith" });
|
||||||
|
const artist2 = anArtist({ name: "Bob Marley" });
|
||||||
|
const artist3 = anArtist({ name: "Beatles" });
|
||||||
|
const artist4 = anArtist({ name: "Cat Empire" });
|
||||||
|
const artist5 = anArtist({ name: "Metallica" });
|
||||||
|
const artist6 = anArtist({ name: "Yellow Brick Road" });
|
||||||
|
|
||||||
|
const artists = [artist1, artist2, artist3, artist4, artist5, artist6];
|
||||||
|
const artistsWithSortName = artists.map(it => ({ ...it, sortName: it.name }));
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
ws = await createClientAsync(`${service.uri}?wsdl`, {
|
||||||
|
endpoint: service.uri,
|
||||||
|
httpClient: supersoap(server),
|
||||||
|
});
|
||||||
|
setupAuthenticatedRequest(ws);
|
||||||
|
musicLibrary.artists.mockResolvedValue({
|
||||||
|
results: artistsWithSortName,
|
||||||
|
total: 6
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should return paging information", async () => {
|
||||||
|
const root = await ws.getScrollIndicesAsync({
|
||||||
|
id: `artists`,
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(root[0]).toEqual({
|
||||||
|
getScrollIndicesResult: scrollIndicesFrom(artistsWithSortName)
|
||||||
|
});
|
||||||
|
expect(musicService.login).toHaveBeenCalledWith(serviceToken);
|
||||||
|
expect(apiTokens.mint).toHaveBeenCalledWith(serviceToken);
|
||||||
|
expect(musicLibrary.artists).toHaveBeenCalledWith({ _index: 0, _count: undefined });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
describe("createContainer", () => {
|
describe("createContainer", () => {
|
||||||
let ws: Client;
|
let ws: Client;
|
||||||
|
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
4713
tests/subsonic/generic.test.ts
Normal file
4713
tests/subsonic/generic.test.ts
Normal file
File diff suppressed because it is too large
Load Diff
78
tests/subsonic/navidrome.test.ts
Normal file
78
tests/subsonic/navidrome.test.ts
Normal file
@@ -0,0 +1,78 @@
|
|||||||
|
import { v4 as uuid } from "uuid";
|
||||||
|
import { DODGY_IMAGE_NAME } from "../../src/subsonic";
|
||||||
|
import { artistImageURN } from "../../src/subsonic/generic";
|
||||||
|
import { artistSummaryFromNDArtist } from "../../src/subsonic/navidrome";
|
||||||
|
|
||||||
|
|
||||||
|
describe("artistSummaryFromNDArtist", () => {
|
||||||
|
describe("when the orderArtistName is undefined", () => {
|
||||||
|
it("should use name", () => {
|
||||||
|
const artist = {
|
||||||
|
id: uuid(),
|
||||||
|
name: `name ${uuid()}`,
|
||||||
|
orderArtistName: undefined,
|
||||||
|
largeImageUrl: 'http://example.com/something.jpg'
|
||||||
|
}
|
||||||
|
expect(artistSummaryFromNDArtist(artist)).toEqual({
|
||||||
|
id: artist.id,
|
||||||
|
name: artist.name,
|
||||||
|
sortName: artist.name,
|
||||||
|
image: artistImageURN({ artistId: artist.id, artistImageURL: artist.largeImageUrl })
|
||||||
|
})
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("when the artist image is valid", () => {
|
||||||
|
it("should create an ArtistSummary with Sortable", () => {
|
||||||
|
const artist = {
|
||||||
|
id: uuid(),
|
||||||
|
name: `name ${uuid()}`,
|
||||||
|
orderArtistName: `orderArtistName ${uuid()}`,
|
||||||
|
largeImageUrl: 'http://example.com/something.jpg'
|
||||||
|
}
|
||||||
|
expect(artistSummaryFromNDArtist(artist)).toEqual({
|
||||||
|
id: artist.id,
|
||||||
|
name: artist.name,
|
||||||
|
sortName: artist.orderArtistName,
|
||||||
|
image: artistImageURN({ artistId: artist.id, artistImageURL: artist.largeImageUrl })
|
||||||
|
})
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("when the artist image is not valid", () => {
|
||||||
|
it("should create an ArtistSummary with Sortable", () => {
|
||||||
|
const artist = {
|
||||||
|
id: uuid(),
|
||||||
|
name: `name ${uuid()}`,
|
||||||
|
orderArtistName: `orderArtistName ${uuid()}`,
|
||||||
|
largeImageUrl: `http://example.com/${DODGY_IMAGE_NAME}`
|
||||||
|
}
|
||||||
|
|
||||||
|
expect(artistSummaryFromNDArtist(artist)).toEqual({
|
||||||
|
id: artist.id,
|
||||||
|
name: artist.name,
|
||||||
|
sortName: artist.orderArtistName,
|
||||||
|
image: artistImageURN({ artistId: artist.id, artistImageURL: artist.largeImageUrl })
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("when the artist image is missing", () => {
|
||||||
|
it("should create an ArtistSummary with Sortable", () => {
|
||||||
|
const artist = {
|
||||||
|
id: uuid(),
|
||||||
|
name: `name ${uuid()}`,
|
||||||
|
orderArtistName: `orderArtistName ${uuid()}`,
|
||||||
|
largeImageUrl: undefined
|
||||||
|
}
|
||||||
|
|
||||||
|
expect(artistSummaryFromNDArtist(artist)).toEqual({
|
||||||
|
id: artist.id,
|
||||||
|
name: artist.name,
|
||||||
|
sortName: artist.orderArtistName,
|
||||||
|
image: artistImageURN({ artistId: artist.id, artistImageURL: artist.largeImageUrl })
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
@@ -1,4 +1,50 @@
|
|||||||
import { takeWithRepeats } from "../src/utils";
|
import { asURLSearchParams, mask, takeWithRepeats } from "../src/utils";
|
||||||
|
|
||||||
|
describe("asURLSearchParams", () => {
|
||||||
|
describe("empty q", () => {
|
||||||
|
it("should return empty params", () => {
|
||||||
|
const q = {};
|
||||||
|
const expected = new URLSearchParams();
|
||||||
|
expect(asURLSearchParams(q)).toEqual(expected);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("singular params", () => {
|
||||||
|
it("should append each", () => {
|
||||||
|
const q = {
|
||||||
|
a: 1,
|
||||||
|
b: "bee",
|
||||||
|
c: false,
|
||||||
|
d: true,
|
||||||
|
};
|
||||||
|
const expected = new URLSearchParams();
|
||||||
|
expected.append("a", "1");
|
||||||
|
expected.append("b", "bee");
|
||||||
|
expected.append("c", "false");
|
||||||
|
expected.append("d", "true");
|
||||||
|
|
||||||
|
expect(asURLSearchParams(q)).toEqual(expected);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("list params", () => {
|
||||||
|
it("should append each", () => {
|
||||||
|
const q = {
|
||||||
|
a: [1, "two", false, true],
|
||||||
|
b: "yippee",
|
||||||
|
};
|
||||||
|
|
||||||
|
const expected = new URLSearchParams();
|
||||||
|
expected.append("a", "1");
|
||||||
|
expected.append("a", "two");
|
||||||
|
expected.append("a", "false");
|
||||||
|
expected.append("a", "true");
|
||||||
|
expected.append("b", "yippee");
|
||||||
|
|
||||||
|
expect(asURLSearchParams(q)).toEqual(expected);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
describe("takeWithRepeat", () => {
|
describe("takeWithRepeat", () => {
|
||||||
describe("when there is nothing in the input", () => {
|
describe("when there is nothing in the input", () => {
|
||||||
@@ -29,7 +75,32 @@ describe("takeWithRepeat", () => {
|
|||||||
describe("when there more than the amount required", () => {
|
describe("when there more than the amount required", () => {
|
||||||
it("should return the first n items", () => {
|
it("should return the first n items", () => {
|
||||||
expect(takeWithRepeats(["a", "b", "c"], 2)).toEqual(["a", "b"]);
|
expect(takeWithRepeats(["a", "b", "c"], 2)).toEqual(["a", "b"]);
|
||||||
expect(takeWithRepeats(["a", undefined, "c"], 2)).toEqual(["a", undefined]);
|
expect(takeWithRepeats(["a", undefined, "c"], 2)).toEqual([
|
||||||
|
"a",
|
||||||
|
undefined,
|
||||||
|
]);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe("mask", () => {
|
||||||
|
it.each([
|
||||||
|
[{}, ["a", "b"], {}],
|
||||||
|
[{ foo: "bar" }, ["a", "b"], { foo: "bar" }],
|
||||||
|
[{ a: 1 }, ["a", "b"], { a: "****" }],
|
||||||
|
[{ a: 1, b: "dog" }, ["a", "b"], { a: "****", b: "****" }],
|
||||||
|
[
|
||||||
|
{ a: 1, b: "dog", foo: "bar" },
|
||||||
|
["a", "b"],
|
||||||
|
{ a: "****", b: "****", foo: "bar" },
|
||||||
|
],
|
||||||
|
])(
|
||||||
|
"masking of %s, keys = %s, should result in %s",
|
||||||
|
(original: any, keys: string[], expected: any) => {
|
||||||
|
const copyOfOrig = JSON.parse(JSON.stringify(original));
|
||||||
|
const masked = mask(original, keys);
|
||||||
|
expect(masked).toEqual(expected);
|
||||||
|
expect(original).toEqual(copyOfOrig);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|||||||
@@ -54,7 +54,7 @@
|
|||||||
]
|
]
|
||||||
/* List of folders to include type definitions from. */,
|
/* List of folders to include type definitions from. */,
|
||||||
// "types": ["src/customTypes/scale-that-svg.d.ts"], /* Type declaration files to be included in compilation. */
|
// "types": ["src/customTypes/scale-that-svg.d.ts"], /* Type declaration files to be included in compilation. */
|
||||||
// "allowSyntheticDefaultImports": true, /* Allow default imports from modules with no default export. This does not affect code emit, just typechecking. */
|
"allowSyntheticDefaultImports": true, /* Allow default imports from modules with no default export. This does not affect code emit, just typechecking. */
|
||||||
"esModuleInterop": true /* Enables emit interoperability between CommonJS and ES Modules via creation of namespace objects for all imports. Implies 'allowSyntheticDefaultImports'. */,
|
"esModuleInterop": true /* Enables emit interoperability between CommonJS and ES Modules via creation of namespace objects for all imports. Implies 'allowSyntheticDefaultImports'. */,
|
||||||
// "preserveSymlinks": true, /* Do not resolve the real path of symlinks. */
|
// "preserveSymlinks": true, /* Do not resolve the real path of symlinks. */
|
||||||
// "allowUmdGlobalAccess": true, /* Allow accessing UMD globals from modules. */
|
// "allowUmdGlobalAccess": true, /* Allow accessing UMD globals from modules. */
|
||||||
|
|||||||
Reference in New Issue
Block a user