Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(blog): authors page #10216

Merged
merged 125 commits into from
Aug 1, 2024
Merged
Show file tree
Hide file tree
Changes from 26 commits
Commits
Show all changes
125 commits
Select commit Hold shift + click to select a range
269eaaa
feat(blog): authors page
OzakIOne Jun 13, 2024
f734a90
fix build
OzakIOne Jun 13, 2024
29b3a64
refactor: apply lint autofix
OzakIOne Jun 13, 2024
f1a4427
fix typo
OzakIOne Jun 14, 2024
1792c78
refactor: apply lint autofix
OzakIOne Jun 14, 2024
e1c0e5e
fix tests
OzakIOne Jun 14, 2024
20e5f29
fix tests
OzakIOne Jun 14, 2024
d018eb6
fix typecheck
OzakIOne Jun 14, 2024
f29b69a
update styles
OzakIOne Jun 14, 2024
cb4cc78
fix test?
OzakIOne Jun 16, 2024
6e66bc7
revert
OzakIOne Jun 16, 2024
fc7f7d0
remove socials & support unnamed author
OzakIOne Jun 17, 2024
cac2f99
remove duplication and inline check
OzakIOne Jun 17, 2024
65db409
feat: permalink support
OzakIOne Jun 18, 2024
a6dbf71
feat: author description
OzakIOne Jun 18, 2024
2f725cb
spread properties
OzakIOne Jun 19, 2024
8bf3e4f
refactor
OzakIOne Jun 19, 2024
2f69b6d
refactor: image base url
OzakIOne Jun 19, 2024
80f368a
refactor: types
OzakIOne Jun 19, 2024
574d45a
style: use blog post author ui
OzakIOne Jun 19, 2024
ff32178
fix bug when permalink is other author key with slash
OzakIOne Jun 19, 2024
d4f892f
refactor
OzakIOne Jun 19, 2024
788b198
refactor: apply lint autofix
OzakIOne Jun 19, 2024
16b0dbf
refactor
OzakIOne Jun 20, 2024
5c789b3
refactor remove pageAuthor prop, use author instead
OzakIOne Jun 20, 2024
151f49f
Revert "refactor remove pageAuthor prop, use author instead"
OzakIOne Jun 20, 2024
0a35323
move authorsUtils to blog
OzakIOne Jun 23, 2024
ce545b0
Merge branch 'main' into ozaki/mediaShareButtons
slorber Jun 24, 2024
da37991
Refactor blog author pages code and types
slorber Jun 24, 2024
d6df47c
move useBlogPostsPlural & listByLetters
OzakIOne Jun 25, 2024
6258f80
move for consistency with tags components
OzakIOne Jun 25, 2024
cf5f2ea
update theme path
OzakIOne Jun 25, 2024
6fcd67f
fix test
OzakIOne Jun 25, 2024
6b62c98
fix tests
OzakIOne Jun 25, 2024
485435d
Merge branch 'main' into ozaki/mediaShareButtons
OzakIOne Jun 27, 2024
ba12369
fix type
OzakIOne Jun 27, 2024
ad66da7
typecheck
OzakIOne Jun 28, 2024
f538f5f
refactor: move logic in contentLoaded
OzakIOne Jul 1, 2024
a4fdaf1
move code
OzakIOne Jul 1, 2024
ece683a
Merge branch 'main' into ozaki/mediaShareButtons
OzakIOne Jul 1, 2024
aec2154
fix typecheck
OzakIOne Jul 1, 2024
44c426a
fix authorinput type mismatch
slorber Jul 2, 2024
051f473
Merge remote-tracking branch 'origin/ozaki/mediaShareButtons' into oz…
slorber Jul 2, 2024
e85a803
fix types
slorber Jul 2, 2024
a9bf820
little refactors
slorber Jul 2, 2024
6a1ff0a
refactor (pagination not working)
OzakIOne Jul 8, 2024
ae795a3
pagination
OzakIOne Jul 9, 2024
d4a64bb
refactor
OzakIOne Jul 9, 2024
66d457b
refactor list letter & permalink collision
OzakIOne Jul 9, 2024
b8a5ae9
remove authorKey fallback for name
OzakIOne Jul 9, 2024
9d6d8d6
review
OzakIOne Jul 9, 2024
5893259
fix test
OzakIOne Jul 9, 2024
c280f02
refactor: apply lint autofix
OzakIOne Jul 9, 2024
78ab5ca
refactor review
OzakIOne Jul 9, 2024
12962f4
refactor
OzakIOne Jul 10, 2024
5693caf
fix tests
OzakIOne Jul 10, 2024
2e1fc03
refactor
OzakIOne Jul 10, 2024
232c4ac
refactor
OzakIOne Jul 10, 2024
bef4e4d
remove unused key
OzakIOne Jul 10, 2024
bb8e554
refactor: apply lint autofix
OzakIOne Jul 10, 2024
e29650d
autolint
OzakIOne Jul 10, 2024
8e51453
refactor: apply lint autofix
OzakIOne Jul 10, 2024
e5c2f38
Merge branch 'main' into ozaki/mediaShareButtons
OzakIOne Jul 11, 2024
72c3923
refactor
OzakIOne Jul 12, 2024
a3c75ed
refactor
OzakIOne Jul 12, 2024
0a739e1
Merge branch 'main' into ozaki/mediaShareButtons
OzakIOne Jul 12, 2024
4d370b9
refactor: apply lint autofix
OzakIOne Jul 12, 2024
1c410e1
refactor
OzakIOne Jul 15, 2024
3823d22
refactor: apply lint autofix
OzakIOne Jul 15, 2024
36ad47e
refactor
OzakIOne Jul 15, 2024
45e7c38
UI test
OzakIOne Jul 17, 2024
0c5a122
fix test
OzakIOne Jul 17, 2024
00fc21d
ui test
OzakIOne Jul 22, 2024
e2d6aab
Merge branch 'main' into ozaki/mediaShareButtons
slorber Jul 25, 2024
9d7ee53
move useBlogPostsPlural to internal
slorber Jul 25, 2024
a092d1d
add useful groupBlogPostsByAuthorKey comment
slorber Jul 25, 2024
6044010
add tests for groupBlogPostsByAuthorKey
slorber Jul 25, 2024
0260a3b
narrow typing for AuthorsMap
slorber Jul 25, 2024
3a86873
rename checkAuthorsMapPermalinkCollisions
slorber Jul 25, 2024
4f48e9e
refactor getDataFileData to readDataFile
slorber Jul 25, 2024
12d72c6
move relevant tests to authorsMap test file
slorber Jul 25, 2024
9553099
remove useless function
slorber Jul 25, 2024
ca2afb3
rename toAuthorItemProp
slorber Jul 25, 2024
7268707
revert generic listByLetters generic util
slorber Jul 25, 2024
493f3d7
revert generic listByLetters generic util
slorber Jul 25, 2024
30d076b
bad import
slorber Jul 26, 2024
e4da374
useless label
slorber Jul 26, 2024
50f209c
rename test file to match impl
slorber Jul 26, 2024
9d47ed1
remove useless lastUpdated in test output
slorber Jul 26, 2024
bd1c3d1
remove useless lastUpdated in test output
slorber Jul 26, 2024
64fa38d
add integration test for localized blog author page permalink
slorber Jul 26, 2024
519d2a9
refactor: apply lint autofix
slorber Jul 26, 2024
7927b5d
commit typo
slorber Jul 26, 2024
3b6845d
Merge remote-tracking branch 'origin/ozaki/mediaShareButtons' into oz…
slorber Jul 26, 2024
ccfc506
refactor: apply lint autofix
slorber Jul 26, 2024
befb552
refactor blog translations - introduce AuthorWithKey type
slorber Jul 26, 2024
7376f6f
docs: add authors page section
OzakIOne Jul 26, 2024
c6b2695
Merge remote-tracking branch 'origin/ozaki/mediaShareButtons' into oz…
slorber Jul 26, 2024
1025362
simplify routes => sync code
slorber Jul 26, 2024
295f7a1
add authorsListPath to global blog metadata bundle
slorber Jul 26, 2024
cbd14e3
rename things for consistency
slorber Jul 26, 2024
8620ef4
rename things for consistency
slorber Jul 26, 2024
de0119f
remove useless typedefs
slorber Jul 26, 2024
9d9a845
refactor blog theme components organization
slorber Jul 26, 2024
7aaa68b
paginateBlogPosts should generate first page even when there's 0 items
slorber Jul 26, 2024
2b26fbd
Dogfood custom author page permalink
slorber Jul 26, 2024
2425144
remove unused css
slorber Jul 26, 2024
ad929c0
rename ThemeClassNames for consistency
slorber Jul 26, 2024
12b3956
move count outside link
slorber Jul 26, 2024
0ebe271
polish display code
slorber Jul 26, 2024
66bb85b
swizzle config
slorber Jul 29, 2024
ea494ca
fix changelog plugin
slorber Jul 29, 2024
3cd4f9f
feat: Add authors page configuration option to the blog plugin
OzakIOne Jul 29, 2024
63f8785
docs: clarify explanation
OzakIOne Jul 29, 2024
a23f8ec
refactor: apply lint autofix
OzakIOne Jul 29, 2024
ab48912
Merge branch 'main' into ozaki/mediaShareButtons
slorber Aug 1, 2024
de556c9
docs, rename authorsPagePath to authorsBasePath
slorber Aug 1, 2024
75dd81f
useful comment on BlopAuthor
slorber Aug 1, 2024
b3196f5
simplify authors pages doc
slorber Aug 1, 2024
25546d4
simplify authors pages doc
slorber Aug 1, 2024
35e704d
simplify authors pages doc
slorber Aug 1, 2024
e19765d
simplify authors pages doc
slorber Aug 1, 2024
c632270
Add missing plugin options
slorber Aug 1, 2024
8b97d27
Add Authors File api ref
slorber Aug 1, 2024
0b8b103
Polish CSS of blog authors pages
slorber Aug 1, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -147,6 +147,7 @@ exports[`blog plugin process blog posts load content 2`] = `
"permalink": "/blog/another/tags",
"title": "Another With Tag",
},
"pageAuthors": [],
"permalink": "/blog/simple/slug/another",
"readingTime": 0.015,
"source": "@site/blog/another-simple-slug-with-tags.md",
Expand Down Expand Up @@ -186,6 +187,7 @@ exports[`blog plugin process blog posts load content 2`] = `
"permalink": "/blog/another/tags2",
"title": "Another With Tag",
},
"pageAuthors": [],
"permalink": "/blog/another/tags",
"prevItem": {
"permalink": "/blog/simple/slug/another",
Expand Down Expand Up @@ -231,6 +233,7 @@ exports[`blog plugin process blog posts load content 2`] = `
"hasTruncateMarker": false,
"lastUpdatedAt": undefined,
"lastUpdatedBy": undefined,
"pageAuthors": [],
"permalink": "/blog/another/tags2",
"prevItem": {
"permalink": "/blog/another/tags",
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -202,7 +202,13 @@ describe('getBlogPostAuthors', () => {
authorsMap: undefined,
baseUrl: '/',
}),
).toEqual([{name: 'Sébastien Lorber', title: 'maintainer'}]);
).toEqual([
{
name: 'Sébastien Lorber',
title: 'maintainer',
imageURL: undefined,
},
]);
});

it('can read authors Author[]', () => {
Expand All @@ -218,8 +224,12 @@ describe('getBlogPostAuthors', () => {
baseUrl: '/',
}),
).toEqual([
{name: 'Sébastien Lorber', title: 'maintainer'},
{name: 'Yangshun Tay'},
{
name: 'Sébastien Lorber',
title: 'maintainer',
imageURL: undefined,
},
{name: 'Yangshun Tay', imageURL: undefined},
]);
});

Expand Down Expand Up @@ -250,8 +260,9 @@ describe('getBlogPostAuthors', () => {
name: 'Yangshun Tay',
title: 'Yangshun title local override',
extra: 42,
imageURL: undefined,
},
{name: 'Alexey'},
{name: 'Alexey', imageURL: undefined},
]);
});

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -196,6 +196,7 @@ describe('blog plugin', () => {
permalink: '/blog/tags/date',
},
],
pageAuthors: [],
nextItem: {
permalink: '/blog/2018/12/14/Happy-First-Birthday-Slash',
title: 'Happy 1st Birthday Slash! (translated)',
Expand All @@ -220,12 +221,14 @@ describe('blog plugin', () => {
authors: [
{
name: 'Yangshun Tay (translated)',
imageURL: undefined,
},
{
email: 'lorber.sebastien@gmail.com',
key: 'slorber',
name: 'Sébastien Lorber (translated)',
title: 'Docusaurus maintainer (translated)',
imageURL: undefined,
},
],
date: new Date('2018-12-14'),
Expand Down Expand Up @@ -253,6 +256,7 @@ describe('blog plugin', () => {
permalink: '/blog/tags/global-tag-permalink (en)',
},
],
pageAuthors: [],
prevItem: {
permalink: '/blog/date-matter',
title: 'date-matter',
Expand Down Expand Up @@ -299,6 +303,7 @@ describe('blog plugin', () => {
permalink: '/blog/tags/complex',
},
],
pageAuthors: [],
hasTruncateMarker: false,
unlisted: false,
});
Expand Down Expand Up @@ -336,6 +341,7 @@ describe('blog plugin', () => {
title: 'Simple Slug',
},
tags: [],
pageAuthors: [],
hasTruncateMarker: false,
unlisted: false,
});
Expand All @@ -357,6 +363,7 @@ describe('blog plugin', () => {
},
prevItem: undefined,
tags: [],
pageAuthors: [],
nextItem: {
permalink: '/blog/date-matter',
title: 'date-matter',
Expand Down Expand Up @@ -481,6 +488,7 @@ describe('blog plugin', () => {
date: noDateSourceTime,
frontMatter: {},
tags: [],
pageAuthors: [],
prevItem: undefined,
nextItem: undefined,
hasTruncateMarker: false,
Expand Down
65 changes: 47 additions & 18 deletions packages/docusaurus-plugin-content-blog/src/authors.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,11 +5,11 @@
* LICENSE file in the root directory of this source tree.
*/

import {getDataFileData, normalizeUrl} from '@docusaurus/utils';
import _ from 'lodash';
import {getDataFileData, normalizeUrl, type Author} from '@docusaurus/utils';
import {Joi, URISchema} from '@docusaurus/utils-validation';
import type {BlogContentPaths} from './types';
import type {
Author,
BlogPostFrontMatter,
BlogPostFrontMatterAuthor,
BlogPostFrontMatterAuthors,
Expand All @@ -26,6 +26,9 @@ const AuthorsMapSchema = Joi.object<AuthorsMap>()
imageURL: URISchema,
title: Joi.string(),
email: Joi.string(),
generateAuthorPage: Joi.bool(),
permalink: Joi.string(),
description: Joi.string(),
})
.rename('image_url', 'imageURL')
.or('name', 'imageURL')
Expand Down Expand Up @@ -133,7 +136,7 @@ function normalizeFrontMatterAuthors(
}

function getFrontMatterAuthors(params: AuthorsParam): Author[] {
const {authorsMap} = params;
const {authorsMap, baseUrl} = params;
const frontMatterAuthors = normalizeFrontMatterAuthors(
params.frontMatter.authors,
);
Expand All @@ -158,36 +161,29 @@ ${Object.keys(authorsMap)
}

function toAuthor(frontMatterAuthor: BlogPostFrontMatterAuthor): Author {
return {
const author = {
// Author def from authorsMap can be locally overridden by front matter
...getAuthorsMapAuthor(frontMatterAuthor.key),
...frontMatterAuthor,
};

return {
...author,
imageURL: normalizeImageUrl({imageURL: author.imageURL, baseUrl}),
};
}

return frontMatterAuthors.map(toAuthor);
}

function fixAuthorImageBaseURL(
authors: Author[],
{baseUrl}: {baseUrl: string},
) {
return authors.map((author) => ({
...author,
imageURL: normalizeImageUrl({imageURL: author.imageURL, baseUrl}),
}));
}

export function getBlogPostAuthors(params: AuthorsParam): Author[] {
const authorLegacy = getFrontMatterAuthorLegacy(params);
const authors = getFrontMatterAuthors(params);

const updatedAuthors = fixAuthorImageBaseURL(authors, params);

if (authorLegacy) {
// Technically, we could allow mixing legacy/authors front matter, but do we
// really want to?
if (updatedAuthors.length > 0) {
if (authors.length > 0) {
throw new Error(
`To declare blog post authors, use the 'authors' front matter in priority.
Don't mix 'authors' with other existing 'author_*' front matter. Choose one or the other, not both at the same time.`,
Expand All @@ -196,5 +192,38 @@ Don't mix 'authors' with other existing 'author_*' front matter. Choose one or t
return [authorLegacy];
}

