-
Notifications
You must be signed in to change notification settings - Fork 38
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
57 changed files
with
2,001 additions
and
1,384 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,18 +1,17 @@ | ||
import {composeFrontMatter, extractFrontMatter} from '@diplodoc/transform/lib/frontmatter'; | ||
|
||
export function addSourcePath(fileContent: string, sourcePath: string) { | ||
const [frontMatter, strippedContent] = extractFrontMatter( | ||
fileContent, | ||
sourcePath, | ||
); | ||
const [frontMatter, strippedContent] = extractFrontMatter(fileContent, sourcePath); | ||
|
||
if (frontMatter.sourcePath) { | ||
return fileContent; | ||
} | ||
|
||
return composeFrontMatter({ | ||
...frontMatter, | ||
sourcePath, | ||
}, strippedContent); | ||
return composeFrontMatter( | ||
{ | ||
...frontMatter, | ||
sourcePath, | ||
}, | ||
strippedContent, | ||
); | ||
} | ||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,249 @@ | ||
import type {BuildConfig, Run} from '~/commands/build'; | ||
import type {Includer, IncluderOptions, RawToc, RawTocItem, Toc, TocItem, WithItems} from './types'; | ||
|
||
import {ok} from 'node:assert'; | ||
import {basename, dirname, join} from 'node:path'; | ||
import {dump, load} from 'js-yaml'; | ||
import {AsyncParallelHook, AsyncSeriesWaterfallHook, HookMap} from 'tapable'; | ||
|
||
import {freeze, intercept, isExternalHref, normalizePath, own} from '~/utils'; | ||
import {Stage} from '~/constants'; | ||
|
||
import loader, {IncludeMode, LoaderContext} from './loader'; | ||
|
||
export type TocServiceConfig = { | ||
ignore: BuildConfig['ignore']; | ||
ignoreStage: BuildConfig['ignoreStage']; | ||
template: BuildConfig['template']; | ||
removeHiddenTocItems: BuildConfig['removeHiddenTocItems']; | ||
}; | ||
|
||
type WalkStepResult<I> = I | I[] | null | undefined; | ||
|
||
type TocServiceHooks = { | ||
/** | ||
* Called before item data processing (but after data interpolation) | ||
*/ | ||
Item: AsyncSeriesWaterfallHook<[RawTocItem, RelativePath]>; | ||
Includer: HookMap<AsyncSeriesWaterfallHook<[Toc, IncluderOptions, RelativePath]>>; | ||
Resolved: AsyncParallelHook<[Toc, RelativePath]>; | ||
Included: AsyncParallelHook<[Toc, RelativePath, IncludeInfo]>; | ||
}; | ||
|
||
type IncludeInfo = { | ||
from: RelativePath; | ||
mode: IncludeMode; | ||
mergeBase?: RelativePath; | ||
}; | ||
|
||
export class TocService { | ||
hooks: TocServiceHooks; | ||
|
||
get entries() { | ||
return [...this._entries]; | ||
} | ||
|
||
private run: Run; | ||
|
||
private logger: Run['logger']; | ||
|
||
private vars: Run['vars']; | ||
|
||
private config: TocServiceConfig; | ||
|
||
private tocs: Map<NormalizedPath, Toc> = new Map(); | ||
|
||
private _entries: Set<NormalizedPath> = new Set(); | ||
|
||
private processed: Hash<boolean> = {}; | ||
|
||
constructor(run: Run) { | ||
this.run = run; | ||
this.logger = run.logger; | ||
this.vars = run.vars; | ||
this.config = run.config; | ||
this.hooks = intercept('TocService', { | ||
Item: new AsyncSeriesWaterfallHook(['item', 'path']), | ||
Includer: new HookMap(() => new AsyncSeriesWaterfallHook(['toc', 'options', 'path'])), | ||
Resolved: new AsyncParallelHook(['toc', 'path']), | ||
Included: new AsyncParallelHook(['toc', 'path', 'info']), | ||
}); | ||
} | ||
|
||
async init() { | ||
const tocs = await this.run.glob('**/toc.yaml', { | ||
cwd: this.run.input, | ||
ignore: this.config.ignore, | ||
}); | ||
|
||
for (const toc of tocs) { | ||
await this.load(toc); | ||
} | ||
} | ||
|
||
async load(path: RelativePath, include?: IncludeInfo): Promise<Toc | undefined> { | ||
path = normalizePath(path); | ||
|
||
// There is no error. We really skip toc processing, if it was processed previously in any way. | ||
// For example toc can be processed as include of some other toc. | ||
if (!include && this.processed[path]) { | ||
return; | ||
} | ||
|
||
this.processed[path] = true; | ||
|
||
this.logger.proc(path); | ||
|
||
const file = join(this.run.input, path); | ||
|
||
ok(file.startsWith(this.run.input), `Requested toc '${file}' is out of project scope.`); | ||
|
||
const context: LoaderContext = { | ||
root: this.run.input, | ||
mode: include?.mode || IncludeMode.RootMerge, | ||
base: include?.from || dirname(path), | ||
mergeBase: include?.mergeBase, | ||
path, | ||
vars: await this.vars.load(path), | ||
toc: this, | ||
options: { | ||
resolveConditions: this.config.template.features.conditions, | ||
resolveSubstitutions: this.config.template.features.substitutions, | ||
removeHiddenItems: this.config.removeHiddenTocItems, | ||
}, | ||
}; | ||
|
||
const content = load(await this.run.read(file)) as RawToc; | ||
|
||
// Should ignore included toc with tech-preview stage. | ||
// TODO(major): remove this | ||
if (content && content.stage === Stage.TECH_PREVIEW) { | ||
return; | ||
} | ||
|
||
const {ignoreStage} = this.config; | ||
if (content.stage && ignoreStage.length && ignoreStage.includes(content.stage)) { | ||
return; | ||
} | ||
|
||
if (include && [IncludeMode.RootMerge, IncludeMode.Merge].includes(include.mode)) { | ||
const from = dirname(file); | ||
const to = join(this.run.input, include.mergeBase || dirname(include.from)); | ||
await this.run.copy(from, to, { | ||
sourcePath: (file: string) => file.endsWith('.md'), | ||
ignore: [basename(file), '**/toc.yaml'], | ||
}); | ||
} | ||
|
||
const toc = (await loader.call(context, content)) as Toc; | ||
|
||
// If this is a part of other toc.yaml | ||
if (include) { | ||
await this.hooks.Included.promise(toc, path, include); | ||
} else { | ||
// TODO: we don't need to store tocs in future | ||
// All processing should subscribe on toc.hooks.Resolved | ||
this.tocs.set(path as NormalizedPath, toc); | ||
await this.walkItems([toc], (item: TocItem | Toc) => { | ||
if (own<string>(item, 'href') && !isExternalHref(item.href)) { | ||
this._entries.add(normalizePath(join(dirname(path), item.href))); | ||
} | ||
|
||
return item; | ||
}); | ||
|
||
await this.hooks.Resolved.promise(freeze(toc), path); | ||
} | ||
|
||
// eslint-disable-next-line consistent-return | ||
return toc; | ||
} | ||
|
||
dump(toc: Toc | undefined) { | ||
ok(toc, 'Toc is empty.'); | ||
|
||
return dump(toc); | ||
} | ||
|
||
async walkItems<T extends WithItems<I>, I extends WithItems<I>>( | ||
items: T[] | undefined, | ||
actor: (item: T) => Promise<WalkStepResult<T>> | WalkStepResult<T>, | ||
): Promise<T[] | undefined> { | ||
if (!items || !items.length) { | ||
return items; | ||
} | ||
|
||
const results: T[] = []; | ||
const queue = [...items]; | ||
while (queue.length) { | ||
const item = queue.shift() as T; | ||
|
||
const result = await actor(item); | ||
if (result) { | ||
if (Array.isArray(result)) { | ||
results.push(...result); | ||
} else { | ||
results.push(result); | ||
} | ||
} | ||
|
||
if (hasItems<T>(result)) { | ||
result.items = await this.walkItems(result.items, actor); | ||
} | ||
} | ||
|
||
return results; | ||
} | ||
|
||
/** | ||
* Resolves toc path and data for any page path | ||
* | ||
* @param {RelativePath} path - any page path | ||
* | ||
* @returns [RelativePath, Toc] | ||
*/ | ||
for(path: RelativePath): [RelativePath, Toc] { | ||
path = normalizePath(path); | ||
|
||
if (!path) { | ||
throw new Error('Error while finding toc dir.'); | ||
} | ||
|
||
const tocPath = join(dirname(path), 'toc.yaml'); | ||
|
||
if (this.tocs.has(tocPath as NormalizedPath)) { | ||
return [tocPath, this.tocs.get(tocPath as NormalizedPath) as Toc]; | ||
} | ||
|
||
return this.for(dirname(path)); | ||
} | ||
|
||
async applyIncluders( | ||
toc: Toc, | ||
path: RelativePath, | ||
includers: Includer[], | ||
tocPath: RelativePath, | ||
) { | ||
for (const includer of includers) { | ||
ok(includer.name, 'Includer name should be a string.'); | ||
|
||
const hook = this.hooks.Includer.get(includer.name); | ||
|
||
ok(hook, `Includer with name '${includer.name}' is not registered.`); | ||
|
||
const options = { | ||
...includer, | ||
path: tocPath, | ||
}; | ||
|
||
toc = await hook.promise(toc, options, path); | ||
} | ||
|
||
return toc; | ||
} | ||
} | ||
|
||
// eslint-disable-next-line @typescript-eslint/no-explicit-any | ||
function hasItems<I>(item: any): item is WithItems<I> { | ||
return item && typeof item === 'object' && item.items && item.items.length; | ||
} |
Oops, something went wrong.