2020-03-16 18:04:09 +01:00
|
|
|
import { field, logger } from "@coder/logger"
|
2020-03-02 19:43:02 +01:00
|
|
|
import * as http from "http"
|
2020-03-16 18:04:09 +01:00
|
|
|
import * as path from "path"
|
|
|
|
import { Readable } from "stream"
|
|
|
|
import * as tarFs from "tar-fs"
|
|
|
|
import * as zlib from "zlib"
|
2020-03-02 19:43:02 +01:00
|
|
|
import { HttpProvider, HttpResponse, Route } from "../http"
|
2020-05-18 19:57:50 +02:00
|
|
|
import { pathToFsPath } from "../util"
|
2020-03-02 19:43:02 +01:00
|
|
|
|
|
|
|
/**
|
2020-07-29 22:02:14 +02:00
|
|
|
* Static file HTTP provider. Static requests do not require authentication if
|
|
|
|
* the resource is in the application's directory except requests to serve a
|
|
|
|
* directory as a tar which always requires authentication.
|
2020-03-02 19:43:02 +01:00
|
|
|
*/
|
|
|
|
export class StaticHttpProvider extends HttpProvider {
|
|
|
|
public async handleRequest(route: Route, request: http.IncomingMessage): Promise<HttpResponse> {
|
|
|
|
this.ensureMethod(request)
|
2020-03-16 18:04:09 +01:00
|
|
|
|
|
|
|
if (typeof route.query.tar === "string") {
|
|
|
|
this.ensureAuthenticated(request)
|
2020-05-18 19:57:50 +02:00
|
|
|
return this.getTarredResource(request, pathToFsPath(route.query.tar))
|
2020-03-16 18:04:09 +01:00
|
|
|
}
|
|
|
|
|
2020-07-29 22:02:14 +02:00
|
|
|
const response = await this.getReplacedResource(request, route)
|
2020-03-02 19:43:02 +01:00
|
|
|
if (!this.isDev) {
|
|
|
|
response.cache = true
|
|
|
|
}
|
|
|
|
return response
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return a resource with variables replaced where necessary.
|
|
|
|
*/
|
2020-07-29 22:02:14 +02:00
|
|
|
protected async getReplacedResource(request: http.IncomingMessage, route: Route): Promise<HttpResponse> {
|
2020-03-02 19:43:02 +01:00
|
|
|
// The first part is always the commit (for caching purposes).
|
|
|
|
const split = route.requestPath.split("/").slice(1)
|
|
|
|
|
2020-07-29 22:02:14 +02:00
|
|
|
const resourcePath = path.resolve("/", ...split)
|
|
|
|
|
|
|
|
// Make sure it's in code-server or a plugin.
|
|
|
|
const validPaths = [this.rootPath, process.env.PLUGIN_DIR]
|
2020-08-05 19:45:50 +02:00
|
|
|
if (!validPaths.find((p) => p && resourcePath.startsWith(p))) {
|
2020-07-29 22:02:14 +02:00
|
|
|
this.ensureAuthenticated(request)
|
|
|
|
}
|
|
|
|
|
2020-03-02 19:43:02 +01:00
|
|
|
switch (split[split.length - 1]) {
|
2020-03-03 00:44:16 +01:00
|
|
|
case "manifest.json": {
|
2020-07-29 22:02:14 +02:00
|
|
|
const response = await this.getUtf8Resource(resourcePath)
|
2020-03-02 19:43:02 +01:00
|
|
|
return this.replaceTemplates(route, response)
|
|
|
|
}
|
|
|
|
}
|
2020-07-29 22:02:14 +02:00
|
|
|
return this.getResource(resourcePath)
|
2020-03-02 19:43:02 +01:00
|
|
|
}
|
2020-03-16 18:04:09 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Tar up and stream a directory.
|
|
|
|
*/
|
|
|
|
private async getTarredResource(request: http.IncomingMessage, ...parts: string[]): Promise<HttpResponse> {
|
|
|
|
const filePath = path.join(...parts)
|
|
|
|
let stream: Readable = tarFs.pack(filePath)
|
|
|
|
const headers: http.OutgoingHttpHeaders = {}
|
|
|
|
if (request.headers["accept-encoding"] && request.headers["accept-encoding"].includes("gzip")) {
|
|
|
|
logger.debug("gzipping tar", field("filePath", filePath))
|
|
|
|
const compress = zlib.createGzip()
|
|
|
|
stream.pipe(compress)
|
|
|
|
stream.on("error", (error) => compress.destroy(error))
|
|
|
|
stream.on("close", () => compress.end())
|
|
|
|
stream = compress
|
|
|
|
headers["content-encoding"] = "gzip"
|
|
|
|
}
|
|
|
|
return { stream, filePath, mime: "application/x-tar", cache: true, headers }
|
|
|
|
}
|
2020-03-02 19:43:02 +01:00
|
|
|
}
|