150 lines
5.4 KiB
TypeScript
150 lines
5.4 KiB
TypeScript
import express from 'express';
|
||
import React from 'react';
|
||
import { renderToString } from 'react-dom/server';
|
||
import { StaticRouter } from 'react-router-dom';
|
||
import serialize from 'serialize-javascript';
|
||
|
||
import {
|
||
getBuildingById,
|
||
getBuildingUPRNsById,
|
||
getBuildingPlanningDataById,
|
||
getLatestRevisionId,
|
||
getUserVerifiedAttributes
|
||
} from './api/services/building/base';
|
||
import { getUserById } from './api/services/user';
|
||
import { App } from './frontend/app';
|
||
import { parseBuildingURL } from './parse';
|
||
import asyncController from './api/routes/asyncController';
|
||
|
||
|
||
// reference packed assets
|
||
const assets = require(process.env.RAZZLE_ASSETS_MANIFEST);
|
||
|
||
|
||
const frontendRoute = asyncController(async (req: express.Request, res: express.Response) => {
|
||
const context: any = {}; // TODO: remove any
|
||
const data: any = {}; // TODO: remove any
|
||
context.status = 200;
|
||
|
||
const userId = req.session.user_id;
|
||
const buildingId = parseBuildingURL(req.url);
|
||
const isBuilding = (typeof (buildingId) !== 'undefined');
|
||
if (isBuilding && isNaN(buildingId)) {
|
||
context.status = 404;
|
||
}
|
||
|
||
try {
|
||
let [user, building, uprns, planningData, userVerified, latestRevisionId] = await Promise.all([
|
||
userId ? getUserById(userId) : undefined,
|
||
isBuilding ? getBuildingById(
|
||
buildingId,
|
||
{ userDataOptions: userId ? { userId, userAttributes: true } : null }
|
||
) : undefined,
|
||
isBuilding ? getBuildingUPRNsById(buildingId) : undefined,
|
||
isBuilding ? getBuildingPlanningDataById(buildingId) : undefined,
|
||
(isBuilding && userId) ? getUserVerifiedAttributes(buildingId, userId) : {},
|
||
getLatestRevisionId()
|
||
]);
|
||
|
||
if (isBuilding && typeof (building) === 'undefined') {
|
||
context.status = 404;
|
||
}
|
||
data.user = user;
|
||
data.building = building;
|
||
data.user_verified = userVerified;
|
||
if (data.building != null) {
|
||
data.building.uprns = uprns;
|
||
}
|
||
if (data.building != null) {
|
||
data.building.planning_data = planningData;
|
||
}
|
||
data.latestRevisionId = latestRevisionId;
|
||
renderHTML(context, data, req, res);
|
||
} catch(error) {
|
||
console.error(error);
|
||
data.user = undefined;
|
||
data.building = undefined;
|
||
data.user_verified = {}
|
||
data.latestRevisionId = 0;
|
||
context.status = 500;
|
||
renderHTML(context, data, req, res);
|
||
}
|
||
});
|
||
|
||
function renderHTML(context, data, req, res) {
|
||
const markup = renderToString(
|
||
<StaticRouter context={context} location={req.url}>
|
||
<App
|
||
user={data.user}
|
||
building={data.building}
|
||
revisionId={data.latestRevisionId}
|
||
user_verified={data.userVerified}
|
||
/>
|
||
</StaticRouter>
|
||
);
|
||
|
||
if (context.url) {
|
||
res.redirect(context.url);
|
||
} else {
|
||
res.status(context.status).send(
|
||
`<!doctype html>
|
||
<html lang="">
|
||
<head>
|
||
<meta http-equiv="X-UA-Compatible" content="IE=edge" />
|
||
<meta charset="utf-8" />
|
||
|
||
<meta name="twitter:card" content="summary" />
|
||
<meta name="twitter:site" content="@colouringlondon" />
|
||
|
||
<meta property="og:url" content="https://colouring.london" />
|
||
<meta property="og:type" content="website" />
|
||
<meta property="og:title" content="Colouring London" />
|
||
<meta property="og:description" content="Colouring London is a knowledge exchange platform collecting information on every building in London, to help make the city more sustainable. We’re building it at The Bartlett Centre for Advanced Spatial Analysis, University College London." />
|
||
<meta property="og:locale" content="en_GB" />
|
||
<meta property="og:image" content="https://colouring.london/images/logo-cl-square.png" />
|
||
|
||
<link rel="manifest" href="site.webmanifest">
|
||
|
||
<meta name="apple-mobile-web-app-capable" content="yes">
|
||
<meta name="apple-mobile-web-app-status-bar-style" content="black">
|
||
<meta name="apple-mobile-web-app-title" content="Colouring London">
|
||
<link rel="apple-touch-icon" href="icon-192x192.png">
|
||
|
||
<meta name="mobile-web-app-capable" content="yes">
|
||
<link rel="icon" sizes="192x192" href="icon-192x192.png">
|
||
|
||
<title>Colouring London</title>
|
||
<meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no">
|
||
<style>
|
||
@font-face {
|
||
font-family: 'glacial_cl';
|
||
src: url('/fonts/glacialindifference-regular-webfont.woff2') format('woff2'),
|
||
url('/fonts/glacialindifference-regular-webfont.woff') format('woff');
|
||
font-weight: normal;
|
||
font-style: normal;
|
||
}
|
||
</style>
|
||
${
|
||
assets.client.css
|
||
? `<link rel="stylesheet" href="${assets.client.css}">`
|
||
: ''
|
||
}
|
||
${
|
||
process.env.NODE_ENV === 'production'
|
||
? `<script src="${assets.client.js}" defer></script>`
|
||
: `<script src="${assets.client.js}" defer crossorigin></script>`
|
||
}
|
||
</head>
|
||
<body>
|
||
<div id="root">${markup}</div>
|
||
<script>
|
||
window.__PRELOADED_STATE__ = ${serialize(data)}
|
||
</script>
|
||
</body>
|
||
</html>`
|
||
);
|
||
}
|
||
}
|
||
|
||
export default frontendRoute;
|