-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmod.ts
189 lines (162 loc) · 5.7 KB
/
mod.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
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
const decoder = new TextDecoder();
const encoder = new TextEncoder();
// deno-lint-ignore no-explicit-any
declare var Deno: any;
// deno-lint-ignore no-explicit-any
declare var require: any;
/**
* The structure of items in the json_output property.
*
* file
*
* The file in question.
*
* members
*
* The members in the file in question.
*/
interface IJsonOutput {
file: string;
members: string[];
}
/**
* Parse data member doc blocks and signatures and place them in a minimalistic,
* JSON format.
*
* Regex Note 1: The initial expression which matches /**\n.
* Regex Note 2: After the initial expression, keep going until one of the
* following groups is matched. For example, "Hey regex, find /** and keep going
* until you find {}. Stop at {} and do not include it in what you have
* matched." The groups to stop at are as follows:
*
* - (\n\n) --> double NL
* - ( {}\n) --> {} followed by a NL
* - ( {\n$) --> { followed by a NL where the NL is the end of the line
* - ( = {) --> = {
* - (\n$) --> a new line where the NL is the end of the line
*/
export class Docable {
/**
* A property to hold the list of file paths containing doc block info that
* will be written in the json_output property.
*/
protected filepaths: string[];
/**
* A property to hold the JSON that is written in the output file.
*/
protected json_output: { [k: string]: IJsonOutput } = {};
///////////////////////////////////////////////////////////////////////////////
// FILE MARKER - CONSTRUCTOR //////////////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////
/**
* Construct an object of this class.
*
* @param filepaths - An array of filepaths to check for doc blocks.
* @param outputFilepath - The filepath that this script will write to.
*/
constructor(filepaths: string[]) {
this.filepaths = filepaths;
}
///////////////////////////////////////////////////////////////////////////////
// FILE MARKER - METHODS - PUBLIC /////////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////
/**
* Run this script.
*
* @returns Doc blocks in JSON format.
*/
public run(): string | void {
for (const index in this.filepaths) {
const filepath = this.filepaths[index];
const fileContents = this.getFileContents(filepath);
// Get the full member name and only continue with the script if it's found
const fullMemberName = this.getFullMemberName(fileContents);
if (!fullMemberName) {
console.log(
`File "${filepath}" is missing the "// docable-member-namespace:" comment at the top of the file.`,
);
return;
}
const members = this.getAllDataMembers(fileContents);
if (!members) {
console.log(`File "${filepath}" does not have any doc blocks.`);
return;
}
this.json_output[fullMemberName as string] = {
file: filepath,
members: [],
};
(members as string[]).forEach((member: string) => {
// We want to clean up the output of the JSON so we remove unnecessary
// whitespace here
member = member
.replace(/\s+\*/g, " *")
.replace(/\ \*/g, "\n *")
.replace(/\s+protected/g, "protected")
.replace(/\s+private/g, "private")
.replace(/\s+public/g, "public")
.replace(/\s+constructor/g, "constructor");
this.json_output[fullMemberName as string].members.push(member);
});
}
return JSON.stringify(this.json_output, null, 2);
}
///////////////////////////////////////////////////////////////////////////////
// FILE MARKER - METHODS - PROTECTED //////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////
/**
* Get all data members in the file contents in question.
*
* @param fileContents - The file contents containing the data members.
*
* @returns False if there are no members; an array of members if any.
*/
protected getAllDataMembers(fileContents: string): boolean | string[] {
const members = fileContents
.match(/\/\*\*\n[\s\S]*?(?=((\n\n)|( {}\n)|( {\n$)|( = {)|(\n$)))/g);
// \_______________/\_______________________________________/
// | |
// v v
// See Regex Note 1 See Regex Note 2
// at top of file at top of file
if (!members) {
return false;
}
return members;
}
/**
* Get the file contents of the filepath in question.
*
* @param filepath - A filepath (e.g., /some/file/path.extension).
*
* @returns The contents.
*/
protected getFileContents(filepath: string): string {
let ret = "";
if (globalThis.Deno) {
ret = decoder.decode(Deno.readFileSync(filepath));
} else {
const fs = require("fs");
ret = decoder.decode(fs.readFileSync(filepath));
}
return ret;
}
/**
* Get the member's full name from the contents in question (e.g.,
* Drash.Http.Server).
*
* @param fileContents - The file contents containing the member's full name.
* The member's full name should be in a comment like the following:
*
* /// Member: Drash.Http.Server
*
* @returns The member's full name.
*/
protected getFullMemberName(fileContents: string): boolean | string {
const fullMemberNameMatch = fileContents.match(/\/\/ docable-member-namespace:.+/g);
if (!fullMemberNameMatch) {
return false;
}
return fullMemberNameMatch[0].replace("// docable-member-namespace: ", "");
}
}