2014-12-01 10:43:01 +00:00
|
|
|
var _ = require('lodash'),
|
|
|
|
xml = require('xml'),
|
|
|
|
moment = require('moment'),
|
|
|
|
api = require('../../api'),
|
|
|
|
config = require('../../config'),
|
|
|
|
utils = require('./utils'),
|
2014-10-27 19:41:18 -05:00
|
|
|
Promise = require('bluebird'),
|
|
|
|
CHANGE_FREQ = 'weekly',
|
|
|
|
XMLNS_DECLS;
|
|
|
|
|
|
|
|
// Sitemap specific xml namespace declarations that should not change
|
|
|
|
XMLNS_DECLS = {
|
|
|
|
_attr: {
|
|
|
|
xmlns: 'http://www.sitemaps.org/schemas/sitemap/0.9',
|
|
|
|
'xmlns:image': 'http://www.google.com/schemas/sitemap-image/1.1'
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
function BaseSiteMapGenerator() {
|
|
|
|
this.lastModified = 0;
|
|
|
|
this.nodeLookup = {};
|
2014-12-09 12:41:31 -08:00
|
|
|
this.nodeTimeLookup = {};
|
2014-10-27 19:41:18 -05:00
|
|
|
this.siteMapContent = '';
|
|
|
|
}
|
|
|
|
|
|
|
|
_.extend(BaseSiteMapGenerator.prototype, {
|
|
|
|
init: function () {
|
|
|
|
return this.refreshAll();
|
|
|
|
},
|
|
|
|
|
|
|
|
getData: function () {
|
|
|
|
return Promise.resolve([]);
|
|
|
|
},
|
|
|
|
|
|
|
|
refreshAll: function () {
|
|
|
|
var self = this;
|
|
|
|
|
|
|
|
// Load all data
|
|
|
|
return this.getData().then(function (data) {
|
|
|
|
// Generate SiteMap from data
|
|
|
|
return self.generateXmlFromData(data);
|
|
|
|
}).then(function (generatedXml) {
|
|
|
|
self.siteMapContent = generatedXml;
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
|
|
|
generateXmlFromData: function (data) {
|
|
|
|
// This has to be async because of the permalinks retrieval
|
|
|
|
var self = this;
|
|
|
|
|
|
|
|
// Fetch the permalinks value only once for all the urlFor calls
|
|
|
|
return this.getPermalinksValue().then(function (permalinks) {
|
|
|
|
// Create all the url elements in JSON
|
|
|
|
return _.map(data, function (datum) {
|
|
|
|
var node = self.createUrlNodeFromDatum(datum, permalinks);
|
|
|
|
self.updateLastModified(datum);
|
2014-12-09 12:41:31 -08:00
|
|
|
self.updateLookups(datum, node);
|
2014-10-27 19:41:18 -05:00
|
|
|
|
|
|
|
return node;
|
|
|
|
});
|
2014-12-09 12:41:31 -08:00
|
|
|
}).then(this.generateXmlFromNodes.bind(this));
|
2014-10-27 19:41:18 -05:00
|
|
|
},
|
|
|
|
|
|
|
|
getPermalinksValue: function () {
|
|
|
|
var self = this;
|
|
|
|
|
|
|
|
if (this.permalinks) {
|
|
|
|
return Promise.resolve(this.permalinks);
|
|
|
|
}
|
|
|
|
|
|
|
|
return api.settings.read('permalinks').then(function (response) {
|
|
|
|
self.permalinks = response.settings[0];
|
|
|
|
return self.permalinks;
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
|
|
|
updatePermalinksValue: function (permalinks) {
|
|
|
|
this.permalinks = permalinks;
|
|
|
|
|
|
|
|
// Re-generate xml with new permalinks values
|
2014-12-09 12:41:31 -08:00
|
|
|
this.updateXmlFromNodes();
|
|
|
|
},
|
|
|
|
|
|
|
|
generateXmlFromNodes: function () {
|
|
|
|
var self = this,
|
|
|
|
// Get a mapping of node to timestamp
|
|
|
|
timedNodes = _.map(this.nodeLookup, function (node, id) {
|
|
|
|
return {
|
|
|
|
id: id,
|
|
|
|
// Using negative here to sort newest to oldest
|
|
|
|
ts: -(self.nodeTimeLookup[id] || 0),
|
|
|
|
node: node
|
|
|
|
};
|
|
|
|
}, []),
|
|
|
|
// Sort nodes by timestamp
|
|
|
|
sortedNodes = _.sortBy(timedNodes, 'ts'),
|
|
|
|
// Grab just the nodes
|
|
|
|
urlElements = _.pluck(sortedNodes, 'node'),
|
|
|
|
data = {
|
|
|
|
// Concat the elements to the _attr declaration
|
|
|
|
urlset: [XMLNS_DECLS].concat(urlElements)
|
|
|
|
};
|
2014-10-27 19:41:18 -05:00
|
|
|
|
|
|
|
// Return the xml
|
2014-12-01 10:43:01 +00:00
|
|
|
return utils.getDeclarations() + xml(data);
|
2014-10-27 19:41:18 -05:00
|
|
|
},
|
|
|
|
|
|
|
|
updateXmlFromNodes: function (urlElements) {
|
|
|
|
var content = this.generateXmlFromNodes(urlElements);
|
|
|
|
|
|
|
|
this.setSiteMapContent(content);
|
|
|
|
|
|
|
|
return content;
|
|
|
|
},
|
|
|
|
|
|
|
|
addUrl: function (datum) {
|
|
|
|
var self = this;
|
|
|
|
return this.getPermalinksValue().then(function (permalinks) {
|
|
|
|
var node = self.createUrlNodeFromDatum(datum, permalinks);
|
|
|
|
self.updateLastModified(datum);
|
2014-12-09 12:41:31 -08:00
|
|
|
self.updateLookups(datum, node);
|
2014-10-27 19:41:18 -05:00
|
|
|
|
2014-12-09 12:41:31 -08:00
|
|
|
return self.updateXmlFromNodes();
|
2014-10-27 19:41:18 -05:00
|
|
|
});
|
|
|
|
},
|
|
|
|
|
|
|
|
removeUrl: function (datum) {
|
2014-12-09 12:41:31 -08:00
|
|
|
this.removeFromLookups(datum);
|
2014-10-27 19:41:18 -05:00
|
|
|
|
|
|
|
this.lastModified = Date.now();
|
|
|
|
|
2014-12-09 12:41:31 -08:00
|
|
|
return this.updateXmlFromNodes();
|
2014-10-27 19:41:18 -05:00
|
|
|
},
|
|
|
|
|
|
|
|
updateUrl: function (datum) {
|
|
|
|
var self = this;
|
|
|
|
return this.getPermalinksValue().then(function (permalinks) {
|
|
|
|
var node = self.createUrlNodeFromDatum(datum, permalinks);
|
|
|
|
self.updateLastModified(datum);
|
|
|
|
// TODO: Check if the node values changed, and if not don't regenerate
|
2014-12-09 12:41:31 -08:00
|
|
|
self.updateLookups(datum, node);
|
2014-10-27 19:41:18 -05:00
|
|
|
|
2014-12-09 12:41:31 -08:00
|
|
|
return self.updateXmlFromNodes();
|
2014-10-27 19:41:18 -05:00
|
|
|
});
|
|
|
|
},
|
|
|
|
|
|
|
|
getUrlForDatum: function () {
|
|
|
|
return config.urlFor('home', true);
|
|
|
|
},
|
|
|
|
|
|
|
|
getUrlForImage: function (image) {
|
|
|
|
return config.urlFor('image', {image: image}, true);
|
|
|
|
},
|
|
|
|
|
|
|
|
getPriorityForDatum: function () {
|
|
|
|
return 1.0;
|
|
|
|
},
|
|
|
|
|
2014-12-09 12:41:31 -08:00
|
|
|
getLastModifiedForDatum: function (datum) {
|
|
|
|
return datum.updated_at || datum.published_at || datum.created_at;
|
|
|
|
},
|
|
|
|
|
2014-10-27 19:41:18 -05:00
|
|
|
createUrlNodeFromDatum: function (datum, permalinks) {
|
|
|
|
var url = this.getUrlForDatum(datum, permalinks),
|
|
|
|
priority = this.getPriorityForDatum(datum);
|
|
|
|
|
|
|
|
return {
|
|
|
|
url: [
|
|
|
|
{loc: url},
|
2014-12-09 12:41:31 -08:00
|
|
|
{lastmod: moment(this.getLastModifiedForDatum(datum)).toISOString()},
|
2014-10-27 19:41:18 -05:00
|
|
|
{changefreq: CHANGE_FREQ},
|
|
|
|
{priority: priority}
|
|
|
|
]
|
|
|
|
};
|
|
|
|
},
|
|
|
|
|
|
|
|
setSiteMapContent: function (content) {
|
|
|
|
this.siteMapContent = content;
|
|
|
|
},
|
|
|
|
|
|
|
|
updateLastModified: function (datum) {
|
2014-12-09 12:41:31 -08:00
|
|
|
var lastModified = this.getLastModifiedForDatum(datum);
|
2014-10-27 19:41:18 -05:00
|
|
|
|
|
|
|
if (lastModified > this.lastModified) {
|
|
|
|
this.lastModified = lastModified;
|
|
|
|
}
|
2014-12-09 12:41:31 -08:00
|
|
|
},
|
|
|
|
|
|
|
|
updateLookups: function (datum, node) {
|
|
|
|
this.nodeLookup[datum.id] = node;
|
|
|
|
this.nodeTimeLookup[datum.id] = this.getLastModifiedForDatum(datum);
|
|
|
|
},
|
|
|
|
|
|
|
|
removeFromLookups: function (datum) {
|
|
|
|
var lookup = this.nodeLookup;
|
|
|
|
delete lookup[datum.id];
|
|
|
|
|
|
|
|
lookup = this.nodeTimeLookup;
|
|
|
|
delete lookup[datum.id];
|
2014-10-27 19:41:18 -05:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
module.exports = BaseSiteMapGenerator;
|