mirror of
https://github.com/facebook/docusaurus.git
synced 2025-05-06 13:47:29 +02:00
chore: prepare for docs sidebar
This commit is contained in:
parent
2141e6ea90
commit
b477863a30
9 changed files with 350 additions and 2 deletions
34
lib/load/docs/order.js
Normal file
34
lib/load/docs/order.js
Normal file
|
@ -0,0 +1,34 @@
|
||||||
|
// build the docs meta such as next, previous, category and sidebar
|
||||||
|
module.exports = function createOrder(allSidebars = {}) {
|
||||||
|
const order = {};
|
||||||
|
if (!allSidebars) {
|
||||||
|
return order;
|
||||||
|
}
|
||||||
|
Object.keys(allSidebars).forEach(sidebar => {
|
||||||
|
const categories = allSidebars[sidebar];
|
||||||
|
|
||||||
|
let ids = [];
|
||||||
|
const categoryOrder = [];
|
||||||
|
Object.keys(categories).forEach(category => {
|
||||||
|
ids = ids.concat(categories[category]);
|
||||||
|
for (let i = 0; i < categories[category].length; i++) {
|
||||||
|
categoryOrder.push(category);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
for (let i = 0; i < ids.length; i++) {
|
||||||
|
const id = ids[i];
|
||||||
|
let previous;
|
||||||
|
let next;
|
||||||
|
if (i > 0) previous = ids[i - 1];
|
||||||
|
if (i < ids.length - 1) next = ids[i + 1];
|
||||||
|
order[id] = {
|
||||||
|
previous,
|
||||||
|
next,
|
||||||
|
sidebar,
|
||||||
|
category: categoryOrder[i]
|
||||||
|
};
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return order;
|
||||||
|
};
|
34
lib/load/docs/sidebars.js
Normal file
34
lib/load/docs/sidebars.js
Normal file
|
@ -0,0 +1,34 @@
|
||||||
|
const fs = require('fs-extra');
|
||||||
|
const path = require('path');
|
||||||
|
const {idx} = require('../utils');
|
||||||
|
|
||||||
|
module.exports = function loadSidebars({siteDir, env}) {
|
||||||
|
let allSidebars = {};
|
||||||
|
|
||||||
|
// current sidebars
|
||||||
|
const sidebarsJSONFile = path.join(siteDir, 'sidebars.json');
|
||||||
|
if (fs.existsSync(sidebarsJSONFile)) {
|
||||||
|
allSidebars = require(sidebarsJSONFile);
|
||||||
|
}
|
||||||
|
|
||||||
|
// versioned sidebars
|
||||||
|
if (idx(env, ['versioning', 'enabled'])) {
|
||||||
|
const versions = idx(env, ['versioning', 'versions']);
|
||||||
|
versions &&
|
||||||
|
versions.forEach(version => {
|
||||||
|
const versionedSidebarsJSONFile = path.join(
|
||||||
|
siteDir,
|
||||||
|
'versioned_sidebars',
|
||||||
|
`version-${version}-sidebars.json`
|
||||||
|
);
|
||||||
|
if (fs.existsSync(versionedSidebarsJSONFile)) {
|
||||||
|
const sidebar = require(versionedSidebarsJSONFile);
|
||||||
|
Object.assign(allSidebars, sidebar);
|
||||||
|
} else {
|
||||||
|
const missingFile = path.relative(siteDir, versionedSidebarsJSONFile);
|
||||||
|
throw new Error(`Failed to load ${missingFile}. It does not exist.`);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return allSidebars;
|
||||||
|
};
|
|
@ -1,4 +1,6 @@
|
||||||
const path = require('path');
|
const path = require('path');
|
||||||
|
const fm = require('front-matter');
|
||||||
|
const escapeStringRegexp = require('escape-string-regexp');
|
||||||
const fs = require('fs-extra');
|
const fs = require('fs-extra');
|
||||||
|
|
||||||
const genPath = path.resolve(__dirname, '../core/generated');
|
const genPath = path.resolve(__dirname, '../core/generated');
|
||||||
|
@ -40,9 +42,40 @@ function fileToComponentName(file) {
|
||||||
return ext ? ext.toUpperCase() + str : str;
|
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};
|
||||||
|
}
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
encodePath,
|
encodePath,
|
||||||
generate,
|
generate,
|
||||||
fileToPath,
|
fileToPath,
|
||||||
fileToComponentName
|
fileToComponentName,
|
||||||
|
getSubFolder,
|
||||||
|
idx,
|
||||||
|
parse
|
||||||
};
|
};
|
||||||
|
|
|
@ -55,6 +55,7 @@
|
||||||
"connect-history-api-fallback": "^1.5.0",
|
"connect-history-api-fallback": "^1.5.0",
|
||||||
"css-loader": "^1.0.0",
|
"css-loader": "^1.0.0",
|
||||||
"escape-html": "^1.0.3",
|
"escape-html": "^1.0.3",
|
||||||
|
"escape-string-regexp": "^1.0.5",
|
||||||
"front-matter": "^2.3.0",
|
"front-matter": "^2.3.0",
|
||||||
"fs-extra": "^7.0.0",
|
"fs-extra": "^7.0.0",
|
||||||
"globby": "^8.0.1",
|
"globby": "^8.0.1",
|
||||||
|
|
59
test/load/docs/__snapshots__/order.test.js.snap
Normal file
59
test/load/docs/__snapshots__/order.test.js.snap
Normal file
|
@ -0,0 +1,59 @@
|
||||||
|
// Jest Snapshot v1, https://goo.gl/fbAQLP
|
||||||
|
|
||||||
|
exports[`createOrder should populate docs index from multiple sidebars 1`] = `
|
||||||
|
Object {
|
||||||
|
"doc1": Object {
|
||||||
|
"category": "Category1",
|
||||||
|
"next": "doc2",
|
||||||
|
"previous": undefined,
|
||||||
|
"sidebar": "docs",
|
||||||
|
},
|
||||||
|
"doc2": Object {
|
||||||
|
"category": "Category1",
|
||||||
|
"next": "doc3",
|
||||||
|
"previous": "doc1",
|
||||||
|
"sidebar": "docs",
|
||||||
|
},
|
||||||
|
"doc3": Object {
|
||||||
|
"category": "Category2",
|
||||||
|
"next": "doc4",
|
||||||
|
"previous": "doc2",
|
||||||
|
"sidebar": "docs",
|
||||||
|
},
|
||||||
|
"doc4": Object {
|
||||||
|
"category": "Category2",
|
||||||
|
"next": undefined,
|
||||||
|
"previous": "doc3",
|
||||||
|
"sidebar": "docs",
|
||||||
|
},
|
||||||
|
"doc5": Object {
|
||||||
|
"category": "Category1",
|
||||||
|
"next": undefined,
|
||||||
|
"previous": undefined,
|
||||||
|
"sidebar": "otherDocs",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
|
||||||
|
exports[`createOrder should resolve docs from older versions 1`] = `
|
||||||
|
Object {
|
||||||
|
"doc1": Object {
|
||||||
|
"category": "Category1",
|
||||||
|
"next": undefined,
|
||||||
|
"previous": undefined,
|
||||||
|
"sidebar": "docs",
|
||||||
|
},
|
||||||
|
"version-1.2.3-doc1": Object {
|
||||||
|
"category": "Category2",
|
||||||
|
"next": undefined,
|
||||||
|
"previous": "version-1.2.3-doc2",
|
||||||
|
"sidebar": "version-1.2.3-docs",
|
||||||
|
},
|
||||||
|
"version-1.2.3-doc2": Object {
|
||||||
|
"category": "Category1",
|
||||||
|
"next": "version-1.2.3-doc1",
|
||||||
|
"previous": undefined,
|
||||||
|
"sidebar": "version-1.2.3-docs",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
`;
|
45
test/load/docs/__snapshots__/sidebars.test.js.snap
Normal file
45
test/load/docs/__snapshots__/sidebars.test.js.snap
Normal file
|
@ -0,0 +1,45 @@
|
||||||
|
// Jest Snapshot v1, https://goo.gl/fbAQLP
|
||||||
|
|
||||||
|
exports[`loadSidebars normal site with sidebars 1`] = `
|
||||||
|
Object {
|
||||||
|
"docs": Object {
|
||||||
|
"Getting Started": Array [
|
||||||
|
"installation",
|
||||||
|
],
|
||||||
|
"Guides": Array [
|
||||||
|
"blog",
|
||||||
|
],
|
||||||
|
},
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
|
||||||
|
exports[`loadSidebars site with sidebars & versioned sidebars 1`] = `
|
||||||
|
Object {
|
||||||
|
"docs": Object {
|
||||||
|
"Getting Started": Array [
|
||||||
|
"installation",
|
||||||
|
],
|
||||||
|
"Guides": Array [
|
||||||
|
"blog",
|
||||||
|
],
|
||||||
|
},
|
||||||
|
"version-1.0.0-docs": Object {
|
||||||
|
"Getting Started": Array [
|
||||||
|
"version-1.0.0-installation",
|
||||||
|
],
|
||||||
|
"Guides": Array [
|
||||||
|
"version-1.0.0-blog",
|
||||||
|
],
|
||||||
|
},
|
||||||
|
"version-1.0.1-docs": Object {
|
||||||
|
"Getting Started": Array [
|
||||||
|
"version-1.0.1-installation",
|
||||||
|
],
|
||||||
|
"Guides": Array [
|
||||||
|
"version-1.0.1-blog",
|
||||||
|
],
|
||||||
|
},
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
|
||||||
|
exports[`loadSidebars site without sidebars 1`] = `Object {}`;
|
35
test/load/docs/order.test.js
Normal file
35
test/load/docs/order.test.js
Normal file
|
@ -0,0 +1,35 @@
|
||||||
|
import createOrder from '@lib/load/docs/order';
|
||||||
|
|
||||||
|
describe('createOrder', () => {
|
||||||
|
test('should populate docs index from multiple sidebars', () => {
|
||||||
|
const result = createOrder({
|
||||||
|
docs: {
|
||||||
|
Category1: ['doc1', 'doc2'],
|
||||||
|
Category2: ['doc3', 'doc4']
|
||||||
|
},
|
||||||
|
otherDocs: {
|
||||||
|
Category1: ['doc5']
|
||||||
|
}
|
||||||
|
});
|
||||||
|
expect(result).toMatchSnapshot();
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should resolve docs from older versions', () => {
|
||||||
|
const result = createOrder({
|
||||||
|
docs: {
|
||||||
|
Category1: ['doc1']
|
||||||
|
},
|
||||||
|
'version-1.2.3-docs': {
|
||||||
|
Category1: ['version-1.2.3-doc2'],
|
||||||
|
Category2: ['version-1.2.3-doc1']
|
||||||
|
}
|
||||||
|
});
|
||||||
|
expect(result).toMatchSnapshot();
|
||||||
|
});
|
||||||
|
|
||||||
|
test('edge cases', () => {
|
||||||
|
expect(createOrder({})).toEqual({});
|
||||||
|
expect(createOrder(null)).toEqual({});
|
||||||
|
expect(createOrder(undefined)).toEqual({});
|
||||||
|
});
|
||||||
|
});
|
46
test/load/docs/sidebars.test.js
Normal file
46
test/load/docs/sidebars.test.js
Normal file
|
@ -0,0 +1,46 @@
|
||||||
|
import path from 'path';
|
||||||
|
import loadSidebars from '@lib/load/docs/sidebars';
|
||||||
|
|
||||||
|
describe('loadSidebars', () => {
|
||||||
|
const fixtures = path.join(__dirname, '..', '__fixtures__');
|
||||||
|
test('normal site with sidebars', () => {
|
||||||
|
const env = {};
|
||||||
|
const siteDir = path.join(fixtures, 'simple-site');
|
||||||
|
const result = loadSidebars({siteDir, env});
|
||||||
|
expect(result).toMatchSnapshot();
|
||||||
|
});
|
||||||
|
|
||||||
|
test('site without sidebars', () => {
|
||||||
|
const env = {};
|
||||||
|
const siteDir = path.join(fixtures, 'bad-site');
|
||||||
|
const result = loadSidebars({siteDir, env});
|
||||||
|
expect(result).toMatchSnapshot();
|
||||||
|
});
|
||||||
|
|
||||||
|
test('site with sidebars & versioned sidebars', () => {
|
||||||
|
const env = {
|
||||||
|
versioning: {
|
||||||
|
enabled: true,
|
||||||
|
versions: ['1.0.1', '1.0.0']
|
||||||
|
}
|
||||||
|
};
|
||||||
|
const siteDir = path.join(fixtures, 'versioned-site');
|
||||||
|
const result = loadSidebars({siteDir, env});
|
||||||
|
expect(result).toMatchSnapshot();
|
||||||
|
});
|
||||||
|
|
||||||
|
test('site with missing versioned sidebars', () => {
|
||||||
|
const env = {
|
||||||
|
versioning: {
|
||||||
|
enabled: true,
|
||||||
|
versions: ['2.0.0']
|
||||||
|
}
|
||||||
|
};
|
||||||
|
const siteDir = path.join(fixtures, 'versioned-site');
|
||||||
|
expect(() => {
|
||||||
|
loadSidebars({siteDir, env});
|
||||||
|
}).toThrowErrorMatchingInlineSnapshot(
|
||||||
|
`"Failed to load versioned_sidebars/version-2.0.0-sidebars.json. It does not exist."`
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
|
@ -1,4 +1,10 @@
|
||||||
import {fileToPath, fileToComponentName} from '@lib/load/utils';
|
import path from 'path';
|
||||||
|
import {
|
||||||
|
fileToPath,
|
||||||
|
fileToComponentName,
|
||||||
|
idx,
|
||||||
|
getSubFolder
|
||||||
|
} from '@lib/load/utils';
|
||||||
|
|
||||||
describe('load utils', () => {
|
describe('load utils', () => {
|
||||||
test('fileToComponentName', () => {
|
test('fileToComponentName', () => {
|
||||||
|
@ -34,4 +40,59 @@ describe('load utils', () => {
|
||||||
expect(fileToPath(file)).toBe(asserts[file]);
|
expect(fileToPath(file)).toBe(asserts[file]);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test('idx', () => {
|
||||||
|
const a = {};
|
||||||
|
const b = {hello: 'world'};
|
||||||
|
const env = {
|
||||||
|
translation: {
|
||||||
|
enabled: true,
|
||||||
|
enabledLanguages: [
|
||||||
|
{
|
||||||
|
enabled: true,
|
||||||
|
name: 'English',
|
||||||
|
tag: 'en'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
enabled: true,
|
||||||
|
name: '日本語',
|
||||||
|
tag: 'ja'
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
versioning: {
|
||||||
|
enabled: false,
|
||||||
|
versions: []
|
||||||
|
}
|
||||||
|
};
|
||||||
|
const variable = 'enabledLanguages';
|
||||||
|
expect(idx(a, [('b', 'c')])).toBeUndefined();
|
||||||
|
expect(idx(b, ['hello'])).toEqual('world');
|
||||||
|
expect(idx(b, 'hello')).toEqual('world');
|
||||||
|
expect(idx(env, 'typo')).toBeUndefined();
|
||||||
|
expect(idx(env, 'versioning')).toEqual({
|
||||||
|
enabled: false,
|
||||||
|
versions: []
|
||||||
|
});
|
||||||
|
expect(idx(env, ['translation', 'enabled'])).toEqual(true);
|
||||||
|
expect(idx(env, ['translation', variable]).map(lang => lang.tag)).toEqual([
|
||||||
|
'en',
|
||||||
|
'ja'
|
||||||
|
]);
|
||||||
|
expect(idx(undefined)).toBeUndefined();
|
||||||
|
expect(idx(null)).toBeNull();
|
||||||
|
});
|
||||||
|
|
||||||
|
test('getSubFolder', () => {
|
||||||
|
const testA = path.join('folder', 'en', 'test.md');
|
||||||
|
const testB = path.join('folder', 'ja', 'test.md');
|
||||||
|
const testC = path.join('folder', 'ja', 'en', 'test.md');
|
||||||
|
const testD = path.join('docs', 'ro', 'test.md');
|
||||||
|
const testE = path.join('docs', 'test.md');
|
||||||
|
expect(getSubFolder(testA, 'folder')).toBe('en');
|
||||||
|
expect(getSubFolder(testB, 'folder')).toBe('ja');
|
||||||
|
expect(getSubFolder(testC, 'folder')).toBe('ja');
|
||||||
|
expect(getSubFolder(testD, 'docs')).toBe('ro');
|
||||||
|
expect(getSubFolder(testE, 'docs')).toBeNull();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
Loading…
Add table
Reference in a new issue