.tidal gist url
This commit is contained in:
@@ -1,6 +1,6 @@
|
||||
import { Button } from "@/components/ui/button";
|
||||
import { SearchCheck, CheckCircle2, XCircle, Loader2 } from "lucide-react";
|
||||
import { TidalIcon, QobuzIcon, AmazonIcon, LrclibIcon, MusicBrainzIcon } from "./PlatformIcons";
|
||||
import { TidalIcon, QobuzIcon, AmazonIcon, MusicBrainzIcon } from "./PlatformIcons";
|
||||
import { useApiStatus } from "@/hooks/useApiStatus";
|
||||
export function ApiStatusTab() {
|
||||
const { sources, statuses, isCheckingAll, checkAll } = useApiStatus();
|
||||
@@ -17,7 +17,7 @@ export function ApiStatusTab() {
|
||||
const status = statuses[source.id] || "idle";
|
||||
return (<div key={source.id} className="flex items-center justify-between p-4 border rounded-lg bg-card text-card-foreground shadow-sm">
|
||||
<div className="flex items-center gap-3">
|
||||
{source.type === "tidal" ? <TidalIcon className="w-5 h-5 shrink-0 text-muted-foreground"/> : source.type === "amazon" ? <AmazonIcon className="w-5 h-5 shrink-0 text-muted-foreground"/> : source.type === "lrclib" ? <LrclibIcon className="w-5 h-5 shrink-0 text-muted-foreground"/> : source.type === "musicbrainz" ? <MusicBrainzIcon className="w-5 h-5 shrink-0 text-muted-foreground"/> : <QobuzIcon className="w-5 h-5 shrink-0 text-muted-foreground"/>}
|
||||
{source.type === "tidal" ? <TidalIcon className="w-5 h-5 shrink-0 text-muted-foreground"/> : source.type === "amazon" ? <AmazonIcon className="w-5 h-5 shrink-0 text-muted-foreground"/> : source.type === "musicbrainz" ? <MusicBrainzIcon className="w-5 h-5 shrink-0 text-muted-foreground"/> : <QobuzIcon className="w-5 h-5 shrink-0 text-muted-foreground"/>}
|
||||
<p className="font-medium leading-none">{source.name}</p>
|
||||
</div>
|
||||
|
||||
|
||||
@@ -1,100 +1,72 @@
|
||||
import { CheckAPIStatus, FetchUnifiedAPIStatus } from "../../wailsjs/go/main/App";
|
||||
import { CheckAPIStatus } from "../../wailsjs/go/main/App";
|
||||
import { CHECK_TIMEOUT_MS, withTimeout } from "@/lib/async-timeout";
|
||||
|
||||
export type ApiCheckStatus = "checking" | "online" | "offline" | "idle";
|
||||
|
||||
export interface ApiSource {
|
||||
id: string;
|
||||
type: string;
|
||||
name: string;
|
||||
url: string;
|
||||
}
|
||||
|
||||
export const API_SOURCES: ApiSource[] = [
|
||||
{ id: "tidal1", type: "tidal", name: "Tidal A", url: "https://hifi-one.spotisaver.net" },
|
||||
{ id: "tidal2", type: "tidal", name: "Tidal B", url: "https://hifi-two.spotisaver.net" },
|
||||
{ id: "tidal3", type: "tidal", name: "Tidal C", url: "https://eu-central.monochrome.tf" },
|
||||
{ id: "tidal4", type: "tidal", name: "Tidal D", url: "https://us-west.monochrome.tf" },
|
||||
{ id: "tidal5", type: "tidal", name: "Tidal E", url: "https://api.monochrome.tf" },
|
||||
{ id: "tidal6", type: "tidal", name: "Tidal F", url: "https://monochrome-api.samidy.com" },
|
||||
{ id: "tidal7", type: "tidal", name: "Tidal G", url: "https://tidal.kinoplus.online" },
|
||||
{ id: "qobuz1", type: "qobuz", name: "Qobuz A", url: "https://dab.yeet.su" },
|
||||
{ id: "qobuz2", type: "qobuz", name: "Qobuz B", url: "https://dabmusic.xyz" },
|
||||
{ id: "qobuz3", type: "qbz", name: "Qobuz C", url: "https://qobuz.spotbye.qzz.io" },
|
||||
{ id: "amazon1", type: "amazon", name: "Amazon Music", url: "https://amazon.spotbye.qzz.io" },
|
||||
{ id: "lrclib", type: "lrclib", name: "LRCLIB", url: "https://lrclib.net" },
|
||||
{ id: "tidal", type: "tidal", name: "Tidal", url: "" },
|
||||
{ id: "qobuz", type: "qobuz", name: "Qobuz", url: "" },
|
||||
{ id: "amazon", type: "amazon", name: "Amazon Music", url: "" },
|
||||
{ id: "musicbrainz", type: "musicbrainz", name: "MusicBrainz", url: "https://musicbrainz.org" },
|
||||
];
|
||||
|
||||
type ApiStatusState = {
|
||||
isCheckingAll: boolean;
|
||||
statuses: Record<string, ApiCheckStatus>;
|
||||
};
|
||||
|
||||
let apiStatusState: ApiStatusState = {
|
||||
isCheckingAll: false,
|
||||
statuses: {},
|
||||
};
|
||||
|
||||
let activeCheckAll: Promise<void> | null = null;
|
||||
const listeners = new Set<() => void>();
|
||||
type SpotiFLACUnifiedStatusResponse = {
|
||||
tidal?: string;
|
||||
qobuz_a?: string;
|
||||
qobuz_b?: string;
|
||||
qobuz_c?: string;
|
||||
amazon?: string;
|
||||
lrclib?: string;
|
||||
};
|
||||
|
||||
function emitApiStatusChange() {
|
||||
for (const listener of listeners) {
|
||||
listener();
|
||||
}
|
||||
}
|
||||
|
||||
function setApiStatusState(updater: (current: ApiStatusState) => ApiStatusState) {
|
||||
apiStatusState = updater(apiStatusState);
|
||||
emitApiStatusChange();
|
||||
}
|
||||
function statusFromUnifiedValue(value: string | undefined): ApiCheckStatus {
|
||||
return value === "up" ? "online" : "offline";
|
||||
}
|
||||
async function fetchUnifiedStatuses(forceRefresh: boolean): Promise<Pick<ApiStatusState, "statuses">> {
|
||||
const response = await FetchUnifiedAPIStatus(forceRefresh);
|
||||
const payload = JSON.parse(response) as SpotiFLACUnifiedStatusResponse;
|
||||
const tidalStatus = statusFromUnifiedValue(payload.tidal);
|
||||
return {
|
||||
statuses: {
|
||||
tidal1: tidalStatus,
|
||||
tidal2: tidalStatus,
|
||||
tidal3: tidalStatus,
|
||||
tidal4: tidalStatus,
|
||||
tidal5: tidalStatus,
|
||||
tidal6: tidalStatus,
|
||||
tidal7: tidalStatus,
|
||||
qobuz1: statusFromUnifiedValue(payload.qobuz_a),
|
||||
qobuz2: statusFromUnifiedValue(payload.qobuz_b),
|
||||
qobuz3: statusFromUnifiedValue(payload.qobuz_c),
|
||||
amazon1: statusFromUnifiedValue(payload.amazon),
|
||||
lrclib: statusFromUnifiedValue(payload.lrclib),
|
||||
},
|
||||
};
|
||||
}
|
||||
async function checkMusicBrainzStatus(): Promise<ApiCheckStatus> {
|
||||
|
||||
async function checkSourceStatus(source: ApiSource): Promise<ApiCheckStatus> {
|
||||
try {
|
||||
const isOnline = await withTimeout(CheckAPIStatus("musicbrainz", "https://musicbrainz.org"), CHECK_TIMEOUT_MS, "API status check timed out after 10 seconds for MusicBrainz");
|
||||
const isOnline = await withTimeout(CheckAPIStatus(source.type, source.url), CHECK_TIMEOUT_MS, `API status check timed out after 10 seconds for ${source.name}`);
|
||||
return isOnline ? "online" : "offline";
|
||||
}
|
||||
catch {
|
||||
return "offline";
|
||||
}
|
||||
}
|
||||
|
||||
export function getApiStatusState(): ApiStatusState {
|
||||
return apiStatusState;
|
||||
}
|
||||
|
||||
export function subscribeApiStatus(listener: () => void): () => void {
|
||||
listeners.add(listener);
|
||||
return () => {
|
||||
listeners.delete(listener);
|
||||
};
|
||||
}
|
||||
export async function checkAllApiStatuses(forceRefresh: boolean = false): Promise<void> {
|
||||
|
||||
export async function checkAllApiStatuses(_forceRefresh: boolean = false): Promise<void> {
|
||||
if (activeCheckAll) {
|
||||
return activeCheckAll;
|
||||
}
|
||||
|
||||
activeCheckAll = (async () => {
|
||||
const checkingStatuses = Object.fromEntries(API_SOURCES.map((source) => [source.id, "checking" as ApiCheckStatus]));
|
||||
setApiStatusState((current) => ({
|
||||
@@ -105,37 +77,20 @@ export async function checkAllApiStatuses(forceRefresh: boolean = false): Promis
|
||||
...checkingStatuses,
|
||||
},
|
||||
}));
|
||||
|
||||
try {
|
||||
const [unifiedResult, musicBrainzStatus] = await Promise.allSettled([
|
||||
withTimeout(fetchUnifiedStatuses(forceRefresh), CHECK_TIMEOUT_MS, "Unified SpotiFLAC status check timed out after 10 seconds"),
|
||||
checkMusicBrainzStatus(),
|
||||
]);
|
||||
setApiStatusState((current) => {
|
||||
const nextStatuses = { ...current.statuses };
|
||||
if (unifiedResult.status === "fulfilled") {
|
||||
Object.assign(nextStatuses, unifiedResult.value.statuses);
|
||||
}
|
||||
else {
|
||||
nextStatuses.tidal1 = "offline";
|
||||
nextStatuses.tidal2 = "offline";
|
||||
nextStatuses.tidal3 = "offline";
|
||||
nextStatuses.tidal4 = "offline";
|
||||
nextStatuses.tidal5 = "offline";
|
||||
nextStatuses.tidal6 = "offline";
|
||||
nextStatuses.tidal7 = "offline";
|
||||
nextStatuses.qobuz1 = "offline";
|
||||
nextStatuses.qobuz2 = "offline";
|
||||
nextStatuses.qobuz3 = "offline";
|
||||
nextStatuses.amazon1 = "offline";
|
||||
nextStatuses.lrclib = "offline";
|
||||
}
|
||||
nextStatuses.musicbrainz =
|
||||
musicBrainzStatus.status === "fulfilled" ? musicBrainzStatus.value : "offline";
|
||||
return {
|
||||
...current,
|
||||
statuses: nextStatuses,
|
||||
};
|
||||
});
|
||||
const results = await Promise.all(API_SOURCES.map(async (source) => ({
|
||||
id: source.id,
|
||||
status: await checkSourceStatus(source),
|
||||
})));
|
||||
|
||||
setApiStatusState((current) => ({
|
||||
...current,
|
||||
statuses: results.reduce<Record<string, ApiCheckStatus>>((acc, result) => {
|
||||
acc[result.id] = result.status;
|
||||
return acc;
|
||||
}, { ...current.statuses }),
|
||||
}));
|
||||
}
|
||||
finally {
|
||||
setApiStatusState((current) => ({
|
||||
@@ -145,5 +100,6 @@ export async function checkAllApiStatuses(forceRefresh: boolean = false): Promis
|
||||
activeCheckAll = null;
|
||||
}
|
||||
})();
|
||||
|
||||
return activeCheckAll;
|
||||
}
|
||||
|
||||
Vendored
-2
@@ -53,8 +53,6 @@ export function DownloadTrack(arg1:main.DownloadRequest):Promise<main.DownloadRe
|
||||
|
||||
export function ExportFailedDownloads():Promise<string>;
|
||||
|
||||
export function FetchUnifiedAPIStatus(arg1:boolean):Promise<string>;
|
||||
|
||||
export function GetBrewPath():Promise<string>;
|
||||
|
||||
export function GetConfigPath():Promise<string>;
|
||||
|
||||
@@ -102,10 +102,6 @@ export function ExportFailedDownloads() {
|
||||
return window['go']['main']['App']['ExportFailedDownloads']();
|
||||
}
|
||||
|
||||
export function FetchUnifiedAPIStatus(arg1) {
|
||||
return window['go']['main']['App']['FetchUnifiedAPIStatus'](arg1);
|
||||
}
|
||||
|
||||
export function GetBrewPath() {
|
||||
return window['go']['main']['App']['GetBrewPath']();
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user