mirror of
https://github.com/facebook/docusaurus.git
synced 2025-06-04 11:52:39 +02:00
* refactor(utils): reorganize functions; move authors file resolution to utils * More refactor
420 lines
11 KiB
TypeScript
420 lines
11 KiB
TypeScript
/**
|
|
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
*
|
|
* This source code is licensed under the MIT license found in the
|
|
* LICENSE file in the root directory of this source tree.
|
|
*/
|
|
|
|
import {
|
|
type AuthorsMap,
|
|
getAuthorsMap,
|
|
getBlogPostAuthors,
|
|
validateAuthorsMap,
|
|
} from '../authors';
|
|
import path from 'path';
|
|
|
|
describe('getBlogPostAuthors', () => {
|
|
test('can read no authors', () => {
|
|
expect(
|
|
getBlogPostAuthors({
|
|
frontMatter: {},
|
|
authorsMap: undefined,
|
|
}),
|
|
).toEqual([]);
|
|
expect(
|
|
getBlogPostAuthors({
|
|
frontMatter: {
|
|
authors: [],
|
|
},
|
|
authorsMap: undefined,
|
|
}),
|
|
).toEqual([]);
|
|
});
|
|
|
|
test('can read author from legacy frontmatter', () => {
|
|
expect(
|
|
getBlogPostAuthors({
|
|
frontMatter: {
|
|
author: 'Sébastien Lorber',
|
|
},
|
|
authorsMap: undefined,
|
|
}),
|
|
).toEqual([{name: 'Sébastien Lorber'}]);
|
|
expect(
|
|
getBlogPostAuthors({
|
|
frontMatter: {
|
|
authorTitle: 'maintainer',
|
|
},
|
|
authorsMap: undefined,
|
|
}),
|
|
).toEqual([{title: 'maintainer'}]);
|
|
expect(
|
|
getBlogPostAuthors({
|
|
frontMatter: {
|
|
authorImageURL: 'https://github.com/slorber.png',
|
|
},
|
|
authorsMap: undefined,
|
|
}),
|
|
).toEqual([{imageURL: 'https://github.com/slorber.png'}]);
|
|
expect(
|
|
getBlogPostAuthors({
|
|
frontMatter: {
|
|
author: 'Sébastien Lorber',
|
|
author_title: 'maintainer1',
|
|
authorTitle: 'maintainer2',
|
|
author_image_url: 'https://github.com/slorber1.png',
|
|
authorImageURL: 'https://github.com/slorber2.png',
|
|
author_url: 'https://github.com/slorber1',
|
|
authorURL: 'https://github.com/slorber2',
|
|
},
|
|
authorsMap: undefined,
|
|
}),
|
|
).toEqual([
|
|
{
|
|
name: 'Sébastien Lorber',
|
|
title: 'maintainer1',
|
|
imageURL: 'https://github.com/slorber1.png',
|
|
url: 'https://github.com/slorber1',
|
|
},
|
|
]);
|
|
});
|
|
|
|
test('can read authors string', () => {
|
|
expect(
|
|
getBlogPostAuthors({
|
|
frontMatter: {
|
|
authors: 'slorber',
|
|
},
|
|
authorsMap: {slorber: {name: 'Sébastien Lorber'}},
|
|
}),
|
|
).toEqual([{key: 'slorber', name: 'Sébastien Lorber'}]);
|
|
});
|
|
|
|
test('can read authors string[]', () => {
|
|
expect(
|
|
getBlogPostAuthors({
|
|
frontMatter: {
|
|
authors: ['slorber', 'yangshun'],
|
|
},
|
|
authorsMap: {
|
|
slorber: {name: 'Sébastien Lorber', title: 'maintainer'},
|
|
yangshun: {name: 'Yangshun Tay'},
|
|
},
|
|
}),
|
|
).toEqual([
|
|
{key: 'slorber', name: 'Sébastien Lorber', title: 'maintainer'},
|
|
{key: 'yangshun', name: 'Yangshun Tay'},
|
|
]);
|
|
});
|
|
|
|
test('can read authors Author', () => {
|
|
expect(
|
|
getBlogPostAuthors({
|
|
frontMatter: {
|
|
authors: {name: 'Sébastien Lorber', title: 'maintainer'},
|
|
},
|
|
authorsMap: undefined,
|
|
}),
|
|
).toEqual([{name: 'Sébastien Lorber', title: 'maintainer'}]);
|
|
});
|
|
|
|
test('can read authors Author[]', () => {
|
|
expect(
|
|
getBlogPostAuthors({
|
|
frontMatter: {
|
|
authors: [
|
|
{name: 'Sébastien Lorber', title: 'maintainer'},
|
|
{name: 'Yangshun Tay'},
|
|
],
|
|
},
|
|
authorsMap: undefined,
|
|
}),
|
|
).toEqual([
|
|
{name: 'Sébastien Lorber', title: 'maintainer'},
|
|
{name: 'Yangshun Tay'},
|
|
]);
|
|
});
|
|
|
|
test('can read authors complex (string | Author)[] setup with keys and local overrides', () => {
|
|
expect(
|
|
getBlogPostAuthors({
|
|
frontMatter: {
|
|
authors: [
|
|
'slorber',
|
|
{
|
|
key: 'yangshun',
|
|
title: 'Yangshun title local override',
|
|
extra: 42,
|
|
},
|
|
{name: 'Alexey'},
|
|
],
|
|
},
|
|
authorsMap: {
|
|
slorber: {name: 'Sébastien Lorber', title: 'maintainer'},
|
|
yangshun: {name: 'Yangshun Tay', title: 'Yangshun title original'},
|
|
},
|
|
}),
|
|
).toEqual([
|
|
{key: 'slorber', name: 'Sébastien Lorber', title: 'maintainer'},
|
|
{
|
|
key: 'yangshun',
|
|
name: 'Yangshun Tay',
|
|
title: 'Yangshun title local override',
|
|
extra: 42,
|
|
},
|
|
{name: 'Alexey'},
|
|
]);
|
|
});
|
|
|
|
test('throw when using author key with no authorsMap', () => {
|
|
expect(() =>
|
|
getBlogPostAuthors({
|
|
frontMatter: {
|
|
authors: 'slorber',
|
|
},
|
|
authorsMap: undefined,
|
|
}),
|
|
).toThrowErrorMatchingInlineSnapshot(`
|
|
"Can't reference blog post authors by a key (such as 'slorber') because no authors map file could be loaded.
|
|
Please double-check your blog plugin config (in particular 'authorsMapPath'), ensure the file exists at the configured path, is not empty, and is valid!"
|
|
`);
|
|
});
|
|
|
|
test('throw when using author key with empty authorsMap', () => {
|
|
expect(() =>
|
|
getBlogPostAuthors({
|
|
frontMatter: {
|
|
authors: 'slorber',
|
|
},
|
|
authorsMap: {},
|
|
}),
|
|
).toThrowErrorMatchingInlineSnapshot(`
|
|
"Can't reference blog post authors by a key (such as 'slorber') because no authors map file could be loaded.
|
|
Please double-check your blog plugin config (in particular 'authorsMapPath'), ensure the file exists at the configured path, is not empty, and is valid!"
|
|
`);
|
|
});
|
|
|
|
test('throw when using bad author key in string', () => {
|
|
expect(() =>
|
|
getBlogPostAuthors({
|
|
frontMatter: {
|
|
authors: 'slorber',
|
|
},
|
|
authorsMap: {
|
|
yangshun: {name: 'Yangshun Tay'},
|
|
jmarcey: {name: 'Joel Marcey'},
|
|
},
|
|
}),
|
|
).toThrowErrorMatchingInlineSnapshot(`
|
|
"Blog author with key \\"slorber\\" not found in the authors map file.
|
|
Valid author keys are:
|
|
- yangshun
|
|
- jmarcey"
|
|
`);
|
|
});
|
|
|
|
test('throw when using bad author key in string[]', () => {
|
|
expect(() =>
|
|
getBlogPostAuthors({
|
|
frontMatter: {
|
|
authors: ['yangshun', 'jmarcey', 'slorber'],
|
|
},
|
|
authorsMap: {
|
|
yangshun: {name: 'Yangshun Tay'},
|
|
jmarcey: {name: 'Joel Marcey'},
|
|
},
|
|
}),
|
|
).toThrowErrorMatchingInlineSnapshot(`
|
|
"Blog author with key \\"slorber\\" not found in the authors map file.
|
|
Valid author keys are:
|
|
- yangshun
|
|
- jmarcey"
|
|
`);
|
|
});
|
|
|
|
test('throw when using bad author key in Author[].key', () => {
|
|
expect(() =>
|
|
getBlogPostAuthors({
|
|
frontMatter: {
|
|
authors: [{key: 'yangshun'}, {key: 'jmarcey'}, {key: 'slorber'}],
|
|
},
|
|
authorsMap: {
|
|
yangshun: {name: 'Yangshun Tay'},
|
|
jmarcey: {name: 'Joel Marcey'},
|
|
},
|
|
}),
|
|
).toThrowErrorMatchingInlineSnapshot(`
|
|
"Blog author with key \\"slorber\\" not found in the authors map file.
|
|
Valid author keys are:
|
|
- yangshun
|
|
- jmarcey"
|
|
`);
|
|
});
|
|
|
|
test('throw when mixing legacy/new authors frontmatter', () => {
|
|
expect(() =>
|
|
getBlogPostAuthors({
|
|
frontMatter: {
|
|
authors: [{name: 'Sébastien Lorber'}],
|
|
author: 'Yangshun Tay',
|
|
},
|
|
authorsMap: undefined,
|
|
}),
|
|
).toThrowErrorMatchingInlineSnapshot(`
|
|
"To declare blog post authors, use the 'authors' FrontMatter in priority.
|
|
Don't mix 'authors' with other existing 'author_*' FrontMatter. Choose one or the other, not both at the same time."
|
|
`);
|
|
|
|
expect(() =>
|
|
getBlogPostAuthors({
|
|
frontMatter: {
|
|
authors: [{key: 'slorber'}],
|
|
author_title: 'legacy title',
|
|
},
|
|
authorsMap: {slorber: {name: 'Sébastien Lorber'}},
|
|
}),
|
|
).toThrowErrorMatchingInlineSnapshot(`
|
|
"To declare blog post authors, use the 'authors' FrontMatter in priority.
|
|
Don't mix 'authors' with other existing 'author_*' FrontMatter. Choose one or the other, not both at the same time."
|
|
`);
|
|
});
|
|
});
|
|
|
|
describe('getAuthorsMap', () => {
|
|
const fixturesDir = path.join(__dirname, '__fixtures__/authorsMapFiles');
|
|
const contentPaths = {
|
|
contentPathLocalized: fixturesDir,
|
|
contentPath: fixturesDir,
|
|
};
|
|
|
|
test('getAuthorsMap can read yml file', async () => {
|
|
expect(
|
|
await getAuthorsMap({
|
|
contentPaths,
|
|
authorsMapPath: 'authors.yml',
|
|
}),
|
|
).toBeDefined();
|
|
});
|
|
|
|
test('getAuthorsMap can read json file', async () => {
|
|
expect(
|
|
await getAuthorsMap({
|
|
contentPaths,
|
|
authorsMapPath: 'authors.json',
|
|
}),
|
|
).toBeDefined();
|
|
});
|
|
|
|
test('getAuthorsMap can return undefined if yaml file not found', async () => {
|
|
expect(
|
|
await getAuthorsMap({
|
|
contentPaths,
|
|
authorsMapPath: 'authors_does_not_exist.yml',
|
|
}),
|
|
).toBeUndefined();
|
|
});
|
|
});
|
|
|
|
describe('validateAuthorsMap', () => {
|
|
test('accept valid authors map', () => {
|
|
const authorsMap: AuthorsMap = {
|
|
slorber: {
|
|
name: 'Sébastien Lorber',
|
|
title: 'maintainer',
|
|
url: 'https://sebastienlorber.com',
|
|
imageURL: 'https://github.com/slorber.png',
|
|
},
|
|
yangshun: {
|
|
name: 'Yangshun Tay',
|
|
imageURL: 'https://github.com/yangshun.png',
|
|
randomField: 42,
|
|
},
|
|
jmarcey: {
|
|
name: 'Joel',
|
|
title: 'creator of Docusaurus',
|
|
hello: new Date(),
|
|
},
|
|
};
|
|
expect(validateAuthorsMap(authorsMap)).toEqual(authorsMap);
|
|
});
|
|
|
|
test('rename snake case image_url to camelCase imageURL', () => {
|
|
const authorsMap: AuthorsMap = {
|
|
slorber: {
|
|
name: 'Sébastien Lorber',
|
|
image_url: 'https://github.com/slorber.png',
|
|
},
|
|
};
|
|
expect(validateAuthorsMap(authorsMap)).toEqual({
|
|
slorber: {
|
|
name: 'Sébastien Lorber',
|
|
imageURL: 'https://github.com/slorber.png',
|
|
},
|
|
});
|
|
});
|
|
|
|
test('reject author without name', () => {
|
|
const authorsMap: AuthorsMap = {
|
|
slorber: {
|
|
image_url: 'https://github.com/slorber.png',
|
|
},
|
|
};
|
|
expect(() =>
|
|
validateAuthorsMap(authorsMap),
|
|
).toThrowErrorMatchingInlineSnapshot(`"\\"slorber.name\\" is required"`);
|
|
});
|
|
|
|
test('reject undefined author', () => {
|
|
expect(() =>
|
|
validateAuthorsMap({
|
|
slorber: undefined,
|
|
}),
|
|
).toThrowErrorMatchingInlineSnapshot(`"\\"slorber\\" is required"`);
|
|
});
|
|
|
|
test('reject null author', () => {
|
|
expect(() =>
|
|
validateAuthorsMap({
|
|
slorber: null,
|
|
}),
|
|
).toThrowErrorMatchingInlineSnapshot(
|
|
`"\\"slorber\\" must be of type object"`,
|
|
);
|
|
});
|
|
|
|
test('reject array author', () => {
|
|
expect(() =>
|
|
validateAuthorsMap({slorber: []}),
|
|
).toThrowErrorMatchingInlineSnapshot(
|
|
`"\\"slorber\\" must be of type object"`,
|
|
);
|
|
});
|
|
|
|
test('reject array content', () => {
|
|
expect(() => validateAuthorsMap([])).toThrowErrorMatchingInlineSnapshot(
|
|
// TODO improve this error message
|
|
`"\\"value\\" must be of type object"`,
|
|
);
|
|
});
|
|
|
|
test('reject flat author', () => {
|
|
expect(() =>
|
|
validateAuthorsMap({name: 'Sébastien'}),
|
|
).toThrowErrorMatchingInlineSnapshot(
|
|
// TODO improve this error message
|
|
`"\\"name\\" must be of type object"`,
|
|
);
|
|
});
|
|
|
|
test('reject non-map author', () => {
|
|
const authorsMap: AuthorsMap = {
|
|
// @ts-expect-error: for tests
|
|
slorber: [],
|
|
};
|
|
expect(() =>
|
|
validateAuthorsMap(authorsMap),
|
|
).toThrowErrorMatchingInlineSnapshot(
|
|
`"\\"slorber\\" must be of type object"`,
|
|
);
|
|
});
|
|
});
|