return updatedAuthors;
return authors;
}

export function checkPermalinkCollisions(
authorsMap: AuthorsMap | undefined,
): void {
const pageAuthorsMap = _.pickBy(
authorsMap,
(author) => author.generateAuthorPage === true,
);

const permalinkCounts: {[key: string]: string[]} = {};

for (const [key, author] of Object.entries(pageAuthorsMap)) {
const permalink = normalizeUrl(['/', author.permalink || key, '/']);
if (!permalinkCounts[permalink]) {
permalinkCounts[permalink] = [];
}
permalinkCounts[permalink]?.push(author.name || key);
}

const collisions = Object.entries(permalinkCounts).filter(
([, authors]) => authors.length > 1,
);

if (collisions.length > 0) {
let errorMessage = 'The following permalinks are duplicated:\n';

collisions.forEach(([permalink, authors]) => {
errorMessage += `Permalink: ${permalink}\nAuthors: ${authors.join(', ')}`;
});

throw new Error(errorMessage);
}
OzakIOne marked this conversation as resolved.
Show resolved Hide resolved
}
60 changes: 58 additions & 2 deletions packages/docusaurus-plugin-content-blog/src/blogUtils.ts
Original file line number Diff line number Diff line change
Expand Up @@ -26,10 +26,18 @@ import {
isDraft,
readLastUpdateData,
normalizeTags,
getAuthorVisibility,
groupAuthoredItems,
normalizePageAuthors,
} from '@docusaurus/utils';
import {getTagsFile} from '@docusaurus/utils-validation';
import {validateBlogPostFrontMatter} from './frontMatter';
import {type AuthorsMap, getAuthorsMap, getBlogPostAuthors} from './authors';
import {
type AuthorsMap,
getAuthorsMap,
getBlogPostAuthors,
checkPermalinkCollisions,
} from './authors';
import type {TagsFile} from '@docusaurus/utils';
import type {LoadContext, ParseFrontMatter} from '@docusaurus/types';
import type {
Expand All @@ -38,6 +46,7 @@ import type {
BlogPost,
BlogTags,
BlogPaginated,
BlogPageAuthors,
} from '@docusaurus/plugin-content-blog';
import type {BlogContentPaths} from './types';

Expand Down Expand Up @@ -134,6 +143,41 @@ export function getBlogTags({
});
}

