2016-12-28 15:49:51 -07:00
|
|
|
/**
|
|
|
|
* Core Server
|
|
|
|
*/
|
|
|
|
|
2023-01-12 21:40:33 -07:00
|
|
|
import cluster from "node:cluster";
|
|
|
|
import * as fs from "node:fs";
|
|
|
|
import * as http from "node:http";
|
|
|
|
import Koa from "koa";
|
|
|
|
import Router from "@koa/router";
|
|
|
|
import mount from "koa-mount";
|
|
|
|
import koaLogger from "koa-logger";
|
|
|
|
import * as slow from "koa-slow";
|
|
|
|
|
|
|
|
import { IsNull } from "typeorm";
|
|
|
|
import config from "@/config/index.js";
|
|
|
|
import Logger from "@/services/logger.js";
|
|
|
|
import { UserProfiles, Users } from "@/models/index.js";
|
|
|
|
import { genIdenticon } from "@/misc/gen-identicon.js";
|
|
|
|
import { createTemp } from "@/misc/create-temp.js";
|
|
|
|
import { publishMainStream } from "@/services/stream.js";
|
|
|
|
import * as Acct from "@/misc/acct.js";
|
|
|
|
import { envOption } from "@/env.js";
|
2023-02-11 18:23:30 -07:00
|
|
|
import megalodon, { MegalodonInterface } from "@calckey/megalodon";
|
2023-01-12 21:40:33 -07:00
|
|
|
import activityPub from "./activitypub.js";
|
|
|
|
import nodeinfo from "./nodeinfo.js";
|
|
|
|
import wellKnown from "./well-known.js";
|
|
|
|
import apiServer from "./api/index.js";
|
|
|
|
import fileServer from "./file/index.js";
|
|
|
|
import proxyServer from "./proxy/index.js";
|
|
|
|
import webServer from "./web/index.js";
|
|
|
|
import { initializeStreamingServer } from "./api/streaming.js";
|
2023-02-10 16:41:19 -07:00
|
|
|
import { koaBody } from "koa-body";
|
2023-01-12 21:40:33 -07:00
|
|
|
|
|
|
|
export const serverLogger = new Logger("server", "gray", false);
|
2017-01-16 16:06:39 -07:00
|
|
|
|
2018-04-12 15:06:18 -06:00
|
|
|
// Init app
|
2018-04-12 09:51:55 -06:00
|
|
|
const app = new Koa();
|
|
|
|
app.proxy = true;
|
2017-11-13 03:58:29 -07:00
|
|
|
|
2023-01-12 21:40:33 -07:00
|
|
|
if (!["production", "test"].includes(process.env.NODE_ENV || "")) {
|
2018-04-19 03:03:46 -06:00
|
|
|
// Logger
|
2023-01-12 21:40:33 -07:00
|
|
|
app.use(
|
|
|
|
koaLogger((str) => {
|
|
|
|
serverLogger.info(str);
|
|
|
|
}),
|
|
|
|
);
|
2018-04-25 20:46:42 -06:00
|
|
|
|
|
|
|
// Delay
|
2021-10-08 06:24:05 -06:00
|
|
|
if (envOption.slow) {
|
2023-01-12 21:40:33 -07:00
|
|
|
app.use(
|
|
|
|
slow({
|
|
|
|
delay: 3000,
|
|
|
|
}),
|
|
|
|
);
|
2019-02-03 18:03:49 -07:00
|
|
|
}
|
2018-04-19 03:03:46 -06:00
|
|
|
}
|
|
|
|
|
2018-04-12 09:51:55 -06:00
|
|
|
// HSTS
|
|
|
|
// 6months (15552000sec)
|
2023-01-12 21:40:33 -07:00
|
|
|
if (config.url.startsWith("https") && !config.disableHsts) {
|
2018-04-12 16:34:27 -06:00
|
|
|
app.use(async (ctx, next) => {
|
2023-01-12 21:40:33 -07:00
|
|
|
ctx.set("strict-transport-security", "max-age=15552000; preload");
|
2018-04-12 16:34:27 -06:00
|
|
|
await next();
|
2018-04-11 14:54:54 -06:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-01-12 21:40:33 -07:00
|
|
|
app.use(mount("/api", apiServer));
|
|
|
|
app.use(mount("/files", fileServer));
|
|
|
|
app.use(mount("/proxy", proxyServer));
|
2018-04-12 15:17:14 -06:00
|
|
|
|
2018-04-12 09:51:55 -06:00
|
|
|
// Init router
|
|
|
|
const router = new Router();
|
2023-02-10 16:33:01 -07:00
|
|
|
const mastoRouter = new Router();
|
|
|
|
|
2023-02-11 14:16:45 -07:00
|
|
|
mastoRouter.use(
|
|
|
|
koaBody({
|
|
|
|
urlencoded: true,
|
|
|
|
}),
|
|
|
|
);
|
2017-01-07 07:57:45 -07:00
|
|
|
|
2023-02-13 07:12:04 -07:00
|
|
|
mastoRouter.use(async (ctx, next) => {
|
|
|
|
if (ctx.request.query) {
|
|
|
|
if (!ctx.request.body || Object.keys(ctx.request.body).length === 0) {
|
2023-02-13 12:17:07 -07:00
|
|
|
ctx.request.body = ctx.request.query;
|
2023-02-13 07:12:04 -07:00
|
|
|
} else {
|
2023-02-13 12:17:07 -07:00
|
|
|
ctx.request.body = { ...ctx.request.body, ...ctx.request.query };
|
2023-02-13 07:12:04 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
await next();
|
|
|
|
});
|
|
|
|
|
2018-04-12 09:51:55 -06:00
|
|
|
// Routing
|
|
|
|
router.use(activityPub.routes());
|
2019-02-05 01:42:55 -07:00
|
|
|
router.use(nodeinfo.routes());
|
|
|
|
router.use(wellKnown.routes());
|
2018-04-12 15:17:14 -06:00
|
|
|
|
2023-01-12 21:40:33 -07:00
|
|
|
router.get("/avatar/@:acct", async (ctx) => {
|
2021-11-11 10:02:25 -07:00
|
|
|
const { username, host } = Acct.parse(ctx.params.acct);
|
2021-10-24 06:02:50 -06:00
|
|
|
const user = await Users.findOne({
|
2022-03-26 00:34:00 -06:00
|
|
|
where: {
|
|
|
|
usernameLower: username.toLowerCase(),
|
2023-01-12 21:40:33 -07:00
|
|
|
host: host == null || host === config.host ? IsNull() : host,
|
2022-03-26 00:34:00 -06:00
|
|
|
isSuspended: false,
|
|
|
|
},
|
2023-01-12 21:40:33 -07:00
|
|
|
relations: ["avatar"],
|
2021-10-24 06:02:50 -06:00
|
|
|
});
|
|
|
|
|
|
|
|
if (user) {
|
2022-04-17 06:18:18 -06:00
|
|
|
ctx.redirect(Users.getAvatarUrlSync(user));
|
2021-10-24 06:02:50 -06:00
|
|
|
} else {
|
2023-01-12 21:40:33 -07:00
|
|
|
ctx.redirect("/static-assets/user-unknown.png");
|
2021-10-24 06:02:50 -06:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2023-01-12 21:40:33 -07:00
|
|
|
router.get("/identicon/:x", async (ctx) => {
|
2022-05-25 01:50:22 -06:00
|
|
|
const [temp, cleanup] = await createTemp();
|
2022-01-15 18:45:48 -07:00
|
|
|
await genIdenticon(ctx.params.x, fs.createWriteStream(temp));
|
2023-01-12 21:40:33 -07:00
|
|
|
ctx.set("Content-Type", "image/png");
|
|
|
|
ctx.body = fs.createReadStream(temp).on("close", () => cleanup());
|
2019-04-15 05:37:21 -06:00
|
|
|
});
|
|
|
|
|
2023-01-12 21:40:33 -07:00
|
|
|
router.get("/verify-email/:code", async (ctx) => {
|
2022-03-26 00:34:00 -06:00
|
|
|
const profile = await UserProfiles.findOneBy({
|
2021-12-09 07:58:30 -07:00
|
|
|
emailVerifyCode: ctx.params.code,
|
2019-04-07 06:50:36 -06:00
|
|
|
});
|
2018-11-29 00:23:45 -07:00
|
|
|
|
2019-04-10 00:04:27 -06:00
|
|
|
if (profile != null) {
|
2023-01-12 21:40:33 -07:00
|
|
|
ctx.body = "Verify succeeded!";
|
2018-11-29 00:23:45 -07:00
|
|
|
ctx.status = 200;
|
|
|
|
|
2023-01-12 21:40:33 -07:00
|
|
|
await UserProfiles.update(
|
|
|
|
{ userId: profile.userId },
|
|
|
|
{
|
|
|
|
emailVerified: true,
|
|
|
|
emailVerifyCode: null,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
publishMainStream(
|
|
|
|
profile.userId,
|
|
|
|
"meUpdated",
|
|
|
|
await Users.pack(
|
|
|
|
profile.userId,
|
|
|
|
{ id: profile.userId },
|
|
|
|
{
|
|
|
|
detail: true,
|
|
|
|
includeSecrets: true,
|
|
|
|
},
|
|
|
|
),
|
|
|
|
);
|
2018-11-29 00:23:45 -07:00
|
|
|
} else {
|
|
|
|
ctx.status = 404;
|
|
|
|
}
|
2023-02-07 14:56:07 -07:00
|
|
|
});
|
|
|
|
|
2023-02-10 16:33:01 -07:00
|
|
|
mastoRouter.get("/oauth/authorize", async (ctx) => {
|
2023-02-07 14:56:07 -07:00
|
|
|
const client_id = ctx.request.query.client_id;
|
|
|
|
console.log(ctx.request.req);
|
2023-02-10 16:41:19 -07:00
|
|
|
ctx.redirect(Buffer.from(client_id?.toString() || "", "base64").toString());
|
2023-02-07 14:56:07 -07:00
|
|
|
});
|
|
|
|
|
2023-02-10 16:33:01 -07:00
|
|
|
mastoRouter.post("/oauth/token", async (ctx) => {
|
2023-02-09 15:21:50 -07:00
|
|
|
const body: any = ctx.request.body;
|
2023-02-10 17:17:35 -07:00
|
|
|
let client_id: any = ctx.request.query.client_id;
|
2023-02-10 15:00:15 -07:00
|
|
|
const BASE_URL = `${ctx.request.protocol}://${ctx.request.hostname}`;
|
2023-02-07 14:56:07 -07:00
|
|
|
const generator = (megalodon as any).default;
|
2023-02-10 16:41:19 -07:00
|
|
|
const client = generator("misskey", BASE_URL, null) as MegalodonInterface;
|
2023-02-10 17:07:44 -07:00
|
|
|
let m = null;
|
|
|
|
if (body.code) {
|
|
|
|
m = body.code.match(/^[a-zA-Z0-9-]+/);
|
|
|
|
if (!m.length) {
|
|
|
|
ctx.body = { error: "Invalid code" };
|
|
|
|
return;
|
|
|
|
}
|
2023-02-11 14:16:45 -07:00
|
|
|
}
|
2023-02-10 17:17:35 -07:00
|
|
|
if (client_id instanceof Array) {
|
2023-02-11 14:16:45 -07:00
|
|
|
client_id = client_id.toString();
|
2023-02-10 17:17:35 -07:00
|
|
|
} else if (!client_id) {
|
|
|
|
client_id = null;
|
|
|
|
}
|
2023-02-07 14:56:07 -07:00
|
|
|
try {
|
2023-02-10 16:41:19 -07:00
|
|
|
const atData = await client.fetchAccessToken(
|
2023-02-10 17:17:35 -07:00
|
|
|
client_id,
|
2023-02-10 16:41:19 -07:00
|
|
|
body.client_secret,
|
2023-02-11 14:16:45 -07:00
|
|
|
m ? m[0] : "",
|
2023-02-10 16:41:19 -07:00
|
|
|
);
|
2023-02-09 15:21:50 -07:00
|
|
|
ctx.body = {
|
|
|
|
access_token: atData.accessToken,
|
2023-02-10 16:41:19 -07:00
|
|
|
token_type: "Bearer",
|
|
|
|
scope: "read write follow",
|
2023-02-10 17:51:14 -07:00
|
|
|
created_at: Math.floor(new Date().getTime() / 1000),
|
2023-02-09 15:21:50 -07:00
|
|
|
};
|
2023-02-07 14:56:07 -07:00
|
|
|
} catch (err: any) {
|
|
|
|
console.error(err);
|
|
|
|
ctx.status = 401;
|
|
|
|
ctx.body = err.response.data;
|
|
|
|
}
|
2018-11-29 00:23:45 -07:00
|
|
|
});
|
|
|
|
|
2018-04-12 09:51:55 -06:00
|
|
|
// Register router
|
2023-02-10 16:33:01 -07:00
|
|
|
app.use(mastoRouter.routes());
|
2023-02-10 16:51:45 -07:00
|
|
|
app.use(router.routes());
|
2016-12-28 15:49:51 -07:00
|
|
|
|
2022-02-26 19:07:39 -07:00
|
|
|
app.use(mount(webServer));
|
2018-04-12 16:34:27 -06:00
|
|
|
|
2018-03-28 10:20:40 -06:00
|
|
|
function createServer() {
|
2022-03-08 07:23:18 -07:00
|
|
|
return http.createServer(app.callback());
|
2018-03-28 10:20:40 -06:00
|
|
|
}
|
2016-12-28 15:49:51 -07:00
|
|
|
|
2019-01-22 21:35:22 -07:00
|
|
|
// For testing
|
|
|
|
export const startServer = () => {
|
|
|
|
const server = createServer();
|
|
|
|
|
2022-02-26 19:07:39 -07:00
|
|
|
initializeStreamingServer(server);
|
2019-01-22 21:35:22 -07:00
|
|
|
|
|
|
|
server.listen(config.port);
|
|
|
|
|
|
|
|
return server;
|
|
|
|
};
|
|
|
|
|
2023-01-12 21:40:33 -07:00
|
|
|
export default () =>
|
|
|
|
new Promise((resolve) => {
|
|
|
|
const server = createServer();
|
|
|
|
|
|
|
|
initializeStreamingServer(server);
|
|
|
|
|
|
|
|
server.on("error", (e) => {
|
|
|
|
switch ((e as any).code) {
|
|
|
|
case "EACCES":
|
|
|
|
serverLogger.error(
|
|
|
|
`You do not have permission to listen on port ${config.port}.`,
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
case "EADDRINUSE":
|
|
|
|
serverLogger.error(
|
|
|
|
`Port ${config.port} is already in use by another process.`,
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
serverLogger.error(e);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cluster.isWorker) {
|
|
|
|
process.send!("listenFailed");
|
|
|
|
} else {
|
|
|
|
// disableClustering
|
|
|
|
process.exit(1);
|
|
|
|
}
|
|
|
|
});
|
2017-01-16 15:51:27 -07:00
|
|
|
|
2023-01-12 21:40:33 -07:00
|
|
|
// @ts-ignore
|
|
|
|
server.listen(config.port, resolve);
|
2022-05-18 20:49:07 -06:00
|
|
|
});
|