2019-03-20 03:45:23 +08:00
|
|
|
import * as parser from "./parser.js";
|
|
|
|
|
|
|
|
let ws;
|
|
|
|
let commandQueue = [];
|
2019-03-20 23:20:17 +08:00
|
|
|
let current;
|
2019-03-20 03:45:23 +08:00
|
|
|
|
|
|
|
function onMessage(e) {
|
2019-03-20 23:20:17 +08:00
|
|
|
if (current) {
|
|
|
|
let lines = JSON.parse(e.data);
|
|
|
|
let last = lines.pop();
|
|
|
|
if (last.startsWith("OK")) {
|
|
|
|
current.resolve(lines);
|
|
|
|
} else {
|
|
|
|
current.reject(last);
|
|
|
|
}
|
|
|
|
current = null;
|
2019-03-20 03:45:23 +08:00
|
|
|
}
|
|
|
|
processQueue();
|
|
|
|
}
|
|
|
|
|
|
|
|
function onError(e) {
|
|
|
|
console.error(e);
|
|
|
|
ws = null; // fixme
|
|
|
|
}
|
|
|
|
|
|
|
|
function onClose(e) {
|
|
|
|
console.warn(e);
|
|
|
|
ws = null; // fixme
|
|
|
|
}
|
|
|
|
|
|
|
|
function processQueue() {
|
2019-03-20 23:20:17 +08:00
|
|
|
if (current || commandQueue.length == 0) { return; }
|
|
|
|
current = commandQueue.shift();
|
|
|
|
ws.send(current.cmd);
|
2019-03-20 03:45:23 +08:00
|
|
|
}
|
|
|
|
|
2019-03-25 22:49:23 +08:00
|
|
|
function serializeFilter(filter) {
|
|
|
|
let tokens = ["("];
|
|
|
|
Object.entries(filter).forEach(([key, value], index) => {
|
|
|
|
index && tokens.push(" AND ");
|
|
|
|
tokens.push(`(${key} == "${value}")`);
|
|
|
|
});
|
|
|
|
tokens.push(")");
|
|
|
|
|
|
|
|
let filterStr = tokens.join("");
|
|
|
|
return `"${escape(filterStr)}"`;
|
|
|
|
}
|
|
|
|
|
|
|
|
function escape(str) {
|
2019-03-20 05:56:39 +08:00
|
|
|
return str.replace(/(['"\\])/g, "\\$1");
|
|
|
|
}
|
2019-03-20 03:45:23 +08:00
|
|
|
|
2019-03-20 05:56:39 +08:00
|
|
|
export async function command(cmd) {
|
2019-03-20 23:20:17 +08:00
|
|
|
if (cmd instanceof Array) { cmd = ["command_list_begin", ...cmd, "command_list_end"].join("\n"); }
|
|
|
|
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
commandQueue.push({cmd, resolve, reject});
|
2019-03-20 05:56:39 +08:00
|
|
|
processQueue();
|
|
|
|
});
|
2019-03-20 03:45:23 +08:00
|
|
|
}
|
|
|
|
|
2019-03-20 23:20:17 +08:00
|
|
|
export async function commandAndStatus(cmd) {
|
|
|
|
let lines = await command([cmd, "status", "currentsong"]);
|
|
|
|
return parser.linesToStruct(lines);
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function status() {
|
2019-03-20 03:45:23 +08:00
|
|
|
let lines = await command(["status", "currentsong"]);
|
2019-03-25 22:49:23 +08:00
|
|
|
let status = parser.linesToStruct(lines);
|
|
|
|
// duration returned 2x => arrayfied
|
|
|
|
if ("duration" in status) { status["duration"] = status["duration"][0]; }
|
|
|
|
return status;
|
2019-03-20 03:45:23 +08:00
|
|
|
}
|
|
|
|
|
2019-03-22 22:35:04 +08:00
|
|
|
export async function listQueue() {
|
|
|
|
let lines = await command("playlistinfo");
|
|
|
|
return parser.songList(lines);
|
|
|
|
}
|
|
|
|
|
2019-03-28 22:23:28 +08:00
|
|
|
export async function listPlaylists() {
|
|
|
|
let lines = await command("listplaylists");
|
|
|
|
let parsed = parser.linesToStruct(lines);
|
|
|
|
|
|
|
|
let list = parsed["playlist"];
|
|
|
|
if (!list) { return []; }
|
|
|
|
return (list instanceof Array ? list : [list]);
|
|
|
|
}
|
|
|
|
|
2019-03-26 17:09:26 +08:00
|
|
|
export async function enqueue(urlOrFilter, sort = null) {
|
|
|
|
if (typeof(urlOrFilter) == "string") {
|
|
|
|
return command(`add "${escape(urlOrFilter)}"`);
|
2019-03-25 22:49:23 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
let tokens = ["findadd"];
|
2019-03-26 17:09:26 +08:00
|
|
|
tokens.push(serializeFilter(urlOrFilter));
|
2019-03-25 22:49:23 +08:00
|
|
|
// sort && tokens.push("sort", sort); FIXME not implemented in MPD
|
|
|
|
return command(tokens.join(" "));
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function listPath(path) {
|
|
|
|
let lines = await command(`lsinfo "${escape(path)}"`);
|
|
|
|
return parser.pathContents(lines);
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function listTags(tag, filter = null) {
|
|
|
|
let tokens = ["list", tag];
|
|
|
|
if (filter) {
|
|
|
|
tokens.push(serializeFilter(filter));
|
|
|
|
|
|
|
|
let fakeGroup = Object.keys(filter)[0]; // FIXME hack for MPD < 0.21.6
|
|
|
|
tokens.push("group", fakeGroup);
|
|
|
|
}
|
|
|
|
let lines = await command(tokens.join(" "));
|
|
|
|
let parsed = parser.linesToStruct(lines);
|
|
|
|
return [].concat(parsed[tag] || []);
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function listSongs(filter) {
|
|
|
|
let tokens = ["find"];
|
|
|
|
tokens.push(serializeFilter(filter));
|
|
|
|
let lines = await command(tokens.join(" "));
|
|
|
|
return parser.songList(lines);
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function albumArt(songUrl) {
|
|
|
|
let data = [];
|
|
|
|
let offset = 0;
|
|
|
|
while (1) {
|
|
|
|
let params = ["albumart", `"${escape(songUrl)}"`, offset];
|
|
|
|
let lines = await command(params.join(" "));
|
|
|
|
data = data.concat(lines[2]);
|
|
|
|
let metadata = parser.linesToStruct(lines.slice(0, 2));
|
|
|
|
if (data.length >= Number(metadata["size"])) { return data; }
|
|
|
|
offset += Number(metadata["binary"]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-28 22:23:28 +08:00
|
|
|
export async function save(name) {
|
|
|
|
return command(`save "${escape(name)}"`);
|
|
|
|
}
|
|
|
|
|
2019-03-20 03:45:23 +08:00
|
|
|
export async function init() {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
try {
|
2019-03-20 23:20:17 +08:00
|
|
|
ws = new WebSocket("ws://localhost:8080");
|
2019-03-20 03:45:23 +08:00
|
|
|
} catch (e) { reject(e); }
|
2019-03-20 23:20:17 +08:00
|
|
|
current = {resolve, reject};
|
2019-03-20 03:45:23 +08:00
|
|
|
|
|
|
|
ws.addEventListener("error", onError);
|
|
|
|
ws.addEventListener("message", onMessage);
|
|
|
|
ws.addEventListener("close", onClose);
|
|
|
|
});
|
|
|
|
}
|