Code of the https://kraut.world/ server. Map at https://github.com/HackspaceJena/krautspace-rc3-map - Fork from https://github.com/thecodingmachine/workadventure
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
119 lines
5.0 KiB
119 lines
5.0 KiB
import { HttpRequest, HttpResponse, TemplatedApp } from "uWebSockets.js"; |
|
import { BaseController } from "./BaseController"; |
|
import { parse } from "query-string"; |
|
import { adminApi } from "../Services/AdminApi"; |
|
import { ADMIN_API_URL, DISABLE_ANONYMOUS, FRONT_URL } from "../Enum/EnvironmentVariable"; |
|
import { GameRoomPolicyTypes } from "../Model/PusherRoom"; |
|
import { isMapDetailsData, MapDetailsData } from "../Messages/JsonMessages/MapDetailsData"; |
|
import { socketManager } from "../Services/SocketManager"; |
|
import { AuthTokenData, jwtTokenManager } from "../Services/JWTTokenManager"; |
|
import { v4 } from "uuid"; |
|
import { InvalidTokenError } from "./InvalidTokenError"; |
|
|
|
export class MapController extends BaseController { |
|
constructor(private App: TemplatedApp) { |
|
super(); |
|
this.App = App; |
|
this.getMapUrl(); |
|
} |
|
|
|
// Returns a map mapping map name to file name of the map |
|
getMapUrl() { |
|
this.App.options("/map", (res: HttpResponse, req: HttpRequest) => { |
|
this.addCorsHeaders(res); |
|
res.end(); |
|
}); |
|
|
|
this.App.get("/map", (res: HttpResponse, req: HttpRequest) => { |
|
res.onAborted(() => { |
|
console.warn("/map request was aborted"); |
|
}); |
|
|
|
const query = parse(req.getQuery()); |
|
|
|
if (typeof query.playUri !== "string") { |
|
console.error("Expected playUri parameter in /map endpoint"); |
|
res.writeStatus("400 Bad request"); |
|
this.addCorsHeaders(res); |
|
res.end("Expected playUri parameter"); |
|
return; |
|
} |
|
|
|
// If no admin URL is set, let's react on '/_/[instance]/[map url]' URLs |
|
if (!ADMIN_API_URL) { |
|
const roomUrl = new URL(query.playUri); |
|
|
|
const match = /\/_\/[^/]+\/(.+)/.exec(roomUrl.pathname); |
|
if (!match) { |
|
res.writeStatus("404 Not Found"); |
|
this.addCorsHeaders(res); |
|
res.writeHeader("Content-Type", "application/json"); |
|
res.end(JSON.stringify({})); |
|
return; |
|
} |
|
|
|
const mapUrl = roomUrl.protocol + "//" + match[1]; |
|
|
|
res.writeStatus("200 OK"); |
|
this.addCorsHeaders(res); |
|
res.writeHeader("Content-Type", "application/json"); |
|
res.end( |
|
JSON.stringify({ |
|
mapUrl, |
|
policy_type: GameRoomPolicyTypes.ANONYMOUS_POLICY, |
|
roomSlug: null, // Deprecated |
|
group: null, |
|
tags: [], |
|
textures: [], |
|
contactPage: null, |
|
authenticationMandatory: DISABLE_ANONYMOUS, |
|
} as MapDetailsData) |
|
); |
|
|
|
return; |
|
} |
|
|
|
(async () => { |
|
try { |
|
let userId: string | undefined = undefined; |
|
if (query.authToken != undefined) { |
|
let authTokenData: AuthTokenData; |
|
try { |
|
authTokenData = jwtTokenManager.verifyJWTToken(query.authToken as string); |
|
userId = authTokenData.identifier; |
|
} catch (e) { |
|
try { |
|
// Decode token, in this case we don't need to create new token. |
|
authTokenData = jwtTokenManager.verifyJWTToken(query.authToken as string, true); |
|
userId = authTokenData.identifier; |
|
console.info("JWT expire, but decoded", userId); |
|
} catch (e) { |
|
if (e instanceof InvalidTokenError) { |
|
// The token was not good, redirect user on login page |
|
res.writeStatus("401 Unauthorized"); |
|
res.writeHeader("Access-Control-Allow-Origin", FRONT_URL); |
|
res.end("Token decrypted error"); |
|
return; |
|
} else { |
|
return this.errorToResponse(e, res); |
|
} |
|
} |
|
} |
|
} |
|
const mapDetails = await adminApi.fetchMapDetails(query.playUri as string, userId); |
|
|
|
if (isMapDetailsData(mapDetails) && DISABLE_ANONYMOUS) { |
|
mapDetails.authenticationMandatory = true; |
|
} |
|
|
|
res.writeStatus("200 OK"); |
|
this.addCorsHeaders(res); |
|
res.writeHeader("Content-Type", "application/json"); |
|
res.end(JSON.stringify(mapDetails)); |
|
} catch (e) { |
|
this.errorToResponse(e, res); |
|
} |
|
})(); |
|
}); |
|
} |
|
}
|
|
|