-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsearch.js
114 lines (101 loc) · 3.36 KB
/
search.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
'use strict'
module.exports = exports = search
const npm = require('./npm.js')
const allPackageSearch = require('./search/all-package-search')
const figgyPudding = require('figgy-pudding')
const formatPackageStream = require('./search/format-package-stream.js')
const libSearch = require('libnpm/search')
const log = require('npmlog')
const ms = require('mississippi')
const npmConfig = require('./config/figgy-config.js')
const output = require('./utils/output.js')
const usage = require('./utils/usage')
search.usage = usage(
'search',
'npm search [--long] [search terms ...]'
)
search.completion = function (opts, cb) {
cb(null, [])
}
const SearchOpts = figgyPudding({
description: {},
exclude: {},
include: {},
limit: {},
log: {},
staleness: {},
unicode: {}
})
function search (args, cb) {
const opts = SearchOpts(npmConfig()).concat({
description: npm.config.get('description'),
exclude: prepareExcludes(npm.config.get('searchexclude')),
include: prepareIncludes(args, npm.config.get('searchopts')),
limit: npm.config.get('searchlimit') || 20,
log: log,
staleness: npm.config.get('searchstaleness'),
unicode: npm.config.get('unicode')
})
if (opts.include.length === 0) {
return cb(new Error('search must be called with arguments'))
}
// Used later to figure out whether we had any packages go out
let anyOutput = false
const entriesStream = ms.through.obj()
let esearchWritten = false
libSearch.stream(opts.include, opts).on('data', pkg => {
entriesStream.write(pkg)
!esearchWritten && (esearchWritten = true)
}).on('error', err => {
if (esearchWritten) {
// If esearch errored after already starting output, we can't fall back.
return entriesStream.emit('error', err)
}
log.warn('search', 'fast search endpoint errored. Using old search.')
allPackageSearch(opts)
.on('data', pkg => entriesStream.write(pkg))
.on('error', err => entriesStream.emit('error', err))
.on('end', () => entriesStream.end())
}).on('end', () => entriesStream.end())
// Grab a configured output stream that will spit out packages in the
// desired format.
var outputStream = formatPackageStream({
args: args, // --searchinclude options are not highlighted
long: npm.config.get('long'),
description: npm.config.get('description'),
json: npm.config.get('json'),
parseable: npm.config.get('parseable'),
color: npm.color
})
outputStream.on('data', chunk => {
if (!anyOutput) { anyOutput = true }
output(chunk.toString('utf8'))
})
log.silly('search', 'searching packages')
ms.pipe(entriesStream, outputStream, err => {
if (err) return cb(err)
if (!anyOutput && !npm.config.get('json') && !npm.config.get('parseable')) {
output('No matches found for ' + (args.map(JSON.stringify).join(' ')))
}
log.silly('search', 'search completed')
log.clearProgress()
cb(null, {})
})
}
function prepareIncludes (args, searchopts) {
if (typeof searchopts !== 'string') searchopts = ''
return searchopts.split(/\s+/).concat(args).map(function (s) {
return s.toLowerCase()
}).filter(function (s) { return s })
}
function prepareExcludes (searchexclude) {
var exclude
if (typeof searchexclude === 'string') {
exclude = searchexclude.split(/\s+/)
} else {
exclude = []
}
return exclude.map(function (s) {
return s.toLowerCase()
})
}