Compare commits
No commits in common. "3cb3b76a2b342381388b131fb47513ddf1560174" and "9d6b9e40a7d245825e692cd9018477e270f53842" have entirely different histories.
3cb3b76a2b
...
9d6b9e40a7
14 changed files with 123 additions and 295 deletions
|
@ -15,8 +15,6 @@ BADGE_API_URL=http://localhost:8081
|
||||||
# Required if you want to enable reviews from reviewdb
|
# Required if you want to enable reviews from reviewdb
|
||||||
REVIEW_DB=true
|
REVIEW_DB=true
|
||||||
|
|
||||||
#Timezone api url, aka: https://git.creations.works/creations/timezoneDB
|
|
||||||
TIMEZONE_API_URL=
|
|
||||||
|
|
||||||
# https://www.steamgriddb.com/api/v2, if you want games to have images
|
# https://www.steamgriddb.com/api/v2, if you want games to have images
|
||||||
STEAMGRIDDB_API_KEY=steamgrid_api_key
|
STEAMGRIDDB_API_KEY=steamgrid_api_key
|
||||||
|
|
|
@ -1,26 +0,0 @@
|
||||||
name: Docker Image Release
|
|
||||||
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
tags:
|
|
||||||
- "v*"
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
release:
|
|
||||||
runs-on: docker
|
|
||||||
steps:
|
|
||||||
- name: Checkout
|
|
||||||
uses: actions/checkout@v4
|
|
||||||
|
|
||||||
- name: Set tag
|
|
||||||
run: echo "TAG=${GITHUB_REF##*/}" >> $GITHUB_ENV
|
|
||||||
|
|
||||||
- name: Login to Docker Registry
|
|
||||||
run: echo "${{ secrets.REGISTRY_PASSWORD }}" | docker login -u "${{ secrets.REGISTRY_USERNAME }}" --password-stdin
|
|
||||||
|
|
||||||
- name: Build and Push Docker Image
|
|
||||||
run: |
|
|
||||||
IMAGE=creations/profile-page
|
|
||||||
docker build --target release -t $IMAGE:$TAG -t $IMAGE:latest .
|
|
||||||
docker push $IMAGE:$TAG
|
|
||||||
docker push $IMAGE:latest
|
|
1
.gitignore
vendored
1
.gitignore
vendored
|
@ -4,4 +4,3 @@ bun.lock
|
||||||
logs/
|
logs/
|
||||||
.vscode/
|
.vscode/
|
||||||
robots.txt
|
robots.txt
|
||||||
public/custom
|
|
||||||
|
|
|
@ -63,8 +63,8 @@ cp .env.example .env
|
||||||
| `LANYARD_INSTANCE` | Endpoint of the Lanyard instance |
|
| `LANYARD_INSTANCE` | Endpoint of the Lanyard instance |
|
||||||
| `BADGE_API_URL` | Badge API URL ([badgeAPI](https://git.creations.works/creations/badgeAPI)) |
|
| `BADGE_API_URL` | Badge API URL ([badgeAPI](https://git.creations.works/creations/badgeAPI)) |
|
||||||
| `REVIEW_DB` | Enables showing reviews from reviewdb on user pages |
|
| `REVIEW_DB` | Enables showing reviews from reviewdb on user pages |
|
||||||
| `TIMEZONE_API_URL` | Enables showing times from [timezoneDB](https://git.creations.works/creations/timezoneDB) |
|
|
||||||
| `STEAMGRIDDB_API_KEY` | SteamGridDB API key for fetching game icons |
|
| `STEAMGRIDDB_API_KEY` | SteamGridDB API key for fetching game icons |
|
||||||
|
| `ROBOTS_FILE` | If there it uses the file in /robots.txt route, requires a valid path |
|
||||||
|
|
||||||
#### Optional Lanyard KV Variables (per-user customization)
|
#### Optional Lanyard KV Variables (per-user customization)
|
||||||
|
|
||||||
|
@ -80,7 +80,6 @@ These can be defined in Lanyard's KV store to customize the page:
|
||||||
| `css` | URL to a css to change styles on the page, no import or require allowed |
|
| `css` | URL to a css to change styles on the page, no import or require allowed |
|
||||||
| `optout` | Allows users to stop sharing there profile on the website (`true` / `false`) |
|
| `optout` | Allows users to stop sharing there profile on the website (`true` / `false`) |
|
||||||
| `reviews` | Enables reviews from reviewdb (`true` / `false`) |
|
| `reviews` | Enables reviews from reviewdb (`true` / `false`) |
|
||||||
| `timezone`| Enables the showing of the current time from the timezone db API (`true` / `false`) |
|
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
import { echo } from "@atums/echo";
|
import { resolve } from "node:path";
|
||||||
|
import { logger } from "@creations.works/logger";
|
||||||
|
|
||||||
const environment: Environment = {
|
const environment: Environment = {
|
||||||
port: Number.parseInt(process.env.PORT || "8080", 10),
|
port: Number.parseInt(process.env.PORT || "8080", 10),
|
||||||
|
@ -21,14 +22,16 @@ const reviewDb = {
|
||||||
url: "https://manti.vendicated.dev/api/reviewdb",
|
url: "https://manti.vendicated.dev/api/reviewdb",
|
||||||
};
|
};
|
||||||
|
|
||||||
const timezoneAPIUrl: string | null = process.env.TIMEZONE_API_URL || null;
|
|
||||||
|
|
||||||
const badgeApi: string | null = process.env.BADGE_API_URL || null;
|
const badgeApi: string | null = process.env.BADGE_API_URL || null;
|
||||||
const steamGridDbKey: string | undefined = process.env.STEAMGRIDDB_API_KEY;
|
const steamGridDbKey: string | undefined = process.env.STEAMGRIDDB_API_KEY;
|
||||||
|
|
||||||
const plausibleScript: string | null =
|
const plausibleScript: string | null =
|
||||||
process.env.PLAUSIBLE_SCRIPT_HTML?.trim() || null;
|
process.env.PLAUSIBLE_SCRIPT_HTML?.trim() || null;
|
||||||
|
|
||||||
|
const robotstxtPath: string | null = process.env.ROBOTS_FILE
|
||||||
|
? resolve(process.env.ROBOTS_FILE)
|
||||||
|
: null;
|
||||||
|
|
||||||
function verifyRequiredVariables(): void {
|
function verifyRequiredVariables(): void {
|
||||||
const requiredVariables = [
|
const requiredVariables = [
|
||||||
"HOST",
|
"HOST",
|
||||||
|
@ -43,7 +46,7 @@ function verifyRequiredVariables(): void {
|
||||||
for (const key of requiredVariables) {
|
for (const key of requiredVariables) {
|
||||||
const value = process.env[key];
|
const value = process.env[key];
|
||||||
if (value === undefined || value.trim() === "") {
|
if (value === undefined || value.trim() === "") {
|
||||||
echo.error(`Missing or empty environment variable: ${key}`);
|
logger.error(`Missing or empty environment variable: ${key}`);
|
||||||
hasError = true;
|
hasError = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -58,9 +61,9 @@ export {
|
||||||
lanyardConfig,
|
lanyardConfig,
|
||||||
redisTtl,
|
redisTtl,
|
||||||
reviewDb,
|
reviewDb,
|
||||||
timezoneAPIUrl,
|
|
||||||
badgeApi,
|
badgeApi,
|
||||||
steamGridDbKey,
|
steamGridDbKey,
|
||||||
plausibleScript,
|
plausibleScript,
|
||||||
|
robotstxtPath,
|
||||||
verifyRequiredVariables,
|
verifyRequiredVariables,
|
||||||
};
|
};
|
||||||
|
|
39
logger.json
39
logger.json
|
@ -1,39 +0,0 @@
|
||||||
{
|
|
||||||
"directory": "logs",
|
|
||||||
"level": "debug",
|
|
||||||
"disableFile": false,
|
|
||||||
|
|
||||||
"rotate": true,
|
|
||||||
"maxFiles": 3,
|
|
||||||
|
|
||||||
"console": true,
|
|
||||||
"consoleColor": true,
|
|
||||||
|
|
||||||
"dateFormat": "yyyy-MM-dd HH:mm:ss.SSS",
|
|
||||||
"timezone": "local",
|
|
||||||
|
|
||||||
"silent": false,
|
|
||||||
|
|
||||||
"pattern": "{color:gray}{pretty-timestamp}{reset} {color:levelColor}[{level-name}]{reset} {color:gray}({reset}{file-name}:{color:blue}{line}{reset}:{color:blue}{column}{color:gray}){reset} {data}",
|
|
||||||
"levelColor": {
|
|
||||||
"debug": "blue",
|
|
||||||
"info": "green",
|
|
||||||
"warn": "yellow",
|
|
||||||
"error": "red",
|
|
||||||
"fatal": "red"
|
|
||||||
},
|
|
||||||
|
|
||||||
"customPattern": "{color:gray}{pretty-timestamp}{reset} {color:tagColor}[{tag}]{reset} {color:contextColor}({context}){reset} {data}",
|
|
||||||
"customColors": {
|
|
||||||
"GET": "green",
|
|
||||||
"POST": "blue",
|
|
||||||
"PUT": "yellow",
|
|
||||||
"DELETE": "red",
|
|
||||||
"PATCH": "cyan",
|
|
||||||
"HEAD": "magenta",
|
|
||||||
"OPTIONS": "white",
|
|
||||||
"TRACE": "gray"
|
|
||||||
},
|
|
||||||
|
|
||||||
"prettyPrint": true
|
|
||||||
}
|
|
|
@ -18,7 +18,7 @@
|
||||||
"typescript": "^5.8.3"
|
"typescript": "^5.8.3"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@atums/echo": "^1.0.3",
|
"@creations.works/logger": "^1.0.3",
|
||||||
"marked": "^15.0.7"
|
"marked": "^15.0.7"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -995,37 +995,3 @@ ul {
|
||||||
height: 16px;
|
height: 16px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* timezone display */
|
|
||||||
|
|
||||||
.timezone-wrapper {
|
|
||||||
position: fixed;
|
|
||||||
top: 1rem;
|
|
||||||
right: 1rem;
|
|
||||||
background-color: var(--card-bg);
|
|
||||||
color: var(--text-color);
|
|
||||||
font-size: 0.9rem;
|
|
||||||
padding: 0.4rem 0.8rem;
|
|
||||||
border-radius: 6px;
|
|
||||||
border: 1px solid var(--border-color);
|
|
||||||
box-shadow: 0 0 8px rgba(0, 0, 0, 0.2);
|
|
||||||
z-index: 100;
|
|
||||||
user-select: none;
|
|
||||||
opacity: 0.85;
|
|
||||||
transition: opacity 0.2s ease;
|
|
||||||
}
|
|
||||||
|
|
||||||
.timezone-wrapper:hover {
|
|
||||||
opacity: 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
.timezone-label {
|
|
||||||
color: var(--text-muted);
|
|
||||||
margin-right: 0.4rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media (max-width: 600px) {
|
|
||||||
.timezone-label {
|
|
||||||
display: none;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
|
@ -3,7 +3,6 @@ const userId = head?.dataset.userId;
|
||||||
const activityProgressMap = new Map();
|
const activityProgressMap = new Map();
|
||||||
|
|
||||||
const reviewURL = head?.dataset.reviewDb;
|
const reviewURL = head?.dataset.reviewDb;
|
||||||
const timezoneApiUrl = head?.dataset.timezoneApi;
|
|
||||||
let instanceUri = head?.dataset.instanceUri;
|
let instanceUri = head?.dataset.instanceUri;
|
||||||
let badgeURL = head?.dataset.badgeUrl;
|
let badgeURL = head?.dataset.badgeUrl;
|
||||||
let socket;
|
let socket;
|
||||||
|
@ -210,60 +209,6 @@ async function populateReviews(userId) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function populateTimezone(userId) {
|
|
||||||
if (!userId || !timezoneApiUrl) return;
|
|
||||||
|
|
||||||
let currentTimezone = null;
|
|
||||||
|
|
||||||
async function fetchTimezone() {
|
|
||||||
try {
|
|
||||||
const res = await fetch(
|
|
||||||
`${timezoneApiUrl}/get?id=${encodeURIComponent(userId)}`,
|
|
||||||
);
|
|
||||||
if (!res.ok) throw new Error("Failed to fetch timezone");
|
|
||||||
|
|
||||||
const json = await res.json();
|
|
||||||
if (!json || typeof json.timezone !== "string") return;
|
|
||||||
|
|
||||||
currentTimezone = json.timezone;
|
|
||||||
updateTime();
|
|
||||||
} catch (err) {
|
|
||||||
console.error("Failed to populate timezone", err);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function updateTime() {
|
|
||||||
if (!currentTimezone) return;
|
|
||||||
|
|
||||||
const timezoneEl = document.querySelector(".timezone-value");
|
|
||||||
if (!timezoneEl) return;
|
|
||||||
|
|
||||||
const now = new Date();
|
|
||||||
|
|
||||||
const time24 = now.toLocaleTimeString("en-GB", {
|
|
||||||
timeZone: currentTimezone,
|
|
||||||
hour12: false,
|
|
||||||
hour: "2-digit",
|
|
||||||
minute: "2-digit",
|
|
||||||
second: "2-digit",
|
|
||||||
});
|
|
||||||
|
|
||||||
const time12 = now.toLocaleTimeString("en-US", {
|
|
||||||
timeZone: currentTimezone,
|
|
||||||
hour12: true,
|
|
||||||
hour: "2-digit",
|
|
||||||
minute: "2-digit",
|
|
||||||
second: "2-digit",
|
|
||||||
});
|
|
||||||
|
|
||||||
timezoneEl.textContent = time24;
|
|
||||||
timezoneEl.title = `${time12} (${currentTimezone})`;
|
|
||||||
}
|
|
||||||
|
|
||||||
fetchTimezone();
|
|
||||||
setInterval(updateTime, 1000);
|
|
||||||
}
|
|
||||||
|
|
||||||
function setupReviewScrollObserver(userId) {
|
function setupReviewScrollObserver(userId) {
|
||||||
const sentinel = document.createElement("div");
|
const sentinel = document.createElement("div");
|
||||||
sentinel.className = "review-scroll-sentinel";
|
sentinel.className = "review-scroll-sentinel";
|
||||||
|
@ -648,14 +593,6 @@ async function updatePresence(initialData) {
|
||||||
setupReviewScrollObserver(userId);
|
setupReviewScrollObserver(userId);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (kv.timezone !== "false" && userId && timezoneApiUrl) {
|
|
||||||
populateTimezone(userId);
|
|
||||||
const timezoneEl = document.querySelector(".timezone-value");
|
|
||||||
if (timezoneEl) {
|
|
||||||
timezoneEl.classList.remove("hidden");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const platform = {
|
const platform = {
|
||||||
mobile: data.active_on_discord_mobile,
|
mobile: data.active_on_discord_mobile,
|
||||||
web: data.active_on_discord_web,
|
web: data.active_on_discord_web,
|
||||||
|
@ -818,7 +755,7 @@ function updateClanBadge(data) {
|
||||||
const userInfoInner = document.querySelector(".user-info-inner");
|
const userInfoInner = document.querySelector(".user-info-inner");
|
||||||
if (!userInfoInner) return;
|
if (!userInfoInner) return;
|
||||||
|
|
||||||
const clan = data?.discord_user?.primary_guild;
|
const clan = data?.discord_user?.clan;
|
||||||
if (!clan || !clan.tag || !clan.identity_guild_id || !clan.badge) return;
|
if (!clan || !clan.tag || !clan.identity_guild_id || !clan.badge) return;
|
||||||
|
|
||||||
const existing = userInfoInner.querySelector(".clan-badge");
|
const existing = userInfoInner.querySelector(".clan-badge");
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
import { serverHandler } from "@/server";
|
import { serverHandler } from "@/server";
|
||||||
import { echo } from "@atums/echo";
|
|
||||||
import { verifyRequiredVariables } from "@config/environment";
|
import { verifyRequiredVariables } from "@config/environment";
|
||||||
|
import { logger } from "@creations.works/logger";
|
||||||
|
|
||||||
async function main(): Promise<void> {
|
async function main(): Promise<void> {
|
||||||
verifyRequiredVariables();
|
verifyRequiredVariables();
|
||||||
|
@ -8,7 +8,7 @@ async function main(): Promise<void> {
|
||||||
}
|
}
|
||||||
|
|
||||||
main().catch((error: Error) => {
|
main().catch((error: Error) => {
|
||||||
echo.error({ message: "Error initializing the server", error });
|
logger.error(["Error initializing the server:", error]);
|
||||||
process.exit(1);
|
process.exit(1);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -4,7 +4,6 @@ import {
|
||||||
lanyardConfig,
|
lanyardConfig,
|
||||||
plausibleScript,
|
plausibleScript,
|
||||||
reviewDb,
|
reviewDb,
|
||||||
timezoneAPIUrl,
|
|
||||||
} from "@config/environment";
|
} from "@config/environment";
|
||||||
import { file } from "bun";
|
import { file } from "bun";
|
||||||
|
|
||||||
|
@ -34,10 +33,6 @@ async function handler(request: ExtendedRequest): Promise<Response> {
|
||||||
head.setAttribute("data-review-db", reviewDb.url);
|
head.setAttribute("data-review-db", reviewDb.url);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (timezoneAPIUrl) {
|
|
||||||
head.setAttribute("data-timezone-api", timezoneAPIUrl);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (plausibleScript) {
|
if (plausibleScript) {
|
||||||
head.append(plausibleScript, { html: true });
|
head.append(plausibleScript, { html: true });
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
import { resolve } from "node:path";
|
import { resolve } from "node:path";
|
||||||
import { echo } from "@atums/echo";
|
import { environment, robotstxtPath } from "@config/environment";
|
||||||
import { environment } from "@config/environment";
|
import { logger } from "@creations.works/logger";
|
||||||
import {
|
import {
|
||||||
type BunFile,
|
type BunFile,
|
||||||
FileSystemRouter,
|
FileSystemRouter,
|
||||||
|
@ -43,14 +43,16 @@ class ServerHandler {
|
||||||
`http://127.0.0.1:${server.port}`,
|
`http://127.0.0.1:${server.port}`,
|
||||||
];
|
];
|
||||||
|
|
||||||
echo.info(`Server running at ${accessUrls[0]}`);
|
logger.info(`Server running at ${accessUrls[0]}`);
|
||||||
echo.info(`Access via: ${accessUrls[1]} or ${accessUrls[2]}`);
|
logger.info(`Access via: ${accessUrls[1]} or ${accessUrls[2]}`, {
|
||||||
|
breakLine: true,
|
||||||
|
});
|
||||||
|
|
||||||
this.logRoutes();
|
this.logRoutes();
|
||||||
}
|
}
|
||||||
|
|
||||||
private logRoutes(): void {
|
private logRoutes(): void {
|
||||||
echo.info("Available routes:");
|
logger.info("Available routes:");
|
||||||
|
|
||||||
const sortedRoutes: [string, string][] = Object.entries(
|
const sortedRoutes: [string, string][] = Object.entries(
|
||||||
this.router.routes,
|
this.router.routes,
|
||||||
|
@ -59,7 +61,7 @@ class ServerHandler {
|
||||||
);
|
);
|
||||||
|
|
||||||
for (const [path, filePath] of sortedRoutes) {
|
for (const [path, filePath] of sortedRoutes) {
|
||||||
echo.info(`Route: ${path}, File: ${filePath}`);
|
logger.info(`Route: ${path}, File: ${filePath}`);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -88,14 +90,11 @@ class ServerHandler {
|
||||||
headers: { "Content-Type": contentType },
|
headers: { "Content-Type": contentType },
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
echo.warn(`File not found: ${filePath}`);
|
logger.warn(`File not found: ${filePath}`);
|
||||||
response = new Response("Not Found", { status: 404 });
|
response = new Response("Not Found", { status: 404 });
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
echo.error({
|
logger.error([`Error serving static file: ${pathname}`, error as Error]);
|
||||||
message: `Error serving static file: ${pathname}`,
|
|
||||||
error: error as Error,
|
|
||||||
});
|
|
||||||
response = new Response("Internal Server Error", { status: 500 });
|
response = new Response("Internal Server Error", { status: 500 });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -108,23 +107,16 @@ class ServerHandler {
|
||||||
response: Response,
|
response: Response,
|
||||||
ip: string | undefined,
|
ip: string | undefined,
|
||||||
): void {
|
): void {
|
||||||
const pathname = new URL(request.url).pathname;
|
logger.custom(
|
||||||
|
`[${request.method}]`,
|
||||||
const ignoredStartsWith: string[] = ["/public"];
|
`(${response.status})`,
|
||||||
const ignoredPaths: string[] = ["/favicon.ico"];
|
[
|
||||||
|
request.url,
|
||||||
if (
|
`${(performance.now() - request.startPerf).toFixed(2)}ms`,
|
||||||
ignoredStartsWith.some((prefix) => pathname.startsWith(prefix)) ||
|
ip || "unknown",
|
||||||
ignoredPaths.includes(pathname)
|
],
|
||||||
) {
|
"90",
|
||||||
return;
|
);
|
||||||
}
|
|
||||||
|
|
||||||
echo.custom(`${request.method}`, `${response.status}`, [
|
|
||||||
request.url,
|
|
||||||
`${(performance.now() - request.startPerf).toFixed(2)}ms`,
|
|
||||||
ip || "unknown",
|
|
||||||
]);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private async handleRequest(
|
private async handleRequest(
|
||||||
|
@ -147,21 +139,29 @@ class ServerHandler {
|
||||||
}
|
}
|
||||||
|
|
||||||
const pathname: string = new URL(request.url).pathname;
|
const pathname: string = new URL(request.url).pathname;
|
||||||
|
if (pathname === "/robots.txt" && robotstxtPath) {
|
||||||
|
try {
|
||||||
|
const file: BunFile = Bun.file(robotstxtPath);
|
||||||
|
|
||||||
const baseDir = resolve("public/custom");
|
if (await file.exists()) {
|
||||||
const customPath = resolve(baseDir, pathname.slice(1));
|
const fileContent: ArrayBuffer = await file.arrayBuffer();
|
||||||
|
const contentType: string = file.type || "text/plain";
|
||||||
|
|
||||||
if (!customPath.startsWith(baseDir)) {
|
response = new Response(fileContent, {
|
||||||
return new Response("Forbidden", { status: 403 });
|
headers: { "Content-Type": contentType },
|
||||||
}
|
});
|
||||||
|
} else {
|
||||||
|
logger.warn(`File not found: ${robotstxtPath}`);
|
||||||
|
response = new Response("Not Found", { status: 404 });
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
logger.error([
|
||||||
|
`Error serving robots.txt: ${robotstxtPath}`,
|
||||||
|
error as Error,
|
||||||
|
]);
|
||||||
|
response = new Response("Internal Server Error", { status: 500 });
|
||||||
|
}
|
||||||
|
|
||||||
const customFile = Bun.file(customPath);
|
|
||||||
if (await customFile.exists()) {
|
|
||||||
const content = await customFile.arrayBuffer();
|
|
||||||
const type = customFile.type || "application/octet-stream";
|
|
||||||
response = new Response(content, {
|
|
||||||
headers: { "Content-Type": type },
|
|
||||||
});
|
|
||||||
this.logRequest(extendedRequest, response, ip);
|
this.logRequest(extendedRequest, response, ip);
|
||||||
return response;
|
return response;
|
||||||
}
|
}
|
||||||
|
@ -269,10 +269,7 @@ class ServerHandler {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} catch (error: unknown) {
|
} catch (error: unknown) {
|
||||||
echo.error({
|
logger.error([`Error handling route ${request.url}:`, error as Error]);
|
||||||
message: `Error handling route ${request.url}`,
|
|
||||||
error: error,
|
|
||||||
});
|
|
||||||
|
|
||||||
response = Response.json(
|
response = Response.json(
|
||||||
{
|
{
|
||||||
|
@ -294,11 +291,9 @@ class ServerHandler {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
this.logRequest(extendedRequest, response, ip);
|
|
||||||
return response;
|
return response;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const serverHandler: ServerHandler = new ServerHandler(
|
const serverHandler: ServerHandler = new ServerHandler(
|
||||||
environment.port,
|
environment.port,
|
||||||
environment.host,
|
environment.host,
|
||||||
|
|
|
@ -1,73 +1,74 @@
|
||||||
<!doctype html>
|
<!doctype html>
|
||||||
<html lang="en">
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8" />
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||||||
|
<meta name="description" content="View real-time Discord presence, activities, and badges with open-source integration." />
|
||||||
|
<meta name="color-scheme" content="dark" />
|
||||||
|
|
||||||
<head>
|
<title>Discord Presence</title>
|
||||||
<meta charset="UTF-8" />
|
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
|
||||||
<meta name="description"
|
|
||||||
content="View real-time Discord presence, activities, and badges with open-source integration." />
|
|
||||||
<meta name="color-scheme" content="dark" />
|
|
||||||
|
|
||||||
<title>Discord Presence</title>
|
<link rel="icon" id="site-icon" type="image/png" href="/public/assets/favicon.png" />
|
||||||
|
<link rel="stylesheet" href="/public/css/index.css" />
|
||||||
|
<link rel="stylesheet" href="/public/css/root.css" />
|
||||||
|
</head>
|
||||||
|
|
||||||
<link rel="icon" id="site-icon" type="image/png" href="/public/assets/favicon.png" />
|
<body>
|
||||||
<link rel="stylesheet" href="/public/css/index.css" />
|
<div id="loading-overlay" role="status" aria-live="polite">
|
||||||
<link rel="stylesheet" href="/public/css/root.css" />
|
<div class="loading-spinner"></div>
|
||||||
</head>
|
|
||||||
|
|
||||||
<body>
|
|
||||||
<div id="loading-overlay" role="status" aria-live="polite">
|
|
||||||
<div class="loading-spinner"></div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<header>
|
|
||||||
<a href="https://git.creations.works/creations/profilePage" target="_blank" rel="noopener noreferrer"
|
|
||||||
title="View source code on Forgejo">
|
|
||||||
<img class="open-source-logo" src="/public/assets/forgejo_logo.svg" alt="Forgejo open-source logo"
|
|
||||||
style="opacity: 0.5" loading="lazy" />
|
|
||||||
</a>
|
|
||||||
|
|
||||||
<div class="timezone-wrapper" id="timezone-wrapper" aria-label="Timezone Information">
|
|
||||||
<span class="timezone-label">Users Time:</span>
|
|
||||||
<span class="timezone-value"></span>
|
|
||||||
</div>
|
</div>
|
||||||
</header>
|
|
||||||
|
|
||||||
<main>
|
<header>
|
||||||
<section class="user-card">
|
<a
|
||||||
<div class="avatar-status-wrapper">
|
href="https://git.creations.works/creations/profilePage"
|
||||||
<div class="avatar-wrapper">
|
target="_blank"
|
||||||
<img class="avatar hidden" />
|
rel="noopener noreferrer"
|
||||||
<img class="decoration hidden" />
|
title="View source code on Forgejo"
|
||||||
<div class="status-indicator offline hidden"></div>
|
>
|
||||||
</div>
|
<img
|
||||||
<div class="user-info">
|
class="open-source-logo"
|
||||||
<div class="user-info-inner">
|
src="/public/assets/forgejo_logo.svg"
|
||||||
<h1 class="username"></h1>
|
alt="Forgejo open-source logo"
|
||||||
|
style="opacity: 0.5"
|
||||||
|
loading="lazy"
|
||||||
|
/>
|
||||||
|
</a>
|
||||||
|
</header>
|
||||||
|
|
||||||
|
<main>
|
||||||
|
<section class="user-card">
|
||||||
|
<div class="avatar-status-wrapper">
|
||||||
|
<div class="avatar-wrapper">
|
||||||
|
<img class="avatar hidden"/>
|
||||||
|
<img class="decoration hidden"/>
|
||||||
|
<div class="status-indicator offline hidden"></div>
|
||||||
|
</div>
|
||||||
|
<div class="user-info">
|
||||||
|
<div class="user-info-inner">
|
||||||
|
<h1 class="username"></h1>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</section>
|
||||||
|
|
||||||
|
<section id="badges" class="badges hidden" aria-label="User Badges"></section>
|
||||||
|
|
||||||
|
<section aria-label="Discord Activities" class="activities-section">
|
||||||
|
<h2 class="activity-block-header hidden">Activities</h2>
|
||||||
|
<ul class="activities"></ul>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section class="readme hidden" aria-label="Profile README">
|
||||||
|
<div class="markdown-body"></div>
|
||||||
|
</section>
|
||||||
|
</main>
|
||||||
|
|
||||||
|
<section class="reviews hidden" aria-label="User Reviews">
|
||||||
|
<h2>User Reviews</h2>
|
||||||
|
<ul class="reviews-list">
|
||||||
|
</ul>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
<section id="badges" class="badges hidden" aria-label="User Badges"></section>
|
<script src="/public/js/index.js" type="module"></script>
|
||||||
|
</body>
|
||||||
<section aria-label="Discord Activities" class="activities-section">
|
|
||||||
<h2 class="activity-block-header hidden">Activities</h2>
|
|
||||||
<ul class="activities"></ul>
|
|
||||||
</section>
|
|
||||||
|
|
||||||
<section class="readme hidden" aria-label="Profile README">
|
|
||||||
<div class="markdown-body"></div>
|
|
||||||
</section>
|
|
||||||
</main>
|
|
||||||
|
|
||||||
<section class="reviews hidden" aria-label="User Reviews">
|
|
||||||
<h2>User Reviews</h2>
|
|
||||||
<ul class="reviews-list">
|
|
||||||
</ul>
|
|
||||||
</section>
|
|
||||||
|
|
||||||
<script src="/public/js/index.js" type="module"></script>
|
|
||||||
</body>
|
|
||||||
|
|
||||||
</html>
|
</html>
|
|
@ -1,27 +1,27 @@
|
||||||
import { echo } from "@atums/echo";
|
import { logger } from "@creations.works/logger";
|
||||||
import type { ServerWebSocket } from "bun";
|
import type { ServerWebSocket } from "bun";
|
||||||
|
|
||||||
class WebSocketHandler {
|
class WebSocketHandler {
|
||||||
public handleMessage(ws: ServerWebSocket, message: string): void {
|
public handleMessage(ws: ServerWebSocket, message: string): void {
|
||||||
echo.info(`WebSocket received: ${message}`);
|
logger.info(`WebSocket received: ${message}`);
|
||||||
try {
|
try {
|
||||||
ws.send(`You said: ${message}`);
|
ws.send(`You said: ${message}`);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
echo.error({ message: "WebSocket send error", error: error });
|
logger.error(["WebSocket send error", error as Error]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public handleOpen(ws: ServerWebSocket): void {
|
public handleOpen(ws: ServerWebSocket): void {
|
||||||
echo.info("WebSocket connection opened.");
|
logger.info("WebSocket connection opened.");
|
||||||
try {
|
try {
|
||||||
ws.send("Welcome to the WebSocket server!");
|
ws.send("Welcome to the WebSocket server!");
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
echo.error({ message: "WebSocket send error", error: error });
|
logger.error(["WebSocket send error", error as Error]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public handleClose(ws: ServerWebSocket, code: number, reason: string): void {
|
public handleClose(ws: ServerWebSocket, code: number, reason: string): void {
|
||||||
echo.warn(`WebSocket closed with code ${code}, reason: ${reason}`);
|
logger.warn(`WebSocket closed with code ${code}, reason: ${reason}`);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue