summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--includes/external/jobs/index.js26
-rw-r--r--includes/external/matrix/index.js40
-rw-r--r--includes/fragments/metadata.inc228
-rw-r--r--includes/util/functions.inc6
-rw-r--r--pages/api/plex-thumb.php11
-rw-r--r--pages/api/plex.php88
-rw-r--r--pages/jobs.inc11
7 files changed, 271 insertions, 139 deletions
diff --git a/includes/external/jobs/index.js b/includes/external/jobs/index.js
index 39b7de5..0f44cc9 100644
--- a/includes/external/jobs/index.js
+++ b/includes/external/jobs/index.js
@@ -1,13 +1,21 @@
const fs = require('fs');
const child_process = require('child_process');
-let jobs = require('../../data/jobs.json');
+let jobsList = [];
+let jobs = [];
let history = require('../../data/history.json');
+fs.writeFileSync("../../data/running.json", "null");
+
setInterval(() => {
- if (JSON.stringify(JSON.parse(fs.readFileSync("../../data/jobs.json").toString())) !== JSON.stringify(jobs)) {
+ if (JSON.stringify(fs.readdirSync("../../data/jobs")) !== JSON.stringify(jobsList)) {
console.log("Updating the jobs list...");
- jobs = JSON.parse(fs.readFileSync("../../data/jobs.json").toString());
+ jobs = fs.readdirSync("../../data/jobs").map((i) => {
+ let obj = JSON.parse(fs.readFileSync("../../data/jobs/" + i).toString());
+ obj["_id"] = i;
+ return obj;
+ });
+ jobsList = fs.readdirSync("../../data/jobs");
}
}, 1000);
@@ -20,11 +28,13 @@ setTimeout(() => {
console.log("\nRunning jobs:");
for (let job of jobs) {
- console.log(" " + job.name);
+ console.log(" " + job.name + " [" + job._id + "]");
let output;
let start;
let end;
+ fs.writeFileSync("../../data/running.json", JSON.stringify(job._id));
+
try {
start = new Date();
output = child_process.execFileSync("php", [job.name + ".php", JSON.stringify(job.options)], { cwd: "../../jobs" });
@@ -37,8 +47,7 @@ setTimeout(() => {
description = "," + name + "=" + JSON.stringify(job.options[name]);
}
- jobs.splice(jobs.indexOf(job), 1);
- fs.writeFileSync("../../data/jobs.json", JSON.stringify(jobs));
+ fs.unlinkSync("../../data/jobs/" + job._id);
history.unshift({
completed: true,
@@ -58,6 +67,7 @@ setTimeout(() => {
history = history.slice(0, 200);
fs.writeFileSync("../../data/history.json", JSON.stringify(history));
+ fs.writeFileSync("../../data/running.json", "null");
} catch (e) {
end = start ? new Date() : null;
console.log(" Failed to process job");
@@ -70,8 +80,7 @@ setTimeout(() => {
description = "," + name + "=" + JSON.stringify(job.options[name]);
}
- jobs.splice(jobs.indexOf(job), 1);
- fs.writeFileSync("../../data/jobs.json", JSON.stringify(jobs));
+ fs.unlinkSync("../../data/jobs/" + job._id);
history.unshift({
completed: false,
@@ -91,6 +100,7 @@ setTimeout(() => {
history = history.slice(0, 200);
fs.writeFileSync("../../data/history.json", JSON.stringify(history));
+ fs.writeFileSync("../../data/running.json", "null");
}
}
diff --git a/includes/external/matrix/index.js b/includes/external/matrix/index.js
index 5149005..1bfb8d4 100644
--- a/includes/external/matrix/index.js
+++ b/includes/external/matrix/index.js
@@ -1,5 +1,6 @@
const user = process.argv[2];
const info = require('../../app.json')['matrix'][user];
+const fs = require('fs');
if (typeof info !== "object") throw new Error("Unable to find info");
@@ -7,6 +8,7 @@ const token = info.token;
const sdk = require("matrix-js-sdk");
const axios = require("axios");
+const child_process = require("child_process");
const client = sdk.createClient({
baseUrl: "https://chat.equestria.dev",
accessToken: token,
@@ -24,13 +26,16 @@ client.once("sync", async function (state, prevState, res) {
console.log(fronters.members.length + " member(s) at front");
let name = system.name;
- let avatar = system.avatar_url;
+ let avatar1 = system.avatar_url;
+ let avatar2 = null;
- if (fronters.members.length === 1) {
+ if (fronters.members.length === 1 || (fronters.members.length === 2 && (fronters.members[0].name === "scootaloo" && fronters.members[1].name === "rainbowdash"))) {
name = fronters.members[0].display_name ?? fronters.members[0].name;
- avatar = fronters.members[0].avatar_url ?? avatar;
- } else if (fronters.members.length >= 2) {
+ avatar1 = fronters.members[0].avatar_url ?? avatar1;
+ } else if (fronters.members.length >= 2 && !(fronters.members[0].name === "scootaloo" && fronters.members[1].name === "rainbowdash")) {
name = (fronters.members[0].display_name ?? fronters.members[0].name) + " and " + (fronters.members[1].display_name ?? fronters.members[1].name);
+ avatar1 = fronters.members[0].avatar_url ?? avatar1;
+ avatar2 = fronters.members[1].avatar_url ?? avatar2;
}
await client.setDeviceDetails(info.device, {
@@ -52,7 +57,32 @@ client.once("sync", async function (state, prevState, res) {
});
console.log("Uploading avatar");
- let res = await client.uploadContent((await axios.get(avatar, {responseType: "arraybuffer"})).data);
+ let pfp;
+
+ if (avatar2) {
+ let img1 = (await axios.get(avatar1, {responseType: "arraybuffer"})).data;
+ let img2 = (await axios.get(avatar2, {responseType: "arraybuffer"})).data;
+
+ fs.writeFileSync("/tmp/chm-" + user + "-1", img1);
+ fs.writeFileSync("/tmp/chm-" + user + "-2", img2);
+
+ child_process.execSync("convert /tmp/chm-" + user + "-1 -gravity center -crop 50%x100% /tmp/chm-" + user + "-1.webp");
+ child_process.execSync("convert /tmp/chm-" + user + "-2 -gravity center -crop 50%x100% /tmp/chm-" + user + "-2.webp");
+ child_process.execSync("montage -mode concatenate -tile x1 /tmp/chm-" + user + "-1.webp /tmp/chm-" + user + "-2.webp /tmp/chm-" + user + ".webp");
+
+ pfp = fs.readFileSync("/tmp/chm-" + user + ".webp");
+
+ fs.unlinkSync("/tmp/chm-" + user + "-1");
+ fs.unlinkSync("/tmp/chm-" + user + "-2");
+ fs.unlinkSync("/tmp/chm-" + user + "-1.webp");
+ fs.unlinkSync("/tmp/chm-" + user + "-2.webp");
+ fs.unlinkSync("/tmp/chm-" + user + ".webp");
+ } else {
+ pfp = (await axios.get(avatar1, {responseType: "arraybuffer"})).data;
+ }
+
+ let res = await client.uploadContent(pfp);
+
await axios.put("https://chat.equestria.dev/_matrix/client/r0/profile/%40" + user + "%3Aequestria.dev/avatar_url", {
avatar_url: res.content_uri
}, {
diff --git a/includes/fragments/metadata.inc b/includes/fragments/metadata.inc
index 0d2187c..719abbc 100644
--- a/includes/fragments/metadata.inc
+++ b/includes/fragments/metadata.inc
@@ -6,9 +6,51 @@ if (file_exists($_SERVER['DOCUMENT_ROOT'] . "/includes/data/metadata/" . $member
?>
+<style>
+ .modal-header {
+ border-bottom: 1px solid #353738;
+ }
+
+ .modal-content {
+ border: 1px solid rgba(255, 255, 255, .2);
+ background-color: #111;
+ }
+
+ .btn-close {
+ filter: invert(1);
+ }
+
+ .list-group-item {
+ color: #fff;
+ background-color: #222;
+ border: 1px solid rgba(255, 255, 255, .125);
+ }
+
+ .list-group-item.disabled {
+ color: #fff;
+ background-color: #222;
+ border-color: rgba(255, 255, 255, .125);
+ opacity: .75;
+ }
+
+ .list-group-item:hover {
+ background-color: #252525;
+ color: #ddd;
+ }
+
+ .list-group-item:active, .list-group-item:focus {
+ background-color: #272727;
+ color: #bbb;
+ }
+
+ .member-link, .list-group-item-action {
+ cursor: pointer !important;
+ }
+</style>
+
<br>
-<div class="container" id="page-content">
- <h2><a href="/<?= $memberData["name"] ?>"><?= $memberData["display_name"] ?? $memberData["name"] ?></a></h2>
+<div class="container">
+ <h2 id="page-content"><a href="/<?= $memberData["name"] ?>"><?= $memberData["display_name"] ?? $memberData["name"] ?></a></h2>
<?php if (file_exists($_SERVER['DOCUMENT_ROOT'] . "/includes/data/metadata/" . $memberID . ".json")): ?>
<form>
<input name="submit" type="hidden">
@@ -94,28 +136,166 @@ if (file_exists($_SERVER['DOCUMENT_ROOT'] . "/includes/data/metadata/" . $member
<h3>Relationships</h3>
- <?php if ($systemID !== $app["other"]["id"]): ?>
- <p>
- <b>Sexfriends (full IDs, comma-separated)</b><br>
- <input name="sexfriends" class="form-control" style="filter: invert(1) hue-rotate(180deg);" type="text" value="<?= implode(", ", $metadata["sexfriends"]) ?>">
- </p>
- <?php endif; ?>
- <p>
- <b>Marefriends (full IDs, comma-separated)</b><br>
- <input name="marefriends" class="form-control" style="filter: invert(1) hue-rotate(180deg);" type="text" value="<?= implode(", ", $metadata["marefriends"]) ?>">
- </p>
- <p>
- <b>Sisters (full IDs, comma-separated)</b><br>
- <input name="sisters" class="form-control" style="filter: invert(1) hue-rotate(180deg);" type="text" value="<?= implode(", ", $metadata["sisters"]) ?>">
- </p>
- <p>
- <b>Caretakers (full IDs, comma-separated)</b><br>
- <input name="caretakers" class="form-control" style="filter: invert(1) hue-rotate(180deg);" type="text" value="<?= implode(", ", $metadata["caretakers"]) ?>">
- </p>
- <p>
- <b>Friends (full IDs, comma-separated)</b><br>
- <input name="friends" class="form-control" style="filter: invert(1) hue-rotate(180deg);" type="text" value="<?= implode(", ", $metadata["friends"] ?? []) ?>">
- </p>
+ <div class="modal" id="new-relation" data-bs-backdrop="static" data-bs-keyboard="false">
+ <div class="modal-dialog">
+ <div class="modal-content">
+
+ <div class="modal-header">
+ <h4 class="modal-title">Add a new <span id="new-relation-type">-</span> for <?= $memberData["display_name"] ?? $memberData["name"] ?></h4>
+ <button id="new-fronter-close" type="button" class="btn-close" data-bs-dismiss="modal"></button>
+ </div>
+
+ <div class="modal-body">
+ <input type="text" placeholder="Search for a pony..." class="form-control" style="margin-bottom:15px;color:white;background:#111;border-color:#222;" id="search">
+
+ <div id="list">
+ <div class="list-group">
+ <?php foreach (scoreOrderGlobal() as $member): ?>
+ <a id="list-pony-<?= $member['id'] ?>" onclick="confirmAdd('<?= $member['system'] ?>/<?= $member['id'] ?>');" class="new-fronter-link list-group-item list-group-item-action"><img src="<?= getAsset($member["system"], $member["id"], "heads") ?>" style="width:24px;"> <?= $member["display_name"] ?? $member["name"] ?></a>
+ <?php endforeach; ?>
+ </div>
+ </div>
+
+ <div id="search-results" class="list-group"></div>
+ </div>
+
+ </div>
+ </div>
+ </div>
+
+ <script src="/assets/editor/fuse.js"></script>
+ <script>
+ window.modal = new bootstrap.Modal(document.getElementById("new-relation"));
+ window.currentSelectedValue = null;
+
+ function openAddDialog(value, text) {
+ modal.show();
+ window.currentSelectedValue = value;
+ document.getElementById("new-relation-type").innerText = text;
+ document.getElementById("search").value = ""; search({});
+ document.getElementById("search").focus();
+ }
+
+ window.poniesList = JSON.parse(atob(`<?= base64_encode(json_encode(array_values(scoreOrderGlobal()))) ?>`));
+ window.avatars = JSON.parse(atob(`<?php
+
+ $avatars = [];
+
+ foreach (scoreOrderGlobal() as $pony) {
+ $avatars[$pony["system"] . "/" . $pony["id"]] = getAsset($pony["system"], $pony["id"]);
+ }
+
+ echo(base64_encode(json_encode($avatars))); ?>`))
+
+ const fuse = new Fuse(window.poniesList, {
+ includeScore: true,
+ keys: [
+ {
+ name: 'name',
+ weight: 1
+ },
+ {
+ name: 'display_name',
+ weight: 1
+ },
+ {
+ name: 'id',
+ weight: 0.7
+ },
+ {
+ name: 'species',
+ weight: 0.5
+ }
+ ]
+ })
+
+ function search(event) {
+ if (event.key === "Enter") {
+ return;
+ }
+
+ let query = document.getElementById("search").value;
+ let results = fuse.search(query).map((i) => {
+ return {
+ id: i.item.id,
+ score: i.score
+ };
+ });
+
+ let unfiltered = results;
+
+ results = results.filter((i) => {
+ return i.score < 0.7;
+ });
+
+ console.log("Before:", unfiltered, "After:", results);
+
+ document.getElementById("list").style.display = "none";
+ document.getElementById("search-results").style.display = "block";
+ document.getElementById("search-results").innerHTML = "";
+
+ for (let result of results) {
+ if (document.getElementById("list-pony-" + result.id)) document.getElementById("search-results").innerHTML += document.getElementById("list-pony-" + result.id).outerHTML;
+ }
+
+ console.log(results);
+
+ if (query.trim() === "") {
+ document.getElementById("list").style.display = "block";
+ document.getElementById("search-results").style.display = "none";
+ }
+ }
+
+ document.getElementById("search").onchange = document.getElementById("search").onkeyup = document.getElementById("search").onkeydown = search;
+
+ document.getElementById("search").addEventListener("keydown", (event) => {
+ if (event.key === "Enter") {
+ document.querySelector("#search-results .new-fronter-link").click();
+ event.preventDefault();
+ }
+ });
+
+ function update(type) {
+ document.getElementById(type + "-list").innerHTML = document.getElementsByName(type)[0].value.split(",").map(i => i.trim()).filter(i => i && i.trim() !== "").map(i => `
+ <a style="margin-top: 4px;" class="btn btn-dark" onclick="deleteRelation('${type}', '${i}');">
+ <img src="${window.avatars[i]}" style="width: 24px; vertical-align: middle; border-radius: 999px;">
+ <span style="vertical-align: middle; margin-left: 5px;">${window.poniesList.filter(j => j.id === i.split("/")[1])[0]['display_name'] ?? window.poniesList.filter(j => j.id === i.split("/")[1])[0]['name']}</span>
+ </a>
+ `).join("");
+ }
+
+ function confirmAdd(item) {
+ document.getElementsByName(window.currentSelectedValue)[0].value += ", " + item;
+ update(window.currentSelectedValue);
+ window.currentSelectedValue = null;
+ modal.hide();
+ }
+
+ function deleteRelation(type, relation) {
+ let relations = document.getElementsByName(type)[0].value.split(",").map(i => i.trim()).filter(i => i && i !== "");
+ relations = relations.filter(i => i !== relation);
+ document.getElementsByName(type)[0].value = relations.join(", ");
+ update(type);
+ }
+ </script>
+
+ <?php foreach ([
+ $systemID !== $app["other"]["id"] ? [ "sexfriends", "Sexfriends" ] : null,
+ [ "marefriends", "Marefriends", "marefriend" ],
+ [ "sisters", "Sisters", "sister" ],
+ [ "caretakers", "Caretakers", "caretaker" ],
+ [ "friends", "Friends", "friend" ],
+ ] as $relations): if (isset($relations)): ?>
+ <p>
+ <b><?= $relations[1] ?></b><br>
+ <input name="<?= $relations[0] ?>" type="hidden" value="<?= implode(", ", $metadata[$relations[0]] ?? []) ?>">
+ <span id="<?= $relations[0] ?>-list"></span>
+ <a style="margin-top: 4px;" class="btn btn-secondary" onclick="openAddDialog('<?= $relations[0] ?>', '<?= $relations[2] ?>')">+</a>
+ <script>
+ update('<?= $relations[0] ?>');
+ </script>
+ </p>
+ <?php endif; endforeach; ?>
<hr>
diff --git a/includes/util/functions.inc b/includes/util/functions.inc
index 79a47c6..fe97980 100644
--- a/includes/util/functions.inc
+++ b/includes/util/functions.inc
@@ -3,17 +3,17 @@
require_once $_SERVER['DOCUMENT_ROOT'] . "/includes/util/score.inc";
require_once $_SERVER['DOCUMENT_ROOT'] . "/includes/util/bitset.inc";
require_once $_SERVER['DOCUMENT_ROOT'] . "/includes/util/homepage.inc";
+require_once $_SERVER['DOCUMENT_ROOT'] . "/includes/util/random.inc";
if (!function_exists("createJob")) {
function createJob($title, $options) {
- $jobs = json_decode(file_get_contents($_SERVER['DOCUMENT_ROOT'] . "/includes/data/jobs.json"), true);
- $jobs[] = [
+ $job = [
"name" => $title,
"options" => $options,
"date" => date('c')
];
- file_put_contents($_SERVER['DOCUMENT_ROOT'] . "/includes/data/jobs.json", json_encode($jobs));
+ file_put_contents($_SERVER['DOCUMENT_ROOT'] . "/includes/data/jobs/" . round(microtime(true) * 1000) . "-" . random() . ".json", json_encode($job));
}
}
diff --git a/pages/api/plex-thumb.php b/pages/api/plex-thumb.php
deleted file mode 100644
index 4009871..0000000
--- a/pages/api/plex-thumb.php
+++ /dev/null
@@ -1,11 +0,0 @@
-<?php
-
-$app = $GLOBALS["ColdHazeApp"];
-
-if (isset($_GET["library"]) && isset($_GET["id"]) && is_numeric($_GET["library"]) && is_numeric($_GET["id"])) {
- header("Content-Type: image/jpg");
- die(file_get_contents("https://plex.equestria.dev/library/metadata/" . $_GET["library"] . "/thumb/" . $_GET["id"] . "?X-Plex-Token=" . $app["plex"]));
-} else {
- header("HTTP/1.1 400 Invalid request");
- die("Invalid request");
-} \ No newline at end of file
diff --git a/pages/api/plex.php b/pages/api/plex.php
deleted file mode 100644
index 00164c7..0000000
--- a/pages/api/plex.php
+++ /dev/null
@@ -1,88 +0,0 @@
-<?php
-
-require_once $_SERVER['DOCUMENT_ROOT'] . "/includes/util/random.inc";
-$app = $GLOBALS["ColdHazeApp"];
-
-function formatTitle($metadata) {
- if ($metadata['grandparentTitle']) {
- return $metadata['grandparentTitle'];
- } else {
- $ret = $metadata['title'];
- if ($metadata['year']) {
- $ret .= " (" . $metadata['year'] . ")";
- }
- return $ret;
- }
-}
-
-function formatSubtitle($metadata) {
- $ret = '';
-
- if ($metadata['grandparentTitle']) {
- if ($metadata['type'] === 'track') {
- $ret = $metadata['parentTitle'];
- } else if ($metadata['index'] && $metadata['parentIndex']) {
- $ret = "S" . $metadata['parentIndex'] . " E" . $metadata['index'];
- } else if ($metadata['originallyAvailableAt']) {
- $ret = $metadata['originallyAvailableAt'];
- }
-
- if ($metadata['title']) {
- $ret .= ' - ' . $metadata['title'];
- }
- } else if ($metadata['type'] === 'movie') {
- $ret = $metadata['tagline'];
- }
-
- return $ret;
-}
-
-$payload = json_decode($_POST["payload"], true);
-
-if (!file_exists($_SERVER['DOCUMENT_ROOT'] . "/assets/cache")) mkdir($_SERVER['DOCUMENT_ROOT'] . "/assets/cache");
-$id = random(32);
-
-if ($payload["Metadata"]["type"] === "track") {
- file_put_contents($_SERVER['DOCUMENT_ROOT'] . "/assets/cache/" . $id . ".jpg", file_get_contents("https://plex.equestria.dev" . $payload["Metadata"]["thumb"] . "?X-Plex-Token=" . $app["plex"]));
-} else {
- file_put_contents($_SERVER['DOCUMENT_ROOT'] . "/assets/cache/" . $id . ".jpg", file_get_contents("https://plex.equestria.dev" . $payload["Metadata"]["grandparentThumb"] . "?X-Plex-Token=" . $app["plex"]));
-}
-
-if ($payload["event"] === "playback.started" || $payload["event"] === "media.play") {
- $hookObject = json_encode([
- "username" => "Plex",
- "avatar_url" => "https://support.plex.tv/wp-content/themes/plex/assets/img/favicons/plex-192.png",
- "embeds" => [
- [
- "title" => formatTitle($payload["Metadata"]),
- "type" => "rich",
- "description" => formatSubtitle($payload["Metadata"]),
- "color" => hexdec( "2b2d31" ),
- "thumbnail" => [
- "url" => "https://ponies.equestria.horse/assets/cache/" . $id . ".jpg"
- ],
- "footer" => [
- "text" => $payload["Account"]["title"] . " ยท Playing from " . $payload["Player"]["title"],
- "icon_url" => $payload["Account"]["thumb"]
- ]
- ]
- ]
-
- ], JSON_UNESCAPED_SLASHES | JSON_UNESCAPED_UNICODE );
-}
-
-if (isset($hookObject)) {
- $ch = curl_init();
-
- curl_setopt_array( $ch, [
- CURLOPT_URL => $app["webhook"]["plex"],
- CURLOPT_POST => true,
- CURLOPT_POSTFIELDS => $hookObject,
- CURLOPT_HTTPHEADER => [
- "Content-Type: application/json"
- ]
- ]);
-
- $response = curl_exec( $ch );
- curl_close( $ch );
-} \ No newline at end of file
diff --git a/pages/jobs.inc b/pages/jobs.inc
index 6723175..c00e35c 100644
--- a/pages/jobs.inc
+++ b/pages/jobs.inc
@@ -4,6 +4,7 @@ require_once $_SERVER['DOCUMENT_ROOT'] . "/includes/init.inc"; global $title; gl
require_once $_SERVER['DOCUMENT_ROOT'] . '/includes/util/functions.inc';
$history = json_decode(file_get_contents($_SERVER['DOCUMENT_ROOT'] . "/includes/data/history.json"), true);
+$running = json_decode(file_get_contents($_SERVER['DOCUMENT_ROOT'] . "/includes/data/running.json"), true);
if (isset($_GET["enqueue"]) && isset($history[(int)$_GET["enqueue"]])) {
createJob(explode("(", $history[(int)$_GET["enqueue"]]["name"])[0], $history[(int)$_GET["enqueue"]]["options"]);
@@ -73,6 +74,16 @@ require_once $_SERVER['DOCUMENT_ROOT'] . '/includes/components/header.inc';
<?php endif; ?>
<div class="list-group">
+ <?php foreach (array_reverse(array_filter(scandir($_SERVER['DOCUMENT_ROOT'] . "/includes/data/jobs"), function ($i) {
+ return !str_starts_with($i, ".");
+ })) as $index => $file): $item = json_decode(file_get_contents($_SERVER['DOCUMENT_ROOT'] . "/includes/data/jobs/" . $file), true); ?>
+ <div class="list-group-item disabled" style="list-style: none; display: grid; grid-template-columns: 3fr repeat(2, 1fr);">
+ <div><?= $item["name"] ?>(<?php foreach ($item["options"] as $name => $value): echo($name . "=" . json_encode($value)); endforeach; ?>)</div>
+ <div><?= timeAgo($item["date"]) ?></div>
+ <div><?php if (isset($running) && $file === $running): ?><span class="text-info">Running</span><?php else: ?><span class="text-warning">Pending</span><?php endif; ?></div>
+ </div>
+ <?php endforeach; ?>
+
<?php foreach ($history as $index => $item): ?>
<details class="list-group-item">
<summary style="list-style: none; display: grid; grid-template-columns: 3fr repeat(2, 1fr);">