const _ = require('lodash'); const xml = require('xml'); const config = require('../../shared/config'); const urlService = require('../../frontend/services/url'); const errors = require('@tryghost/errors'); const {events, i18n} = require('../lib/common'); const logging = require('../../shared/logging'); const request = require('../lib/request'); const settingsCache = require('./settings/cache'); const defaultPostSlugs = [ 'welcome', 'the-editor', 'using-tags', 'managing-users', 'private-sites', 'advanced-markdown', 'themes' ]; // ToDo: Make this configurable const pingList = [ { url: 'http://rpc.pingomatic.com' } ]; function ping(post) { let pingXML; const title = post.title; const url = urlService.getUrlByResourceId(post.id, {absolute: true}); if (post.type === 'page' || config.isPrivacyDisabled('useRpcPing') || settingsCache.get('is_private')) { return; } // Don't ping for the default posts. // This also handles the case where during ghost's first run // models.init() inserts this post but permissions.init() hasn't // (can't) run yet. if (defaultPostSlugs.indexOf(post.slug) > -1) { return; } // Build XML object. pingXML = xml({ methodCall: [{ methodName: 'weblogUpdates.ping' }, { params: [{ param: [{ value: [{ string: title }] }] }, { param: [{ value: [{ string: url }] }] }] }] }, {declaration: true}); // Ping each of the defined services. _.each(pingList, function (pingHost) { const options = { body: pingXML, timeout: 2 * 1000 }; const goodResponse = /[\s]*flerror<\/name>[\s]*[\s]*0<\/boolean><\/value><\/member>/; const errorMessage = /(?:faultString|message)<\/name>[\s]*[\s]*([^<]+)/; request(pingHost.url, options) .then(function (res) { if (!goodResponse.test(res.body)) { const matches = res.body.match(errorMessage); const message = matches ? matches[1] : res.body; throw new Error(message); } }) .catch(function (err) { if (err.statusCode === 429) { logging.error(new errors.TooManyRequestsError({ err, message: err.message, context: i18n.t('errors.services.ping.requestFailed.error', {service: 'xmlrpc'}), help: i18n.t('errors.services.ping.requestFailed.help', {url: 'https://ghost.org/docs/'}) })); } else { logging.error(new errors.GhostError({ err: err, message: err.message, context: i18n.t('errors.services.ping.requestFailed.error', {service: 'xmlrpc'}), help: i18n.t('errors.services.ping.requestFailed.help', {url: 'https://ghost.org/docs/'}) })); } }); }); } function listener(model, options) { // CASE: do not rpc ping if we import a database // TODO: refactor post.published events to never fire on importing if (options && options.importing) { return; } ping(model.toJSON()); } function listen() { events.on('post.published', listener); } module.exports = { listen: listen };