2020-02-13 22:38:05 +00:00
|
|
|
import * as http from "http"
|
|
|
|
import * as querystring from "querystring"
|
|
|
|
import { Application } from "../../common/api"
|
|
|
|
import { HttpCode, HttpError } from "../../common/http"
|
|
|
|
import { HttpProvider, HttpProviderOptions, HttpResponse, Route } from "../http"
|
|
|
|
import { ApiHttpProvider } from "./api"
|
2020-02-14 21:57:51 +00:00
|
|
|
import { UpdateHttpProvider } from "./update"
|
2020-02-13 22:38:05 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Top-level and fallback HTTP provider.
|
|
|
|
*/
|
|
|
|
export class MainHttpProvider extends HttpProvider {
|
2020-02-14 21:57:51 +00:00
|
|
|
public constructor(
|
|
|
|
options: HttpProviderOptions,
|
|
|
|
private readonly api: ApiHttpProvider,
|
2020-02-18 17:52:29 +00:00
|
|
|
private readonly update: UpdateHttpProvider,
|
2020-02-14 21:57:51 +00:00
|
|
|
) {
|
2020-02-13 22:38:05 +00:00
|
|
|
super(options)
|
|
|
|
}
|
|
|
|
|
|
|
|
public async handleRequest(route: Route, request: http.IncomingMessage): Promise<HttpResponse | undefined> {
|
|
|
|
switch (route.base) {
|
|
|
|
case "/static": {
|
|
|
|
this.ensureMethod(request)
|
2020-02-18 18:57:45 +00:00
|
|
|
const response = await this.getReplacedResource(route)
|
2020-02-13 22:38:05 +00:00
|
|
|
if (!this.isDev) {
|
|
|
|
response.cache = true
|
|
|
|
}
|
|
|
|
return response
|
|
|
|
}
|
|
|
|
|
|
|
|
case "/delete": {
|
|
|
|
this.ensureMethod(request, "POST")
|
|
|
|
const data = await this.getData(request)
|
|
|
|
const p = data ? querystring.parse(data) : {}
|
|
|
|
this.api.deleteSession(p.sessionId as string)
|
|
|
|
return { redirect: "/" }
|
|
|
|
}
|
|
|
|
|
|
|
|
case "/": {
|
|
|
|
this.ensureMethod(request)
|
|
|
|
if (route.requestPath !== "/index.html") {
|
|
|
|
throw new HttpError("Not found", HttpCode.NotFound)
|
|
|
|
} else if (!this.authenticated(request)) {
|
|
|
|
return { redirect: "/login" }
|
|
|
|
}
|
|
|
|
return this.getRoot(route)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Run an existing app, but if it doesn't exist go ahead and start it.
|
|
|
|
let app = this.api.getRunningApplication(route.base)
|
|
|
|
let sessionId = app && app.sessionId
|
|
|
|
if (!app) {
|
|
|
|
app = (await this.api.installedApplications()).find((a) => a.path === route.base)
|
|
|
|
if (app) {
|
|
|
|
sessionId = await this.api.createSession(app)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sessionId) {
|
2020-02-27 20:56:14 +00:00
|
|
|
return this.getAppRoot(route, (app && app.name) || "", sessionId)
|
2020-02-13 22:38:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return this.getErrorRoot(route, "404", "404", "Application not found")
|
|
|
|
}
|
|
|
|
|
2020-02-18 18:57:45 +00:00
|
|
|
/**
|
|
|
|
* Return a resource with variables replaced where necessary.
|
|
|
|
*/
|
|
|
|
protected async getReplacedResource(route: Route): Promise<HttpResponse> {
|
2020-02-27 20:56:14 +00:00
|
|
|
const split = route.requestPath.split("/")
|
|
|
|
switch (split[split.length - 1]) {
|
|
|
|
case "manifest.json": {
|
|
|
|
const response = await this.getUtf8Resource(this.rootPath, route.requestPath)
|
|
|
|
return this.replaceTemplates(route, response)
|
|
|
|
}
|
2020-02-18 18:57:45 +00:00
|
|
|
}
|
|
|
|
return this.getResource(this.rootPath, route.requestPath)
|
|
|
|
}
|
|
|
|
|
2020-02-13 22:38:05 +00:00
|
|
|
public async getRoot(route: Route): Promise<HttpResponse> {
|
2020-02-18 20:13:22 +00:00
|
|
|
const running = await this.api.running()
|
2020-02-13 22:38:05 +00:00
|
|
|
const apps = await this.api.installedApplications()
|
|
|
|
const response = await this.getUtf8Resource(this.rootPath, "src/browser/pages/home.html")
|
|
|
|
response.content = response.content
|
2020-02-14 21:57:51 +00:00
|
|
|
.replace(/{{UPDATE:NAME}}/, await this.getUpdate())
|
2020-02-18 20:13:22 +00:00
|
|
|
.replace(/{{APP_LIST:RUNNING}}/, this.getAppRows(running.applications))
|
2020-02-13 22:38:05 +00:00
|
|
|
.replace(
|
2020-02-14 21:57:51 +00:00
|
|
|
/{{APP_LIST:EDITORS}}/,
|
2020-02-15 00:46:00 +00:00
|
|
|
this.getAppRows(apps.filter((app) => app.categories && app.categories.includes("Editor"))),
|
2020-02-13 22:38:05 +00:00
|
|
|
)
|
|
|
|
.replace(
|
2020-02-14 21:57:51 +00:00
|
|
|
/{{APP_LIST:OTHER}}/,
|
2020-02-15 00:46:00 +00:00
|
|
|
this.getAppRows(apps.filter((app) => !app.categories || !app.categories.includes("Editor"))),
|
2020-02-13 22:38:05 +00:00
|
|
|
)
|
2020-02-27 20:56:14 +00:00
|
|
|
return this.replaceTemplates(route, response)
|
2020-02-13 22:38:05 +00:00
|
|
|
}
|
|
|
|
|
2020-02-27 20:56:14 +00:00
|
|
|
public async getAppRoot(route: Route, name: string, sessionId: string): Promise<HttpResponse> {
|
2020-02-13 22:38:05 +00:00
|
|
|
const response = await this.getUtf8Resource(this.rootPath, "src/browser/pages/app.html")
|
2020-02-27 20:56:14 +00:00
|
|
|
response.content = response.content.replace(/{{APP_NAME}}/, name)
|
|
|
|
return this.replaceTemplates(route, response, sessionId)
|
2020-02-13 22:38:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public async handleWebSocket(): Promise<undefined> {
|
|
|
|
return undefined
|
|
|
|
}
|
|
|
|
|
|
|
|
private getAppRows(apps: ReadonlyArray<Application>): string {
|
2020-02-27 18:04:23 +00:00
|
|
|
return apps.length > 0
|
|
|
|
? apps.map((app) => this.getAppRow(app)).join("\n")
|
|
|
|
: `<div class="none">No applications are currently running.</div>`
|
2020-02-13 22:38:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private getAppRow(app: Application): string {
|
2020-02-14 21:57:51 +00:00
|
|
|
return `<div class="block-row">
|
2020-02-14 22:41:42 +00:00
|
|
|
<a class="item -row -link" href=".${app.path}">
|
2020-02-13 22:38:05 +00:00
|
|
|
${
|
|
|
|
app.icon
|
|
|
|
? `<img class="icon" src="data:image/png;base64,${app.icon}"></img>`
|
|
|
|
: `<div class="icon -missing"></div>`
|
|
|
|
}
|
|
|
|
<div class="name">${app.name}</div>
|
|
|
|
</a>
|
|
|
|
${
|
|
|
|
app.sessionId
|
|
|
|
? `<form class="kill-form" action="./delete" method="POST">
|
|
|
|
<input type="hidden" name="sessionId" value="${app.sessionId}">
|
2020-02-27 18:04:23 +00:00
|
|
|
<button class="kill -button" type="submit">Kill</button>
|
2020-02-13 22:38:05 +00:00
|
|
|
</form>`
|
|
|
|
: ""
|
|
|
|
}
|
|
|
|
</div>`
|
|
|
|
}
|
2020-02-14 21:57:51 +00:00
|
|
|
|
|
|
|
private async getUpdate(): Promise<string> {
|
|
|
|
if (!this.update.enabled) {
|
2020-02-18 18:24:12 +00:00
|
|
|
return `<div class="block-row"><div class="item"><div class="sub">Updates are disabled</div></div></div>`
|
2020-02-14 21:57:51 +00:00
|
|
|
}
|
|
|
|
|
2020-02-14 22:41:42 +00:00
|
|
|
const humanize = (time: number): string => {
|
|
|
|
const d = new Date(time)
|
|
|
|
const pad = (t: number): string => (t < 10 ? "0" : "") + t
|
|
|
|
return (
|
|
|
|
`${d.getFullYear()}-${pad(d.getMonth() + 1)}-${pad(d.getDate())}` +
|
|
|
|
` ${pad(d.getHours())}:${pad(d.getMinutes())}`
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2020-02-14 21:57:51 +00:00
|
|
|
const update = await this.update.getUpdate()
|
2020-02-14 22:41:42 +00:00
|
|
|
if (this.update.isLatestVersion(update)) {
|
2020-02-14 21:57:51 +00:00
|
|
|
return `<div class="block-row">
|
2020-02-14 22:41:42 +00:00
|
|
|
<div class="item">
|
2020-02-26 18:02:20 +00:00
|
|
|
Latest: ${update.version}
|
2020-02-14 22:41:42 +00:00
|
|
|
<div class="sub">Up to date</div>
|
|
|
|
</div>
|
|
|
|
<div class="item">
|
|
|
|
${humanize(update.checked)}
|
|
|
|
<a class="sub -link" href="./update/check">Check now</a>
|
|
|
|
</div>
|
|
|
|
<div class="item" >Current: ${this.update.currentVersion}</div>
|
2020-02-14 21:57:51 +00:00
|
|
|
</div>`
|
|
|
|
}
|
|
|
|
|
|
|
|
return `<div class="block-row">
|
2020-02-26 18:02:20 +00:00
|
|
|
<div class="item">
|
|
|
|
Latest: ${update.version}
|
2020-02-14 22:41:42 +00:00
|
|
|
<div class="sub">Out of date</div>
|
2020-02-26 18:02:20 +00:00
|
|
|
</div>
|
2020-02-14 22:41:42 +00:00
|
|
|
<div class="item">
|
|
|
|
${humanize(update.checked)}
|
2020-02-26 18:02:20 +00:00
|
|
|
<a class="sub -link" href="./update">Update now</a>
|
2020-02-14 22:41:42 +00:00
|
|
|
</div>
|
|
|
|
<div class="item" >Current: ${this.update.currentVersion}</div>
|
2020-02-14 21:57:51 +00:00
|
|
|
</div>`
|
|
|
|
}
|
2020-02-13 22:38:05 +00:00
|
|
|
}
|