// # Ghost Head Helper // Usage: `{{ghost_head}}` // // Outputs scripts and other assets at the top of a Ghost theme // // We use the name ghost_head to match the helper for consistency: // jscs:disable requireCamelCaseOrUpperCaseIdentifiers var hbs = require('express-hbs'), moment = require('moment'), _ = require('lodash'), Promise = require('bluebird'), fs = require('fs'), path = require('path'), config = require('../config'), filters = require('../filters'), api = require('../api'), urlHelper = require('./url'), meta_description = require('./meta_description'), meta_title = require('./meta_title'), excerpt = require('./excerpt'), tagsHelper = require('./tags'), imageHelper = require('./image'), labs = require('../utils/labs'), blog, ghost_head; function getClient() { if (labs.isSet('publicAPI') === true) { return api.clients.read({slug: 'ghost-frontend'}).then(function (client) { client = client.clients[0]; if (client.status === 'enabled') { return { id: client.slug, secret: client.secret }; } return {}; }); } return {}; } function writeMetaTag(property, content, type) { type = type || property.substring(0, 7) === 'twitter' ? 'name' : 'property'; return ''; } function getImage(props, context, contextObject) { if (context === 'home' || context === 'author') { contextObject.image = contextObject.cover; } props.image = imageHelper.call(contextObject, {hash: {absolute: true}}); if (context === 'post' && contextObject.author) { props.author_image = imageHelper.call(contextObject.author, {hash: {absolute: true}}); } } function getPaginationUrls(pagination, relativeUrl, secure, head) { var trimmedUrl, next, prev, trimmedUrlpattern = /.+(?=\/page\/\d*\/)/, tagOrAuthorPattern = /\/(tag)|(author)\//; trimmedUrl = relativeUrl.match(trimmedUrlpattern); if (pagination.prev) { prev = (pagination.prev > 1 ? '/page/' + pagination.prev + '/' : '/'); prev = (trimmedUrl) ? trimmedUrl + prev : prev; head.push('' ); } if (pagination.next) { next = '/page/' + pagination.next + '/'; if (trimmedUrl) { next = trimmedUrl + next; } else if (tagOrAuthorPattern.test(relativeUrl)) { next = relativeUrl.slice(0, -1) + next; } head.push('' ); } return head; } function addContextMetaData(context, data, metaData) { // escaped data metaData.metaTitle = hbs.handlebars.Utils.escapeExpression(metaData.metaTitle); metaData.metaDescription = metaData.metaDescription ? hbs.handlebars.Utils.escapeExpression(metaData.metaDescription) : null; if (context === 'author') { metaData.authorUrl = hbs.handlebars.Utils.escapeExpression(blog.url + '/author/' + data.author.slug); metaData.ogType = 'profile'; } else if (context === 'post') { metaData.publishedDate = moment(data.post.published_at).toISOString(); metaData.modifiedDate = moment(data.post.updated_at).toISOString(); metaData.authorUrl = hbs.handlebars.Utils.escapeExpression(blog.url + '/author/' + data.post.author.slug); metaData.ogType = 'article'; } return metaData; } function initMetaData(context, data, results) { var metaData = { url: results.url, metaDescription: results.meta_description || null, metaTitle: results.meta_title, coverImage: results.image, authorImage: results.author_image, publishedDate: null, modifiedDate: null, tags: null, card: 'summary', authorUrl: null, ogType: 'website', keywords: null, blog: blog, title: blog.title, clientId: results.client.id, clientSecret: results.client.secret }; if (!metaData.metaDescription) { if (context === 'post') { metaData.metaDescription = excerpt.call(data.post, {hash: {words: '40'}}).string + '...'; } else if (context === 'tag') { metaData.metaDescription = data.tag.description ? data.tag.description : null; } } return addContextMetaData(context, data, metaData); } function getStructuredData(metaData) { var structuredData; if (metaData.coverImage) { metaData.card = 'summary_large_image'; } structuredData = { 'og:site_name': metaData.title, 'og:type': metaData.ogType, 'og:title': metaData.metaTitle, 'og:description': metaData.metaDescription, 'og:url': metaData.url, 'og:image': metaData.coverImage, 'article:published_time': metaData.publishedDate, 'article:modified_time': metaData.modifiedDate, 'article:tag': metaData.tags, 'twitter:card': metaData.card, 'twitter:title': metaData.metaTitle, 'twitter:description': metaData.metaDescription, 'twitter:url': metaData.url, 'twitter:image:src': metaData.coverImage }; return structuredData; } // Creates the final schema object with values that are not null function trimSchema(schema) { var schemaObject = {}; _.each(schema, function (value, key) { if (value !== null && value !== undefined) { schemaObject[key] = value; } }); return schemaObject; } function getPostSchema(metaData, data) { var schema = { '@context': 'http://schema.org', '@type': 'Article', publisher: metaData.title, author: { '@type': 'Person', name: data.post.author.name, image: metaData.authorImage, url: metaData.authorUrl, sameAs: data.post.author.website || null, description: data.post.author.bio || null }, headline: metaData.metaTitle, url: metaData.url, datePublished: metaData.publishedDate, dateModified: metaData.modifiedDate, image: metaData.coverImage, keywords: metaData.keywords, description: metaData.metaDescription }; return trimSchema(schema); } function getTagSchema(metaData, data) { var schema = { '@context': 'http://schema.org', '@type': 'Series', publisher: metaData.title, url: metaData.url, image: metaData.coverImage, name: data.tag.name, description: metaData.metaDescription }; return trimSchema(schema); } function getAuthorSchema(metaData, data) { var schema = { '@context': 'http://schema.org', '@type': 'Person', sameAs: data.author.website || null, publisher: metaData.title, name: data.author.name, url: metaData.authorUrl, image: metaData.coverImage, description: metaData.metaDescription }; return trimSchema(schema); } function getHomeSchema(metaData) { var schema = { '@context': 'http://schema.org', '@type': 'Website', publisher: metaData.title, url: metaData.url, image: metaData.coverImage, description: metaData.metaDescription }; return trimSchema(schema); } function chooseSchema(metaData, context, data) { if (context === 'post') { return getPostSchema(metaData, data); } else if (context === 'home') { return getHomeSchema(metaData); } else if (context === 'tag') { return getTagSchema(metaData, data); } else if (context === 'author') { return getAuthorSchema(metaData, data); } } function finaliseStructuredData(structuredData, tags, head) { _.each(structuredData, function (content, property) { if (property === 'article:tag') { _.each(tags, function (tag) { if (tag !== '') { tag = hbs.handlebars.Utils.escapeExpression(tag.trim()); head.push(writeMetaTag(property, tag)); } }); head.push(''); } else if (content !== null && content !== undefined) { head.push(writeMetaTag(property, content)); } }); return head; } function finaliseSchema(schema, head) { head.push('\n' ); return head; } function getAjaxHelper() { var ghostUrlScript = fs.readFileSync(path.join(config.paths.corePath, 'shared', 'ghost-url.js'), 'utf8'), template = hbs.compile(ghostUrlScript, path.join(config.paths.subdir || '/', 'shared', 'ghost-url.js')), apiPath = require('../routes').apiBaseUri, url, useOrigin; if (config.forceAdminSSL) { url = 'https://' + (config.urlSSL || config.url).replace(/.*?:\/\//g, '').replace(/\/$/, '') + apiPath; useOrigin = false; } else { url = config.paths.subdir + apiPath; useOrigin = true; } return ''; } ghost_head = function (options) { // create a shortcut for theme config blog = config.theme; /*jshint unused:false*/ var self = this, useStructuredData = !config.isPrivacyDisabled('useStructuredData'), head = [], safeVersion = this.safeVersion, props = {}, structuredData, schema, title = hbs.handlebars.Utils.escapeExpression(blog.title), context = self.context ? self.context[0] : null, contextObject = self[context] || blog; // Store Async calls in an object of named promises props.url = urlHelper.call(self, {hash: {absolute: true}}); props.meta_description = meta_description.call(self, options); props.meta_title = meta_title.call(self, options); props.client = getClient(); getImage(props, context, contextObject); // Resolves promises then push pushes meta data into ghost_head return Promise.props(props).then(function (results) { if (context) { var metaData = initMetaData(context, self, results), tags = tagsHelper.call(self.post, {hash: {autolink: 'false'}}).string.split(','); // If there are tags - build the keywords metaData string if (tags[0] !== '') { metaData.keywords = hbs.handlebars.Utils.escapeExpression(tagsHelper.call(self.post, {hash: {autolink: 'false', separator: ', '}} ).string); } // head is our main array that holds our meta data head.push(''); head.push(''); // Generate context driven pagination urls if (self.pagination) { getPaginationUrls(self.pagination, self.relativeUrl, self.secure, head); } // Test to see if we are on a post page and that Structured data has not been disabled in config.js if (context !== 'paged' && context !== 'page' && useStructuredData) { // Create context driven OpenGraph and Twitter meta data structuredData = getStructuredData(metaData); // Create context driven JSONLD object schema = chooseSchema(metaData, context, self); head.push(''); // Formats structured data and pushes to head array finaliseStructuredData(structuredData, tags, head); head.push(''); // Formats schema script/JSONLD data and pushes to head array finaliseSchema(schema, head); } if (metaData.clientId && metaData.clientSecret) { head.push(writeMetaTag('ghost:client_id', metaData.clientId)); head.push(writeMetaTag('ghost:client_secret', metaData.clientSecret)); head.push(getAjaxHelper()); } } head.push(''); head.push(''); }).then(function () { return api.settings.read({key: 'ghost_head'}); }).then(function (response) { head.push(response.settings[0].value); return filters.doFilter('ghost_head', head); }).then(function (head) { var headString = _.reduce(head, function (memo, item) { return memo + '\n ' + item; }, ''); return new hbs.handlebars.SafeString(headString.trim()); }); }; module.exports = ghost_head;