mirror of
https://github.com/facebook/docusaurus.git
synced 2025-04-30 02:37:59 +02:00
147 lines
3.9 KiB
JavaScript
147 lines
3.9 KiB
JavaScript
/**
|
|
* 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.
|
|
*/
|
|
|
|
const path = require('path');
|
|
const fm = require('front-matter');
|
|
const escapeStringRegexp = require('escape-string-regexp');
|
|
const fs = require('fs-extra');
|
|
|
|
const genCache = new Map();
|
|
async function generate(generatedFilesDir, file, content) {
|
|
const cached = genCache.get(file);
|
|
if (cached !== content) {
|
|
await fs.writeFile(path.join(generatedFilesDir, file), content);
|
|
genCache.set(file, content);
|
|
}
|
|
}
|
|
|
|
const indexRE = /(^|.*\/)index\.(md|js)$/i;
|
|
const extRE = /\.(md|js)$/;
|
|
|
|
function fileToPath(file) {
|
|
if (indexRE.test(file)) {
|
|
return file.replace(indexRE, '/$1');
|
|
}
|
|
return `/${file.replace(extRE, '').replace(/\\/g, '/')}`;
|
|
}
|
|
|
|
function encodePath(userpath) {
|
|
return userpath
|
|
.split('/')
|
|
.map(item => encodeURIComponent(item))
|
|
.join('/');
|
|
}
|
|
|
|
function fileToComponentName(file) {
|
|
const ext = extRE.exec(file)[1];
|
|
let str = file.replace(extRE, '');
|
|
str = str.replace(/([A-Z])/g, ' $1');
|
|
str = str.replace(/^[\W_]+|[\W_]+$/g, '').toLowerCase();
|
|
str = str.charAt(0).toUpperCase() + str.slice(1);
|
|
str = str.replace(/[\W_]+(\w|$)/g, (_, ch) => ch.toUpperCase());
|
|
return ext ? ext.toUpperCase() + str : str;
|
|
}
|
|
|
|
function idx(target, keyPaths) {
|
|
return (
|
|
target &&
|
|
(Array.isArray(keyPaths)
|
|
? keyPaths.reduce((obj, key) => obj && obj[key], target)
|
|
: target[keyPaths])
|
|
);
|
|
}
|
|
|
|
function getSubFolder(file, refDir) {
|
|
const separator = escapeStringRegexp(path.sep);
|
|
const baseDir = escapeStringRegexp(path.basename(refDir));
|
|
const regexSubFolder = new RegExp(
|
|
`${baseDir}${separator}(.*?)${separator}.*`,
|
|
);
|
|
const match = regexSubFolder.exec(file);
|
|
return match && match[1];
|
|
}
|
|
|
|
function parse(fileString) {
|
|
if (!fm.test(fileString)) {
|
|
return {metadata: null, content: fileString};
|
|
}
|
|
const {attributes: metadata, body: content} = fm(fileString);
|
|
|
|
return {metadata, content};
|
|
}
|
|
|
|
function normalizeUrl(rawUrls) {
|
|
const urls = rawUrls;
|
|
const resultArray = [];
|
|
|
|
// If the first part is a plain protocol, we combine it with the next part.
|
|
if (urls[0].match(/^[^/:]+:\/*$/) && urls.length > 1) {
|
|
const first = urls.shift();
|
|
urls[0] = first + urls[0];
|
|
}
|
|
|
|
// There must be two or three slashes in the file protocol, two slashes in anything else.
|
|
if (urls[0].match(/^file:\/\/\//)) {
|
|
urls[0] = urls[0].replace(/^([^/:]+):\/*/, '$1:///');
|
|
} else {
|
|
urls[0] = urls[0].replace(/^([^/:]+):\/*/, '$1://');
|
|
}
|
|
|
|
// eslint-disable-next-line
|
|
for (let i = 0; i < urls.length; i++) {
|
|
let component = urls[i];
|
|
|
|
if (typeof component !== 'string') {
|
|
throw new TypeError(`Url must be a string. Received ${component}`);
|
|
}
|
|
|
|
if (component === '') {
|
|
// eslint-disable-next-line
|
|
continue;
|
|
}
|
|
|
|
if (i > 0) {
|
|
// Removing the starting slashes for each component but the first.
|
|
component = component.replace(/^[/]+/, '');
|
|
}
|
|
if (i < urls.length - 1) {
|
|
// Removing the ending slashes for each component but the last.
|
|
component = component.replace(/[/]+$/, '');
|
|
} else {
|
|
// For the last component we will combine multiple slashes to a single one.
|
|
component = component.replace(/[/]+$/, '/');
|
|
}
|
|
|
|
resultArray.push(component);
|
|
}
|
|
|
|
let str = resultArray.join('/');
|
|
// Each input component is now separated by a single slash except the possible first plain protocol part.
|
|
|
|
// remove trailing slash before parameters or hash
|
|
str = str.replace(/\/(\?|&|#[^!])/g, '$1');
|
|
|
|
// replace ? in parameters with &
|
|
const parts = str.split('?');
|
|
str = parts.shift() + (parts.length > 0 ? '?' : '') + parts.join('&');
|
|
|
|
// dedupe forward slashes
|
|
str = str.replace(/^\/+/, '/');
|
|
|
|
return str;
|
|
}
|
|
|
|
module.exports = {
|
|
encodePath,
|
|
generate,
|
|
fileToPath,
|
|
fileToComponentName,
|
|
getSubFolder,
|
|
idx,
|
|
normalizeUrl,
|
|
parse,
|
|
};
|