-
-
Notifications
You must be signed in to change notification settings - Fork 13
/
Copy pathindex.js
executable file
Β·225 lines (208 loc) Β· 8.43 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
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
#!/usr/bin/env node
import { program } from 'commander'
import kleur from 'kleur'
import ora from 'ora'
import { linkspector } from './linkspector.js'
import { createRequire } from 'module'
const require = createRequire(import.meta.url)
const pkg = require('./package.json')
program
.version(pkg.version)
.description('π Uncover broken links in your content.')
.command('check')
.description('Check hyperlinks based on the configuration file.')
.option('-c, --config <path>', 'Specify a custom configuration file path')
.option('-j, --json', 'Output the results in JSON format')
.option('-s, --showstat', 'Display statistics about the links checked')
.action(async (cmd) => {
// Validate that -j and -s options are not used together
if (cmd.json && cmd.showstat) {
console.error(
kleur.red(
'Error: The --json and --showstat options cannot be used together.'
)
)
process.exit(1)
}
const configFile = cmd.config || '.linkspector.yml' // Use custom config file path if provided
let currentFile = '' // Variable to store the current file name
let results = [] // Array to store the results if json is true
// Initialize statistics counters
let stats = {
filesChecked: 0,
totalLinks: 0,
httpLinks: 0,
fileLinks: 0,
emailLinks: 0,
correctLinks: 0,
failedLinks: 0,
}
const spinner = cmd.json ? null : ora().start()
try {
let hasErrorLinks = false
// Initialize the results object
let results = {
source: {
name: 'linkspector',
url: 'https://github.com/UmbrellaDocs/linkspector',
},
severity: 'ERROR',
diagnostics: [],
}
for await (const { file, result } of linkspector(configFile, cmd)) {
// Update the current file name
currentFile = file
if (!cmd.json) {
spinner.text = `Checking ${currentFile}...`
}
// Increment file count for statistics
stats.filesChecked++
for (const linkStatusObj of result) {
// Count total links
stats.totalLinks++
// Count links by type
if (linkStatusObj.link && linkStatusObj.link.match(/^https?:\/\//)) {
stats.httpLinks++
} else if (
linkStatusObj.link &&
linkStatusObj.link.startsWith('mailto:')
) {
stats.emailLinks++
} else if (
linkStatusObj.link &&
(linkStatusObj.link.startsWith('#') ||
linkStatusObj.link.includes('.md') ||
linkStatusObj.link.includes('#'))
) {
stats.fileLinks++
} else if (linkStatusObj.link) {
// Count any remaining links as file links
stats.fileLinks++
}
// Count correct vs failed links - Updated to handle skipped links
if (linkStatusObj.status === 'error') {
stats.failedLinks++
if (cmd.json) {
results.diagnostics.push({
message: `Cannot reach ${linkStatusObj.link} Status: ${linkStatusObj.status_code}${linkStatusObj.error_message ? ` ${linkStatusObj.error_message}` : ''}`,
location: {
path: currentFile,
range: {
start: {
line: linkStatusObj.line_number,
column: linkStatusObj.position.start.column,
},
end: {
line: linkStatusObj.position.end.line,
column: linkStatusObj.position.end.column,
},
},
},
severity: linkStatusObj.status.toUpperCase(),
})
} else {
// If json is false, print the results in the console
spinner.stop()
console.log(
kleur.red(
`${currentFile}:${linkStatusObj.line_number}:${linkStatusObj.position.start.column}: π« ${linkStatusObj.link} Status:${linkStatusObj.status_code}${linkStatusObj.error_message ? ` ${linkStatusObj.error_message}` : ' Cannot reach link'}`
)
)
spinner.start(`Checking ${currentFile}...`)
}
hasErrorLinks = true
} else if (
linkStatusObj.status === 'alive' ||
linkStatusObj.status === 'assumed alive'
) {
stats.correctLinks++
} else if (linkStatusObj.status === 'skipped') {
// Skipped links don't count towards failed links
} else {
// Count other status as failed
stats.failedLinks++
}
}
}
if (cmd.json) {
// If there are no links with a status of "error", print a blank object
if (results.diagnostics.length === 0) {
console.log('{}')
} else {
console.log(JSON.stringify(results, null, 2))
}
}
// Display statistics if --showstat option is used
if (cmd.showstat) {
spinner.stop()
console.log('\n' + kleur.bold('ππ Linkspector check stats'))
console.log('βββββββββββββββββββββββββββββββββ¬βββββββββ')
console.log(
`β π° ${kleur.bold('Total files checked')} β ${kleur.cyan(padNumber(stats.filesChecked))} β`
)
console.log('βββββββββββββββββββββββββββββββββΌβββββββββ€')
console.log(
`β π ${kleur.bold('Total links checked')} β ${kleur.cyan(padNumber(stats.totalLinks))} β`
)
console.log('βββββββββββββββββββββββββββββββββΌβββββββββ€')
console.log(
`β π ${kleur.bold('Hyperlinks')} β ${kleur.cyan(padNumber(stats.httpLinks))} β`
)
console.log('βββββββββββββββββββββββββββββββββΌβββββββββ€')
console.log(
`β π ${kleur.bold('File and header links')} β ${kleur.cyan(padNumber(stats.fileLinks))} β`
)
console.log('βββββββββββββββββββββββββββββββββΌβββββββββ€')
console.log(
`β βοΈ ${kleur.bold('Email links (Skipped)')} β ${kleur.cyan(padNumber(stats.emailLinks))} β`
)
console.log('βββββββββββββββββββββββββββββββββΌβββββββββ€')
console.log(
`β β
${kleur.bold('Working links')} β ${kleur.green(padNumber(stats.correctLinks))} β`
)
console.log('βββββββββββββββββββββββββββββββββΌβββββββββ€')
console.log(
`β π« ${kleur.bold('Failed links')} β ${kleur.red(padNumber(stats.failedLinks))} β`
)
console.log('βββββββββββββββββββββββββββββββββ΄βββββββββ')
console.log('')
}
if (!hasErrorLinks) {
if (!cmd.json && !cmd.showstat) {
spinner.stop()
console.log(
kleur.green(
'β¨ Success: All hyperlinks in the specified files are valid.'
)
)
}
process.exit(0)
} else {
if (!cmd.json && !cmd.showstat) {
spinner.stop()
console.error(
kleur.red(
'π₯ Error: Some hyperlinks in the specified files are invalid.'
)
)
} else if (cmd.showstat) {
console.error(
kleur.red(
'π₯ Error: Some hyperlinks in the specified files are invalid.'
)
)
}
process.exit(1)
}
} catch (error) {
if (spinner) spinner.stop()
console.error(kleur.red(`π₯ Main error: ${error.message}`))
process.exit(1)
}
// Helper function to pad numbers for consistent table formatting
function padNumber(num) {
return num.toString().padStart(6, ' ')
}
})
// Parse the command line arguments
program.parse(process.argv)