experimental media domain + temp allow node while bun has node:http issues
This commit is contained in:
parent
fb3096ed03
commit
81afe50526
7 changed files with 123 additions and 29 deletions
BIN
bun.lockb
BIN
bun.lockb
Binary file not shown.
|
@ -6,6 +6,7 @@ export default class Domain {
|
|||
return this.HasHttps ? "Yes" : "No";
|
||||
}
|
||||
public Domain: string = "";
|
||||
public MediaDomain: string = "";
|
||||
public Active: boolean = false;
|
||||
public get ActiveString() {
|
||||
return this.Active ? "Yes" : "No";
|
||||
|
|
108
index.ts
108
index.ts
|
@ -1,7 +1,10 @@
|
|||
if (!process.versions.bun) {
|
||||
console.log("EUS is only designed to run on Bun, sorry!");
|
||||
process.exit(1);
|
||||
}
|
||||
// HSNOTE: It is designed to be used with bun but there is an issue with
|
||||
// bun's implementation of node:http so i've temp allowed this to
|
||||
// run with node again. very disapointed.
|
||||
// if (!process.versions.bun) {
|
||||
// console.log("EUS is only designed to run on Bun, sorry!");
|
||||
// process.exit(1);
|
||||
// }
|
||||
|
||||
import Fastify, { type FastifyReply, type FastifyRequest } from "fastify";
|
||||
import FastifyFormBody from "@fastify/formbody";
|
||||
|
@ -9,6 +12,7 @@ import FastifyMultipart from "@fastify/multipart";
|
|||
import FastifyCookie from "@fastify/cookie";
|
||||
import FastifyView from "@fastify/view";
|
||||
import FastifyStatic from "@fastify/static";
|
||||
//import FastifySend from "@fastify/send";
|
||||
import Config from "./objects/Config";
|
||||
import EJS from "ejs";
|
||||
import { Console } from "hsconsole";
|
||||
|
@ -27,6 +31,7 @@ import HeaderUtility from "./utilities/HeaderUtility";
|
|||
import { createReadStream } from "fs";
|
||||
import ApiController from "./controllers/ApiController";
|
||||
import UploadController from "./controllers/UploadController";
|
||||
import DomainService from "./services/DomainService";
|
||||
|
||||
Console.customHeader(`EUS server started at ${new Date()}`);
|
||||
|
||||
|
@ -82,32 +87,40 @@ fastify.addHook("preHandler", (req, res, done) => {
|
|||
HeaderUtility.AddBakedHeaders(res);
|
||||
return done();
|
||||
} else {
|
||||
const urlParts = req.url.split("/");
|
||||
if (urlParts.length === 2 && urlParts[1].length <= 16) {
|
||||
let media = hashLookupCache.get(urlParts[1]) ?? null;
|
||||
if (!media) {
|
||||
media = await MediaService.GetByTag(urlParts[1]);
|
||||
if (media) {
|
||||
hashLookupCache.set(urlParts[1], media);
|
||||
const domain = await DomainService.LoadDomainByHost(req.headers.host ?? "");
|
||||
if (domain) {
|
||||
const urlParts = req.url.split("/");
|
||||
if (urlParts.length === 2 && urlParts[1].length <= 16) {
|
||||
let media = hashLookupCache.get(urlParts[1]) ?? null;
|
||||
if (!media) {
|
||||
media = await MediaService.GetByTag(urlParts[1]);
|
||||
if (media) {
|
||||
hashLookupCache.set(urlParts[1], media);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (media) {
|
||||
// @ts-ignore
|
||||
req.logType = cyan("MEDIA");
|
||||
const fileStore = HashFS.GetHashFSInstance("images");
|
||||
const readStream = createReadStream(join(fileStore.path, fileStore.GetRelativePath(media.Hash)));
|
||||
res.raw.writeHead(200, HeaderUtility.CombineHeaders({
|
||||
"content-type": media.MediaType,
|
||||
"content-length": media.FileSize,
|
||||
}));
|
||||
readStream.pipe(res.raw);
|
||||
if (media) {
|
||||
// @ts-ignore
|
||||
/*req.logType = "REDIRECT_TO_MEDIA";
|
||||
res.redirect(`${domain.HasHttps ? "https" : "http"}://${domain.MediaDomain}/${media.MediaTag}`);
|
||||
return;*/
|
||||
|
||||
printReqInfo(req, res);
|
||||
return;
|
||||
// @ts-ignore
|
||||
req.logType = cyan("MEDIA");
|
||||
const fileStore = HashFS.GetHashFSInstance("images");
|
||||
const readStream = createReadStream(join(fileStore.path, fileStore.GetRelativePath(media.Hash)));
|
||||
res.raw.writeHead(200, HeaderUtility.CombineHeaders({
|
||||
"content-type": media.MediaType,
|
||||
"content-length": media.FileSize,
|
||||
}));
|
||||
readStream.pipe(res.raw);
|
||||
|
||||
printReqInfo(req, res);
|
||||
return;
|
||||
}
|
||||
} else {
|
||||
HeaderUtility.AddBakedHeaders(res);
|
||||
}
|
||||
} else {
|
||||
HeaderUtility.AddBakedHeaders(res);
|
||||
}
|
||||
|
||||
// @ts-ignore
|
||||
|
@ -119,7 +132,10 @@ fastify.addHook("preHandler", (req, res, done) => {
|
|||
});
|
||||
|
||||
fastify.addHook("onSend", (req, res, _payload, done) => {
|
||||
printReqInfo(req, res);
|
||||
// @ts-ignore
|
||||
if (req.logType !== "REDIRECT_TO_MEDIA") {
|
||||
printReqInfo(req, res);
|
||||
}
|
||||
|
||||
done();
|
||||
});
|
||||
|
@ -158,4 +174,40 @@ fastify.listen({ port: Config.hosts.webPort, host: Config.hosts.webHost }, (err,
|
|||
|
||||
Console.printWarn(`[ ${green("MAIN")} ] The EUS rewrite is currently beta software, use at your own risk!`);
|
||||
Console.printInfo(`[ ${green("MAIN")} ] Listening at ${address.replace("0.0.0.0", "localhost").replace("127.0.0.1", "localhost")}`);
|
||||
});
|
||||
});
|
||||
|
||||
/*const mediaServer = createServer(async (req, res) => {
|
||||
const startTime = Date.now();
|
||||
const urlParts = req.url?.split("/") ?? "";
|
||||
if (urlParts.length === 2 && urlParts[1].length <= 16) {
|
||||
let media = hashLookupCache.get(urlParts[1]) ?? null;
|
||||
if (!media) {
|
||||
media = await MediaService.GetByTag(urlParts[1]);
|
||||
if (media) {
|
||||
hashLookupCache.set(urlParts[1], media);
|
||||
}
|
||||
}
|
||||
|
||||
if (media) {
|
||||
const fileStore = HashFS.GetHashFSInstance("images");
|
||||
|
||||
const { statusCode, headers, stream } = await FastifySend(req, fileStore.GetRelativePath(media.Hash), { root: fileStore.path });
|
||||
headers["content-type"] = media.MediaType;
|
||||
res.writeHead(statusCode, HeaderUtility.CombineHeaders(headers));
|
||||
stream.pipe(res);
|
||||
|
||||
Console.printInfo(`[ ${cyan("MEDIA")} ] [ ${req.method?.toUpperCase()} ] [ ${ConsoleUtility.StatusColor(res.statusCode)} ] [ ${blue(`${Date.now() - startTime}ms`)} ] > ${req.url}`);
|
||||
return;
|
||||
} else {
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
res.statusCode = 404;
|
||||
res.write("EUS Media Server");
|
||||
return res.end();
|
||||
});
|
||||
|
||||
mediaServer.listen(Config.hosts.webPort + 1, () => {
|
||||
Console.printInfo(`[ ${cyan("MEDIA")} ] Listening at http://localhost:${(Config.hosts.webPort + 1)}`);
|
||||
});*/
|
|
@ -29,6 +29,7 @@
|
|||
"@fastify/cookie": "^11.0.2",
|
||||
"@fastify/formbody": "^8.0.2",
|
||||
"@fastify/multipart": "^9.0.3",
|
||||
"@fastify/send": "^4.0.0",
|
||||
"@fastify/static": "^8.1.1",
|
||||
"@fastify/view": "^11.0.0",
|
||||
"dyetty": "^1.0.1",
|
||||
|
|
|
@ -50,6 +50,17 @@ export default class DomainRepo {
|
|||
}
|
||||
}
|
||||
|
||||
public static async SelectByMediaDomain(domain: string) {
|
||||
const dbDomain = await Database.Instance.query("SELECT * FROM Domain WHERE MediaDomain = ? AND IsDeleted = 0 LIMIT 1", [domain]);
|
||||
if (dbDomain == null || dbDomain.length === 0) {
|
||||
return null;
|
||||
} else {
|
||||
const domain = new Domain();
|
||||
PopulateDomainFromDB(domain, dbDomain[0]);
|
||||
return domain;
|
||||
}
|
||||
}
|
||||
|
||||
public static async InsertUpdate(domain: Domain) {
|
||||
if (domain.Id === Number.MIN_VALUE) {
|
||||
domain.Id = (await Database.Instance.query("INSERT Domain (UserId, DomainId, FileName, MediaTag, MediaType, Hash, FileSize, CreatedByUserId, CreatedDatetime, LastModifiedByUserId, LastModifiedDatetime, DeletedByUserId, DeletedDatetime, IsDeleted) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?) RETURNING Id;", [
|
||||
|
@ -70,6 +81,7 @@ function PopulateDomainFromDB(domain: Domain, dbDomain: any) {
|
|||
domain.UserId = dbDomain.UserId;
|
||||
domain.HasHttps = dbDomain.HasHttps[0] === 1;
|
||||
domain.Domain = dbDomain.Domain;
|
||||
domain.MediaDomain = dbDomain.MediaDomain;
|
||||
domain.Active = dbDomain.Active[0] === 1;
|
||||
domain.CreatedByUserId = dbDomain.CreatedByUserId;
|
||||
domain.CreatedDatetime = dbDomain.CreatedDatetime;
|
||||
|
|
|
@ -10,4 +10,22 @@ export default abstract class DomainService {
|
|||
throw e;
|
||||
}
|
||||
}
|
||||
|
||||
public static async LoadDomainByHost(host: string) {
|
||||
try {
|
||||
return await DomainRepo.SelectByDomain(host);
|
||||
} catch (e) {
|
||||
Console.printError(`EUS server service error:\n${e}`);
|
||||
throw e;
|
||||
}
|
||||
}
|
||||
|
||||
public static async LoadDomainByMediaHost(host: string) {
|
||||
try {
|
||||
return await DomainRepo.SelectByMediaDomain(host);
|
||||
} catch (e) {
|
||||
Console.printError(`EUS server service error:\n${e}`);
|
||||
throw e;
|
||||
}
|
||||
}
|
||||
}
|
|
@ -10,8 +10,11 @@ export default abstract class HeaderUtility {
|
|||
"Referrer-Policy": "strict-origin-when-cross-origin",
|
||||
"Content-Security-Policy": "block-all-mixed-content;frame-ancestors 'self'",
|
||||
"X-Frame-Options": "SAMEORIGIN",
|
||||
"X-Content-Type-Options": "nosniff"
|
||||
"X-Content-Type-Options": "nosniff",
|
||||
"Connection": "keep-alive",
|
||||
"Keep-Alive": "timeout=5"
|
||||
};
|
||||
public static BakedHeadersKeys = Object.keys(this.BakedHeaders);
|
||||
|
||||
public static AddBakedHeaders(res: FastifyReply) {
|
||||
/*res.header("x-powered-by", "EUS");
|
||||
|
@ -26,6 +29,13 @@ export default abstract class HeaderUtility {
|
|||
res.headers(this.BakedHeaders);
|
||||
}
|
||||
|
||||
public static AddBakedHeadersExpress(res: any) {
|
||||
for (const key of this.BakedHeadersKeys) {
|
||||
// @ts-ignore
|
||||
res.header(key, this.BakedHeaders[key]);
|
||||
}
|
||||
}
|
||||
|
||||
public static CombineHeaders(headers: any) {
|
||||
// for (const header of Object.keys(headers)) {
|
||||
// res.header(header, headers[header]);
|
||||
|
|
Loading…
Add table
Reference in a new issue