Files
LingChair/server/main.ts
CrescentLeaf 8b3b32422f refactor: 使用表单进行文件上传!
* 可以上传大文件啦
* 最大限制 2GB
* 后端方法重置
2025-11-01 01:12:50 +08:00

149 lines
4.6 KiB
TypeScript

import ApiManager from "./api/ApiManager.ts"
// @ts-types="npm:@types/express"
import express from 'express'
import * as SocketIo from 'socket.io'
import HttpServerLike from "./typedef/HttpServerLike.ts"
import config from './config.ts'
import http from 'node:http'
import https from 'node:https'
import readline from 'node:readline'
import process from "node:process"
import chalk from "chalk"
import child_process from "node:child_process"
import FileManager from "./data/FileManager.ts"
import TokenManager from "./api/TokenManager.ts"
import UserChatLinker from "./data/UserChatLinker.ts"
import path from "node:path"
import cookieParser from 'cookie-parser'
import fs from 'node:fs/promises'
// @ts-types="npm:@types/express-fileupload"
import fileUpload from 'express-fileupload'
const app = express()
app.use('/', express.static(config.data_path + '/page_compiled'))
app.use(cookieParser())
app.get('/uploaded_files/:hash', (req, res) => {
const hash = req.params.hash as string
if (hash == null) {
res.status(404).send({
msg: "404 Not Found",
})
return
}
const file = FileManager.findByHash(hash)
if (file == null) {
res.status(404).send({
msg: "404 Not Found",
})
return
}
if (file.getChatId() != null) {
const userToken = TokenManager.decode(req.headers.token || req.cookies.token)
if (!TokenManager.checkToken(userToken, req.headers['device-id'] || req.cookies.device_id)) {
res.status(401).send({
msg: "401 UnAuthorized",
})
return
}
if (!UserChatLinker.checkUserIsLinkedToChat(userToken.author, file.getChatId() as string)) {
res.status(403).send({
msg: "403 Forbidden",
})
return
}
}
const fileName = encodeURIComponent(file!.getName()?.replaceAll('"', ''))
res.setHeader('Content-Disposition', `inline; filename="${fileName}"`)
res.setHeader('Content-Type', file!.getMime())
res.sendFile(path.resolve(file!.getFilePath()))
file.updateLastUsedTime()
})
await fs.mkdir(config.data_path + '/upload_cache', { recursive: true })
app.use(fileUpload({
limits: { fileSize: 2 * 1024 * 1024 * 1024 },
useTempFiles: true,
tempFileDir: config.data_path + '/upload_cache',
abortOnLimit: true,
}))
app.post('/upload_file', async (req, res) => {
const userToken = TokenManager.decode(req.headers.token || req.cookies.token)
if (!TokenManager.checkToken(userToken, req.headers['device-id'] || req.cookies.device_id)) {
res.status(401).send({
msg: "401 UnAuthorized",
})
return
}
if (req.body.chat_id && !UserChatLinker.checkUserIsLinkedToChat(userToken.author, req.body.chat_id)) {
res.status(403).send({
msg: "403 Forbidden",
})
return
}
const file = req.files?.file as fileUpload.UploadedFile
if (file?.data == null) {
res.status(400).send({
msg: "No file was found or multiple files were uploaded",
})
return
}
if (req.body.file_name == null) {
res.status(400).send({
msg: "Filename is required",
})
return
}
const hash = (await FileManager.uploadFile(req.body.file_name, await fs.readFile(file.tempFilePath), req.body.chat_id)).getHash()
res.status(200).send({
msg: "success",
data: {
file_hash: hash,
},
})
})
const httpServer: HttpServerLike = (
((config.server.use == 'http') && http.createServer(app)) ||
((config.server.use == 'https') && https.createServer({
...config.server.https,
key: await fs.readFile(config.server.https.key),
cert: await fs.readFile(config.server.https.cert),
}, app)) ||
http.createServer(app)
)
const io = new SocketIo.Server(httpServer, {
transports: ["polling", "websocket", "webtransport"],
})
ApiManager.initServer(httpServer, io)
ApiManager.initEvents()
ApiManager.initAllApis()
httpServer.listen(config.server.listen)
console.log(chalk.green("API & Web 服务已启动"))
function help() {
console.log(chalk.yellow("===== LingChair Server ====="))
console.log(chalk.yellow("b - 重新编译前端"))
}
help()
const rl = readline.createInterface({
input: process.stdin,
output: process.stdout
})
rl.on('line', (text) => {
if (text == 'b') {
console.log(chalk.green("重新编译..."))
child_process.spawnSync("deno", ["task", "build"], {
stdio: [process.stdin, process.stdout, process.stderr]
})
help()
}
})