docusaurus/lib/server/server.js

489 lines
15 KiB
JavaScript
Raw Normal View History

2017-07-07 13:28:29 -04:00
/**
* Copyright (c) 2017-present, Facebook, Inc.
*
* This source code is licensed under the MIT license found in the
* LICENSE file in the root directory of this source tree.
2017-07-07 13:28:29 -04:00
*/
function execute(port) {
2017-07-10 19:38:35 -04:00
const translation = require("./translation.js");
const express = require("express");
const React = require("react");
const request = require("request");
const renderToStaticMarkup = require("react-dom/server").renderToStaticMarkup;
const fs = require("fs-extra");
const os = require("os");
const path = require("path");
const toSlug = require("../core/toSlug.js");
const mkdirp = require("mkdirp");
const glob = require("glob");
2017-08-10 19:03:43 -04:00
const chalk = require("chalk");
const translate = require("./translate.js");
2017-08-03 13:25:01 -04:00
const versionFallback = require("./versionFallback");
2017-08-02 18:13:10 -04:00
const feed = require("./feed.js");
2017-08-02 18:13:10 -04:00
const CWD = process.cwd();
2017-07-31 19:19:02 -04:00
const ENABLE_TRANSLATION = fs.existsSync(CWD + "/languages.js");
const ENABLE_VERSIONING = fs.existsSync(CWD + "/versions.json");
2017-07-07 13:28:29 -04:00
2017-08-02 18:13:10 -04:00
let siteConfig = require(CWD + "/siteConfig.js");
2017-08-11 14:39:04 -04:00
// remove a module and child modules from require cache, so server does not have
// to be restarted
function removeModuleAndChildrenFromCache(moduleName) {
2017-07-07 13:28:29 -04:00
let mod = require.resolve(moduleName);
2017-08-11 14:39:04 -04:00
if (mod && (mod = require.cache[mod])) {
mod.children.forEach(child => {
delete require.cache[child.id];
removeModulePathFromCache(mod.id);
});
delete require.cache[mod.id];
removeModulePathFromCache(mod.id);
2017-07-07 13:28:29 -04:00
}
}
2017-08-11 14:39:04 -04:00
function removeModulePathFromCache(moduleName) {
2017-08-11 14:39:04 -04:00
Object.keys(module.constructor._pathCache).forEach(function(cacheKey) {
if (cacheKey.indexOf(moduleName) > 0) {
delete module.constructor._pathCache[cacheKey];
}
});
2017-07-10 19:38:35 -04:00
}
2017-07-07 13:28:29 -04:00
/****************************************************************************/
let readMetadata;
2017-07-07 13:28:29 -04:00
let Metadata;
2017-07-25 19:30:49 -04:00
function reloadMetadata() {
removeModuleAndChildrenFromCache("./readMetadata.js");
readMetadata = require("./readMetadata.js");
2017-07-07 13:28:29 -04:00
readMetadata.generateDocsMetadata();
removeModuleAndChildrenFromCache("../core/metadata.js");
2017-07-10 19:38:35 -04:00
Metadata = require("../core/metadata.js");
2017-07-07 13:28:29 -04:00
}
/****************************************************************************/
2017-07-10 19:38:35 -04:00
const TABLE_OF_CONTENTS_TOKEN = "<AUTOGENERATED_TABLE_OF_CONTENTS>";
2017-07-07 13:28:29 -04:00
const insertTableOfContents = rawContent => {
const regexp = /\n###\s+(`.*`.*)\n/g;
let match;
const headers = [];
while ((match = regexp.exec(rawContent))) {
headers.push(match[1]);
}
const tableOfContents = headers
.map(header => ` - [${header}](#${toSlug(header)})`)
2017-07-10 19:38:35 -04:00
.join("\n");
2017-07-07 13:28:29 -04:00
return rawContent.replace(TABLE_OF_CONTENTS_TOKEN, tableOfContents);
};
/****************************************************************************/
function isSeparateCss(file) {
if (!siteConfig.separateCss) {
return false;
}
for (let i = 0; i < siteConfig.separateCss.length; i++) {
if (file.includes(siteConfig.separateCss[i])) {
return true;
}
}
return false;
}
/****************************************************************************/
2017-07-10 19:38:35 -04:00
console.log("server.js triggered...");
2017-07-07 13:28:29 -04:00
2017-07-25 19:30:49 -04:00
reloadMetadata();
2017-07-07 13:28:29 -04:00
2017-08-15 19:55:38 -04:00
// handle all requests for document pages
2017-07-31 19:19:02 -04:00
const app = express().get(/docs\/.*html$/, (req, res, next) => {
removeModuleAndChildrenFromCache(CWD + "/siteConfig.js");
2017-07-10 19:38:35 -04:00
siteConfig = require(CWD + "/siteConfig.js");
2017-07-07 13:28:29 -04:00
2017-08-03 13:25:01 -04:00
let url = req.path.toString().replace(siteConfig.baseUrl, "");
2017-07-25 19:30:49 -04:00
reloadMetadata();
2017-08-03 13:25:01 -04:00
2017-08-15 19:55:38 -04:00
// links is a map from a permalink to an id for each document
2017-07-10 19:38:35 -04:00
let links = {};
2017-08-03 13:25:01 -04:00
Object.keys(Metadata).forEach(id => {
const metadata = Metadata[id];
links[metadata.permalink] = id;
});
2017-08-15 19:55:38 -04:00
// mdToHtml is a map from a markdown file name to its html link, used to
// change relative markdown links that work on GitHub into actual site links
2017-08-03 13:25:01 -04:00
const mdToHtml = {};
Object.keys(Metadata).forEach(id => {
const metadata = Metadata[id];
if (metadata.language !== "en" || metadata.original_id) {
2017-08-03 13:25:01 -04:00
return;
}
let htmlLink =
siteConfig.baseUrl + metadata.permalink.replace("/next/", "/");
2017-08-03 13:25:01 -04:00
if (htmlLink.includes("/docs/en/")) {
htmlLink = htmlLink.replace("/docs/en/", "/docs/en/VERSION/");
} else {
htmlLink = htmlLink.replace("/docs/", "/docs/VERSION/");
}
mdToHtml[metadata.source] = htmlLink;
});
const metadata = Metadata[links[url]];
2017-08-09 18:43:30 -04:00
if (!metadata) {
next();
return;
}
2017-08-03 13:25:01 -04:00
const language = metadata.language;
2017-08-15 19:55:38 -04:00
// determine what file to use according to its id
2017-08-03 13:25:01 -04:00
let file;
if (metadata.original_id) {
if (ENABLE_TRANSLATION && metadata.language !== "en") {
2017-08-03 13:25:01 -04:00
file =
CWD + "/translated_docs/" + metadata.language + "/" + metadata.source;
2017-08-03 13:25:01 -04:00
} else {
file = CWD + "/versioned_docs/" + metadata.source;
}
} else {
2017-07-31 19:19:02 -04:00
if (metadata.language === "en") {
2017-08-03 13:25:01 -04:00
file = CWD + "/../docs/" + metadata.source;
2017-07-31 19:19:02 -04:00
} else {
2017-08-03 13:25:01 -04:00
file =
2017-07-31 19:19:02 -04:00
CWD + "/translated_docs/" + metadata.language + "/" + metadata.source;
}
2017-07-10 19:38:35 -04:00
}
2017-08-02 18:13:10 -04:00
2017-07-18 19:52:47 -04:00
if (!fs.existsSync(file)) {
next();
2017-07-19 13:50:32 -04:00
return;
2017-07-18 19:52:47 -04:00
}
2017-07-07 13:28:29 -04:00
2017-08-03 13:25:01 -04:00
let rawContent = readMetadata.extractMetadata(fs.readFileSync(file, "utf8"))
.rawContent;
2017-07-07 13:28:29 -04:00
2017-08-15 19:55:38 -04:00
// generate table of contents if appropriate
2017-07-10 19:38:35 -04:00
if (rawContent && rawContent.indexOf(TABLE_OF_CONTENTS_TOKEN) !== -1) {
rawContent = insertTableOfContents(rawContent);
}
2017-07-07 13:28:29 -04:00
let latestVersion;
if (ENABLE_VERSIONING) {
latestVersion = JSON.parse(
fs.readFileSync(CWD + "/versions.json", "utf8")
)[0];
}
2017-08-15 19:55:38 -04:00
// replace any links to markdown files to their website html links
2017-07-10 19:38:35 -04:00
Object.keys(mdToHtml).forEach(function(key, index) {
2017-08-03 13:25:01 -04:00
let link = mdToHtml[key];
link = link.replace("/en/", "/" + language + "/");
link = link.replace(
"/VERSION/",
2017-08-08 18:44:51 -04:00
metadata.version && metadata.version !== latestVersion
? "/" + metadata.version + "/"
: "/"
2017-07-31 19:19:02 -04:00
);
2017-08-16 19:17:33 -04:00
// replace relative links without "./"
2017-08-08 18:44:51 -04:00
rawContent = rawContent.replace(
new RegExp("\\]\\(" + key, "g"),
"](" + link
);
2017-08-16 19:17:33 -04:00
// replace relative links with "./"
rawContent = rawContent.replace(
new RegExp("\\]\\(\\./" + key, "g"),
"](" + link
);
2017-07-10 19:38:35 -04:00
});
2017-07-07 13:28:29 -04:00
2017-08-15 19:55:38 -04:00
// replace any relative links to static assets to absolute links
2017-07-31 19:19:02 -04:00
rawContent = rawContent.replace(
/\]\(assets\//g,
"](" + siteConfig.baseUrl + "docs/assets/"
);
removeModuleAndChildrenFromCache("../core/DocsLayout.js");
2017-07-10 19:38:35 -04:00
const DocsLayout = require("../core/DocsLayout.js");
const docComp = (
<DocsLayout metadata={metadata} language={language} config={siteConfig}>
{rawContent}
</DocsLayout>
);
2017-07-07 13:28:29 -04:00
2017-07-10 19:38:35 -04:00
res.send(renderToStaticMarkup(docComp));
});
2017-08-03 13:25:01 -04:00
app.get(/blog\/.*xml$/, (req, res) => {
res.set("Content-Type", "application/rss+xml");
let parts = req.path.toString().split("blog/");
if (parts[1].toLowerCase() == "atom.xml") {
res.send(feed("atom"));
return;
}
res.send(feed("rss"));
});
2017-08-15 19:55:38 -04:00
// handle all requests for blog pages and posts
2017-07-31 19:19:02 -04:00
app.get(/blog\/.*html$/, (req, res) => {
removeModuleAndChildrenFromCache(CWD + "/siteConfig.js");
2017-07-10 19:38:35 -04:00
siteConfig = require(CWD + "/siteConfig.js");
2017-08-08 17:58:41 -04:00
if (fs.existsSync(__dirname + "/../core/MetadataBlog.js")) {
removeModuleAndChildrenFromCache("../core/MetadataBlog.js");
2017-08-08 17:58:41 -04:00
fs.removeSync(__dirname + "/../core/MetadataBlog.js");
2017-07-07 13:28:29 -04:00
}
readMetadata.generateBlogMetadata();
2017-08-08 17:58:41 -04:00
const MetadataBlog = require("../core/MetadataBlog.js");
2017-07-07 13:28:29 -04:00
2017-08-15 19:55:38 -04:00
// generate all of the blog pages
removeModuleAndChildrenFromCache("../core/BlogPageLayout.js");
2017-07-10 19:38:35 -04:00
const BlogPageLayout = require("../core/BlogPageLayout.js");
2017-07-07 13:28:29 -04:00
const blogPages = {};
2017-08-15 19:55:38 -04:00
// make blog pages with 10 posts per page
2017-07-07 13:28:29 -04:00
const perPage = 10;
for (
let page = 0;
page < Math.ceil(MetadataBlog.length / perPage);
page++
) {
2017-07-10 19:38:35 -04:00
let language = "en";
const metadata = { page: page, perPage: perPage };
2017-07-10 19:38:35 -04:00
const blogPageComp = (
<BlogPageLayout
metadata={metadata}
language={language}
config={siteConfig}
/>
);
2017-07-07 13:28:29 -04:00
const str = renderToStaticMarkup(blogPageComp);
2017-07-10 19:38:35 -04:00
let path = (page > 0 ? "page" + (page + 1) : "") + "/index.html";
2017-07-07 13:28:29 -04:00
blogPages[path] = str;
}
2017-07-10 19:38:35 -04:00
let parts = req.path.toString().split("blog/");
2017-07-07 13:28:29 -04:00
// send corresponding blog page if appropriate
2017-07-10 19:38:35 -04:00
if (parts[1] === "index.html") {
res.send(blogPages["/index.html"]);
} else if (parts[1].endsWith("/index.html")) {
2017-07-07 13:28:29 -04:00
res.send(blogPages[parts[1]]);
2017-07-10 19:38:35 -04:00
} else if (parts[1].match(/page([0-9]+)/)) {
if (parts[1].endsWith("/")) {
res.send(blogPages[parts[1] + "index.html"]);
2017-07-07 13:28:29 -04:00
} else {
2017-07-10 19:38:35 -04:00
res.send(blogPages[parts[1] + "/index.html"]);
2017-07-07 13:28:29 -04:00
}
2017-07-10 19:38:35 -04:00
} else {
// else send corresponding blog post
2017-07-07 13:28:29 -04:00
let file = parts[1];
2017-07-10 19:38:35 -04:00
file = file.replace(/\.html$/, ".md");
file = file.replace(new RegExp("/", "g"), "-");
2017-07-31 19:19:02 -04:00
file = CWD + "/blog/" + file;
2017-07-07 13:28:29 -04:00
2017-07-10 19:38:35 -04:00
const result = readMetadata.extractMetadata(
fs.readFileSync(file, { encoding: "utf8" })
2017-07-10 19:38:35 -04:00
);
2017-07-31 19:19:02 -04:00
let rawContent = result.rawContent;
rawContent = rawContent.replace(
/\]\(assets\//g,
"](" + siteConfig.baseUrl + "blog/assets/"
);
2017-07-07 13:28:29 -04:00
const metadata = Object.assign(
{ path: req.path.toString().split("blog/")[1], content: rawContent },
2017-07-07 13:28:29 -04:00
result.metadata
);
metadata.id = metadata.title;
2017-07-10 19:38:35 -04:00
let language = "en";
removeModuleAndChildrenFromCache("../core/BlogPostLayout.js");
2017-07-10 19:38:35 -04:00
const BlogPostLayout = require("../core/BlogPostLayout.js");
const blogPostComp = (
<BlogPostLayout
metadata={metadata}
language={language}
config={siteConfig}
>
{rawContent}
</BlogPostLayout>
);
2017-07-07 13:28:29 -04:00
res.send(renderToStaticMarkup(blogPostComp));
}
});
2017-08-15 19:55:38 -04:00
// handle all other main pages
2017-07-19 13:50:32 -04:00
app.get("*.html", (req, res, next) => {
removeModuleAndChildrenFromCache(CWD + "/siteConfig.js");
2017-07-10 19:38:35 -04:00
siteConfig = require(CWD + "/siteConfig.js");
2017-07-07 13:28:29 -04:00
2017-08-15 19:55:38 -04:00
// look for user provided html file first
2017-07-10 19:38:35 -04:00
let htmlFile = req.path.toString().replace(siteConfig.baseUrl, "");
htmlFile = CWD + "/pages/" + htmlFile;
if (
fs.existsSync(htmlFile) ||
fs.existsSync(
(htmlFile = htmlFile.replace(
path.basename(htmlFile),
"en/" + path.basename(htmlFile)
))
)
) {
res.send(fs.readFileSync(htmlFile, { encoding: "utf8" }));
2017-07-07 13:28:29 -04:00
return;
}
2017-08-15 19:55:38 -04:00
// look for user provided react file either in specified path or in path for english files
2017-07-10 19:38:35 -04:00
let file = req.path.toString().replace(/\.html$/, ".js");
file = file.replace(siteConfig.baseUrl, "");
let userFile = CWD + "/pages/" + file;
2017-07-07 13:28:29 -04:00
2017-07-10 19:38:35 -04:00
let language = "en";
2017-07-07 13:28:29 -04:00
const regexLang = /(.*)\/.*\.html$/;
const match = regexLang.exec(req.path);
2017-07-10 19:38:35 -04:00
const parts = match[1].split("/");
2017-07-07 13:28:29 -04:00
const enabledLangTags = [];
for (let i = 0; i < translation["languages"].length; i++) {
enabledLangTags.push(translation["languages"][i].tag);
2017-07-07 13:28:29 -04:00
}
for (let i = 0; i < parts.length; i++) {
if (enabledLangTags.indexOf(parts[i]) !== -1) {
language = parts[i];
}
}
let englishFile = CWD + "/pages/" + file;
if (language !== "en") {
englishFile = englishFile.replace("/" + language + "/", "/en/");
}
// check for: a file for the page, an english file for page with unspecified language,
2017-08-15 19:55:38 -04:00
// english file for the page
2017-07-10 19:38:35 -04:00
if (
fs.existsSync(userFile) ||
fs.existsSync(
(userFile = userFile.replace(
path.basename(userFile),
"en/" + path.basename(userFile)
))
) ||
fs.existsSync((userFile = englishFile))
2017-07-10 19:38:35 -04:00
) {
2017-08-15 19:55:38 -04:00
// copy into docusaurus so require paths work
2017-07-10 19:38:35 -04:00
let parts = userFile.split("pages/");
let tempFile = __dirname + "/../pages/" + parts[1];
tempFile = tempFile.replace(
path.basename(file),
"temp" + path.basename(file)
);
mkdirp.sync(tempFile.replace(new RegExp("/[^/]*$"), ""));
2017-07-07 13:28:29 -04:00
fs.copySync(userFile, tempFile);
2017-08-15 19:55:38 -04:00
// render into a string
removeModuleAndChildrenFromCache(tempFile);
2017-07-07 13:28:29 -04:00
const ReactComp = require(tempFile);
removeModuleAndChildrenFromCache("../core/Site.js");
2017-07-10 19:38:35 -04:00
const Site = require("../core/Site.js");
translate.setLanguage(language);
2017-07-10 19:38:35 -04:00
const str = renderToStaticMarkup(
<Site language={language} config={siteConfig}>
<ReactComp language={language} />
</Site>
);
2017-07-07 13:28:29 -04:00
fs.removeSync(tempFile);
res.send(str);
2017-07-10 19:38:35 -04:00
} else {
2017-07-19 13:50:32 -04:00
next();
return;
2017-07-07 13:28:29 -04:00
}
});
2017-08-15 19:55:38 -04:00
// generate the main.css file by concatenating user provided css to the end
2017-07-10 19:38:35 -04:00
app.get(/main\.css$/, (req, res) => {
const mainCssPath =
__dirname +
"/../static/" +
req.path.toString().replace(siteConfig.baseUrl, "/");
let cssContent = fs.readFileSync(mainCssPath, { encoding: "utf8" });
2017-07-07 13:28:29 -04:00
2017-07-10 19:38:35 -04:00
let files = glob.sync(CWD + "/static/**/*.css");
2017-07-07 13:28:29 -04:00
files.forEach(file => {
if (isSeparateCss(file)) {
return;
}
2017-07-10 19:38:35 -04:00
cssContent =
cssContent + "\n" + fs.readFileSync(file, { encoding: "utf8" });
2017-07-07 13:28:29 -04:00
});
2017-08-10 19:03:43 -04:00
if (
2017-08-10 19:10:30 -04:00
!siteConfig.colors ||
2017-08-10 19:03:43 -04:00
!siteConfig.colors.primaryColor ||
!siteConfig.colors.secondaryColor ||
!siteConfig.colors.prismColor
) {
console.error(
`${chalk.yellow(
"Missing color configuration."
)} Make sure siteConfig.colors includes primaryColor, secondaryColor, and prismColor fields.`
2017-07-10 19:38:35 -04:00
);
2017-08-10 19:03:43 -04:00
}
Object.keys(siteConfig.colors).forEach(key => {
const color = siteConfig.colors[key];
cssContent = cssContent.replace(new RegExp("\\$" + key, "g"), color);
});
2017-07-07 13:28:29 -04:00
res.send(cssContent);
});
2017-08-15 19:55:38 -04:00
// serve static assets from these locations
2017-07-31 19:19:02 -04:00
app.use(
siteConfig.baseUrl + "docs/assets/",
express.static(CWD + "/../docs/assets")
);
app.use(
siteConfig.baseUrl + "blog/assets/",
express.static(CWD + "/blog/assets")
);
2017-07-10 19:38:35 -04:00
app.use(siteConfig.baseUrl, express.static(CWD + "/static"));
app.use(siteConfig.baseUrl, express.static(__dirname + "/../static"));
2017-07-07 13:28:29 -04:00
2017-08-15 19:55:38 -04:00
// "redirect" requests to pages ending with "/" or no extension so that
// request to "...blog" returns same result as "...blog/index.html"
2017-07-07 13:28:29 -04:00
app.get(/\/[^\.]*\/?$/, (req, res) => {
2017-07-10 19:38:35 -04:00
if (req.path.toString().endsWith("/")) {
request.get(
2017-07-12 17:43:24 -04:00
"http://localhost:" + port + req.path + "index.html",
2017-07-10 19:38:35 -04:00
(err, response, body) => {
if (!err) {
res.send(body);
}
2017-07-07 13:28:29 -04:00
}
2017-07-10 19:38:35 -04:00
);
2017-07-07 13:28:29 -04:00
} else {
2017-07-10 19:38:35 -04:00
request.get(
2017-07-12 17:43:24 -04:00
"http://localhost:" + port + req.path + "/index.html",
2017-07-10 19:38:35 -04:00
(err, response, body) => {
if (!err) {
res.send(body);
}
2017-07-07 13:28:29 -04:00
}
2017-07-10 19:38:35 -04:00
);
2017-07-07 13:28:29 -04:00
}
});
app.listen(port);
2017-07-10 19:38:35 -04:00
console.log("listening on port: " + port);
console.log("Open http://localhost:" + port + "/");
2017-07-07 13:28:29 -04:00
}
module.exports = execute;