updated the API to support the new calendar service and added a smoke test for the API.

This commit is contained in:
2026-05-20 16:53:26 -07:00
parent 745d0a2909
commit c423f1191d
4 changed files with 92 additions and 23 deletions
+18 -11
View File
@@ -1,7 +1,6 @@
(function () {
const apiBaseUrlStorageKey = "tarot-time-api-base-url";
const apiKeyStorageKey = "tarot-time-api-key";
const defaultApiBaseUrl = "";
function normalizeBaseUrl(value) {
return String(value || "")
@@ -20,22 +19,33 @@
};
}
function stripLegacyCredentialParams() {
try {
const currentUrl = new URL(window.location.href);
const hadApiKeyParam = currentUrl.searchParams.has("apiKey") || currentUrl.searchParams.has("api_key");
if (!hadApiKeyParam) {
return;
}
currentUrl.searchParams.delete("apiKey");
currentUrl.searchParams.delete("api_key");
const nextUrl = `${currentUrl.pathname}${currentUrl.search}${currentUrl.hash}`;
window.history.replaceState(window.history.state, "", nextUrl);
} catch (_error) {
}
}
function persistConnectionSettings(settings) {
let didPersist = true;
try {
const params = new URLSearchParams(window.location.search || "");
const queryValue = String(params.get("apiBaseUrl") || "").trim();
const queryApiKey = String(params.get("apiKey") || params.get("api_key") || "").trim();
if (queryValue) {
window.localStorage.setItem(apiBaseUrlStorageKey, normalizeBaseUrl(queryValue));
}
if (queryApiKey) {
window.localStorage.setItem(apiKeyStorageKey, normalizeApiKey(queryApiKey));
}
if (settings.apiBaseUrl) {
window.localStorage.setItem(apiBaseUrlStorageKey, settings.apiBaseUrl);
} else {
@@ -58,16 +68,11 @@
try {
const params = new URLSearchParams(window.location.search || "");
const queryValue = String(params.get("apiBaseUrl") || "").trim();
const queryApiKey = String(params.get("apiKey") || params.get("api_key") || "").trim();
if (queryValue) {
window.localStorage.setItem(apiBaseUrlStorageKey, normalizeBaseUrl(queryValue));
}
if (queryApiKey) {
window.localStorage.setItem(apiKeyStorageKey, normalizeApiKey(queryApiKey));
}
const storedBaseUrl = String(window.localStorage.getItem(apiBaseUrlStorageKey) || "").trim();
const storedApiKey = String(window.localStorage.getItem(apiKeyStorageKey) || "").trim();
return normalizeConnectionSettings({
@@ -82,6 +87,8 @@
}
}
stripLegacyCredentialParams();
const initialConnectionSettings = readConfiguredConnectionSettings();
window.TarotAppConfig = {
+1 -12
View File
@@ -192,11 +192,6 @@
url.searchParams.set(key, normalizedValue);
});
const apiKey = getApiKey();
if (apiKey && !url.searchParams.has("api_key")) {
url.searchParams.set("api_key", apiKey);
}
return url.toString();
}
@@ -211,13 +206,7 @@
return "";
}
const url = new URL(`/api/v1/assets/${encodePathSegments(normalizedAssetPath)}`, `${apiBaseUrl}/`);
const apiKey = getApiKey();
if (apiKey) {
url.searchParams.set("api_key", apiKey);
}
return url.toString();
return new URL(`/api/v1/assets/${encodePathSegments(normalizedAssetPath)}`, `${apiBaseUrl}/`).toString();
}
function resetCaches() {
+2
View File
@@ -20,6 +20,8 @@
"@tootallnate/once": "3.0.1"
},
"scripts": {
"check": "npm run check:syntax",
"check:syntax": "node scripts/check-syntax.js app.js app scripts",
"start": "npx http-server . -o index.html",
"dev": "npm run start"
}
+71
View File
@@ -0,0 +1,71 @@
const fs = require("fs");
const path = require("path");
const { spawnSync } = require("child_process");
const skippedDirectoryNames = new Set(["node_modules", ".git"]);
function collectJavaScriptFiles(targetPath, results = []) {
const resolvedPath = path.resolve(targetPath);
if (!fs.existsSync(resolvedPath)) {
return results;
}
const stats = fs.statSync(resolvedPath);
if (stats.isDirectory()) {
const entries = fs.readdirSync(resolvedPath, { withFileTypes: true });
for (const entry of entries) {
if (entry.isDirectory() && skippedDirectoryNames.has(entry.name)) {
continue;
}
collectJavaScriptFiles(path.join(resolvedPath, entry.name), results);
}
return results;
}
if (stats.isFile() && resolvedPath.endsWith(".js")) {
results.push(resolvedPath);
}
return results;
}
function main() {
const targets = process.argv.slice(2);
if (!targets.length) {
console.error("Provide at least one file or directory to validate.");
process.exit(1);
}
const files = Array.from(new Set(
targets.flatMap((targetPath) => collectJavaScriptFiles(targetPath))
)).sort();
if (!files.length) {
console.error("No JavaScript files were found for validation.");
process.exit(1);
}
let hasFailures = false;
for (const filePath of files) {
const result = spawnSync(process.execPath, ["--check", filePath], {
encoding: "utf8"
});
if (result.status === 0) {
continue;
}
hasFailures = true;
process.stderr.write(result.stderr || result.stdout || `Syntax check failed: ${filePath}\n`);
}
if (hasFailures) {
process.exit(1);
}
console.log(`Syntax check passed for ${files.length} files.`);
}
main();