log.js 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400
  1. 'use strict'
  2. var Progress = require('are-we-there-yet')
  3. var Gauge = require('gauge')
  4. var EE = require('events').EventEmitter
  5. var log = exports = module.exports = new EE()
  6. var util = require('util')
  7. var setBlocking = require('set-blocking')
  8. var consoleControl = require('console-control-strings')
  9. setBlocking(true)
  10. var stream = process.stderr
  11. Object.defineProperty(log, 'stream', {
  12. set: function (newStream) {
  13. stream = newStream
  14. if (this.gauge) {
  15. this.gauge.setWriteTo(stream, stream)
  16. }
  17. },
  18. get: function () {
  19. return stream
  20. },
  21. })
  22. // by default, decide based on tty-ness.
  23. var colorEnabled
  24. log.useColor = function () {
  25. return colorEnabled != null ? colorEnabled : stream.isTTY
  26. }
  27. log.enableColor = function () {
  28. colorEnabled = true
  29. this.gauge.setTheme({ hasColor: colorEnabled, hasUnicode: unicodeEnabled })
  30. }
  31. log.disableColor = function () {
  32. colorEnabled = false
  33. this.gauge.setTheme({ hasColor: colorEnabled, hasUnicode: unicodeEnabled })
  34. }
  35. // default level
  36. log.level = 'info'
  37. log.gauge = new Gauge(stream, {
  38. enabled: false, // no progress bars unless asked
  39. theme: { hasColor: log.useColor() },
  40. template: [
  41. { type: 'progressbar', length: 20 },
  42. { type: 'activityIndicator', kerning: 1, length: 1 },
  43. { type: 'section', default: '' },
  44. ':',
  45. { type: 'logline', kerning: 1, default: '' },
  46. ],
  47. })
  48. log.tracker = new Progress.TrackerGroup()
  49. // we track this separately as we may need to temporarily disable the
  50. // display of the status bar for our own loggy purposes.
  51. log.progressEnabled = log.gauge.isEnabled()
  52. var unicodeEnabled
  53. log.enableUnicode = function () {
  54. unicodeEnabled = true
  55. this.gauge.setTheme({ hasColor: this.useColor(), hasUnicode: unicodeEnabled })
  56. }
  57. log.disableUnicode = function () {
  58. unicodeEnabled = false
  59. this.gauge.setTheme({ hasColor: this.useColor(), hasUnicode: unicodeEnabled })
  60. }
  61. log.setGaugeThemeset = function (themes) {
  62. this.gauge.setThemeset(themes)
  63. }
  64. log.setGaugeTemplate = function (template) {
  65. this.gauge.setTemplate(template)
  66. }
  67. log.enableProgress = function () {
  68. if (this.progressEnabled || this._paused) {
  69. return
  70. }
  71. this.progressEnabled = true
  72. this.tracker.on('change', this.showProgress)
  73. this.gauge.enable()
  74. }
  75. log.disableProgress = function () {
  76. if (!this.progressEnabled) {
  77. return
  78. }
  79. this.progressEnabled = false
  80. this.tracker.removeListener('change', this.showProgress)
  81. this.gauge.disable()
  82. }
  83. var trackerConstructors = ['newGroup', 'newItem', 'newStream']
  84. var mixinLog = function (tracker) {
  85. // mixin the public methods from log into the tracker
  86. // (except: conflicts and one's we handle specially)
  87. Object.keys(log).forEach(function (P) {
  88. if (P[0] === '_') {
  89. return
  90. }
  91. if (trackerConstructors.filter(function (C) {
  92. return C === P
  93. }).length) {
  94. return
  95. }
  96. if (tracker[P]) {
  97. return
  98. }
  99. if (typeof log[P] !== 'function') {
  100. return
  101. }
  102. var func = log[P]
  103. tracker[P] = function () {
  104. return func.apply(log, arguments)
  105. }
  106. })
  107. // if the new tracker is a group, make sure any subtrackers get
  108. // mixed in too
  109. if (tracker instanceof Progress.TrackerGroup) {
  110. trackerConstructors.forEach(function (C) {
  111. var func = tracker[C]
  112. tracker[C] = function () {
  113. return mixinLog(func.apply(tracker, arguments))
  114. }
  115. })
  116. }
  117. return tracker
  118. }
  119. // Add tracker constructors to the top level log object
  120. trackerConstructors.forEach(function (C) {
  121. log[C] = function () {
  122. return mixinLog(this.tracker[C].apply(this.tracker, arguments))
  123. }
  124. })
  125. log.clearProgress = function (cb) {
  126. if (!this.progressEnabled) {
  127. return cb && process.nextTick(cb)
  128. }
  129. this.gauge.hide(cb)
  130. }
  131. log.showProgress = function (name, completed) {
  132. if (!this.progressEnabled) {
  133. return
  134. }
  135. var values = {}
  136. if (name) {
  137. values.section = name
  138. }
  139. var last = log.record[log.record.length - 1]
  140. if (last) {
  141. values.subsection = last.prefix
  142. var disp = log.disp[last.level] || last.level
  143. var logline = this._format(disp, log.style[last.level])
  144. if (last.prefix) {
  145. logline += ' ' + this._format(last.prefix, this.prefixStyle)
  146. }
  147. logline += ' ' + last.message.split(/\r?\n/)[0]
  148. values.logline = logline
  149. }
  150. values.completed = completed || this.tracker.completed()
  151. this.gauge.show(values)
  152. }.bind(log) // bind for use in tracker's on-change listener
  153. // temporarily stop emitting, but don't drop
  154. log.pause = function () {
  155. this._paused = true
  156. if (this.progressEnabled) {
  157. this.gauge.disable()
  158. }
  159. }
  160. log.resume = function () {
  161. if (!this._paused) {
  162. return
  163. }
  164. this._paused = false
  165. var b = this._buffer
  166. this._buffer = []
  167. b.forEach(function (m) {
  168. this.emitLog(m)
  169. }, this)
  170. if (this.progressEnabled) {
  171. this.gauge.enable()
  172. }
  173. }
  174. log._buffer = []
  175. var id = 0
  176. log.record = []
  177. log.maxRecordSize = 10000
  178. log.log = function (lvl, prefix, message) {
  179. var l = this.levels[lvl]
  180. if (l === undefined) {
  181. return this.emit('error', new Error(util.format(
  182. 'Undefined log level: %j', lvl)))
  183. }
  184. var a = new Array(arguments.length - 2)
  185. var stack = null
  186. for (var i = 2; i < arguments.length; i++) {
  187. var arg = a[i - 2] = arguments[i]
  188. // resolve stack traces to a plain string.
  189. if (typeof arg === 'object' && arg instanceof Error && arg.stack) {
  190. Object.defineProperty(arg, 'stack', {
  191. value: stack = arg.stack + '',
  192. enumerable: true,
  193. writable: true,
  194. })
  195. }
  196. }
  197. if (stack) {
  198. a.unshift(stack + '\n')
  199. }
  200. message = util.format.apply(util, a)
  201. var m = {
  202. id: id++,
  203. level: lvl,
  204. prefix: String(prefix || ''),
  205. message: message,
  206. messageRaw: a,
  207. }
  208. this.emit('log', m)
  209. this.emit('log.' + lvl, m)
  210. if (m.prefix) {
  211. this.emit(m.prefix, m)
  212. }
  213. this.record.push(m)
  214. var mrs = this.maxRecordSize
  215. var n = this.record.length - mrs
  216. if (n > mrs / 10) {
  217. var newSize = Math.floor(mrs * 0.9)
  218. this.record = this.record.slice(-1 * newSize)
  219. }
  220. this.emitLog(m)
  221. }.bind(log)
  222. log.emitLog = function (m) {
  223. if (this._paused) {
  224. this._buffer.push(m)
  225. return
  226. }
  227. if (this.progressEnabled) {
  228. this.gauge.pulse(m.prefix)
  229. }
  230. var l = this.levels[m.level]
  231. if (l === undefined) {
  232. return
  233. }
  234. if (l < this.levels[this.level]) {
  235. return
  236. }
  237. if (l > 0 && !isFinite(l)) {
  238. return
  239. }
  240. // If 'disp' is null or undefined, use the lvl as a default
  241. // Allows: '', 0 as valid disp
  242. var disp = log.disp[m.level] != null ? log.disp[m.level] : m.level
  243. this.clearProgress()
  244. m.message.split(/\r?\n/).forEach(function (line) {
  245. var heading = this.heading
  246. if (heading) {
  247. this.write(heading, this.headingStyle)
  248. this.write(' ')
  249. }
  250. this.write(disp, log.style[m.level])
  251. var p = m.prefix || ''
  252. if (p) {
  253. this.write(' ')
  254. }
  255. this.write(p, this.prefixStyle)
  256. this.write(' ' + line + '\n')
  257. }, this)
  258. this.showProgress()
  259. }
  260. log._format = function (msg, style) {
  261. if (!stream) {
  262. return
  263. }
  264. var output = ''
  265. if (this.useColor()) {
  266. style = style || {}
  267. var settings = []
  268. if (style.fg) {
  269. settings.push(style.fg)
  270. }
  271. if (style.bg) {
  272. settings.push('bg' + style.bg[0].toUpperCase() + style.bg.slice(1))
  273. }
  274. if (style.bold) {
  275. settings.push('bold')
  276. }
  277. if (style.underline) {
  278. settings.push('underline')
  279. }
  280. if (style.inverse) {
  281. settings.push('inverse')
  282. }
  283. if (settings.length) {
  284. output += consoleControl.color(settings)
  285. }
  286. if (style.beep) {
  287. output += consoleControl.beep()
  288. }
  289. }
  290. output += msg
  291. if (this.useColor()) {
  292. output += consoleControl.color('reset')
  293. }
  294. return output
  295. }
  296. log.write = function (msg, style) {
  297. if (!stream) {
  298. return
  299. }
  300. stream.write(this._format(msg, style))
  301. }
  302. log.addLevel = function (lvl, n, style, disp) {
  303. // If 'disp' is null or undefined, use the lvl as a default
  304. if (disp == null) {
  305. disp = lvl
  306. }
  307. this.levels[lvl] = n
  308. this.style[lvl] = style
  309. if (!this[lvl]) {
  310. this[lvl] = function () {
  311. var a = new Array(arguments.length + 1)
  312. a[0] = lvl
  313. for (var i = 0; i < arguments.length; i++) {
  314. a[i + 1] = arguments[i]
  315. }
  316. return this.log.apply(this, a)
  317. }.bind(this)
  318. }
  319. this.disp[lvl] = disp
  320. }
  321. log.prefixStyle = { fg: 'magenta' }
  322. log.headingStyle = { fg: 'white', bg: 'black' }
  323. log.style = {}
  324. log.levels = {}
  325. log.disp = {}
  326. log.addLevel('silly', -Infinity, { inverse: true }, 'sill')
  327. log.addLevel('verbose', 1000, { fg: 'cyan', bg: 'black' }, 'verb')
  328. log.addLevel('info', 2000, { fg: 'green' })
  329. log.addLevel('timing', 2500, { fg: 'green', bg: 'black' })
  330. log.addLevel('http', 3000, { fg: 'green', bg: 'black' })
  331. log.addLevel('notice', 3500, { fg: 'cyan', bg: 'black' })
  332. log.addLevel('warn', 4000, { fg: 'black', bg: 'yellow' }, 'WARN')
  333. log.addLevel('error', 5000, { fg: 'red', bg: 'black' }, 'ERR!')
  334. log.addLevel('silent', Infinity)
  335. // allow 'error' prefix
  336. log.on('error', function () {})