export function getBlogPageAuthors({
OzakIOne marked this conversation as resolved.
Show resolved Hide resolved
blogPosts,
...params
}: {
blogPosts: BlogPost[];
blogTitle: string;
blogDescription: string;
postsPerPageOption: number | 'ALL';
pageBasePath: string;
}): BlogPageAuthors {
const getPostPageAuthors = (blogPost: BlogPost) =>
blogPost.metadata.pageAuthors;

const groups = groupAuthoredItems(blogPosts, getPostPageAuthors);

return _.mapValues(groups, ({author, items: authorBlogPosts}) => {
const authorVisibility = getAuthorVisibility({
items: authorBlogPosts,
isUnlisted: (item: BlogPost) => item.metadata.unlisted,
});
return {
...author,
items: authorVisibility.listedItems.map((item: BlogPost) => item.id),
pages: author.permalink
? paginateBlogPosts({
blogPosts: authorVisibility.listedItems,
basePageUrl: author.permalink,
...params,
})
: [],
unlisted: authorVisibility.unlisted,
};
});
}

const DATE_FILENAME_REGEX =
/^(?<folder>.*)(?<date>\d{4}[-/]\d{1,2}[-/]\d{1,2})[-/]?(?<text>.*?)(?:\/index)?.mdx?$/;

