forked from Codecademy/docs
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcontent.test.ts
139 lines (122 loc) · 4.05 KB
/
content.test.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
import frontmatter, { FrontMatterResult } from 'front-matter';
import fs from 'fs';
import glob from 'glob';
import path from 'path';
/*
* the content/ directory is organized like so:
*
* content/
* | <child>/
* | | <node>/
* | | | <node>.md
* | | | <child>/
* | | | | <node>/
* | | | | | <node>.md
* | | | | ...
* | | <node>/
* | | | <node>.md
* | <child>/
* | | <node>/
* | | | <node>.md
* | | ...
*
*
* node: directory containing one .md file with the same name as the node and optionally one child directory
*
* child: directory containing an arbitrary number of nodes
*/
describe('Codecademy Docs Content', () => {
it('adheres to content file structure', () => {
// file names can only contain alphanumerics and hyphens
const validateName = (pathName: string, name: string) => {
if (name.match(/[^A-Za-z0-9.-]/g)) {
// format so that test failures are more helpful
expect(
`${pathName} - file name must only include alphanumerics or -`
).toBe('');
}
};
// nodes can contain n children and at most one .md file with the same name as the node
const checkNode = (nodePath: string) => {
const children = fs.readdirSync(nodePath);
children.forEach((child) => {
const childPath = path.join(nodePath, child);
validateName(childPath, child);
if (fs.statSync(childPath).isDirectory()) {
checkChild(childPath); // step into directory and make sure it's a valid child
} else {
const nodeName = nodePath.split(path.sep).slice(-1)[0];
expect(childPath).toBe(path.join(nodePath, `${nodeName}.md`));
}
});
};
// children can only contain nodes
const checkChild = (childPath: string) => {
const nodes = fs.readdirSync(childPath);
nodes.forEach((node) => {
const nodePath = path.join(childPath, node);
validateName(nodePath, node);
// format so that test failures are more helpful
if (!fs.statSync(nodePath).isDirectory()) {
expect(`${nodePath} - expected a directory but got a file`).toBe('');
}
checkNode(nodePath); // step into directory and make sure it's a valid node
});
};
checkChild(path.join(__dirname, 'content'));
});
});
// validate metadata in each markdown file
describe.each(glob.sync('content/**/*.md'))('%s', (file) => {
type FrontMatterAttributes = {
Title: string;
Description?: string;
'Codecademy Hub Page'?: string;
CatalogContent?: string[];
Subjects?: string[];
Tags?: string[];
};
const { attributes }: FrontMatterResult<FrontMatterAttributes> = frontmatter(
fs.readFileSync(file, 'utf8')
);
it('has only valid metadata keys', () => {
const validKeys: Record<string, string> = {
CatalogContent: 'CatalogContent',
'Codecademy Hub Page': 'Codecademy Hub Page',
Description: 'Description',
Subjects: 'Subjects',
Tags: 'Tags',
Title: 'Title',
};
Object.keys(attributes).forEach((key) => {
if (!validKeys[key]) {
expect(`Invalid metadata key: ${key}`).toBe('');
}
});
});
it('has valid metadata values', () => {
const testOptionalStringArray = (key: string, val?: string[]) => {
if (
val !== undefined &&
(!Array.isArray(val) || val.some((item) => typeof item !== 'string'))
) {
expect(
`Expected ${key} to be a string array or undefined but got ${typeof val}`
).toBe('');
}
};
expect(typeof attributes.Title).toBe('string');
expect(typeof attributes.Description).toBe('string');
if (
attributes['Codecademy Hub Page'] !== undefined &&
typeof attributes['Codecademy Hub Page'] !== 'string'
) {
expect('Expected "Codecademy Hub Page" to be a string or undefined').toBe(
''
);
}
testOptionalStringArray('CatalogContent', attributes.CatalogContent);
testOptionalStringArray('Subjects', attributes.Subjects);
testOptionalStringArray('Tags', attributes.Tags);
});
});