// # Update Checking Service // // Makes a request to Ghost.org to check if there is a new version of Ghost available. // The service is provided in return for users opting in to anonymous usage data collection. // // Blog owners can opt-out of update checks by setting `privacy: { useUpdateCheck: false }` in their config.js // // The data collected is as follows: // // - blog id - a hash of the blog hostname, pathname and db_hash. No identifiable info is stored. // - ghost version // - node version // - npm version // - env - production or development // - database type - SQLite, MySQL // - email transport - mail.options.service, or otherwise mail.transport // - created date - database creation date // - post count - total number of posts // - user count - total number of users // - theme - name of the currently active theme // - apps - names of any active apps var crypto = require('crypto'), exec = require('child_process').exec, https = require('https'), http = require('http'), moment = require('moment'), semver = require('semver'), Promise = require('bluebird'), _ = require('lodash'), url = require('url'), api = require('./api'), config = require('./config'), utils = require('./utils'), logging = require('./logging'), errors = require('./errors'), i18n = require('./i18n'), currentVersion = require('./utils/ghost-version').full, internal = {context: {internal: true}}, checkEndpoint = config.get('updateCheckUrl') || 'https://updates.ghost.org'; function updateCheckError(err) { err = errors.utils.deserialize(err); api.settings.edit( {settings: [{key: 'next_update_check', value: Math.round(Date.now() / 1000 + 24 * 3600)}]}, internal ); err.context = i18n.t('errors.update-check.checkingForUpdatesFailed.error'); err.help = i18n.t('errors.update-check.checkingForUpdatesFailed.help', {url: 'https://docs.ghost.org/v1.0.0'}); logging.error(err); } /** * If the custom message is intended for current version, create and store a custom notification. * @param {Object} message {id: uuid, version: '0.9.x', content: '' } * @return {*|Promise} */ function createCustomNotification(message) { if (!semver.satisfies(currentVersion, message.version)) { return Promise.resolve(); } var notification = { status: 'alert', type: 'info', custom: true, uuid: message.id, dismissible: true, message: message.content }, getAllNotifications = api.notifications.browse({context: {internal: true}}), getSeenNotifications = api.settings.read(_.extend({key: 'seen_notifications'}, internal)); return Promise.join(getAllNotifications, getSeenNotifications, function joined(all, seen) { var isSeen = _.includes(JSON.parse(seen.settings[0].value || []), notification.id), isDuplicate = _.some(all.notifications, {message: notification.message}); if (!isSeen && !isDuplicate) { return api.notifications.add({notifications: [notification]}, {context: {internal: true}}); } }); } function updateCheckData() { var data = {}, mailConfig = config.get('mail'); data.ghost_version = currentVersion; data.node_version = process.versions.node; data.env = config.get('env'); data.database_type = config.get('database').client; data.email_transport = mailConfig && (mailConfig.options && mailConfig.options.service ? mailConfig.options.service : mailConfig.transport); return Promise.props({ hash: api.settings.read(_.extend({key: 'db_hash'}, internal)).reflect(), theme: api.settings.read(_.extend({key: 'active_theme'}, internal)).reflect(), apps: api.settings.read(_.extend({key: 'active_apps'}, internal)) .then(function (response) { var apps = response.settings[0]; apps = JSON.parse(apps.value); return _.reduce(apps, function (memo, item) { return memo === '' ? memo + item : memo + ', ' + item; }, ''); }).reflect(), posts: api.posts.browse().reflect(), users: api.users.browse(internal).reflect(), npm: Promise.promisify(exec)('npm -v').reflect() }).then(function (descriptors) { var hash = descriptors.hash.value().settings[0], theme = descriptors.theme.value().settings[0], apps = descriptors.apps.value(), posts = descriptors.posts.value(), users = descriptors.users.value(), npm = descriptors.npm.value(), blogUrl = url.parse(utils.url.urlFor('home', true)), blogId = blogUrl.hostname + blogUrl.pathname.replace(/\//, '') + hash.value; data.blog_id = crypto.createHash('md5').update(blogId).digest('hex'); data.theme = theme ? theme.value : ''; data.apps = apps || ''; data.post_count = posts && posts.meta && posts.meta.pagination ? posts.meta.pagination.total : 0; data.user_count = users && users.users && users.users.length ? users.users.length : 0; data.blog_created_at = users && users.users && users.users[0] && users.users[0].created_at ? moment(users.users[0].created_at).unix() : ''; data.npm_version = npm.trim(); data.lts = false; return data; }).catch(updateCheckError); } function updateCheckRequest() { return updateCheckData().then(function then(reqData) { var resData = '', headers, req, requestHandler, parsedUrl; reqData = JSON.stringify(reqData); headers = { 'Content-Type': 'application/json', 'Content-Length': Buffer.byteLength(reqData) }; return new Promise(function p(resolve, reject) { requestHandler = checkEndpoint.indexOf('https') === 0 ? https : http; parsedUrl = url.parse(checkEndpoint); req = requestHandler.request({ hostname: parsedUrl.hostname, port: parsedUrl.port, method: 'POST', headers: headers }, function handler(res) { res.on('error', function onError(error) { reject(error); }); res.on('data', function onData(chunk) { resData += chunk; }); res.on('end', function onEnd() { try { resData = JSON.parse(resData); if (this.statusCode >= 400) { return reject(resData); } resolve(resData); } catch (e) { reject(i18n.t('errors.update-check.unableToDecodeUpdateResponse.error')); } }); }); req.on('socket', function onSocket(socket) { // Wait a maximum of 10seconds socket.setTimeout(10000); socket.on('timeout', function onTimeout() { req.abort(); }); }); req.on('error', function onError(error) { reject(error); }); req.write(reqData); req.end(); }); }); } /** * Handles the response from the update check * Does three things with the information received: * 1. Updates the time we can next make a check * 2. Checks if the version in the response is new, and updates the notification setting * 3. Create custom notifications is response from UpdateCheck as "messages" array which has the following structure: * * "messages": [{ * "id": ed9dc38c-73e5-4d72-a741-22b11f6e151a, * "version": "0.5.x", * "content": "
Hey there! 0.6 is available, visit Ghost.org to grab your copy now" * ]} * * @param {Object} response * @return {Promise} */ function updateCheckResponse(response) { return Promise.all([ api.settings.edit({settings: [{key: 'next_update_check', value: response.next_check}]}, internal), api.settings.edit({settings: [{key: 'display_update_notification', value: response.version}]}, internal) ]).then(function () { var messages = response.messages || []; /** * by default the update check service returns messages: [] * but the latest release version get's stored anyway, because we adding the `display_update_notification` ^ */ return Promise.map(messages, createCustomNotification); }); } function updateCheck() { if (config.isPrivacyDisabled('useUpdateCheck')) { return Promise.resolve(); } else { return api.settings.read(_.extend({key: 'next_update_check'}, internal)).then(function then(result) { var nextUpdateCheck = result.settings[0]; if (nextUpdateCheck && nextUpdateCheck.value && nextUpdateCheck.value > moment().unix()) { // It's not time to check yet return; } else { // We need to do a check return updateCheckRequest() .then(updateCheckResponse) .catch(updateCheckError); } }).catch(updateCheckError); } } function showUpdateNotification() { return api.settings.read(_.extend({key: 'display_update_notification'}, internal)).then(function then(response) { var display = response.settings[0]; if (display && display.value && currentVersion && semver.gt(display.value, currentVersion)) { return display.value; } return false; }); } module.exports = updateCheck; module.exports.showUpdateNotification = showUpdateNotification;