computer-scienceangular-roadmapbackend-roadmapblockchain-roadmapdba-roadmapdeveloper-roadmapdevops-roadmapfrontend-roadmapgo-roadmaphactoberfestjava-roadmapjavascript-roadmapnodejs-roadmappython-roadmapqa-roadmapreact-roadmaproadmapstudy-planvue-roadmapweb3-roadmap
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
120 lines
3.1 KiB
120 lines
3.1 KiB
import type { MarkdownFileType } from './file'; |
|
import { type AuthorFileType, getAllAuthors } from './author.ts'; |
|
|
|
export interface GuideFrontmatter { |
|
title: string; |
|
description: string; |
|
authorId: string; |
|
canonicalUrl?: string; |
|
// alternate path where this guide has been published |
|
excludedBySlug?: string; |
|
seo: { |
|
title: string; |
|
description: string; |
|
ogImageUrl?: string; |
|
}; |
|
isNew: boolean; |
|
type: 'visual' | 'textual'; |
|
date: string; |
|
sitemap: { |
|
priority: number; |
|
changefreq: 'daily' | 'weekly' | 'monthly' | 'yearly'; |
|
}; |
|
tags: string[]; |
|
} |
|
|
|
export type GuideFileType = MarkdownFileType<GuideFrontmatter> & { |
|
id: string; |
|
author: AuthorFileType; |
|
}; |
|
|
|
/** |
|
* Generates id from the given guide file |
|
* @param filePath Markdown file path |
|
* |
|
* @returns unique guide identifier |
|
*/ |
|
function guidePathToId(filePath: string): string { |
|
const fileName = filePath.split('/').pop() || ''; |
|
|
|
return fileName.replace('.md', ''); |
|
} |
|
|
|
export async function getGuidesByAuthor( |
|
authorId: string, |
|
): Promise<GuideFileType[]> { |
|
const allGuides = await getAllGuides(); |
|
|
|
return allGuides.filter((guide) => guide.author?.id === authorId); |
|
} |
|
|
|
/** |
|
* Gets all the guides sorted by the publishing date |
|
* @returns Promisifed guide files |
|
*/ |
|
export async function getAllGuides(): Promise<GuideFileType[]> { |
|
// @ts-ignore |
|
const guides = import.meta.glob<GuideFileType>('/src/data/guides/*.md', { |
|
eager: true, |
|
}); |
|
|
|
const allAuthors = await getAllAuthors(); |
|
|
|
const guideFiles = Object.values(guides) as GuideFileType[]; |
|
const enrichedGuides: GuideFileType[] = guideFiles.map((guideFile) => ({ |
|
...guideFile, |
|
id: guidePathToId(guideFile.file), |
|
author: allAuthors.find( |
|
(author) => author.id === guideFile.frontmatter.authorId, |
|
)!, |
|
})); |
|
|
|
return enrichedGuides.sort( |
|
(a, b) => |
|
new Date(b.frontmatter.date).valueOf() - |
|
new Date(a.frontmatter.date).valueOf(), |
|
); |
|
} |
|
|
|
/** |
|
* Gets the guide by the given id |
|
* @param id Guide identifier |
|
* @returns Promisified guide file |
|
*/ |
|
export async function getGuideById( |
|
id: string, |
|
): Promise<GuideFileType | undefined> { |
|
const allGuides = await getAllGuides(); |
|
|
|
return allGuides.find((guide) => guide.id === id); |
|
} |
|
|
|
type HeadingType = ReturnType<MarkdownFileType['getHeadings']>[number]; |
|
export type HeadingGroupType = HeadingType & { children: HeadingType[] }; |
|
|
|
const NUMBERED_LIST_REGEX = /^\d+\.\s+?/; |
|
|
|
export function getGuideTableOfContent(headings: HeadingType[]) { |
|
const tableOfContents: HeadingGroupType[] = []; |
|
let currentGroup: HeadingGroupType | null = null; |
|
|
|
headings |
|
.filter((heading) => heading.depth !== 1) |
|
.forEach((heading) => { |
|
if (heading.depth === 2) { |
|
currentGroup = { |
|
...heading, |
|
text: heading.text.replace(NUMBERED_LIST_REGEX, ''), |
|
children: [], |
|
}; |
|
tableOfContents.push(currentGroup); |
|
} else if (currentGroup && heading.depth === 3) { |
|
currentGroup.children.push({ |
|
...heading, |
|
text: heading.text.replace(NUMBERED_LIST_REGEX, ''), |
|
}); |
|
} |
|
}); |
|
|
|
return tableOfContents; |
|
}
|
|
|