2020-03-17 00:04:09 +07:00
|
|
|
import { field, logger } from "@coder/logger"
|
2020-10-21 06:05:58 +07:00
|
|
|
import { Router } from "express"
|
|
|
|
import { promises as fs } from "fs"
|
2020-03-17 00:04:09 +07:00
|
|
|
import * as path from "path"
|
|
|
|
import { Readable } from "stream"
|
|
|
|
import * as tarFs from "tar-fs"
|
|
|
|
import * as zlib from "zlib"
|
2020-10-21 06:05:58 +07:00
|
|
|
import { HttpCode, HttpError } from "../../common/http"
|
|
|
|
import { rootPath } from "../constants"
|
2020-11-19 06:15:14 +07:00
|
|
|
import { authenticated, ensureAuthenticated, replaceTemplates } from "../http"
|
2020-10-21 06:05:58 +07:00
|
|
|
import { getMediaMime, pathToFsPath } from "../util"
|
2020-03-03 01:43:02 +07:00
|
|
|
|
2020-10-21 06:05:58 +07:00
|
|
|
export const router = Router()
|
2020-03-17 00:04:09 +07:00
|
|
|
|
2020-10-21 06:05:58 +07:00
|
|
|
// The commit is for caching.
|
|
|
|
router.get("/(:commit)(/*)?", async (req, res) => {
|
2020-11-19 06:15:14 +07:00
|
|
|
// Used by VS Code to load extensions into the web worker.
|
|
|
|
const tar = Array.isArray(req.query.tar) ? req.query.tar[0] : req.query.tar
|
|
|
|
if (typeof tar === "string") {
|
|
|
|
ensureAuthenticated(req)
|
|
|
|
let stream: Readable = tarFs.pack(pathToFsPath(tar))
|
|
|
|
if (req.headers["accept-encoding"] && req.headers["accept-encoding"].includes("gzip")) {
|
|
|
|
logger.debug("gzipping tar", field("path", tar))
|
|
|
|
const compress = zlib.createGzip()
|
|
|
|
stream.pipe(compress)
|
|
|
|
stream.on("error", (error) => compress.destroy(error))
|
|
|
|
stream.on("close", () => compress.end())
|
|
|
|
stream = compress
|
|
|
|
res.header("content-encoding", "gzip")
|
|
|
|
}
|
|
|
|
res.set("Content-Type", "application/x-tar")
|
|
|
|
stream.on("close", () => res.end())
|
|
|
|
return stream.pipe(res)
|
|
|
|
}
|
|
|
|
|
|
|
|
// If not a tar use the remainder of the path to load the resource.
|
2020-10-21 06:05:58 +07:00
|
|
|
if (!req.params[0]) {
|
|
|
|
throw new HttpError("Not Found", HttpCode.NotFound)
|
2020-03-03 01:43:02 +07:00
|
|
|
}
|
|
|
|
|
2020-10-21 06:05:58 +07:00
|
|
|
const resourcePath = path.resolve(req.params[0])
|
2020-03-03 01:43:02 +07:00
|
|
|
|
2020-10-21 06:05:58 +07:00
|
|
|
// Make sure it's in code-server if you aren't authenticated. This lets
|
|
|
|
// unauthenticated users load the login assets.
|
|
|
|
if (!resourcePath.startsWith(rootPath) && !authenticated(req)) {
|
|
|
|
throw new HttpError("Unauthorized", HttpCode.Unauthorized)
|
|
|
|
}
|
2020-07-30 03:02:14 +07:00
|
|
|
|
2020-10-21 06:05:58 +07:00
|
|
|
// Don't cache during development. - can also be used if you want to make a
|
|
|
|
// static request without caching.
|
|
|
|
if (req.params.commit !== "development" && req.params.commit !== "-") {
|
|
|
|
res.header("Cache-Control", "public, max-age=31536000")
|
2020-03-03 01:43:02 +07:00
|
|
|
}
|
2020-03-17 00:04:09 +07:00
|
|
|
|
2020-10-21 06:05:58 +07:00
|
|
|
res.set("Content-Type", getMediaMime(resourcePath))
|
|
|
|
|
|
|
|
if (resourcePath.endsWith("manifest.json")) {
|
|
|
|
const content = await fs.readFile(resourcePath, "utf8")
|
|
|
|
return res.send(replaceTemplates(req, content))
|
|
|
|
}
|
|
|
|
|
|
|
|
const content = await fs.readFile(resourcePath)
|
|
|
|
return res.send(content)
|
|
|
|
})
|