Skip to content

Commit 1c7096c

Browse files
authored
refactor: move config validation to backend (#214)
* chore: rebase on main * refactor: remove legacy config type * refactor: move bundler files into own directory * chore: rebase on main * fix: default to not replacing a variable when not declared * fix: tweak type of BundleSuccess config * chore: rebase from main * chore: bump deps * chore: bump codehike deps * chore: bump dependencies * chore: lint and format
1 parent 7b3556a commit 1c7096c

31 files changed

+2465
-24432
lines changed

.nvmrc

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
v18.0.0

api/package.json

+22-23
Original file line numberDiff line numberDiff line change
@@ -1,47 +1,46 @@
11
{
22
"type": "module",
3-
"name": "ts-express",
3+
"name": "docs-page-api",
44
"version": "1.0.0",
55
"description": "template node and express app with typescript using esm ",
66
"main": "src/app.ts",
77
"author": "<[email protected]>",
88
"license": "MIT",
99
"dependencies": {
10-
"@code-hike/mdx": "^v0.3.0-next.20",
10+
"@code-hike/mdx": "^0.7.2",
1111
"@ltd/j-toml": "^1.30.0",
12-
"@octokit/graphql": "^4.8.0",
12+
"@octokit/graphql": "^5.0.0",
1313
"@types/cors": "^2.8.12",
1414
"@types/express": "^4.17.13",
15-
"@types/lodash.get": "^4.4.6",
15+
"@types/lodash.get": "^4.4.7",
1616
"@types/morgan": "^1.9.3",
17-
"@types/node": "^16.7.1",
17+
"@types/node": "^18.0.0",
1818
"a2a": "^0.2.0",
19-
"camelcase": "^6.2.1",
19+
"camelcase": "^7.0.0",
2020
"cors": "^2.8.5",
21-
"dotenv": "^10.0.0",
22-
"esbuild": "^0.14.2",
23-
"express": "^4.17.1",
24-
"express-basic-auth": "^1.1.1",
21+
"dotenv": "^16.0.1",
22+
"esbuild": "^0.14.47",
23+
"express": "^4.18.1",
24+
"express-basic-auth": "^1.2.1",
2525
"hast-util-parse-selector": "^3.1.0",
26-
"http-status": "^1.5.0",
26+
"http-status": "^1.5.2",
2727
"is-badge": "^2.1.0",
2828
"js-yaml": "^4.1.0",
2929
"lodash.get": "^4.4.2",
30-
"mdx-bundler": "^8.0.0",
30+
"mdx-bundler": "^9.0.1",
3131
"morgan": "^1.10.0",
32-
"node-fetch": "^3.2.2",
33-
"probot": "^12.2.1",
34-
"react": "^17.0.2",
32+
"node-fetch": "^3.2.6",
33+
"probot": "^12.2.4",
34+
"react": "^18.2.0",
3535
"rehype-accessible-emojis": "^0.3.2",
3636
"rehype-katex": "^6.0.2",
37-
"rehype-slug": "^5.0.0",
37+
"rehype-slug": "^5.0.1",
3838
"remark-comment": "^1.0.0",
3939
"remark-gfm": "^3.0.1",
4040
"remark-math": "^5.1.1",
4141
"remark-parse": "^10.0.1",
42-
"shiki": "^0.9.15",
42+
"shiki": "^0.10.1",
4343
"tiny-invariant": "^1.2.0",
44-
"typescript": "4.4.4",
4544
"unist-util-visit": "^4.1.0"
4645
},
4746
"scripts": {
@@ -51,10 +50,10 @@
5150
"postinstall": "tsc"
5251
},
5352
"devDependencies": {
54-
"@docs.page/server": "^1.0.0",
55-
"@octokit/webhooks-types": "^5.3.0",
56-
"nodemon": "^2.0.12",
57-
"ts-node": "^10.4.0",
58-
"typescript": "4.4.3"
53+
"@docs.page/server": "^1.0.1",
54+
"@octokit/webhooks-types": "^6.2.4",
55+
"nodemon": "^2.0.18",
56+
"ts-node": "^10.8.1",
57+
"typescript": "4.7.4"
5958
}
6059
}

api/src/utils/bundler.ts api/src/bundler/bundler.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -46,7 +46,7 @@ export async function bundle(
4646

4747
const { code, frontmatter, errors } = await bundleMDX({
4848
source: rawText,
49-
xdmOptions(options) {
49+
mdxOptions(options) {
5050
// @ts-ignore TODO fix types
5151
options.remarkPlugins = [...(options.remarkPlugins ?? []), ...bundleOptions.remarkPlugins];
5252
// @ts-ignore TODO fix types

api/src/utils/getPlugins.ts api/src/bundler/getPlugins.ts

+3-3
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,9 @@
1-
import { theme } from '../utils/plugins/codeHikeTheme.js';
1+
import { theme } from './plugins/codeHikeTheme.js';
22
import { remarkCodeHike } from '@code-hike/mdx';
33
import remarkGfm from 'remark-gfm';
4-
import rehypeCodeBlocks from '../utils/plugins/rehype-code-blocks.js';
4+
import rehypeCodeBlocks from './plugins/rehype-code-blocks.js';
55
import { rehypeAccessibleEmojis } from 'rehype-accessible-emojis';
6-
import rehypeInlineBadges from '../utils/plugins/rehype-inline-badges.js';
6+
import rehypeInlineBadges from './plugins/rehype-inline-badges.js';
77
import rehypeSlug from 'rehype-slug';
88
import remarkMath from 'remark-math';
99
import rehypeKatex from 'rehype-katex';

api/src/controllers/raw.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
import { Request, Response } from 'express';
22
import { BundleResponseData } from '@docs.page/server';
33
import { Bundle, BundleError } from '../utils/bundle.js';
4+
import { formatConfigLocales } from '../utils/config.js';
45
/**
56
* Gets the API information.
67
*
@@ -23,7 +24,7 @@ export const bundleRaw = async (
2324
markdown,
2425
});
2526
if (sourceConfig) {
26-
bundleInstance.formatConfigLocales(sourceConfig);
27+
bundleInstance.config = formatConfigLocales(sourceConfig, path);
2728
}
2829
try {
2930
const data = await bundleInstance.build();

api/src/utils/bundle.ts

+9-51
Original file line numberDiff line numberDiff line change
@@ -1,18 +1,11 @@
1-
import { bundle } from './bundler.js';
2-
import { getPlugins } from './getPlugins.js';
1+
import { bundle } from '../bundler/bundler.js';
2+
import { getPlugins } from '../bundler/getPlugins.js';
33
import { Contents, getConfigs, getGitHubContents } from './github.js';
4-
import { HeadingNode } from './plugins/rehype-headings.js';
4+
import { HeadingNode } from '../bundler/plugins/rehype-headings.js';
55
import { formatSourceAndRef } from './ref.js';
66
import { getRepositorySymLinks } from './symlinks.js';
7-
import {
8-
hasLocales,
9-
InputConfig,
10-
OutputConfig,
11-
defaultConfig,
12-
ErrorReason,
13-
} from '@docs.page/server';
14-
import yaml from 'js-yaml';
15-
import toml from '@ltd/j-toml';
7+
import { OutputConfig, defaultConfig, ErrorReason } from '@docs.page/server';
8+
import { formatConfigLocales } from './config.js';
169

1710
type Frontmatter = Record<string, string>;
1811

@@ -115,7 +108,8 @@ export class Bundle {
115108

116109
this.repositoryFound = githubContents.repositoryFound;
117110

118-
this.formatConfigLocales(githubContents.config);
111+
this.config = formatConfigLocales(githubContents.config, this.path);
112+
119113
await this.matchSymLinks();
120114
if (githubContents.md === null) {
121115
throw new BundleError(404, "Couldn't find file", 'FILE_NOT_FOUND');
@@ -133,7 +127,8 @@ export class Bundle {
133127
if (!repositoryFound || !config) {
134128
throw new BundleError(404, 'Unable to fetch config file.');
135129
}
136-
this.formatConfigLocales(config);
130+
131+
this.config = formatConfigLocales(config, this.path);
137132

138133
return this.config;
139134
}
@@ -190,43 +185,6 @@ export class Bundle {
190185
}
191186
}
192187
}
193-
194-
formatConfigLocales(config?: { configJson?: string; configYaml?: string; configToml?: string }) {
195-
if (!config?.configJson && !config?.configYaml && !config?.configToml) {
196-
throw new BundleError(404, 'Not found: Config file missing', 'MISSING_CONFIG');
197-
}
198-
const { configJson, configYaml, configToml } = config;
199-
200-
let inputConfig: InputConfig = defaultConfig;
201-
// TODO: validation of config?
202-
203-
try {
204-
if (configJson) {
205-
inputConfig = JSON.parse(configJson) as InputConfig;
206-
} else if (configYaml) {
207-
inputConfig = yaml.load(configYaml) as InputConfig;
208-
} else if (configToml) {
209-
//@ts-ignore
210-
inputConfig = Object.assign({}, toml.parse(configToml) as InputConfig);
211-
}
212-
} catch (e) {
213-
console.error(e);
214-
throw new BundleError(500, 'Error parsing config', 'BAD_CONFIG');
215-
}
216-
217-
if (hasLocales(inputConfig)) {
218-
const defaulLocale = inputConfig.locales[0];
219-
220-
const currentLocale = this.path.split('/')[0] || defaulLocale;
221-
222-
this.config = {
223-
...inputConfig,
224-
sidebar: inputConfig?.sidebar[currentLocale],
225-
};
226-
} else {
227-
this.config = inputConfig;
228-
}
229-
}
230188
}
231189

232190
export class BundleError extends Error {

website/app/utils/config.ts api/src/utils/config.ts

+47-65
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,15 @@
1+
import { getBoolean, getNumber, getString, getValue } from './get.js';
12
import get from 'lodash.get';
2-
import { getBoolean, getNumber, getString, getValue } from './get';
3+
4+
import yaml from 'js-yaml';
5+
import toml from '@ltd/j-toml';
6+
import { OutputConfig } from '@docs.page/server';
37

48
// Represents how the sidebar should look in the config file.
59
export type SidebarItem = [string, Array<[string, string]>] | [string, string];
610

711
// Merges in a user sidebar config and ensures all items are valid.
8-
function mergeSidebarConfig(json: Partial<ProjectConfig> | null): SidebarItem[] {
12+
function mergeSidebarConfig(json: Partial<OutputConfig> | null): SidebarItem[] {
913
const sidebar = get(json, 'sidebar', defaultConfig.sidebar);
1014

1115
if (!Array.isArray(sidebar)) {
@@ -57,54 +61,8 @@ function mergeSidebarConfig(json: Partial<ProjectConfig> | null): SidebarItem[]
5761
* This can be provided by creating a `docs.json` file at the root of your
5862
* repository.
5963
*/
60-
export interface ProjectConfig {
61-
// Project name.
62-
name: string;
63-
// URL to project logo.
64-
logo: string;
65-
// URL to project logo for dark mode
66-
logoDark: string;
67-
// URL to the favicon
68-
favicon: string;
69-
// Image to display as the social preview on shared URLs
70-
socialPreview: string;
71-
// Twitter tag for use in the header.
72-
twitter: string;
73-
// Whether the website should be indexable by search bots.
74-
noindex: boolean;
75-
// A color theme used for this project. Defaults to "#00bcd4".
76-
theme: string;
77-
// Docsearch Application ID. If populated, a search box with autocomplete will be rendered.
78-
docsearch?: {
79-
appId?: string;
80-
apiKey: string;
81-
indexName: string;
82-
};
83-
// Header navigation
84-
// navigation: NavigationItem[];
85-
// Sidebar
86-
sidebar: SidebarItem[];
87-
// Locales:
88-
locales?: Record<string, string>;
89-
// The depth to heading tags are linked. Set to 0 to remove any linking.
90-
headerDepth: number;
91-
// Variables which can be injected into the pages content.
92-
variables: Record<string, string>;
93-
// Adds Google Tag Manager to your documentation pages.
94-
googleTagManager: string;
95-
// Adds Google Analytics to your documentation pages.
96-
googleAnalytics: string;
97-
// Whether zoomable images are enabled by default
98-
zoomImages: boolean;
99-
// Whether CodeHike is enabled
100-
experimentalCodehike: boolean;
101-
// Whether Math is enabled
102-
experimentalMath: boolean;
103-
// Whether Next/Previous buttons are showing automatically based on the sidebar
104-
automaticallyInferNextPrevious: boolean;
105-
}
10664

107-
export const defaultConfig: ProjectConfig = {
65+
export const defaultConfig: OutputConfig = {
10866
name: '',
10967
logo: '',
11068
logoDark: '',
@@ -126,7 +84,7 @@ export const defaultConfig: ProjectConfig = {
12684
};
12785

12886
// Merges any user config with default values.
129-
export function mergeConfig(json: Record<string, unknown>): ProjectConfig {
87+
export function mergeConfig(json: Record<string, unknown>): OutputConfig {
13088
return {
13189
name: getString(json, 'name', defaultConfig.name),
13290
logo: getString(json, 'logo', defaultConfig.logo),
@@ -166,25 +124,49 @@ export function mergeConfig(json: Record<string, unknown>): ProjectConfig {
166124
};
167125
}
168126

169-
export async function getConfiguration({
170-
owner,
171-
repo,
172-
ref,
173-
}: Record<string, string>): Promise<ProjectConfig> {
174-
let config: ProjectConfig = defaultConfig;
127+
type Configs = {
128+
configJson?: string;
129+
configYaml?: string;
130+
configToml?: string;
131+
};
132+
133+
export function formatConfigLocales(configFiles: Configs, path: string): OutputConfig {
134+
const { configJson, configYaml, configToml } = configFiles;
175135

176-
const host =
177-
process.env.NODE_ENV === 'production' ? 'https://api.docs.page' : 'http://localhost:8000';
136+
let config: Record<string, unknown> = {};
178137

179-
const endpoint = `${host}/config?owner=${owner}&repository=${repo}${
180-
ref ? `&ref=${encodeURIComponent(ref)}` : ''
181-
}`;
182138
try {
183-
const res = await (await fetch(endpoint)).json();
184-
config = res.config;
139+
if (configJson) {
140+
config = JSON.parse(configJson);
141+
} else if (configYaml) {
142+
config = yaml.load(configYaml) as Record<string, unknown>;
143+
} else if (configToml) {
144+
config = Object.assign({}, toml.parse(configToml));
145+
}
185146
} catch (e) {
186-
console.error(e);
147+
console.error('Error parsing config, using default.');
148+
return defaultConfig;
187149
}
188150

189-
return config;
151+
if (
152+
config.hasOwnProperty('locales') &&
153+
Array.isArray(config.locales) &&
154+
config.hasOwnProperty('sidebar')
155+
) {
156+
// TODO: edge cases of bad configs, e.g what if locales is not an array?
157+
158+
const defaulLocale = config.locales[0];
159+
160+
const currentLocale = path.split('/')[0] || defaulLocale;
161+
162+
const sidebar = (getValue(config, 'sidebar') as Record<string, unknown>)[currentLocale];
163+
164+
config = {
165+
...config,
166+
sidebar,
167+
};
168+
169+
return mergeConfig(config);
170+
}
171+
return mergeConfig(config);
190172
}
File renamed without changes.

api/src/utils/sidebar.ts

-12
This file was deleted.

package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -26,7 +26,7 @@
2626
"eslint-config-prettier": "^8.3.0",
2727
"eslint-plugin-prettier": "^4.0.0",
2828
"eslint-plugin-react": "^7.28.0",
29-
"prettier": "2.5.1",
29+
"prettier": "2.7.1",
3030
"prettier-plugin-tailwindcss": "^0.1.3"
3131
}
3232
}

0 commit comments

Comments
 (0)