Expand Down Expand Up @@ -317,7 +361,6 @@ async function processBlogSourceFile(
routeBasePath,
tagsRouteBasePath,
]);
const authors = getBlogPostAuthors({authorsMap, frontMatter, baseUrl});

const tags = normalizeTags({
options,
Expand All @@ -327,6 +370,16 @@ async function processBlogSourceFile(
tagsFile,
});

const authors = getBlogPostAuthors({authorsMap, frontMatter, baseUrl});
const authorsBaseRoutePath = options.authorsPageBasePath
? normalizeUrl([baseUrl, routeBasePath, options.authorsPageBasePath])
: '';

const pageAuthors = normalizePageAuthors({
authorsBaseRoutePath,
authors,
});
OzakIOne marked this conversation as resolved.
Show resolved Hide resolved

return {
id: slug,
metadata: {
Expand All @@ -350,6 +403,7 @@ async function processBlogSourceFile(
unlisted,
lastUpdatedAt: lastUpdate.lastUpdatedAt,
lastUpdatedBy: lastUpdate.lastUpdatedBy,
pageAuthors,
},
content,
};
Expand All @@ -376,6 +430,8 @@ export async function generateBlogPosts(
authorsMapPath: options.authorsMapPath,
});

checkPermalinkCollisions(authorsMap);

const tagsFile = await getTagsFile({contentPaths, tags: options.tags});

async function doProcessBlogSourceFile(blogSourceFile: string) {
Expand Down
3 changes: 1 addition & 2 deletions packages/docusaurus-plugin-content-blog/src/feed.ts
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@ import fs from 'fs-extra';
import logger from '@docusaurus/logger';
import {Feed, type Author as FeedAuthor} from 'feed';
import * as srcset from 'srcset';
import {normalizeUrl, readOutputHTMLFile} from '@docusaurus/utils';
import {normalizeUrl, readOutputHTMLFile, type Author} from '@docusaurus/utils';
import {
blogPostContainerID,
applyTrailingSlash,
Expand All @@ -20,7 +20,6 @@ import type {DocusaurusConfig, HtmlTags, LoadContext} from '@docusaurus/types';
import type {
FeedType,
PluginOptions,
Author,
BlogPost,
BlogFeedItem,
} from '@docusaurus/plugin-content-blog';
Expand Down
Loading
Loading