init
This commit is contained in:
commit
a0ae1b422d
9 changed files with 371 additions and 0 deletions
9
deno.json
Normal file
9
deno.json
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
{
|
||||||
|
"tasks": {
|
||||||
|
"dev": "deno run --allow-net --allow-read --watch src/main.ts"
|
||||||
|
},
|
||||||
|
"imports": {
|
||||||
|
"@oak/oak": "jsr:@oak/oak@^17.1.4",
|
||||||
|
"@std/assert": "jsr:@std/assert@1"
|
||||||
|
}
|
||||||
|
}
|
97
deno.lock
generated
Normal file
97
deno.lock
generated
Normal file
|
@ -0,0 +1,97 @@
|
||||||
|
{
|
||||||
|
"version": "4",
|
||||||
|
"specifiers": {
|
||||||
|
"jsr:@oak/commons@1": "1.0.0",
|
||||||
|
"jsr:@oak/oak@^17.1.4": "17.1.4",
|
||||||
|
"jsr:@std/assert@1": "1.0.11",
|
||||||
|
"jsr:@std/bytes@1": "1.0.5",
|
||||||
|
"jsr:@std/crypto@1": "1.0.4",
|
||||||
|
"jsr:@std/encoding@1": "1.0.7",
|
||||||
|
"jsr:@std/encoding@^1.0.7": "1.0.7",
|
||||||
|
"jsr:@std/fmt@0.223": "0.223.0",
|
||||||
|
"jsr:@std/http@1": "1.0.13",
|
||||||
|
"jsr:@std/internal@^1.0.5": "1.0.5",
|
||||||
|
"jsr:@std/media-types@1": "1.1.0",
|
||||||
|
"jsr:@std/path@1": "1.0.8",
|
||||||
|
"npm:fast-xml-parser@*": "4.5.2",
|
||||||
|
"npm:path-to-regexp@^6.3.0": "6.3.0"
|
||||||
|
},
|
||||||
|
"jsr": {
|
||||||
|
"@oak/commons@1.0.0": {
|
||||||
|
"integrity": "49805b55603c3627a9d6235c0655aa2b6222d3036b3a13ff0380c16368f607ac",
|
||||||
|
"dependencies": [
|
||||||
|
"jsr:@std/assert",
|
||||||
|
"jsr:@std/bytes",
|
||||||
|
"jsr:@std/crypto",
|
||||||
|
"jsr:@std/encoding@1",
|
||||||
|
"jsr:@std/http",
|
||||||
|
"jsr:@std/media-types"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"@oak/oak@17.1.4": {
|
||||||
|
"integrity": "60530b582bf276ff741e39cc664026781aa08dd5f2bc5134d756cc427bf2c13e",
|
||||||
|
"dependencies": [
|
||||||
|
"jsr:@oak/commons",
|
||||||
|
"jsr:@std/assert",
|
||||||
|
"jsr:@std/bytes",
|
||||||
|
"jsr:@std/http",
|
||||||
|
"jsr:@std/media-types",
|
||||||
|
"jsr:@std/path",
|
||||||
|
"npm:path-to-regexp"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"@std/assert@1.0.11": {
|
||||||
|
"integrity": "2461ef3c368fe88bc60e186e7744a93112f16fd110022e113a0849e94d1c83c1",
|
||||||
|
"dependencies": [
|
||||||
|
"jsr:@std/internal"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"@std/bytes@1.0.5": {
|
||||||
|
"integrity": "4465dd739d7963d964c809202ebea6d5c6b8e3829ef25c6a224290fbb8a1021e"
|
||||||
|
},
|
||||||
|
"@std/crypto@1.0.4": {
|
||||||
|
"integrity": "cee245c453bd5366207f4d8aa25ea3e9c86cecad2be3fefcaa6cb17203d79340"
|
||||||
|
},
|
||||||
|
"@std/encoding@1.0.7": {
|
||||||
|
"integrity": "f631247c1698fef289f2de9e2a33d571e46133b38d042905e3eac3715030a82d"
|
||||||
|
},
|
||||||
|
"@std/fmt@0.223.0": {
|
||||||
|
"integrity": "6deb37794127dfc7d7bded2586b9fc6f5d50e62a8134846608baf71ffc1a5208"
|
||||||
|
},
|
||||||
|
"@std/http@1.0.13": {
|
||||||
|
"integrity": "d29618b982f7ae44380111f7e5b43da59b15db64101198bb5f77100d44eb1e1e",
|
||||||
|
"dependencies": [
|
||||||
|
"jsr:@std/encoding@^1.0.7"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"@std/internal@1.0.5": {
|
||||||
|
"integrity": "54a546004f769c1ac9e025abd15a76b6671ddc9687e2313b67376125650dc7ba"
|
||||||
|
},
|
||||||
|
"@std/media-types@1.1.0": {
|
||||||
|
"integrity": "c9d093f0c05c3512932b330e3cc1fe1d627b301db33a4c2c2185c02471d6eaa4"
|
||||||
|
},
|
||||||
|
"@std/path@1.0.8": {
|
||||||
|
"integrity": "548fa456bb6a04d3c1a1e7477986b6cffbce95102d0bb447c67c4ee70e0364be"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"npm": {
|
||||||
|
"fast-xml-parser@4.5.2": {
|
||||||
|
"integrity": "sha512-xmnYV9o0StIz/0ArdzmWTxn9oDy0lH8Z80/8X/TD2EUQKXY4DHxoT9mYBqgGIG17DgddCJtH1M6DriMbalNsAA==",
|
||||||
|
"dependencies": [
|
||||||
|
"strnum"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"path-to-regexp@6.3.0": {
|
||||||
|
"integrity": "sha512-Yhpw4T9C6hPpgPeA28us07OJeqZ5EzQTkbfwuhsUg0c237RomFoETJgmp2sa3F/41gfLE6G5cqcYwznmeEeOlQ=="
|
||||||
|
},
|
||||||
|
"strnum@1.0.5": {
|
||||||
|
"integrity": "sha512-J8bbNyKKXl5qYcR36TIO8W3mVGVHrmmxsd5PAItGkmyzwJvybiw2IVq5nqd0i4LSNSkB/sx9VHllbfFdr9k1JA=="
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"workspace": {
|
||||||
|
"dependencies": [
|
||||||
|
"jsr:@oak/oak@^17.1.4",
|
||||||
|
"jsr:@std/assert@1"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
61
flake.lock
generated
Normal file
61
flake.lock
generated
Normal file
|
@ -0,0 +1,61 @@
|
||||||
|
{
|
||||||
|
"nodes": {
|
||||||
|
"flake-utils": {
|
||||||
|
"inputs": {
|
||||||
|
"systems": "systems"
|
||||||
|
},
|
||||||
|
"locked": {
|
||||||
|
"lastModified": 1731533236,
|
||||||
|
"narHash": "sha256-l0KFg5HjrsfsO/JpG+r7fRrqm12kzFHyUHqHCVpMMbI=",
|
||||||
|
"owner": "numtide",
|
||||||
|
"repo": "flake-utils",
|
||||||
|
"rev": "11707dc2f618dd54ca8739b309ec4fc024de578b",
|
||||||
|
"type": "github"
|
||||||
|
},
|
||||||
|
"original": {
|
||||||
|
"owner": "numtide",
|
||||||
|
"repo": "flake-utils",
|
||||||
|
"type": "github"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"nixpkgs": {
|
||||||
|
"locked": {
|
||||||
|
"lastModified": 1739736696,
|
||||||
|
"narHash": "sha256-zON2GNBkzsIyALlOCFiEBcIjI4w38GYOb+P+R4S8Jsw=",
|
||||||
|
"owner": "nixos",
|
||||||
|
"repo": "nixpkgs",
|
||||||
|
"rev": "d74a2335ac9c133d6bbec9fc98d91a77f1604c1f",
|
||||||
|
"type": "github"
|
||||||
|
},
|
||||||
|
"original": {
|
||||||
|
"owner": "nixos",
|
||||||
|
"ref": "nixos-unstable",
|
||||||
|
"repo": "nixpkgs",
|
||||||
|
"type": "github"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"root": {
|
||||||
|
"inputs": {
|
||||||
|
"flake-utils": "flake-utils",
|
||||||
|
"nixpkgs": "nixpkgs"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"systems": {
|
||||||
|
"locked": {
|
||||||
|
"lastModified": 1681028828,
|
||||||
|
"narHash": "sha256-Vy1rq5AaRuLzOxct8nz4T6wlgyUR7zLU309k9mBC768=",
|
||||||
|
"owner": "nix-systems",
|
||||||
|
"repo": "default",
|
||||||
|
"rev": "da67096a3b9bf56a91d16901293e51ba5b49a27e",
|
||||||
|
"type": "github"
|
||||||
|
},
|
||||||
|
"original": {
|
||||||
|
"owner": "nix-systems",
|
||||||
|
"repo": "default",
|
||||||
|
"type": "github"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"root": "root",
|
||||||
|
"version": 7
|
||||||
|
}
|
15
flake.nix
Normal file
15
flake.nix
Normal file
|
@ -0,0 +1,15 @@
|
||||||
|
{
|
||||||
|
inputs = {
|
||||||
|
nixpkgs.url = "github:nixos/nixpkgs?ref=nixos-unstable";
|
||||||
|
flake-utils.url = "github:numtide/flake-utils";
|
||||||
|
};
|
||||||
|
|
||||||
|
outputs = { self, nixpkgs, flake-utils }:
|
||||||
|
flake-utils.lib.eachDefaultSystem(system: let
|
||||||
|
pkgs = nixpkgs.legacyPackages.${system};
|
||||||
|
in {
|
||||||
|
# packages.default = pkgs.callPackage ./default.nix {};
|
||||||
|
devShells.default = import ./shell.nix { inherit pkgs; };
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
5
shell.nix
Normal file
5
shell.nix
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
{ pkgs ? import <nixpkgs> {} }:
|
||||||
|
pkgs.mkShell {
|
||||||
|
packages = with pkgs; [deno];
|
||||||
|
}
|
||||||
|
|
106
src/main.ts
Normal file
106
src/main.ts
Normal file
|
@ -0,0 +1,106 @@
|
||||||
|
import { Application, Router } from "@oak/oak";
|
||||||
|
import { XMLParser } from "npm:fast-xml-parser";
|
||||||
|
|
||||||
|
const router = new Router();
|
||||||
|
|
||||||
|
let wsClients: WebSocket[] = [];
|
||||||
|
|
||||||
|
router.get("/", (context) => {
|
||||||
|
if (context.isUpgradable) {
|
||||||
|
const ws = context.upgrade();
|
||||||
|
ws.onopen = () => wsClients.push(ws);
|
||||||
|
ws.onclose = () => wsClients = wsClients.filter((client) => client != ws);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
router
|
||||||
|
.get("/statusline", async (ctx) => {
|
||||||
|
ctx.response.body = await Deno.readFile("web/statusline.html");
|
||||||
|
})
|
||||||
|
.get("/statusline.js", async (ctx) => {
|
||||||
|
ctx.response.body = await Deno.readFile("web/statusline.js");
|
||||||
|
})
|
||||||
|
.get("/style.css", async (ctx) => {
|
||||||
|
ctx.response.body = await Deno.readFile("web/style.css");
|
||||||
|
});
|
||||||
|
|
||||||
|
const app = new Application();
|
||||||
|
app.use(router.routes());
|
||||||
|
app.use(router.allowedMethods());
|
||||||
|
|
||||||
|
app.addEventListener("listen", ({ hostname, port }) => {
|
||||||
|
console.log(`Start listening on ${hostname}:${port}`);
|
||||||
|
});
|
||||||
|
|
||||||
|
app.listen({ port: 8012 });
|
||||||
|
|
||||||
|
const topic =
|
||||||
|
"Something Fun For Everyone With Streamboy!!!!!!!! git.jeevio.xyz/jeeves/streamboy";
|
||||||
|
|
||||||
|
setInterval(async () => {
|
||||||
|
const songinfo = await getVlcSongInfo();
|
||||||
|
const data = {
|
||||||
|
blocks: [
|
||||||
|
{ text: topic, color: "#ffffff" },
|
||||||
|
{ text: `♪ ${songinfo.title} - ${songinfo.artist}`, color: "#ffffff" },
|
||||||
|
],
|
||||||
|
};
|
||||||
|
for (const ws of wsClients) {
|
||||||
|
// ws.send(
|
||||||
|
// `${topic} | ♪ ${songinfo.title} - ${songinfo.artist} | `,
|
||||||
|
// );
|
||||||
|
//
|
||||||
|
ws.send(JSON.stringify(data));
|
||||||
|
}
|
||||||
|
}, 900);
|
||||||
|
|
||||||
|
interface SongInfo {
|
||||||
|
title?: string;
|
||||||
|
artist?: string;
|
||||||
|
album?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getVlcSongInfo(): Promise<SongInfo> {
|
||||||
|
const parser = new XMLParser({ ignoreAttributes: false });
|
||||||
|
const password = "1234";
|
||||||
|
|
||||||
|
const res = await fetch("http://localhost:8080/requests/status.xml", {
|
||||||
|
headers: { authorization: `Basic ${btoa(`:${password}`)}` },
|
||||||
|
});
|
||||||
|
|
||||||
|
const json = parser.parse(await res.text());
|
||||||
|
|
||||||
|
const songinfo: SongInfo = {};
|
||||||
|
|
||||||
|
for (const category of json.root.information.category) {
|
||||||
|
if (category["@_name"] != "meta") continue;
|
||||||
|
for (const property of category.info) {
|
||||||
|
if (property["@_name"] == "title") {
|
||||||
|
songinfo.title = processBadXmlString(property["#text"]);
|
||||||
|
} else if (property["@_name"] == "artist") {
|
||||||
|
songinfo.artist = processBadXmlString(property["#text"]);
|
||||||
|
} else if (property["@_name"] == "album") {
|
||||||
|
songinfo.album = processBadXmlString(property["#text"]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return songinfo;
|
||||||
|
}
|
||||||
|
|
||||||
|
function processBadXmlString(str: string): string {
|
||||||
|
let newStr = str;
|
||||||
|
|
||||||
|
while (true) {
|
||||||
|
const amp = newStr.indexOf("&#");
|
||||||
|
if (amp > 0) {
|
||||||
|
const semi = newStr.indexOf(";", amp);
|
||||||
|
if (semi > 0) {
|
||||||
|
const int = String.fromCharCode(parseInt(newStr.slice(amp + 2, semi)));
|
||||||
|
newStr = newStr.replace(newStr.slice(amp, semi + 1), int);
|
||||||
|
} else break;
|
||||||
|
} else break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return newStr;
|
||||||
|
}
|
12
web/statusline.html
Normal file
12
web/statusline.html
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
<!DOCTYPE html>
|
||||||
|
|
||||||
|
<head>
|
||||||
|
<link rel="stylesheet" href="style.css" />
|
||||||
|
</head>
|
||||||
|
|
||||||
|
<body>
|
||||||
|
<p id="status-line">Streamboy is connecting...</p>
|
||||||
|
|
||||||
|
<script src="statusline.js"></script>
|
||||||
|
</body>
|
||||||
|
|
46
web/statusline.js
Normal file
46
web/statusline.js
Normal file
|
@ -0,0 +1,46 @@
|
||||||
|
const statusLineElement = document.getElementById("status-line");
|
||||||
|
const statusLineElement2 = statusLineElement.cloneNode();
|
||||||
|
statusLineElement2.id = "status-line2";
|
||||||
|
document.body.appendChild(statusLineElement2);
|
||||||
|
|
||||||
|
let status = "Streamboy is connecting...";
|
||||||
|
let scroll = 0;
|
||||||
|
|
||||||
|
setInterval(() => {
|
||||||
|
if (scroll >= status.length) scroll = 0;
|
||||||
|
// if (scroll == 0) {
|
||||||
|
// statusLineElement.textContent = status;
|
||||||
|
// } else {
|
||||||
|
// let string = "";
|
||||||
|
// string += status.slice(scroll);
|
||||||
|
// string += status.slice(0, scroll);
|
||||||
|
// statusLineElement.textContent = string;
|
||||||
|
// }
|
||||||
|
const stringWidth = 8 * status.length;
|
||||||
|
statusLineElement.style.left = `${-8 * scroll}px`;
|
||||||
|
statusLineElement2.style.left = `${-8 * scroll + stringWidth}px`;
|
||||||
|
// console.log(statusLineElement.style.left)
|
||||||
|
statusLineElement.textContent = status;
|
||||||
|
statusLineElement2.textContent = status;
|
||||||
|
scroll += 1;
|
||||||
|
}, 750);
|
||||||
|
|
||||||
|
const socket = new WebSocket("ws://localhost:8012");
|
||||||
|
|
||||||
|
socket.addEventListener("message", (event) => {
|
||||||
|
const data = JSON.parse(event.data);
|
||||||
|
|
||||||
|
let string = "";
|
||||||
|
for (const block of data.blocks) {
|
||||||
|
string += block.text;
|
||||||
|
string += " | ";
|
||||||
|
}
|
||||||
|
status = string;
|
||||||
|
});
|
||||||
|
|
||||||
|
// const status = {
|
||||||
|
// blocks: [
|
||||||
|
// "Something Fun For Everyone With Streamboy!!!!!!!! git.jeevio.xyz/jeeves/streamboy",
|
||||||
|
// "♪ Bonhomme de Neige (EarthBound) - Ridley Snipes feat. Earth Kid",
|
||||||
|
// ],
|
||||||
|
// };
|
20
web/style.css
Normal file
20
web/style.css
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
body {
|
||||||
|
font-family: Unifont;
|
||||||
|
background-color: black;
|
||||||
|
color: white;
|
||||||
|
margin: 0px auto;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
#gone {
|
||||||
|
font-size: 64px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#status-line, #status-line2 {
|
||||||
|
font-size: 16px;
|
||||||
|
margin: 0px;
|
||||||
|
white-space: preserve nowrap;
|
||||||
|
max-width: none;
|
||||||
|
text-align: center;
|
||||||
|
position: absolute;
|
||||||
|
}
|
Loading…
Add table
Reference in a new issue