-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathindex.js
185 lines (166 loc) · 6.34 KB
/
index.js
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
#!/usr/bin/env node
// # `index.js`
// This file exposes functions for using `jdi` APIs directly, without the `jdi`
// command line utility.
'use strict' // jdi-disable-line
// ## module dependencies
// Native node dependencies.
const fs = require('fs')
const path = require('path')
// Dependencies used for processing the read streams of processed source files.
const split = require('split')
const through2 = require('through2')
// ## regular expressions
// ### `isDoc`
// `isDoc` matches comments starting with `//`. Whitespace is being ignored.
const isDoc = /^\s*\/\/\s*/
// ### `isBlank`
// `isBlank` matches lines containing only whitespace characters, such as tabs
// or spaces.
const isBlank = /^\s*$/
// ### `isShebang`
// `isShebang` matches lines that start with the shebang character sequence. See
// [Shebang (Unix)](https://de.wikipedia.org/wiki/Shebang) explaining the syntax
// of the directive itself.
const isShebang = /^#!/
// ### `isIgnored`
// `isIgnored` matches lines that end with "jdi-disable-line". This is inspired
// by `eslint`'s `eslint-disable-line` directive.
const isIgnored = /^.*jdi-disable-line$/
// ## `transformFunction`
function transformFunction (chunk, enc, cb) {
const extname = this.options.extname
// ### shebang
// Are we currently processing the first line of the file in question?
const isFirstLine = !this.isNotFirstLine
this.isNotFirstLine = true
// Check if the file starts with a shebang.
if (isFirstLine && isShebang.test(chunk)) {
// If we are on the first line of the file and the file starts with a
// shebang, e.g. `#!/usr/bin/env node`, we ignore the shebang.
cb()
return
}
// ### blank lines
// We ignore empty lines in order to avoid creating excessive code blocks.
// Empty lines are being preserved in the transformed `.md` file. They can
// be used for separating sections.
if (isBlank.test(chunk)) {
this.push(chunk)
this.push('\n')
cb()
return
}
// ### ignored lines
// Check if line ends with jdi-disable-line.
if (isIgnored.test(chunk)) {
// If jdi should ignore this line, `cb()` without pushing `chunk`.
cb()
return
}
// ### code blocks
// Check if we are currently in a code block. Everything that is **not** a
// comment can be considered to be a code block.
const isCodeBlock = this.isCodeBlock
this.isCodeBlock = !isDoc.test(chunk)
// Did we just start a code block?
if (this.isCodeBlock && !isCodeBlock) {
// If yes, append ``\`\`\`${extname}` to start the code block.
this.push(`\`\`\`${extname}\n`)
}
// Did we just close a code block?
if (!this.isCodeBlock && isCodeBlock) {
// If yes, append `\`\`\`` to close the code block.
this.push('```\n')
}
// Are we currently in a code block?
if (this.isCodeBlock) {
// If yes, just pass the chunk through.
this.push(chunk)
} else {
// Otherwise, we're in a comment (= documentation). We remove the
// comment prefix and trailing whitespace (typically `// `).
// That way `// # title` becomes `# title`.
this.push(String(chunk).replace(isDoc, ''))
}
this.push('\n')
cb()
}
// ## `flushFunction`
// This function is being called just prior to the stream ending. Keep in mind
// that it's a `prototype` function, thuse `flushFunction` and
// `transformFunction` share the same `this` context.
function flushFunction (cb) {
// ### closing code blocks
// Make sure we're closing any open code blocks with `\`\`\``.
if (this.isCodeBlock) {
this.push('```\n')
}
// ### file footer
// Here we append a footer to the generated `.md` file. The footer includes
// the date when the file has been generated, as well as the relative
// filename of the processed file itself.
const file = this.options.file
// The `.md` file will be in the same directory as the source file. We only
// need the basename.
const basename = path.basename(file)
const date = String(new Date())
this.push('------------------------\n')
this.push(`Generated _${date}_ from [Ⓢ ${basename}](${basename} "View in source")\n`)
this.push('\n')
cb()
}
// ## `Transform`
// Instead of returning a `stream.Transform` instance, through2.ctor() returns a
// constructor for our custom Transform that operates on a line-by-line basis.
const Transform = through2.ctor(transformFunction, flushFunction)
// ## `doc`
// `doc` accepts a filename, creates a corresponding read stream, processes the
// file and writes the resulting `.md` file to disk.
const doc = file =>
fs.createReadStream(file)
// [`split`](https://www.npmjs.com/package/split) is a transform that
// generates chunks separated by new-line.
.pipe(split())
// Here we invoke our custom `Transform` instance used for processing
// the separated line-chunks.
.pipe(new Transform({
// If there is no file extension, we assume that the source file is
// a JavaScript file. In some cases, we could also determine the
// extension name from the first line of the file, e.g.
// `#!/usr/bin/env node`.
extname: path.extname(file).substr(1) || 'js',
// The `file` is being used for adding the header of the `.md`
// output.
file
}))
// ## `handleClose`
// We print a message whenever we generated a `.md` file.
const handleClose = cwd => function handler () {
const toPath = path.relative(cwd, this.path)
console.log('wrote', this.bytesWritten, 'bytes to', toPath) // eslint-disable-line no-console
}
// ## `run`
// This is the main function of `jdi`. It accepts an array of `args` and
// generates corresponding `.md` files.
// **Warning:** `run` has persistent side-effects (it overrides files). If you
// just want to create streams of documented source files, use `doc()` instead.
const run = (cwd, args) =>
args
// Files that should be documented are being passed in via `args`, e.g.
// if the user invokes `jdi index.js test.js`, `args` will be an array
// of `['index.js', 'test.js']`.
.map(file => path.join(cwd, file))
// Now we have an array of filenames to be processed.
// We map over those files and create a corresponding stream.
.map(doc)
// Create a corresponding [`WriteStream`](https://nodejs.org/api/fs.html#fs_class_fs_writestream).
.map(out => out.pipe(fs.createWriteStream(`${out.options.file}.md`)))
// .pipe(fs.createWriteStream(`${file}.md`))
// Now we attach the `close` listener.
.forEach(stream => stream.on('close', handleClose(cwd)))
// ## `exports`
// Public APIs:
exports.run = run
exports.doc = doc
exports.Transform = Transform