forked from lix-project/lix-website
120 lines
2.8 KiB
JavaScript
120 lines
2.8 KiB
JavaScript
|
'use strict'
|
||
|
|
||
|
var events = require('events')
|
||
|
var html = require('./compile/html.js')
|
||
|
var parse = require('./parse.js')
|
||
|
var postprocess = require('./postprocess.js')
|
||
|
var preprocess = require('./preprocess.js')
|
||
|
|
||
|
function stream(options) {
|
||
|
var preprocess$1 = preprocess()
|
||
|
var tokenize = parse(options).document().write
|
||
|
var compile = html(options)
|
||
|
var emitter = new events.EventEmitter()
|
||
|
var ended
|
||
|
|
||
|
emitter.writable = emitter.readable = true
|
||
|
emitter.write = write
|
||
|
emitter.end = end
|
||
|
emitter.pipe = pipe
|
||
|
|
||
|
return emitter
|
||
|
|
||
|
// Write a chunk into memory.
|
||
|
function write(chunk, encoding, callback) {
|
||
|
if (typeof encoding === 'function') {
|
||
|
callback = encoding
|
||
|
encoding = undefined
|
||
|
}
|
||
|
|
||
|
if (ended) {
|
||
|
throw new Error('Did not expect `write` after `end`')
|
||
|
}
|
||
|
|
||
|
tokenize(preprocess$1(chunk || '', encoding))
|
||
|
|
||
|
if (callback) {
|
||
|
callback()
|
||
|
}
|
||
|
|
||
|
// Signal succesful write.
|
||
|
return true
|
||
|
}
|
||
|
|
||
|
// End the writing.
|
||
|
// Passes all arguments to a final `write`.
|
||
|
function end(chunk, encoding, callback) {
|
||
|
write(chunk, encoding, callback)
|
||
|
|
||
|
emitter.emit(
|
||
|
'data',
|
||
|
compile(postprocess(tokenize(preprocess$1('', encoding, true))))
|
||
|
)
|
||
|
|
||
|
emitter.emit('end')
|
||
|
ended = true
|
||
|
return true
|
||
|
}
|
||
|
|
||
|
// Pipe the processor into a writable stream.
|
||
|
// Basically `Stream#pipe`, but inlined and simplified to keep the bundled
|
||
|
// size down.
|
||
|
// See: <https://github.com/nodejs/node/blob/43a5170/lib/internal/streams/legacy.js#L13>.
|
||
|
function pipe(dest, options) {
|
||
|
emitter.on('data', ondata)
|
||
|
emitter.on('error', onerror)
|
||
|
emitter.on('end', cleanup)
|
||
|
emitter.on('close', cleanup)
|
||
|
|
||
|
// If the `end` option is not supplied, `dest.end()` will be called when the
|
||
|
// `end` or `close` events are received.
|
||
|
if (!dest._isStdio && (!options || options.end !== false)) {
|
||
|
emitter.on('end', onend)
|
||
|
}
|
||
|
|
||
|
dest.on('error', onerror)
|
||
|
dest.on('close', cleanup)
|
||
|
|
||
|
dest.emit('pipe', emitter)
|
||
|
|
||
|
return dest
|
||
|
|
||
|
// End destination.
|
||
|
function onend() {
|
||
|
if (dest.end) {
|
||
|
dest.end()
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Handle data.
|
||
|
function ondata(chunk) {
|
||
|
if (dest.writable) {
|
||
|
dest.write(chunk)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Clean listeners.
|
||
|
function cleanup() {
|
||
|
emitter.removeListener('data', ondata)
|
||
|
emitter.removeListener('end', onend)
|
||
|
emitter.removeListener('error', onerror)
|
||
|
emitter.removeListener('end', cleanup)
|
||
|
emitter.removeListener('close', cleanup)
|
||
|
|
||
|
dest.removeListener('error', onerror)
|
||
|
dest.removeListener('close', cleanup)
|
||
|
}
|
||
|
|
||
|
// Close dangling pipes and handle unheard errors.
|
||
|
function onerror(error) {
|
||
|
cleanup()
|
||
|
|
||
|
if (!emitter.listenerCount('error')) {
|
||
|
throw error // Unhandled stream error in pipe.
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
module.exports = stream
|