Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feature/pretty print #81

Merged
merged 12 commits into from
Mar 16, 2019
Merged
Show file tree
Hide file tree
Changes from 3 commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
69 changes: 69 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -82,6 +82,8 @@ async function third (instance, opts) {
* <a href="#onClose"><code>instance.<b>onClose()</b></code></a>
* <a href="#close"><code>instance.<b>close()</b></code></a>
* <a href="#express"><code>avvio.<b>express()</b></code></a>
* <a href="#toJSON"><code>avvio.<b>toJSON()</b></code></a>
* <a href="#prettyPrint"><code>avvio.<b>prettyPrint()</b></code></a>

-------------------------------------------------------
<a name="constructor"></a>
Expand Down Expand Up @@ -475,6 +477,73 @@ app.close()

-------------------------------------------------------

<a name="toJSON"></a>

### avvio.toJSON()

Return a JSON tree rappresenting the state of the plugins and the loading time.
Call it on `preReady` to get the complete tree.

```js
const avvio = require('avvio')()
avvio.on('preReady', () => {
avvio.toJSON()
})
```

The ouput is like this:
```json
{
"avvio":{
"label":"avvio",
"start":1550165276637,
"stop":1550165276639,
"diff":2,
"children":[
{
"label":"one",
"start":1550165276639,
"stop":1550165276639,
"diff":0
},
{
"label":"two",
"start":1550165276639,
"stop":1550165276639,
"diff":0
}
]
}
}
```

-------------------------------------------------------

<a name="prettyPrint"></a>

### avvio.prettyPrint()

Print to `console.log` the tree returned by the `toJSON()` method.

```js
const avvio = require('avvio')()
avvio.on('preReady', () => {
avvio.prettyPrint()
})
```

The ouput will be like:
Eomm marked this conversation as resolved.
Show resolved Hide resolved

```
avvio 56 ms
├── first 52 ms
├── second 1 ms
├── bound _after 0 ms
└── third 2 ms
```

-------------------------------------------------------

## Acknowledgements

This project was kindly sponsored by [nearForm](http://nearform.com).
Expand Down
24 changes: 24 additions & 0 deletions boot.js
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@
const fastq = require('fastq')
const EE = require('events').EventEmitter
const inherits = require('util').inherits
const TimeTree = require('./time-tree')
const Plugin = require('./plugin')
const debug = require('debug')('avvio')

Expand Down Expand Up @@ -113,6 +114,8 @@ function Boot (server, opts, done) {

this.started = false
this.booted = false
this.name = 'avvio'
this.pluginTree = new TimeTree(this.name)

this._readyQ = fastq(this, callWithCbOrNextTick, 1)
this._readyQ.pause()
Expand All @@ -133,6 +136,16 @@ function Boot (server, opts, done) {
this._doStart = null
const main = new Plugin(this, root.bind(this), opts, noop, 0)

main.on('enqueue', (serverName, funcName, time) => {
this.pluginTree.add(this.name, null, time)
})
main.on('start', (serverName, funcName, time) => {
this.pluginTree.start(this.name, null, time)
})
main.on('loaded', (serverName, funcName, time) => {
this.pluginTree.stop(this.name, null, time)
})

Plugin.loadPlugin.call(this, main, (err) => {
debug('root plugin ready')
this.emit('preReady')
Expand Down Expand Up @@ -195,6 +208,9 @@ Boot.prototype._addPlugin = function (plugin, opts, isAfter) {
const current = this._current[0]

const obj = new Plugin(this, plugin, opts, isAfter)
obj.on('enqueue', this.pluginTree.add.bind(this.pluginTree))
obj.on('start', this.pluginTree.start.bind(this.pluginTree))
obj.on('loaded', this.pluginTree.stop.bind(this.pluginTree))

if (current.loaded) {
throw new Error(`Impossible to load "${obj.name}" plugin because the parent "${current.name}" was already loaded`)
Expand Down Expand Up @@ -288,6 +304,14 @@ Boot.prototype.ready = function (func) {
})
}

Boot.prototype.prettyPrint = function () {
this.pluginTree.prittyPrint()
}

Boot.prototype.toJSON = function () {
return this.pluginTree.toJSON()
}

function noop () { }

function callWithCbOrNextTick (func, cb, context) {
Expand Down
6 changes: 5 additions & 1 deletion example.js
Original file line number Diff line number Diff line change
Expand Up @@ -27,10 +27,14 @@ avvio
console.log('application booted!')
})

avvio.on('preReady', () => {
avvio.prettyPrint()
})

function first (instance, opts, cb) {
console.log('first loaded', opts)
instance.use(second)
cb()
setTimeout(cb, 42)
}

function second (instance, opts, cb) {
Expand Down
1 change: 1 addition & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -40,6 +40,7 @@
"typescript": "^3.1.1"
},
"dependencies": {
"archy": "^1.0.0",
"debug": "^4.0.0",
"fastq": "^1.6.0"
}
Expand Down
7 changes: 7 additions & 0 deletions plugin.js
Original file line number Diff line number Diff line change
@@ -1,6 +1,8 @@
'use strict'

const fastq = require('fastq')
const EE = require('events').EventEmitter
const inherits = require('util').inherits
const debug = require('debug')('avvio')
const CODE_PLUGIN_TIMEOUT = 'ERR_AVVIO_PLUGIN_TIMEOUT'

Expand Down Expand Up @@ -45,6 +47,8 @@ function Plugin (parent, func, opts, isAfter, timeout) {
// or they will end up at the top of _current
}

inherits(Plugin, EE)

Plugin.prototype.exec = function (server, cb) {
const func = this.func
var completed = false
Expand Down Expand Up @@ -79,6 +83,7 @@ Plugin.prototype.exec = function (server, cb) {
}, this.timeout)
}

this.emit('start', this.server ? this.server.name : null, this.name, Date.now())
var promise = func(this.server, this.opts, done)
if (promise && typeof promise.then === 'function') {
debug('resolving promise', name)
Expand Down Expand Up @@ -111,6 +116,7 @@ Plugin.prototype.exec = function (server, cb) {

Plugin.prototype.enqueue = function (obj, cb) {
debug('enqueue', this.name, obj.name)
this.emit('enqueue', this.server ? this.server.name : null, this.name, Date.now())
this.q.push(obj, cb)
}

Expand All @@ -122,6 +128,7 @@ Plugin.prototype.finish = function (err, cb) {
}

debug('loaded', this.name)
this.emit('loaded', this.server ? this.server.name : null, this.name, Date.now())
this.loaded = true

cb(err)
Expand Down
63 changes: 63 additions & 0 deletions test/to-json.test.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,63 @@
'use strict'

const test = require('tap').test
const boot = require('..')

test('to json', (t) => {
t.plan(4)

const app = boot()
app
.use(one)
.use(two)
.use(three)

const outJson = {
avvio: {
label: 'avvio',
start: /\d*/,
stop: /\d*/,
diff: /\d*/,
children: []
}
}

app.on('preReady', function show () {
const json = app.toJSON()
t.like(json, outJson)
})

function one (s, opts, done) {
const json = app.toJSON()
outJson.avvio.children.push({
label: 'one',
start: /\d*/,
stop: /\d*/,
diff: /\d*/
})
t.like(json, outJson)
done()
}
function two (s, opts, done) {
const json = app.toJSON()
outJson.avvio.children.push({
label: 'two',
start: /\d*/,
stop: /\d*/,
diff: /\d*/
})
t.like(json, outJson)
done()
}
function three (s, opts, done) {
const json = app.toJSON()
outJson.avvio.children.push({
label: 'three',
start: /\d*/,
stop: /\d*/,
diff: /\d*/
})
t.like(json, outJson)
done()
}
})
Eomm marked this conversation as resolved.
Show resolved Hide resolved
81 changes: 81 additions & 0 deletions time-tree.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,81 @@
'use strict'
const archy = require('archy')

function TimeTree (rootName = 'boot') {
this.timeTable = {}
this.root = rootName
}

TimeTree.prototype.get = function (parent = this.root, child) {
const p = this.timeTable[parent]
if (p) {
if (child !== null) {
return p.children.find(_ => _.label === child)
}
return p
}
return undefined
}

TimeTree.prototype.add = function (parent = this.root, track) {
let p = this.timeTable[parent]
if (!p) {
this.timeTable[parent] = {
children: [],
label: parent
}
p = this.timeTable[parent]
}

if (track && !this.get(parent, track)) {
p.children.push({ label: track })
}
}

TimeTree.prototype.start = function (parent, track, start) {
const tm = this.get(parent, track)
if (!tm) {
this.add(parent, track)
}

this.get(parent, track).start = start || Date.now()
}

TimeTree.prototype.stop = function (parent, track, stop) {
const tm = this.get(parent, track)
if (tm) {
tm.stop = stop || Date.now()
tm.diff = (tm.stop - tm.start) || 0
}
}

TimeTree.prototype.toJSON = function () {
return Object.assign({}, this.timeTable)
}

TimeTree.prototype.prittyPrint = function () {
const analyzePlugin = (node) => {
let nodes = []
if (node.children && node.children.length > 0) {
nodes = node.children.map(_ => analyzePlugin(_))
}
return {
label: `${node.label} ${node.diff} ms`,
nodes
}
}

const plugins = Object.keys(this.timeTable).map(_ => analyzePlugin(this.timeTable[_]))
let out
if (plugins.length === 1) {
out = plugins[0]
} else {
out = {
label: this.root,
nodes: plugins
}
}
console.log(archy(out))
Eomm marked this conversation as resolved.
Show resolved Hide resolved
}

module.exports = TimeTree