Skip to content

Commit e9f7c10

Browse files
feat: expose OS/stability tags in the API structure (#8)
1 parent 809c982 commit e9f7c10

5 files changed

+98
-7
lines changed

src/DocsParser.ts

+1
Original file line numberDiff line numberDiff line change
@@ -254,6 +254,7 @@ export class DocsParser {
254254
name: typedKey.key,
255255
description: typedKey.description,
256256
required: typedKey.required,
257+
additionalTags: typedKey.additionalTags,
257258
...typedKey.type,
258259
})),
259260
};

src/ParsedDocumentation.ts

+10
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,12 @@
1+
export enum DocumentationTag {
2+
OS_MACOS = 'os_macos',
3+
OS_MAS = 'os_mas',
4+
OS_WINDOWS = 'os_windows',
5+
OS_LINUX = 'os_linux',
6+
STABILITY_EXPERIMENTAL = 'stability_experimental',
7+
STABILITY_DEPRECATED = 'stability_deprecated',
8+
AVAILABILITY_READONLY = 'availability_readonly',
9+
}
110
export declare type PossibleStringValue = {
211
value: string;
312
description: string;
@@ -43,6 +52,7 @@ export declare type EventParameterDocumentation = {
4352
export declare type DocumentationBlock = {
4453
name: string;
4554
description: string;
55+
additionalTags: DocumentationTag[];
4656
};
4757
export declare type MethodDocumentationBlock = DocumentationBlock & {
4858
signature: string;

src/__tests__/markdown-helpers.spec.ts

+35-1
Original file line numberDiff line numberDiff line change
@@ -2,9 +2,43 @@ import * as fs from 'fs';
22
import * as path from 'path';
33
import MarkdownIt from 'markdown-it';
44

5-
import { safelyJoinTokens, extractStringEnum, rawTypeToTypeInformation } from '../markdown-helpers';
5+
import {
6+
safelyJoinTokens,
7+
extractStringEnum,
8+
rawTypeToTypeInformation,
9+
parseHeadingTags,
10+
} from '../markdown-helpers';
11+
import { DocumentationTag } from '../ParsedDocumentation';
612

713
describe('markdown-helpers', () => {
14+
describe('parseHeadingTags', () => {
15+
it('should return an empty array for null input', () => {
16+
expect(parseHeadingTags(null)).toEqual([]);
17+
});
18+
19+
it('should return an empty array if there are no tags in the input', () => {
20+
expect(parseHeadingTags('String thing no tags')).toEqual([]);
21+
});
22+
23+
it('should return a list of tags if there is one tag', () => {
24+
expect(parseHeadingTags(' _macOS_')).toEqual([DocumentationTag.OS_MACOS]);
25+
});
26+
27+
it('should return a list of tags if there are multiple tags', () => {
28+
expect(parseHeadingTags(' _macOS_ _Windows_ _Experimental_')).toEqual([
29+
DocumentationTag.OS_MACOS,
30+
DocumentationTag.OS_WINDOWS,
31+
DocumentationTag.STABILITY_EXPERIMENTAL,
32+
]);
33+
});
34+
35+
it('should throw an error if there is a tag not on the whitelist', () => {
36+
expect(() => parseHeadingTags(' _Awesome_')).toThrowErrorMatchingInlineSnapshot(
37+
`"heading tags must be from the whitelist: [\\"macOS\\",\\"mas\\",\\"Windows\\",\\"Linux\\",\\"Experimental\\",\\"Deprecated\\",\\"Readonly\\"]: expected [ Array(7) ] to include 'Awesome'"`,
38+
);
39+
});
40+
});
41+
842
describe('safelyJoinTokens', () => {
943
it('should join no tokens to an empty string', () => {
1044
expect(safelyJoinTokens([])).toBe('');

src/block-parsers.ts

+10-6
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@ import { expect } from 'chai';
22
import Token from 'markdown-it/lib/token';
33

44
import {
5+
parseHeadingTags,
56
headingsAndContent,
67
findNextList,
78
convertListToTypedKeys,
@@ -78,14 +79,14 @@ export const _headingToMethodBlock = (
7879
): MethodDocumentationBlock | null => {
7980
if (!heading) return null;
8081

81-
const methodStringRegexp = /`(?:.+\.)?(.+?)(\(.*?\))`/g;
82+
const methodStringRegexp = /`(?:.+\.)?(.+?)(\(.*?\))`( _(?:[^_]+?)_)*/g;
8283
const methodStringMatch = methodStringRegexp.exec(heading.heading)!;
8384
methodStringRegexp.lastIndex = -1;
8485
expect(heading.heading).to.match(
8586
methodStringRegexp,
8687
'each method should have a code blocked method name',
8788
);
88-
const [, methodString, methodSignature] = methodStringMatch;
89+
const [, methodString, methodSignature, headingTags] = methodStringMatch;
8990

9091
let parameters: MethodDocumentationBlock['parameters'] = [];
9192
if (methodSignature !== '()') {
@@ -129,18 +130,19 @@ export const _headingToMethodBlock = (
129130
description: parsedDescription,
130131
parameters,
131132
returns: parsedReturnType,
133+
additionalTags: parseHeadingTags(headingTags),
132134
};
133135
};
134136

135137
export const _headingToPropertyBlock = (heading: HeadingContent): PropertyDocumentationBlock => {
136-
const propertyStringRegexp = /`(?:.+\.)?(.+?)`/g;
138+
const propertyStringRegexp = /`(?:.+\.)?(.+?)`( _(?:[^_]+?)_)*/g;
137139
const propertyStringMatch = propertyStringRegexp.exec(heading.heading)!;
138140
propertyStringRegexp.lastIndex = -1;
139141
expect(heading.heading).to.match(
140142
propertyStringRegexp,
141143
'each property should have a code blocked property name',
142144
);
143-
const [, propertyString] = propertyStringMatch;
145+
const [, propertyString, headingTags] = propertyStringMatch;
144146

145147
const { parsedDescription, parsedReturnType } = extractReturnType(
146148
findContentAfterHeadingClose(heading.content),
@@ -157,16 +159,17 @@ export const _headingToPropertyBlock = (heading: HeadingContent): PropertyDocume
157159
name: propertyString,
158160
description: parsedDescription,
159161
required: !/\(optional\)/i.test(parsedDescription),
162+
additionalTags: parseHeadingTags(headingTags),
160163
...parsedReturnType!,
161164
};
162165
};
163166

164167
export const _headingToEventBlock = (heading: HeadingContent): EventDocumentationBlock => {
165-
const eventNameRegexp = /^Event: '(.+)'/g;
168+
const eventNameRegexp = /^Event: '(.+)'( _(?:[^_]+?)_)*/g;
166169
const eventNameMatch = eventNameRegexp.exec(heading.heading)!;
167170
eventNameRegexp.lastIndex = -1;
168171
expect(heading.heading).to.match(eventNameRegexp, 'each event should have a quoted event name');
169-
const [, eventName] = eventNameMatch;
172+
const [, eventName, headingTags] = eventNameMatch;
170173

171174
expect(eventName).to.not.equal('', 'should have a non-zero-length event name');
172175

@@ -194,6 +197,7 @@ export const _headingToEventBlock = (heading: HeadingContent): EventDocumentatio
194197
name: eventName,
195198
description,
196199
parameters,
200+
additionalTags: parseHeadingTags(headingTags),
197201
};
198202
};
199203

src/markdown-helpers.ts

+42
Original file line numberDiff line numberDiff line change
@@ -5,8 +5,44 @@ import {
55
PropertyDocumentationBlock,
66
MethodParameterDocumentation,
77
PossibleStringValue,
8+
DocumentationTag,
89
} from './ParsedDocumentation';
910

11+
const tagMap = {
12+
macOS: DocumentationTag.OS_MACOS,
13+
mas: DocumentationTag.OS_MAS,
14+
Windows: DocumentationTag.OS_WINDOWS,
15+
Linux: DocumentationTag.OS_LINUX,
16+
Experimental: DocumentationTag.STABILITY_EXPERIMENTAL,
17+
Deprecated: DocumentationTag.STABILITY_DEPRECATED,
18+
Readonly: DocumentationTag.AVAILABILITY_READONLY,
19+
};
20+
21+
const ALLOWED_TAGS = Object.keys(tagMap) as (keyof typeof tagMap)[];
22+
23+
export const parseHeadingTags = (tags: string | null): DocumentationTag[] => {
24+
if (!tags) return [];
25+
26+
const parsedTags: (keyof typeof tagMap)[] = [];
27+
const matcher = / _([^_]+)_/g;
28+
let match: RegExpMatchArray | null;
29+
while ((match = matcher.exec(tags))) {
30+
expect(ALLOWED_TAGS).to.contain(
31+
match[1],
32+
`heading tags must be from the whitelist: ${JSON.stringify(ALLOWED_TAGS)}`,
33+
);
34+
parsedTags.push(match[1] as keyof typeof tagMap);
35+
}
36+
37+
return parsedTags.map(value => {
38+
if (tagMap[value]) return tagMap[value];
39+
40+
throw new Error(
41+
`Impossible scenario detected, "${value}" is not an allowed tag but it got past the allowed tags check`,
42+
);
43+
});
44+
};
45+
1046
export const findNextList = (tokens: Token[]) => {
1147
const start = tokens.findIndex(t => t.type === 'bullet_list_open');
1248
if (start === -1) return null;
@@ -237,6 +273,7 @@ export const rawTypeToTypeInformation = (
237273
name: typedKey.key,
238274
description: typedKey.description,
239275
required: typedKey.required,
276+
additionalTags: typedKey.additionalTags,
240277
...typedKey.type,
241278
}))
242279
: [],
@@ -271,6 +308,7 @@ export const rawTypeToTypeInformation = (
271308
name: typedKey.key,
272309
description: typedKey.description,
273310
required: typedKey.required,
311+
additionalTags: typedKey.additionalTags,
274312
...typedKey.type,
275313
}))
276314
: [],
@@ -514,6 +552,7 @@ type TypedKey = {
514552
type: TypeInformation;
515553
description: string;
516554
required: boolean;
555+
additionalTags: DocumentationTag[];
517556
};
518557

519558
type List = { items: ListItem[] };
@@ -608,6 +647,8 @@ const convertNestedListToTypedKeys = (list: List): TypedKey[] => {
608647
/ ?\(Optional\) ?/,
609648
'optionality should be defined with "(optional)", all lower case, no capital "O"',
610649
);
650+
const tagMatcher = /.+?((?: _(?:[^_]+?)_)+)/g;
651+
const tagMatch = tagMatcher.exec(rawType);
611652
const cleanedType = rawType.replace(/ ?\(optional\) ?/i, '').replace(/_.+?_/g, '');
612653
const subTypedKeys = item.nestedList ? convertNestedListToTypedKeys(item.nestedList) : null;
613654
const type = rawTypeToTypeInformation(cleanedType.trim(), rawDescription, subTypedKeys);
@@ -617,6 +658,7 @@ const convertNestedListToTypedKeys = (list: List): TypedKey[] => {
617658
key: keyToken.content,
618659
description: rawDescription.trim().replace(/^- ?/, ''),
619660
required: !isRootOptional,
661+
additionalTags: tagMatch ? parseHeadingTags(tagMatch[1]) : [],
620662
});
621663
}
622664

0 commit comments

Comments
 (0)