var fs = require('fs') var http = require('http') var qs = require('querystring') var path = require('path') var crypto = require('crypto') var cat = require('pull-cat') var pull = require('pull-stream') var paramap = require('pull-paramap') var marked = require('ssb-marked') var sort = require('ssb-sort') var toPull = require('stream-to-pull-stream') var memo = require('asyncmemo') var lru = require('lrucache') var htime = require('human-time') var emojis = require('emoji-named-characters') var serveEmoji = require('emoji-server')() var emojiDir = path.join(require.resolve('emoji-named-characters'), '../pngs') var appHash = hash([fs.readFileSync(__filename)]) var urlIdRegex = /^(?:\/(([%&]|%25)(?:[A-Za-z0-9\/+]|%2[Ff]|%2[Bb]){43}(?:=|%3D)\.sha256)(?:\.([^?]*))?|(\/.*?))(?:\?(.*))?$/ function MdRenderer(opts) { marked.Renderer.call(this, {}) this.opts = opts } MdRenderer.prototype = new marked.Renderer() MdRenderer.prototype.urltransform = function (href) { if (!href) return false switch (href[0]) { case '%': return this.opts.msg_base + encodeURIComponent(href) case '@': return this.opts.feed_base + encodeURIComponent(href) case '&': return this.opts.blob_base + encodeURIComponent(href) } if (href.indexOf('javascript:') === 0) return false return href } MdRenderer.prototype.image = function (href, title, text) { return '' : '>') } function renderEmoji(emoji) { var opts = this.renderer.opts return emoji in emojis ? '' : ':' + emoji + ':' } exports.name = 'viewer' exports.manifest = {} exports.version = require('./package').version exports.init = function (sbot, config) { var conf = config.viewer || {} var port = conf.port || 8807 var host = conf.host || config.host || '::' var base = conf.base || '/' var defaultOpts = { msg_base: conf.msg_base || base, feed_base: conf.feed_base || '#', blob_base: conf.blob_base || base, img_base: conf.img_base || base, emoji_base: conf.emoji_base || (base + 'emoji/'), } var getMsg = memo({cache: lru(100)}, getMsgWithValue, sbot) var getAbout = memo({cache: lru(100)}, require('./lib/about'), sbot) http.createServer(serve).listen(port, host, function () { console.log('[viewer] Listening on http://' + host + ':' + port) }) function serve(req, res) { if (req.method !== 'GET' && req.method !== 'HEAD') { return respond(res, 405, 'Method must be GET or HEAD') } var m = urlIdRegex.exec(req.url) switch (m[2]) { case '%25': m[2] = '%'; m[1] = decodeURIComponent(m[1]) case '%': return serveId(req, res, m[1], m[3], m[5]) case '&': return serveBlob(req, res, sbot, m[1]) default: return servePath(req, res, m[4]) } } function serveId(req, res, id, ext, query) { var q = query ? qs.parse(query) : {} var includeRoot = !('noroot' in q) var base = q.base || conf.base var baseToken if (!base) { if (ext === 'js') base = baseToken = '__BASE_' + Math.random() + '_' else base = '/' } var opts = { base: base, base_token: baseToken, msg_base: q.msg_base || conf.msg_base || base, feed_base: q.feed_base || conf.feed_base || '#', blob_base: q.blob_base || conf.blob_base || base, img_base: q.img_base || conf.img_base || base, emoji_base: q.emoji_base || conf.emoji_base || (base + 'emoji/'), } opts.marked = { gfm: true, mentions: true, tables: true, breaks: true, pedantic: false, sanitize: true, smartLists: true, smartypants: false, emoji: renderEmoji, renderer: new MdRenderer(opts) } var format = formatMsgs(id, ext, opts) if (format === null) return respond(res, 415, 'Invalid format') pull( sbot.links({dest: id, values: true, rel: 'root'}), includeRoot && prepend(getMsg, id), pull.unique('key'), pull.collect(function (err, links) { if (err) return respond(res, 500, err.stack || err) var etag = hash(sort.heads(links).concat(appHash, ext, qs)) if (req.headers['if-none-match'] === etag) return respond(res, 304) res.writeHead(200, { 'Content-Type': ctype(ext), 'etag': etag }) pull( pull.values(sort(links)), paramap(addAuthorAbout, 8), format, toPull(res, function (err) { if (err) console.error('[viewer]', err) }) ) }) ) } function addAuthorAbout(msg, cb) { getAbout(msg.value.author, function (err, about) { if (err) return cb(err) msg.author = about cb(null, msg) }) } } function serveBlob(req, res, sbot, id) { if (req.headers['if-none-match'] === id) return respond(res, 304) sbot.blobs.has(id, function (err, has) { if (err) { if (/^invalid/.test(err.message)) return respond(res, 400, err.message) else return respond(res, 500, err.message || err) } if (!has) return respond(res, 404, 'Not found') res.writeHead(200, { 'Cache-Control': 'public, max-age=315360000', 'etag': id }) pull( sbot.blobs.get(id), toPull(res, function (err) { if (err) console.error('[viewer]', err) }) ) }) } function getMsgWithValue(sbot, id, cb) { sbot.get(id, function (err, value) { if (err) return cb(err) cb(null, {key: id, value: value}) }) } function escape(str) { return String(str) .replace(/&/g, '&') .replace(//g, '>') .replace(/"/g, '"') } function respond(res, status, message) { res.writeHead(status) res.end(message) } function ctype(name) { switch (name && /[^.\/]*$/.exec(name)[0] || 'html') { case 'html': return 'text/html' case 'js': return 'text/javascript' case 'css': return 'text/css' case 'json': return 'application/json' } } function servePath(req, res, url) { switch (url) { case '/robots.txt': return res.end('User-agent: *') } var m = /^(\/?[^\/]*)(\/.*)?$/.exec(url) switch (m[1]) { case '/static': return serveStatic(req, res, m[2]) case '/emoji': return serveEmoji(req, res, m[2]) } return respond(res, 404, 'Not found') } function ifModified(req, lastMod) { var ifModSince = req.headers['if-modified-since'] if (!ifModSince) return false var d = new Date(ifModSince) return d && Math.floor(d/1000) >= Math.floor(lastMod/1000) } function serveStatic(req, res, file) { serveFile(req, res, path.join(__dirname, 'static', file)) } function serveFile(req, res, file) { fs.stat(file, function (err, stat) { if (err && err.code === 'ENOENT') return respond(res, 404, 'Not found') if (err) return respond(res, 500, err.stack || err) if (!stat.isFile()) return respond(res, 403, 'May only load files') if (ifModified(req, stat.mtime)) return respond(res, 304, 'Not modified') res.writeHead(200, { 'Content-Type': ctype(file), 'Content-Length': stat.size, 'Last-Modified': stat.mtime.toGMTString() }) fs.createReadStream(file).pipe(res) }) } function prepend(fn, arg) { return function (read) { return function (abort, cb) { if (fn && !abort) { var _fn = fn fn = null return _fn(arg, function (err, value) { if (err) return read(err, function (err) { cb(err || true) }) cb(null, value) }) } read(abort, cb) } } } function formatMsgs(id, ext, opts) { switch (ext || 'html') { case 'html': return pull(renderThread(opts), wrapPage(id)) case 'js': return pull(renderThread(opts), wrapJSEmbed(opts)) case 'json': return wrapJSON() default: return null } } function wrap(before, after) { return function (read) { return cat([pull.once(before), read, pull.once(after)]) } } function renderThread(opts) { return pull( pull.map(renderMsg.bind(this, opts)), wrap('
' + JSON.stringify(c, 0, 2) + '' }