From 85496f409afdbac51af5c4e01088eb0dd3d5dc4a Mon Sep 17 00:00:00 2001 From: Kevin Ansfield Date: Wed, 31 May 2017 15:46:29 +0100 Subject: [PATCH] =?UTF-8?q?=F0=9F=94=A5=20remove=20posts.markdown=20field?= =?UTF-8?q?=20(#8497)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit closes #8479 - removes `markdown` field from schema - removes `legacyMarkdown` converter - updates tests to work with `mobiledoc` field instead of `markdown` and adapt for mobiledoc HTML output where necessary --- core/server/data/schema/schema.js | 1 - core/server/models/post.js | 6 +- core/test/functional/routes/api/posts_spec.js | 11 +-- .../integration/model/model_posts_spec.js | 76 ++++++------------- .../scheduling/post-scheduling/index_spec.js | 2 +- .../unit/controllers/frontend/index_spec.js | 13 ++-- core/test/unit/metadata/amp_url_spec.js | 5 +- core/test/unit/metadata/canonical_url_spec.js | 7 +- core/test/unit/metadata/schema_spec.js | 5 +- core/test/unit/metadata/url_spec.js | 7 +- core/test/unit/migration_spec.js | 2 +- core/test/unit/rss_spec.js | 4 +- .../unit/server_helpers/next_post_spec.js | 7 +- .../unit/server_helpers/prev_post_spec.js | 7 +- core/test/unit/server_helpers/url_spec.js | 15 ++-- core/test/utils/api.js | 2 +- core/test/utils/fixtures/data-generator.js | 44 +++++++---- .../test/utils/fixtures/filter-param/index.js | 43 ++++++----- 18 files changed, 127 insertions(+), 130 deletions(-) diff --git a/core/server/data/schema/schema.js b/core/server/data/schema/schema.js index c453b267d4..6616c02786 100644 --- a/core/server/data/schema/schema.js +++ b/core/server/data/schema/schema.js @@ -4,7 +4,6 @@ module.exports = { uuid: {type: 'string', maxlength: 36, nullable: false, validations: {isUUID: true}}, title: {type: 'string', maxlength: 2000, nullable: false}, slug: {type: 'string', maxlength: 191, nullable: false, unique: true}, - markdown: {type: 'text', maxlength: 1000000000, fieldtype: 'long', nullable: true}, mobiledoc: {type: 'text', maxlength: 1000000000, fieldtype: 'long', nullable: true}, html: {type: 'text', maxlength: 1000000000, fieldtype: 'long', nullable: true}, amp: {type: 'text', maxlength: 1000000000, fieldtype: 'long', nullable: true}, diff --git a/core/server/models/post.js b/core/server/models/post.js index 29c00ae893..45765347d0 100644 --- a/core/server/models/post.js +++ b/core/server/models/post.js @@ -5,7 +5,6 @@ var _ = require('lodash'), Promise = require('bluebird'), sequence = require('../utils/sequence'), errors = require('../errors'), - legacyConverter = require('../utils/markdown-converter'), htmlToText = require('html-to-text'), ghostBookshelf = require('./base'), events = require('../events'), @@ -235,9 +234,6 @@ Post = ghostBookshelf.Model.extend({ if (mobiledoc) { this.set('html', utils.mobiledocConverter.render(JSON.parse(mobiledoc))); - } else { - // legacy markdown mode - this.set('html', legacyConverter.render(_.toString(this.get('markdown')))); } if (this.hasChanged('html') || !this.get('plaintext')) { @@ -524,7 +520,7 @@ Post = ghostBookshelf.Model.extend({ return this.isPublicContext() ? 'page:false' : 'page:false+status:published'; } }, { - allowedFormats: ['markdown', 'mobiledoc', 'html', 'plaintext', 'amp'], + allowedFormats: ['mobiledoc', 'html', 'plaintext', 'amp'], orderDefaultOptions: function orderDefaultOptions() { return { diff --git a/core/test/functional/routes/api/posts_spec.js b/core/test/functional/routes/api/posts_spec.js index 4a480e7218..c5eb41309f 100644 --- a/core/test/functional/routes/api/posts_spec.js +++ b/core/test/functional/routes/api/posts_spec.js @@ -5,6 +5,7 @@ var should = require('should'), ObjectId = require('bson-objectid'), config = require('../../../../../core/server/config'), ghost = testUtils.startGhost, + markdownToMobiledoc = testUtils.DataGenerator.markdownToMobiledoc, request; describe('Post API', function () { @@ -525,7 +526,7 @@ describe('Post API', function () { // ## Add describe('Add', function () { it('create and ensure dates are correct', function (done) { - var newPost = {posts: [{status: 'published', published_at: '2016-05-30T07:00:00.000Z'}]}; + var newPost = {posts: [{status: 'published', published_at: '2016-05-30T07:00:00.000Z', mobiledoc: markdownToMobiledoc()}]}; request.post(testUtils.API.getApiQuery('posts')) .set('Authorization', 'Bearer ' + accesstoken) @@ -576,7 +577,7 @@ describe('Post API', function () { newTagName = 'My Tag', publishedState = 'published', newTag = {id: null, name: newTagName}, - newPost = {posts: [{status: 'draft', title: newTitle, markdown: 'my post', tags: [newTag]}]}; + newPost = {posts: [{status: 'draft', title: newTitle, mobiledoc: markdownToMobiledoc('my post'), tags: [newTag]}]}; request.post(testUtils.API.getApiQuery('posts/?include=tags')) .set('Authorization', 'Bearer ' + accesstoken) @@ -711,7 +712,7 @@ describe('Post API', function () { var newTitle = 'My Post', newTagName = 'My Tag', newTag = {id: null, name: newTagName}, - newPost = {posts: [{status: 'draft', title: newTitle, markdown: 'my post', tags: [newTag]}]}; + newPost = {posts: [{status: 'draft', title: newTitle, mobiledoc: markdownToMobiledoc('my post'), tags: [newTag]}]}; request.post(testUtils.API.getApiQuery('posts/?include=tags')) .set('Authorization', 'Bearer ' + accesstoken) @@ -756,7 +757,7 @@ describe('Post API', function () { newTagName = 'My Tag', draftState = 'draft', newTag = {id: null, name: newTagName}, - newPost = {posts: [{status: 'published', title: newTitle, markdown: 'my post', tags: [newTag]}]}; + newPost = {posts: [{status: 'published', title: newTitle, mobiledoc: markdownToMobiledoc('my post'), tags: [newTag]}]}; request.post(testUtils.API.getApiQuery('posts/?include=tags')) .set('Authorization', 'Bearer ' + accesstoken) @@ -1092,7 +1093,7 @@ describe('Post API', function () { it('can delete a new draft', function (done) { var newTitle = 'My Post', publishedState = 'draft', - newPost = {posts: [{status: publishedState, title: newTitle, markdown: 'my post'}]}; + newPost = {posts: [{status: publishedState, title: newTitle, mobiledoc: markdownToMobiledoc('my post')}]}; request.post(testUtils.API.getApiQuery('posts/')) .set('Authorization', 'Bearer ' + accesstoken) diff --git a/core/test/integration/model/model_posts_spec.js b/core/test/integration/model/model_posts_spec.js index 8612f544c0..86e540d51c 100644 --- a/core/test/integration/model/model_posts_spec.js +++ b/core/test/integration/model/model_posts_spec.js @@ -14,7 +14,8 @@ var should = require('should'), configUtils = require('../../utils/configUtils'), DataGenerator = testUtils.DataGenerator, context = testUtils.context.owner, - sandbox = sinon.sandbox.create(); + sandbox = sinon.sandbox.create(), + markdownToMobiledoc = testUtils.DataGenerator.markdownToMobiledoc; /** * IMPORTANT: @@ -62,10 +63,9 @@ describe('Post Model', function () { firstPost.published_by.name.should.equal(DataGenerator.Content.users[0].name); firstPost.tags[0].name.should.equal(DataGenerator.Content.tags[0].name); - // @TODO change / update this for mobiledoc in if (options.formats) { - if (options.formats.indexOf('markdown') !== -1) { - firstPost.markdown.should.match(/HTML Ipsum Presents/); + if (options.formats.indexOf('mobiledoc') !== -1) { + firstPost.mobiledoc.should.match(/HTML Ipsum Presents/); } if (options.formats.indexOf('html') !== -1) { @@ -82,7 +82,7 @@ describe('Post Model', function () { } else { firstPost.html.should.match(/HTML Ipsum Presents/); should.equal(firstPost.plaintext, undefined); - should.equal(firstPost.markdown, undefined); + should.equal(firstPost.mobiledoc, undefined); should.equal(firstPost.amp, undefined); } } @@ -122,7 +122,7 @@ describe('Post Model', function () { it('can findAll, use formats option', function (done) { var options = { - formats: ['markdown', 'plaintext'], + formats: ['mobiledoc', 'plaintext'], include: ['author', 'fields', 'tags', 'created_by', 'updated_by', 'published_by'] }; @@ -182,7 +182,7 @@ describe('Post Model', function () { }); it('returns computed fields when columns are asked for explicitly', function (done) { - PostModel.findPage({columns: ['id', 'slug', 'url', 'markdown']}).then(function (results) { + PostModel.findPage({columns: ['id', 'slug', 'url', 'mobiledoc']}).then(function (results) { should.exist(results); var post = _.find(results.posts, {slug: testUtils.DataGenerator.Content.posts[0].slug}); @@ -457,21 +457,6 @@ describe('Post Model', function () { }).catch(done); }); - it('can change markdown to number', function (done) { - var postId = testUtils.DataGenerator.Content.posts[0].id; - - PostModel.findOne({id: postId}).then(function (results) { - should.exist(results); - var post = results.toJSON(); - post.title.should.not.equal('123'); - return PostModel.edit({markdown: 123}, _.extend({}, context, {id: postId})); - }).then(function (edited) { - should.exist(edited); - edited.attributes.markdown.should.equal('123'); - done(); - }).catch(done); - }); - it('converts html to plaintext', function (done) { var postId = testUtils.DataGenerator.Content.posts[0].id; @@ -966,7 +951,7 @@ describe('Post Model', function () { createdPost.has('uuid').should.equal(true); createdPost.get('status').should.equal('draft'); createdPost.get('title').should.equal(newPost.title, 'title is correct'); - createdPost.get('markdown').should.equal(newPost.markdown, 'markdown is correct'); + createdPost.get('mobiledoc').should.equal(newPost.mobiledoc, 'mobiledoc is correct'); createdPost.has('html').should.equal(true); createdPost.get('html').should.equal(newPostDB.html); createdPost.has('plaintext').should.equal(true); @@ -1023,17 +1008,6 @@ describe('Post Model', function () { }).catch(done); }); - it('can add, with markdown being a number', function (done) { - var newPost = testUtils.DataGenerator.forModel.posts[2]; - - newPost.markdown = 123; - - PostModel.add(newPost, context).then(function (createdPost) { - should.exist(createdPost); - done(); - }).catch(done); - }); - it('can add, with previous published_at date', function (done) { var previousPublishedAtDate = new Date(2013, 8, 21, 12); @@ -1041,7 +1015,7 @@ describe('Post Model', function () { status: 'published', published_at: previousPublishedAtDate, title: 'published_at test', - markdown: 'This is some content' + mobiledoc: markdownToMobiledoc('This is some content') }, context).then(function (newPost) { should.exist(newPost); new Date(newPost.get('published_at')).getTime().should.equal(previousPublishedAtDate.getTime()); @@ -1058,7 +1032,7 @@ describe('Post Model', function () { PostModel.add({ status: 'draft', title: 'draft 1', - markdown: 'This is some content' + mobiledoc: markdownToMobiledoc('This is some content') }, context).then(function (newPost) { should.exist(newPost); should.not.exist(newPost.get('published_at')); @@ -1075,7 +1049,7 @@ describe('Post Model', function () { status: 'draft', published_at: moment().toDate(), title: 'draft 1', - markdown: 'This is some content' + mobiledoc: markdownToMobiledoc('This is some content') }, context).then(function (newPost) { should.exist(newPost); should.exist(newPost.get('published_at')); @@ -1091,7 +1065,7 @@ describe('Post Model', function () { PostModel.add({ status: 'scheduled', title: 'scheduled 1', - markdown: 'This is some content' + mobiledoc: markdownToMobiledoc('This is some content') }, context).catch(function (err) { should.exist(err); (err instanceof errors.ValidationError).should.eql(true); @@ -1105,7 +1079,7 @@ describe('Post Model', function () { status: 'scheduled', published_at: moment().subtract(1, 'minute'), title: 'scheduled 1', - markdown: 'This is some content' + mobiledoc: markdownToMobiledoc('This is some content') }, context).catch(function (err) { should.exist(err); (err instanceof errors.ValidationError).should.eql(true); @@ -1119,7 +1093,7 @@ describe('Post Model', function () { status: 'scheduled', published_at: moment().add(1, 'minute'), title: 'scheduled 1', - markdown: 'This is some content' + mobiledoc: markdownToMobiledoc('This is some content') }, context).catch(function (err) { (err instanceof errors.ValidationError).should.eql(true); Object.keys(eventsTriggered).length.should.eql(0); @@ -1132,7 +1106,7 @@ describe('Post Model', function () { status: 'scheduled', published_at: moment().add(10, 'minute'), title: 'scheduled 1', - markdown: 'This is some content' + mobiledoc: markdownToMobiledoc('This is some content') }, context).then(function (post) { should.exist(post); @@ -1150,7 +1124,7 @@ describe('Post Model', function () { page: 1, published_at: moment().add(10, 'minute'), title: 'scheduled 1', - markdown: 'This is some content' + mobiledoc: markdownToMobiledoc('This is some content') }, context).then(function (post) { should.exist(post); @@ -1164,7 +1138,7 @@ describe('Post Model', function () { it('can add default title, if it\'s missing', function (done) { PostModel.add({ - markdown: 'Content' + mobiledoc: markdownToMobiledoc('Content') }, context).then(function (newPost) { should.exist(newPost); newPost.get('title').should.equal('(Untitled)'); @@ -1178,7 +1152,7 @@ describe('Post Model', function () { untrimmedUpdateTitle = ' test trimmed update title ', newPost = { title: untrimmedCreateTitle, - markdown: 'Test Content' + mobiledoc: markdownToMobiledoc('Test content') }; PostModel.add(newPost, context).then(function (createdPost) { @@ -1207,7 +1181,7 @@ describe('Post Model', function () { return function () { return PostModel.add({ title: 'Test Title', - markdown: 'Test Content ' + (i + 1) + mobiledoc: markdownToMobiledoc('Test Content ' + (i + 1)) }, context); }; })).then(function (createdPosts) { @@ -1225,7 +1199,7 @@ describe('Post Model', function () { } post.get('slug').should.equal('test-title-' + num); - post.get('markdown').should.equal('Test Content ' + num); + JSON.parse(post.get('mobiledoc')).cards[0][1].markdown.should.equal('Test Content ' + num); Object.keys(eventsTriggered).length.should.eql(1); should.exist(eventsTriggered['post.added']); @@ -1239,7 +1213,7 @@ describe('Post Model', function () { it('can generate slugs without duplicate hyphens', function (done) { var newPost = { title: 'apprehensive titles have too many spaces—and m-dashes — – and also n-dashes ', - markdown: 'Test Content 1' + mobiledoc: markdownToMobiledoc('Test Content 1') }; PostModel.add(newPost, context).then(function (createdPost) { @@ -1255,7 +1229,7 @@ describe('Post Model', function () { it('can generate a safe slug when a reserved keyword is used', function (done) { var newPost = { title: 'rss', - markdown: 'Test Content 1' + mobiledoc: markdownToMobiledoc('Test Content 1') }; PostModel.add(newPost, context).then(function (createdPost) { @@ -1271,7 +1245,7 @@ describe('Post Model', function () { it('can generate slugs without non-ascii characters', function (done) { var newPost = { title: 'भुते धडकी भरवणारा आहेत', - markdown: 'Test Content 1' + mobiledoc: markdownToMobiledoc('Test Content 1') }; PostModel.add(newPost, context).then(function (createdPost) { @@ -1283,11 +1257,11 @@ describe('Post Model', function () { it('detects duplicate slugs before saving', function (done) { var firstPost = { title: 'First post', - markdown: 'First content 1' + mobiledoc: markdownToMobiledoc('First content 1') }, secondPost = { title: 'Second post', - markdown: 'Second content 1' + mobiledoc: markdownToMobiledoc('Second content 1') }; // Create the first post diff --git a/core/test/unit/adapters/scheduling/post-scheduling/index_spec.js b/core/test/unit/adapters/scheduling/post-scheduling/index_spec.js index a342a1460a..e552a81eea 100644 --- a/core/test/unit/adapters/scheduling/post-scheduling/index_spec.js +++ b/core/test/unit/adapters/scheduling/post-scheduling/index_spec.js @@ -29,7 +29,7 @@ describe('Scheduling: Post Scheduling', function () { beforeEach(function () { scope.client = models.Client.forge(testUtils.DataGenerator.forKnex.createClient({slug: 'ghost-scheduler'})); - scope.post = models.Post.forge(testUtils.DataGenerator.forKnex.createPost({id: 1337, markdown: 'something'})); + scope.post = models.Post.forge(testUtils.DataGenerator.forKnex.createPost({id: 1337, mobiledoc: testUtils.DataGenerator.markdownToMobiledoc('something')})); scope.adapter = new SchedulingDefault(); diff --git a/core/test/unit/controllers/frontend/index_spec.js b/core/test/unit/controllers/frontend/index_spec.js index e6a41643a6..e3fa093266 100644 --- a/core/test/unit/controllers/frontend/index_spec.js +++ b/core/test/unit/controllers/frontend/index_spec.js @@ -8,6 +8,7 @@ var should = require('should'), configUtils = require('../../../utils/configUtils'), themes = require('../../../../server/themes'), settingsCache = require('../../../../server/settings/cache'), + markdownToMobiledoc = require('../../../utils/fixtures/data-generator').markdownToMobiledoc, sandbox = sinon.sandbox.create(); describe('Frontend Controller', function () { @@ -66,7 +67,7 @@ describe('Frontend Controller', function () { id: 1, title: 'Test static page', slug: 'test-static-page', - markdown: 'Test static page content', + mobiledoc: markdownToMobiledoc('Test static page content'), page: 1, published_at: new Date('2013/12/30').getTime(), author: { @@ -83,7 +84,7 @@ describe('Frontend Controller', function () { id: 2, title: 'Test normal post', slug: 'test-normal-post', - markdown: 'The test normal post content', + mobiledoc: markdownToMobiledoc('The test normal post content'), page: 0, published_at: new Date('2014/1/2').getTime(), author: { @@ -99,7 +100,7 @@ describe('Frontend Controller', function () { id: 3, title: 'About', slug: 'about', - markdown: 'This is the about page content', + mobiledoc: markdownToMobiledoc('This is the about page content'), page: 1, published_at: new Date('2014/1/30').getTime(), author: { @@ -689,7 +690,7 @@ describe('Frontend Controller', function () { id: 1, title: 'Test static page', slug: 'test-static-page', - markdown: 'Test static page content', + mobiledoc: markdownToMobiledoc('Test static page content'), page: 1, author: { id: 1, @@ -706,7 +707,7 @@ describe('Frontend Controller', function () { id: 2, title: 'Test normal post', slug: 'test-normal-post', - markdown: 'The test normal post content', + mobiledoc: markdownToMobiledoc('The test normal post content'), page: 0, author: { id: 1, @@ -722,7 +723,7 @@ describe('Frontend Controller', function () { id: 3, title: 'Getting started', slug: 'about', - markdown: 'This is a blog post', + mobiledoc: markdownToMobiledoc('This is a blog post'), page: 0, published_at: new Date('2014/1/30').getTime(), author: { diff --git a/core/test/unit/metadata/amp_url_spec.js b/core/test/unit/metadata/amp_url_spec.js index 055098ed93..cc016de8e3 100644 --- a/core/test/unit/metadata/amp_url_spec.js +++ b/core/test/unit/metadata/amp_url_spec.js @@ -1,12 +1,13 @@ var should = require('should'), - getAmpUrl = require('../../../server/data/meta/amp_url'); + getAmpUrl = require('../../../server/data/meta/amp_url'), + markdownToMobiledoc = require('../../utils/fixtures/data-generator').markdownToMobiledoc; describe('getAmpUrl', function () { it('should return amp url for post only', function () { var ampUrl = getAmpUrl({ url: '/this-is-a-test-post/', html: '

Test 123

', - markdown: '# Test 123', + markdown: markdownToMobiledoc('# Test 123'), title: 'This is a test post', slug: 'this-is-a-test-post', secure: true, diff --git a/core/test/unit/metadata/canonical_url_spec.js b/core/test/unit/metadata/canonical_url_spec.js index f55a91e325..aa0edf7565 100644 --- a/core/test/unit/metadata/canonical_url_spec.js +++ b/core/test/unit/metadata/canonical_url_spec.js @@ -1,12 +1,13 @@ var should = require('should'), // jshint ignore:line - getCanonicalUrl = require('../../../server/data/meta/canonical_url'); + getCanonicalUrl = require('../../../server/data/meta/canonical_url'), + markdownToMobiledoc = require('../../utils/fixtures/data-generator').markdownToMobiledoc; describe('getCanonicalUrl', function () { it('should return absolute canonical url for post', function () { var canonicalUrl = getCanonicalUrl({ url: '/this-is-a-test-post/', html: '

Test 123

', - markdown: '# Test 123', + mobiledoc: markdownToMobiledoc('# Test 123'), title: 'This is a test post', slug: 'this-is-a-test-post', secure: true @@ -20,7 +21,7 @@ describe('getCanonicalUrl', function () { var canonicalUrl = getCanonicalUrl({ url: '/this-is-a-test-post/amp/', html: '

Test 123

', - markdown: '# Test 123', + mobiledoc: markdownToMobiledoc('# Test 123'), title: 'This is a test post', slug: 'this-is-a-test-post', secure: true diff --git a/core/test/unit/metadata/schema_spec.js b/core/test/unit/metadata/schema_spec.js index 57aaff837f..64c4d9d56e 100644 --- a/core/test/unit/metadata/schema_spec.js +++ b/core/test/unit/metadata/schema_spec.js @@ -1,5 +1,6 @@ var should = require('should'), - getSchema = require('../../../server/data/meta/schema'); + getSchema = require('../../../server/data/meta/schema'), + markdownToMobiledoc = require('../../utils/fixtures/data-generator').markdownToMobiledoc; describe('getSchema', function () { it('should return post schema if context starts with post', function (done) { @@ -142,7 +143,7 @@ describe('getSchema', function () { post: { title: 'Post Title', slug: 'my-amp-post-slug', - markdown: 'some markdown', + mobiledoc: markdownToMobiledoc('some markdown'), html: 'some html', author: { name: 'Post Author', diff --git a/core/test/unit/metadata/url_spec.js b/core/test/unit/metadata/url_spec.js index 6205f1302d..d72375abad 100644 --- a/core/test/unit/metadata/url_spec.js +++ b/core/test/unit/metadata/url_spec.js @@ -1,11 +1,12 @@ var should = require('should'), // jshint ignore:line - getUrl = require('../../../server/data/meta/url'); + getUrl = require('../../../server/data/meta/url'), + markdownToMobiledoc = require('../../utils/fixtures/data-generator').markdownToMobiledoc; describe('getUrl', function () { it('should return url for a post', function () { var url = getUrl({ html: '

Welcome to my post.

', - markdown: 'Welcome to my post.', + mobiledoc: markdownToMobiledoc('Welcome to my post.'), title: 'Welcome Post', slug: 'welcome-post', url: '/post/welcome-post/' @@ -16,7 +17,7 @@ describe('getUrl', function () { it('should return absolute url for a post', function () { var url = getUrl({ html: '

Welcome to my post.

', - markdown: 'Welcome to my post.', + mobiledoc: markdownToMobiledoc('Welcome to my post.'), title: 'Welcome Post', slug: 'welcome-post', url: '/post/welcome-post/' diff --git a/core/test/unit/migration_spec.js b/core/test/unit/migration_spec.js index e2fdc3c614..a1005184bf 100644 --- a/core/test/unit/migration_spec.js +++ b/core/test/unit/migration_spec.js @@ -19,7 +19,7 @@ var should = require('should'), // jshint ignore:line // both of which are required for migrations to work properly. describe('DB version integrity', function () { // Only these variables should need updating - var currentSchemaHash = '0d3a45d3db7f7ae6effb654621ca8ab5', + var currentSchemaHash = 'd10bff0a4dbccbd5e5d66c9ed596301b', currentFixturesHash = 'ec48af84f206fa377214ed4311ba6dfd'; // If this test is failing, then it is likely a change has been made that requires a DB version bump, diff --git a/core/test/unit/rss_spec.js b/core/test/unit/rss_spec.js index 0e55de12e9..f4bef13d82 100644 --- a/core/test/unit/rss_spec.js +++ b/core/test/unit/rss_spec.js @@ -167,7 +167,7 @@ describe('RSS', function () { // item tags xmlData.should.match(/<!\[CDATA\[Short and Sweet\]\]>/); xmlData.should.match(/<description><!\[CDATA\[test stuff/); - xmlData.should.match(/<content:encoded><!\[CDATA\[<h2 id="testing">testing<\/h2>\n/); + xmlData.should.match(/<content:encoded><!\[CDATA\[<div class="kg-card-markdown"><h2 id="testing">testing<\/h2>\n/); xmlData.should.match(/<img src="http:\/\/placekitten.com\/500\/200"/); xmlData.should.match(/<media:content url="http:\/\/placekitten.com\/500\/200" medium="image"\/>/); xmlData.should.match(/<category><!\[CDATA\[public\]\]/); @@ -197,7 +197,7 @@ describe('RSS', function () { // special/optional tags xmlData.should.match(/<title><!\[CDATA\[Short and Sweet\]\]>/); xmlData.should.match(/<description><!\[CDATA\[test stuff/); - xmlData.should.match(/<content:encoded><!\[CDATA\[<h2 id="testing">testing<\/h2>\n/); + xmlData.should.match(/<content:encoded><!\[CDATA\[<div class="kg-card-markdown"><h2 id="testing">testing<\/h2>\n/); xmlData.should.match(/<img src="http:\/\/placekitten.com\/500\/200"/); xmlData.should.match(/<media:content url="http:\/\/placekitten.com\/500\/200" medium="image"\/>/); diff --git a/core/test/unit/server_helpers/next_post_spec.js b/core/test/unit/server_helpers/next_post_spec.js index 9b3b14e89c..bf28f7b86c 100644 --- a/core/test/unit/server_helpers/next_post_spec.js +++ b/core/test/unit/server_helpers/next_post_spec.js @@ -1,6 +1,7 @@ var should = require('should'), // jshint ignore:line sinon = require('sinon'), Promise = require('bluebird'), + markdownToMobiledoc = require('../../utils/fixtures/data-generator').markdownToMobiledoc, // Stuff we are testing helpers = require('../../../server/helpers'), api = require('../../../server/api'), @@ -33,7 +34,7 @@ describe('{{next_post}} helper', function () { helpers.prev_post.call({ html: 'content', status: 'published', - markdown: 'ff', + mobiledoc: markdownToMobiledoc('ff'), title: 'post2', slug: 'current', created_at: new Date(0), @@ -68,7 +69,7 @@ describe('{{next_post}} helper', function () { helpers.prev_post.call({ html: 'content', - markdown: 'ff', + mobiledoc: markdownToMobiledoc('ff'), title: 'post2', slug: 'current', created_at: new Date(0), @@ -127,7 +128,7 @@ describe('{{next_post}} helper', function () { helpers.prev_post.call({ html: 'content', status: 'published', - markdown: 'ff', + mobiledoc: markdownToMobiledoc('ff'), title: 'post2', slug: 'current', created_at: new Date(0), diff --git a/core/test/unit/server_helpers/prev_post_spec.js b/core/test/unit/server_helpers/prev_post_spec.js index 32065accf1..b8ab7bffc1 100644 --- a/core/test/unit/server_helpers/prev_post_spec.js +++ b/core/test/unit/server_helpers/prev_post_spec.js @@ -1,6 +1,7 @@ var should = require('should'), // jshint ignore:line sinon = require('sinon'), Promise = require('bluebird'), + markdownToMobiledoc = require('../../utils/fixtures/data-generator').markdownToMobiledoc, // Stuff we are testing helpers = require('../../../server/helpers'), @@ -34,7 +35,7 @@ describe('{{prev_post}} helper', function () { helpers.prev_post.call({ html: 'content', status: 'published', - markdown: 'ff', + mobiledoc: markdownToMobiledoc('ff'), title: 'post2', slug: 'current', created_at: new Date(0), @@ -71,7 +72,7 @@ describe('{{prev_post}} helper', function () { helpers.prev_post.call({ html: 'content', status: 'published', - markdown: 'ff', + mobiledoc: markdownToMobiledoc('ff'), title: 'post2', slug: 'current', created_at: new Date(0), @@ -134,7 +135,7 @@ describe('{{prev_post}} helper', function () { helpers.prev_post.call({ html: 'content', status: 'draft', - markdown: 'ff', + mobiledoc: markdownToMobiledoc('ff'), title: 'post2', slug: 'current', created_at: new Date(0), diff --git a/core/test/unit/server_helpers/url_spec.js b/core/test/unit/server_helpers/url_spec.js index afd1b3043c..13697035e4 100644 --- a/core/test/unit/server_helpers/url_spec.js +++ b/core/test/unit/server_helpers/url_spec.js @@ -2,6 +2,7 @@ var should = require('should'), // jshint ignore:line sinon = require('sinon'), Promise = require('bluebird'), configUtils = require('../../utils/configUtils'), + markdownToMobiledoc = require('../../utils/fixtures/data-generator').markdownToMobiledoc, // Stuff we are testing helpers = require('../../../server/helpers'), @@ -34,7 +35,7 @@ describe('{{url}} helper', function () { it('should return the slug with a prefix slash if the context is a post', function () { rendered = helpers.url.call({ html: 'content', - markdown: 'ff', + mobiledoc: markdownToMobiledoc('ff'), title: 'title', slug: 'slug', created_at: new Date(0), @@ -47,7 +48,7 @@ describe('{{url}} helper', function () { it('should output an absolute URL if the option is present', function () { rendered = helpers.url.call( - {html: 'content', markdown: 'ff', title: 'title', slug: 'slug', url: '/slug/', created_at: new Date(0)}, + {html: 'content', mobiledoc: markdownToMobiledoc('ff'), title: 'title', slug: 'slug', url: '/slug/', created_at: new Date(0)}, {hash: {absolute: 'true'}} ); @@ -58,7 +59,7 @@ describe('{{url}} helper', function () { it('should output an absolute URL with https if the option is present and secure', function () { rendered = helpers.url.call( { - html: 'content', markdown: 'ff', title: 'title', slug: 'slug', + html: 'content', mobiledoc: markdownToMobiledoc('ff'), title: 'title', slug: 'slug', url: '/slug/', created_at: new Date(0), secure: true }, {hash: {absolute: 'true'}} @@ -71,7 +72,7 @@ describe('{{url}} helper', function () { it('should output an absolute URL with https if secure', function () { rendered = helpers.url.call( { - html: 'content', markdown: 'ff', title: 'title', slug: 'slug', + html: 'content', mobiledoc: markdownToMobiledoc('ff'), title: 'title', slug: 'slug', url: '/slug/', created_at: new Date(0), secure: true }, {hash: {absolute: 'true'}} @@ -94,7 +95,7 @@ describe('{{url}} helper', function () { }); it('should return / if not a post or tag', function () { - rendered = helpers.url.call({markdown: 'ff', title: 'title', slug: 'slug'}); + rendered = helpers.url.call({mobiledoc: markdownToMobiledoc('ff'), title: 'title', slug: 'slug'}); should.exist(rendered); rendered.string.should.equal('/'); @@ -102,11 +103,11 @@ describe('{{url}} helper', function () { should.exist(rendered); rendered.string.should.equal('/'); - rendered = helpers.url.call({html: 'content', markdown: 'ff', slug: 'slug'}); + rendered = helpers.url.call({html: 'content', mobiledoc: markdownToMobiledoc('ff'), slug: 'slug'}); should.exist(rendered); rendered.string.should.equal('/'); - rendered = helpers.url.call({html: 'content', markdown: 'ff', title: 'title'}); + rendered = helpers.url.call({html: 'content', mobiledoc: markdownToMobiledoc('ff'), title: 'title'}); should.exist(rendered); rendered.string.should.equal('/'); }); diff --git a/core/test/utils/api.js b/core/test/utils/api.js index fad8ebd34c..c8837a5f4a 100644 --- a/core/test/utils/api.js +++ b/core/test/utils/api.js @@ -22,7 +22,7 @@ var _ = require('lodash'), // Post API post: _(schema.posts).keys() // does not return all formats by default - .without('markdown', 'mobiledoc', 'amp', 'plaintext') + .without('mobiledoc', 'amp', 'plaintext') // swaps author_id to author, and always returns a computed 'url' property .without('author_id').concat('author', 'url') .value(), diff --git a/core/test/utils/fixtures/data-generator.js b/core/test/utils/fixtures/data-generator.js index e5ad197ae8..e51d2ec107 100644 --- a/core/test/utils/fixtures/data-generator.js +++ b/core/test/utils/fixtures/data-generator.js @@ -5,6 +5,23 @@ var _ = require('lodash'), globalUtils = require('../../../server/utils'), DataGenerator = {}; +DataGenerator.markdownToMobiledoc = function markdownToMobiledoc(content) { + var mobiledoc = { + version: '0.3.1', + markups: [], + atoms: [], + cards: [ + ['card-markdown', { + cardName: 'card-markdown', + markdown: content || '' + }] + ], + sections: [[10, 0]] + }; + + return JSON.stringify(mobiledoc); +}; + /*jshint quotmark:false*/ // jscs:disable validateQuoteMarks, requireCamelCaseOrUpperCaseIdentifiers DataGenerator.Content = { @@ -13,22 +30,22 @@ DataGenerator.Content = { id: ObjectId.generate(), title: "HTML Ipsum", slug: "html-ipsum", - markdown: "<h1>HTML Ipsum Presents</h1><p><strong>Pellentesque habitant morbi tristique</strong> senectus et netus et malesuada fames ac turpis egestas. Vestibulum tortor quam, feugiat vitae, ultricies eget, tempor sit amet, ante. Donec eu libero sit amet quam egestas semper. <em>Aenean ultricies mi vitae est.</em> Mauris placerat eleifend leo. Quisque sit amet est et sapien ullamcorper pharetra. Vestibulum erat wisi, condimentum sed, <code>commodo vitae</code>, ornare sit amet, wisi. Aenean fermentum, elit eget tincidunt condimentum, eros ipsum rutrum orci, sagittis tempus lacus enim ac dui. <a href=\"#\">Donec non enim</a> in turpis pulvinar facilisis. Ut felis.</p><h2>Header Level 2</h2><ol><li>Lorem ipsum dolor sit amet, consectetuer adipiscing elit.</li><li>Aliquam tincidunt mauris eu risus.</li></ol><blockquote><p>Lorem ipsum dolor sit amet, consectetur adipiscing elit. Vivamus magna. Cras in mi at felis aliquet congue. Ut a est eget ligula molestie gravida. Curabitur massa. Donec eleifend, libero at sagittis mollis, tellus est malesuada tellus, at luctus turpis elit sit amet quam. Vivamus pretium ornare est.</p></blockquote><h3>Header Level 3</h3><ul><li>Lorem ipsum dolor sit amet, consectetuer adipiscing elit.</li><li>Aliquam tincidunt mauris eu risus.</li></ul><pre><code>#header h1 a{display: block;width: 300px;height: 80px;}</code></pre>", + mobiledoc: DataGenerator.markdownToMobiledoc("<h1>HTML Ipsum Presents</h1><p><strong>Pellentesque habitant morbi tristique</strong> senectus et netus et malesuada fames ac turpis egestas. Vestibulum tortor quam, feugiat vitae, ultricies eget, tempor sit amet, ante. Donec eu libero sit amet quam egestas semper. <em>Aenean ultricies mi vitae est.</em> Mauris placerat eleifend leo. Quisque sit amet est et sapien ullamcorper pharetra. Vestibulum erat wisi, condimentum sed, <code>commodo vitae</code>, ornare sit amet, wisi. Aenean fermentum, elit eget tincidunt condimentum, eros ipsum rutrum orci, sagittis tempus lacus enim ac dui. <a href=\\\"#\\\">Donec non enim</a> in turpis pulvinar facilisis. Ut felis.</p><h2>Header Level 2</h2><ol><li>Lorem ipsum dolor sit amet, consectetuer adipiscing elit.</li><li>Aliquam tincidunt mauris eu risus.</li></ol><blockquote><p>Lorem ipsum dolor sit amet, consectetur adipiscing elit. Vivamus magna. Cras in mi at felis aliquet congue. Ut a est eget ligula molestie gravida. Curabitur massa. Donec eleifend, libero at sagittis mollis, tellus est malesuada tellus, at luctus turpis elit sit amet quam. Vivamus pretium ornare est.</p></blockquote><h3>Header Level 3</h3><ul><li>Lorem ipsum dolor sit amet, consectetuer adipiscing elit.</li><li>Aliquam tincidunt mauris eu risus.</li></ul><pre><code>#header h1 a{display: block;width: 300px;height: 80px;}</code></pre>"), published_at: new Date("2015-01-01") }, { id: ObjectId.generate(), title: "Ghostly Kitchen Sink", slug: "ghostly-kitchen-sink", - markdown: "<h1>HTML Ipsum Presents</h1><p><strong>Pellentesque habitant morbi tristique</strong> senectus et netus et malesuada fames ac turpis egestas. Vestibulum tortor quam, feugiat vitae, ultricies eget, tempor sit amet, ante. Donec eu libero sit amet quam egestas semper. <em>Aenean ultricies mi vitae est.</em> Mauris placerat eleifend leo. Quisque sit amet est et sapien ullamcorper pharetra. Vestibulum erat wisi, condimentum sed, <code>commodo vitae</code>, ornare sit amet, wisi. Aenean fermentum, elit eget tincidunt condimentum, eros ipsum rutrum orci, sagittis tempus lacus enim ac dui. <a href=\"#\">Donec non enim</a> in turpis pulvinar facilisis. Ut felis.</p><h2>Header Level 2</h2><ol><li>Lorem ipsum dolor sit amet, consectetuer adipiscing elit.</li><li>Aliquam tincidunt mauris eu risus.</li></ol><blockquote><p>Lorem ipsum dolor sit amet, consectetur adipiscing elit. Vivamus magna. Cras in mi at felis aliquet congue. Ut a est eget ligula molestie gravida. Curabitur massa. Donec eleifend, libero at sagittis mollis, tellus est malesuada tellus, at luctus turpis elit sit amet quam. Vivamus pretium ornare est.</p></blockquote><h3>Header Level 3</h3><ul><li>Lorem ipsum dolor sit amet, consectetuer adipiscing elit.</li><li>Aliquam tincidunt mauris eu risus.</li></ul><pre><code>#header h1 a{display: block;width: 300px;height: 80px;}</code></pre>", + mobiledoc: DataGenerator.markdownToMobiledoc("<h1>HTML Ipsum Presents</h1><p><strong>Pellentesque habitant morbi tristique</strong> senectus et netus et malesuada fames ac turpis egestas. Vestibulum tortor quam, feugiat vitae, ultricies eget, tempor sit amet, ante. Donec eu libero sit amet quam egestas semper. <em>Aenean ultricies mi vitae est.</em> Mauris placerat eleifend leo. Quisque sit amet est et sapien ullamcorper pharetra. Vestibulum erat wisi, condimentum sed, <code>commodo vitae</code>, ornare sit amet, wisi. Aenean fermentum, elit eget tincidunt condimentum, eros ipsum rutrum orci, sagittis tempus lacus enim ac dui. <a href=\\\"#\\\">Donec non enim</a> in turpis pulvinar facilisis. Ut felis.</p><h2>Header Level 2</h2><ol><li>Lorem ipsum dolor sit amet, consectetuer adipiscing elit.</li><li>Aliquam tincidunt mauris eu risus.</li></ol><blockquote><p>Lorem ipsum dolor sit amet, consectetur adipiscing elit. Vivamus magna. Cras in mi at felis aliquet congue. Ut a est eget ligula molestie gravida. Curabitur massa. Donec eleifend, libero at sagittis mollis, tellus est malesuada tellus, at luctus turpis elit sit amet quam. Vivamus pretium ornare est.</p></blockquote><h3>Header Level 3</h3><ul><li>Lorem ipsum dolor sit amet, consectetuer adipiscing elit.</li><li>Aliquam tincidunt mauris eu risus.</li></ul><pre><code>#header h1 a{display: block;width: 300px;height: 80px;}</code></pre>"), published_at: new Date("2015-01-02") }, { id: ObjectId.generate(), title: "Short and Sweet", slug: "short-and-sweet", - markdown: "## testing\n\nmctesters\n\n- test\n- line\n- items", - html: "<h2 id=\"testing\">testing</h2>\n<p>mctesters</p>\n<ul>\n<li>test</li>\n<li>line</li>\n<li>items</li>\n</ul>\n", + mobiledoc: DataGenerator.markdownToMobiledoc("## testing\n\nmctesters\n\n- test\n- line\n- items"), + html: "<div class=\"kg-card-markdown\"><h2 id=\"testing\">testing</h2>\n<p>mctesters</p>\n<ul>\n<li>test</li>\n<li>line</li>\n<li>items</li>\n</ul>\n</div>", feature_image: "http://placekitten.com/500/200", meta_description: "test stuff", published_at: new Date("2015-01-03"), @@ -38,7 +55,7 @@ DataGenerator.Content = { id: ObjectId.generate(), title: "Not finished yet", slug: "unfinished", - markdown: "<h1>HTML Ipsum Presents</h1><p><strong>Pellentesque habitant morbi tristique</strong> senectus et netus et malesuada fames ac turpis egestas. Vestibulum tortor quam, feugiat vitae, ultricies eget, tempor sit amet, ante. Donec eu libero sit amet quam egestas semper. <em>Aenean ultricies mi vitae est.</em> Mauris placerat eleifend leo. Quisque sit amet est et sapien ullamcorper pharetra. Vestibulum erat wisi, condimentum sed, <code>commodo vitae</code>, ornare sit amet, wisi. Aenean fermentum, elit eget tincidunt condimentum, eros ipsum rutrum orci, sagittis tempus lacus enim ac dui. <a href=\"#\">Donec non enim</a> in turpis pulvinar facilisis. Ut felis.</p><h2>Header Level 2</h2><ol><li>Lorem ipsum dolor sit amet, consectetuer adipiscing elit.</li><li>Aliquam tincidunt mauris eu risus.</li></ol><blockquote><p>Lorem ipsum dolor sit amet, consectetur adipiscing elit. Vivamus magna. Cras in mi at felis aliquet congue. Ut a est eget ligula molestie gravida. Curabitur massa. Donec eleifend, libero at sagittis mollis, tellus est malesuada tellus, at luctus turpis elit sit amet quam. Vivamus pretium ornare est.</p></blockquote><h3>Header Level 3</h3><ul><li>Lorem ipsum dolor sit amet, consectetuer adipiscing elit.</li><li>Aliquam tincidunt mauris eu risus.</li></ul><pre><code>#header h1 a{display: block;width: 300px;height: 80px;}</code></pre>", + mobiledoc: DataGenerator.markdownToMobiledoc("<h1>HTML Ipsum Presents</h1><p><strong>Pellentesque habitant morbi tristique</strong> senectus et netus et malesuada fames ac turpis egestas. Vestibulum tortor quam, feugiat vitae, ultricies eget, tempor sit amet, ante. Donec eu libero sit amet quam egestas semper. <em>Aenean ultricies mi vitae est.</em> Mauris placerat eleifend leo. Quisque sit amet est et sapien ullamcorper pharetra. Vestibulum erat wisi, condimentum sed, <code>commodo vitae</code>, ornare sit amet, wisi. Aenean fermentum, elit eget tincidunt condimentum, eros ipsum rutrum orci, sagittis tempus lacus enim ac dui. <a href=\\\"#\\\">Donec non enim</a> in turpis pulvinar facilisis. Ut felis.</p><h2>Header Level 2</h2><ol><li>Lorem ipsum dolor sit amet, consectetuer adipiscing elit.</li><li>Aliquam tincidunt mauris eu risus.</li></ol><blockquote><p>Lorem ipsum dolor sit amet, consectetur adipiscing elit. Vivamus magna. Cras in mi at felis aliquet congue. Ut a est eget ligula molestie gravida. Curabitur massa. Donec eleifend, libero at sagittis mollis, tellus est malesuada tellus, at luctus turpis elit sit amet quam. Vivamus pretium ornare est.</p></blockquote><h3>Header Level 3</h3><ul><li>Lorem ipsum dolor sit amet, consectetuer adipiscing elit.</li><li>Aliquam tincidunt mauris eu risus.</li></ul><pre><code>#header h1 a{display: block;width: 300px;height: 80px;}</code></pre>"), status: "draft", uuid: "d52c42ae-2755-455c-80ec-70b2ec55c903", featured: false @@ -47,20 +64,20 @@ DataGenerator.Content = { id: ObjectId.generate(), title: "Not so short, bit complex", slug: "not-so-short-bit-complex", - markdown: "<p><nav><ul><li><a href=\"#nowhere\" title=\"Anchor URL\">Lorem</a></li><li><a href=\"/about#nowhere\" title=\"Relative URL\">Aliquam</a></li><li><a href=\"//somewhere.com/link#nowhere\" title=\"Protocol Relative URL\">Tortor</a></li><li><a href=\"http://somewhere.com/link#nowhere\" title=\"Absolute URL\">Morbi</a></li><li><a href=\"#nowhere\" title=\"Praesent dapibus, neque id cursus faucibus\">Praesent</a></li><li><a href=\"#nowhere\" title=\"Pellentesque fermentum dolor\">Pellentesque</a></li></ul></nav><p>Pellentesque habitant morbi tristique senectus et netus et malesuada fames ac turpis egestas. Vestibulum tortor quam, feugiat vitae, ultricies eget, tempor sit amet, ante. Donec eu libero sit amet quam egestas semper. Aenean ultricies mi vitae est. Mauris placerat eleifend leo.</p><table><thead><tr><th>1</th><th>2</th><th>3</th><th>4</th></tr></thead><tbody><tr><td>a</td><td>b</td><td>c</td><td>d</td></tr><tr><td>e</td><td>f</td><td>g</td><td>h</td></tr><tr><td>i</td><td>j</td><td>k</td><td>l</td></tr></tbody></table><dl><dt>Definition list</dt><dd>Consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat.</dd><dt>Lorem ipsum dolor sit amet</dt><dd>Consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat.</dd></dl><ul><li>Morbi in sem quis dui placerat ornare. Pellentesque odio nisi, euismod in, pharetra a, ultricies in, diam. Sed arcu. Cras consequat.</li><li>Praesent dapibus, neque id cursus faucibus, tortor neque egestas augue, eu vulputate magna eros eu erat. Aliquam erat volutpat. Nam dui mi, tincidunt quis, accumsan porttitor, facilisis luctus, metus.</li><li>Phasellus ultrices nulla quis nibh. Quisque a lectus. Donec consectetuer ligula vulputate sem tristique cursus. Nam nulla quam, gravida non, commodo a, sodales sit amet, nisi.</li><li>Pellentesque fermentum dolor. Aliquam quam lectus, facilisis auctor, ultrices ut, elementum vulputate, nunc.</li></ul></p>" + mobiledoc: DataGenerator.markdownToMobiledoc("<p><nav><ul><li><a href=\"#nowhere\" title=\"Anchor URL\">Lorem</a></li><li><a href=\"/about#nowhere\" title=\"Relative URL\">Aliquam</a></li><li><a href=\"//somewhere.com/link#nowhere\" title=\"Protocol Relative URL\">Tortor</a></li><li><a href=\"http://somewhere.com/link#nowhere\" title=\"Absolute URL\">Morbi</a></li><li><a href=\"#nowhere\" title=\"Praesent dapibus, neque id cursus faucibus\">Praesent</a></li><li><a href=\"#nowhere\" title=\"Pellentesque fermentum dolor\">Pellentesque</a></li></ul></nav><p>Pellentesque habitant morbi tristique senectus et netus et malesuada fames ac turpis egestas. Vestibulum tortor quam, feugiat vitae, ultricies eget, tempor sit amet, ante. Donec eu libero sit amet quam egestas semper. Aenean ultricies mi vitae est. Mauris placerat eleifend leo.</p><table><thead><tr><th>1</th><th>2</th><th>3</th><th>4</th></tr></thead><tbody><tr><td>a</td><td>b</td><td>c</td><td>d</td></tr><tr><td>e</td><td>f</td><td>g</td><td>h</td></tr><tr><td>i</td><td>j</td><td>k</td><td>l</td></tr></tbody></table><dl><dt>Definition list</dt><dd>Consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat.</dd><dt>Lorem ipsum dolor sit amet</dt><dd>Consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat.</dd></dl><ul><li>Morbi in sem quis dui placerat ornare. Pellentesque odio nisi, euismod in, pharetra a, ultricies in, diam. Sed arcu. Cras consequat.</li><li>Praesent dapibus, neque id cursus faucibus, tortor neque egestas augue, eu vulputate magna eros eu erat. Aliquam erat volutpat. Nam dui mi, tincidunt quis, accumsan porttitor, facilisis luctus, metus.</li><li>Phasellus ultrices nulla quis nibh. Quisque a lectus. Donec consectetuer ligula vulputate sem tristique cursus. Nam nulla quam, gravida non, commodo a, sodales sit amet, nisi.</li><li>Pellentesque fermentum dolor. Aliquam quam lectus, facilisis auctor, ultrices ut, elementum vulputate, nunc.</li></ul></p>") }, { id: ObjectId.generate(), title: "This is a static page", slug: "static-page-test", - markdown: "<h1>Static page test is what this is for.</h1><p>Hopefully you don't find it a bore.</p>", + mobiledoc: DataGenerator.markdownToMobiledoc("<h1>Static page test is what this is for.</h1><p>Hopefully you don't find it a bore.</p>"), page: 1 }, { id: ObjectId.generate(), title: "This is a draft static page", slug: "static-page-draft", - markdown: "<h1>Static page test is what this is for.</h1><p>Hopefully you don't find it a bore.</p>", + mobiledoc: DataGenerator.markdownToMobiledoc("<h1>Static page test is what this is for.</h1><p>Hopefully you don't find it a bore.</p>"), page: 1, status: "draft" }, @@ -68,7 +85,7 @@ DataGenerator.Content = { id: ObjectId.generate(), title: "This is a scheduled post!!", slug: "scheduled-post", - markdown: "<h1>Welcome to my invisible post!</h1>", + mobiledoc: DataGenerator.markdownToMobiledoc("<h1>Welcome to my invisible post!</h1>"), status: "scheduled", published_at: moment().add(2, 'days').toDate() } @@ -341,14 +358,15 @@ DataGenerator.forKnex = (function () { } function createPost(overrides) { - var newObj = _.cloneDeep(overrides); + var newObj = _.cloneDeep(overrides), + mobiledoc = JSON.parse(overrides.mobiledoc || '{}'); return _.defaults(newObj, { id: ObjectId.generate(), uuid: uuid.v4(), title: 'title', status: 'published', - html: overrides.markdown, + html: mobiledoc.cards && mobiledoc.cards[0][1].markdown, language: 'en_US', featured: true, page: false, @@ -372,7 +390,7 @@ DataGenerator.forKnex = (function () { title: 'Test Post ' + uniqueInteger, slug: 'ghost-from-fiction-to-function-' + uniqueInteger, author_id: author_id, - markdown: "Three days ago I released a <a title=\"Ghost\" href=\"http:\/\/john.onolan.org\/ghost\/\">concept page<\/a> for a lite version of WordPress that I've been thinking about for a long time, called Ghost. I think it's fair to say that I didn't quite anticipate how strong the reaction would be - and I've hardly had time to catch my breath in the last 72 hours.\n\nThe response was overwhelming, and overwhelmingly positive. In the first 6 hours my site got 35,000 page views after hitting the number 1 slot on <a href=\"http:\/\/news.ycombinator.com\/item?id=4743245\" onclick=\"javascript:_gaq.push(['_trackEvent','outbound-article','http:\/\/news.ycombinator.com']);\">Hacker News<\/a>. As of right now, the traffic count is just over <a href=\"http:\/\/john.onolan.org\/wp-content\/uploads\/2012\/11\/Screen-Shot-2012-11-09-at-17.51.21.png\" rel=\"lightbox\" class=\"cboxElement\">91,000 page views<\/a> - and Ghost has been featured all over the place. Notable mentions so far include Christina Warren from Mashable, who <a href=\"http:\/\/christina.is\/\" onclick=\"javascript:_gaq.push(['_trackEvent','outbound-article','http:\/\/christina.is']);\">wrote about it<\/a>. Michael Carney from PandoDaily <a href=\"http:\/\/pandodaily.com\/2012\/11\/07\/wordpress-guru-designs-a-concept-blogging-platform-that-doesnt-suck-gets-rave-reviews\/\" onclick=\"javascript:_gaq.push(['_trackEvent','outbound-article','http:\/\/pandodaily.com']);\">interviewed me about it<\/a>. Someone even <a href=\"http:\/\/www.voicens.com\/web\/?p=4425\" onclick=\"javascript:_gaq.push(['_trackEvent','outbound-article','http:\/\/www.voicens.com']);\">wrote about it in Chinese<\/a>. That's pretty cool.\n\n\nThe feedback has been amazing, and while it's impossible to reply to all of the messages individually, I'm getting to as many of them as I can and I want to thank each and every one of you who took the time to send me a message or share the concept because you liked it. Now that the initial storm has died down a bit, I wanted to take some time to answer some of the more common questions and talk about what's next.\n<h2>FAQ - Continued...<\/h2>\n\nThe most common question, bizarrely:\n<h5><em><strong>Oh my god, why is that whole page made of images? What's wrong with you? \/\/ I can't take you seriously \/\/ Don't you know anything about the web? \/\/ You are literally Satan re-incarnate.<\/strong><\/em><\/h5>\n\nThis was really the only negativity I got in response to the post, and it surprised me. I put together the concept page as... just that... a concept. It was a way for me to get the ideas out of my head and \"down on paper\" - or so to speak. I used photoshop as a <em>tool<\/em> to write down my idea with text and images. If I used a sketchbook as a <em>tool <\/em>to create images and handwritten notes, then uploaded scans of it, I doubt anyone would complain. The concept page was never supposed to be a finished product because I had no idea if there would be any interest in it. I had no motivation to waste hours coding a custom layout for something might only ever be read by a few people and then forgotten.\n\nHardware manufacturers make hundreds of foam cutout prototypes of products before they build one with working buttons and screens. I'm aware of all the usability problems with a web page made of images, and equally, foam cutouts without buttons or screens aren't particularly user friendly either. They're not supposed to be.\n\nLet's move on.\n<h5><em><strong>What? Why no comments? I need comments.<\/strong><\/em><\/h5>\n\nBecause comments add a layer of complexity that is beyond the core focus of this platform, which is publishing. Again, that's not to say you couldn't have any comments. This could easily be added with a dedicated plugin where you own the data or (as mentioned) there are third party providers such as Disqus, IntenseDebate, Livefyre and Facebook who all have great platforms. The point of this isn't to say \"you can't have comments\" - it's to say \"comments aren't on by default\". It's about simplicity, more than anything else.\n<h5><em><strong>Yeah, but WordPress are already going to revise their dashboard, WordPress.com is experimenting with a potential simplified version... so why bother with this?<\/strong><\/em><\/h5>\n\n<a href=\"http:\/\/john.onolan.org\/wp-content\/uploads\/2012\/11\/newwp.png\" rel=\"lightbox[2102]\" title=\"newwp\" class=\"cboxElement\"><img class=\"alignnone size-large wp-image-2117\" title=\"newwp\" src=\"http:\/\/john.onolan.org\/wp-content\/uploads\/2012\/11\/newwp-550x210.png\" alt=\"\" width=\"550\" height=\"210\"><\/a>\n\nSorry, but Tumblr already did this - it's not the future of blogging, it's the past.\n\nGhost isn't about sharing \"Fuck Yeah [<a href=\"http:\/\/fuckyeahdogs.tumblr.com\/\" onclick=\"javascript:_gaq.push(['_trackEvent','outbound-article','http:\/\/fuckyeahdogs.tumblr.com']);\">Dogs<\/a>\/<a href=\"http:\/\/fuckyeahsharks.tumblr.com\/\" onclick=\"javascript:_gaq.push(['_trackEvent','outbound-article','http:\/\/fuckyeahsharks.tumblr.com']);\" rel=\"lightbox\" class=\"cboxElement\">Sharks<\/a>\/<a href=\"http:\/\/fuckyeahgirlswithtattoos.tumblr.com\/\" onclick=\"javascript:_gaq.push(['_trackEvent','outbound-article','http:\/\/fuckyeahgirlswithtattoos.tumblr.com']);\">Girls with Tattoos<\/a>]\" - it's about publishing - which means writing - rather than mashing a few buttons to make sure that everyone can see and appreciate your latest funny picture\/status, which is surely the most funny picture\/status you've ever posted.\n\nTumblr, Pinterest and Facebook already have this locked down. It's not the future.\n<h5><em><strong>So... are you actually going to build this thing?<\/strong><\/em><\/h5>\n\nThe concept page was a way for me to test demand and interest. To see if anyone actually agreed with my frustrations and, more importantly, my solutions. I plucked a random figure of \"10,000 pageviews\" out of the air before I hit the publish button. If it got less than 10,000 pageviews, I would surrender to the fact that it would only ever be an idea. I've now exceeded that goal 9 times over, so yes, I'm looking at how Ghost can now be made into a reality.\n<h5><em><strong>How can I find out when it's done? \/\/ SHUT UP AND TAKE MY MONEY<\/strong><\/em><\/h5>\n\nOk, ok - there's a holding page up on <a href=\"http:\/\/tryghost.org\" onclick=\"javascript:_gaq.push(['_trackEvent','outbound-article','http:\/\/tryghost.org']);\">http:\/\/TryGhost.org<\/a> - put your email address in.\n<hr>\n<h3>How are you going to do this?<\/h3>\n\nThere's three main ways of going about this, each has merits as well as drawbacks.\n\n<em><strong>1.) Build it from scratch<\/strong><\/em> - Many people (particularly the Hacker News crowd) expressed the sentiment that there was little point in forking WordPress. When you're going to strip out so much, you get to a point where you might as well start from scratch anyway. Take away the crutches of being stuck with older technologies and put together something which is as sophisticated in code as it is in UI\/UX.\n<ul>\n<li><em>Pros:<\/em> The idea of something completely new is exciting, opportunity to build something very sophisticated, complete control over everything.<\/li>\n<li><em>Cons:<\/em> Lose the  WordPress ecosystem which includes millions of users and thousands of developers, potentially spend the next 6 months fighting over whether to use PHP\/RoR\/Django\/Python\/Node\/Whateverthefuck because everyone loves to evangelise the technology they know best.<\/li>\n<\/ul>\n\n<em><strong>2.) Fork WordPress<\/strong><\/em> - This was the original idea I put out. Take the WordPress codebase, as is, and modify it to turn it into something new. Initially the codebase is practically the same, which means developers already know it. Then it can change over time and evolve into its own thing.\n<ul>\n<li><em>Pros:<\/em> Easy start with existing codebase, potential to evolve, doesn't lose WordPress ecosystem initially.<\/li>\n<li><em>Cons:<\/em> Stuck with existing codebase - the good as well as the bad,  eventually needs to be rewritten completely, less control, loses the WordPress ecosystem after a while anyway, makes it complicated to transition from legacy code to new code.<\/li>\n<\/ul>\n\n<em><strong>3.) Make it a plugin\/extension<\/strong><\/em> - Lots of people asked why Ghost couldn't just be a WordPress plugin. It would certainly be the easiest route of the 3, it's possible to completely remove \/wp-admin\/ and replace with with \/ghost\/ ... but I feel like it kind of misses the point. This route bolts Ghost on, but it's still WordPress under the hood. From a UI\/UX standpoint it would function - but it wouldn't revolutionise anything else. It makes WordPress itself about blogging again, rather than creating something new.\n<ul>\n<li><em>Pros:<\/em> Very easy development, very easy deployment, keeps WordPress ecosystem forever, doesn't force anyone to change.<\/li>\n<li><em>Cons:<\/em> The least exciting (for me, personally), much less control, it would be much harder to maintain something like this on a non-profit basis - which loses a piece of what Ghost is about.<\/li>\n<\/ul>\n<h3>What's the answer?<\/h3>\n\nI've spoken to a lot of smart people over the last few days. The one thing that everyone seems to agree on is that a fork is the worst of both worlds. So the one thing that I suggested as a way of making this happen, is the least likely to work in reality. Remember the foam prototype metaphor earlier? Learning and iterating - that's what happening now.\n\nThat leaves a choice between WordPress plugin or fresh build. The answer? Both.\n\nA WordPress plugin will act as a proof of concept and a working prototype, initially, because it's easier to leverage the existing WordPress ecosystem to create it than to go into a cave for 6 months trying to build this amazing thing that everyone will have forgotten about.\n\nThe plugin will not be perfect. It will add the Ghost UI\/UX and as much functionality as we can cram into it. It will completely remove \/wp-admin\/ and replace it with \/ghost\/ - effectively using WordPress core as a basic foundation to build on top of. It will give people who don't want to switch away from WordPress access to the Ghost UX which they want to have, and it will give people who want the full Ghost platform a taste of what's to come.\n\nIt will allow us to develop and learn and iterate on the concept pretty rapidly, which has a great deal of value.\n\nThis is step one. Assuming the plugin is actually used by people - it would then justify exploring building the standalone version of Ghost from the ground up. The plugin would subsequently serve as a great marketing tool for the platform. Think of it as an upgrade path. But that's a long way away. Having the idea is the easy part. Making it happen is what counts.\n\nHappily - amongst the thousands of people talking about Ghost for the last few days - several have been talking about how they've already built some working prototypes of my mockups and turned them into WordPress plugins or just local development sites. These will likely go on to be the starting point of the first Ghost plugin.<\/p>\n\nThere's a lot to do, and I'm amazed by the number of people who have offered their help with this. In the next few days I'll be kicking off work on the plugin properly and start putting together a more organised structure which explains how you can get involved and contribute to the project if you're interested. So... watch this space - and thanks for all your support so far.\n\n<a href=\"http:\/\/twitter.com\/TryGhost\" onclick=\"javascript:_gaq.push(['_trackEvent','outbound-article','http:\/\/twitter.com']);\" class=\"twitter-follow-button\">Follow @TryGhost<\/a>", + mobiledoc: DataGenerator.markdownToMobiledoc("Three days ago I released a <a title=\"Ghost\" href=\"http:\/\/john.onolan.org\/ghost\/\">concept page<\/a> for a lite version of WordPress that I've been thinking about for a long time, called Ghost. I think it's fair to say that I didn't quite anticipate how strong the reaction would be - and I've hardly had time to catch my breath in the last 72 hours.\n\nThe response was overwhelming, and overwhelmingly positive. In the first 6 hours my site got 35,000 page views after hitting the number 1 slot on <a href=\"http:\/\/news.ycombinator.com\/item?id=4743245\" onclick=\"javascript:_gaq.push(['_trackEvent','outbound-article','http:\/\/news.ycombinator.com']);\">Hacker News<\/a>. As of right now, the traffic count is just over <a href=\"http:\/\/john.onolan.org\/wp-content\/uploads\/2012\/11\/Screen-Shot-2012-11-09-at-17.51.21.png\" rel=\"lightbox\" class=\"cboxElement\">91,000 page views<\/a> - and Ghost has been featured all over the place. Notable mentions so far include Christina Warren from Mashable, who <a href=\"http:\/\/christina.is\/\" onclick=\"javascript:_gaq.push(['_trackEvent','outbound-article','http:\/\/christina.is']);\">wrote about it<\/a>. Michael Carney from PandoDaily <a href=\"http:\/\/pandodaily.com\/2012\/11\/07\/wordpress-guru-designs-a-concept-blogging-platform-that-doesnt-suck-gets-rave-reviews\/\" onclick=\"javascript:_gaq.push(['_trackEvent','outbound-article','http:\/\/pandodaily.com']);\">interviewed me about it<\/a>. Someone even <a href=\"http:\/\/www.voicens.com\/web\/?p=4425\" onclick=\"javascript:_gaq.push(['_trackEvent','outbound-article','http:\/\/www.voicens.com']);\">wrote about it in Chinese<\/a>. That's pretty cool.\n\n\nThe feedback has been amazing, and while it's impossible to reply to all of the messages individually, I'm getting to as many of them as I can and I want to thank each and every one of you who took the time to send me a message or share the concept because you liked it. Now that the initial storm has died down a bit, I wanted to take some time to answer some of the more common questions and talk about what's next.\n<h2>FAQ - Continued...<\/h2>\n\nThe most common question, bizarrely:\n<h5><em><strong>Oh my god, why is that whole page made of images? What's wrong with you? \/\/ I can't take you seriously \/\/ Don't you know anything about the web? \/\/ You are literally Satan re-incarnate.<\/strong><\/em><\/h5>\n\nThis was really the only negativity I got in response to the post, and it surprised me. I put together the concept page as... just that... a concept. It was a way for me to get the ideas out of my head and \"down on paper\" - or so to speak. I used photoshop as a <em>tool<\/em> to write down my idea with text and images. If I used a sketchbook as a <em>tool <\/em>to create images and handwritten notes, then uploaded scans of it, I doubt anyone would complain. The concept page was never supposed to be a finished product because I had no idea if there would be any interest in it. I had no motivation to waste hours coding a custom layout for something might only ever be read by a few people and then forgotten.\n\nHardware manufacturers make hundreds of foam cutout prototypes of products before they build one with working buttons and screens. I'm aware of all the usability problems with a web page made of images, and equally, foam cutouts without buttons or screens aren't particularly user friendly either. They're not supposed to be.\n\nLet's move on.\n<h5><em><strong>What? Why no comments? I need comments.<\/strong><\/em><\/h5>\n\nBecause comments add a layer of complexity that is beyond the core focus of this platform, which is publishing. Again, that's not to say you couldn't have any comments. This could easily be added with a dedicated plugin where you own the data or (as mentioned) there are third party providers such as Disqus, IntenseDebate, Livefyre and Facebook who all have great platforms. The point of this isn't to say \"you can't have comments\" - it's to say \"comments aren't on by default\". It's about simplicity, more than anything else.\n<h5><em><strong>Yeah, but WordPress are already going to revise their dashboard, WordPress.com is experimenting with a potential simplified version... so why bother with this?<\/strong><\/em><\/h5>\n\n<a href=\"http:\/\/john.onolan.org\/wp-content\/uploads\/2012\/11\/newwp.png\" rel=\"lightbox[2102]\" title=\"newwp\" class=\"cboxElement\"><img class=\"alignnone size-large wp-image-2117\" title=\"newwp\" src=\"http:\/\/john.onolan.org\/wp-content\/uploads\/2012\/11\/newwp-550x210.png\" alt=\"\" width=\"550\" height=\"210\"><\/a>\n\nSorry, but Tumblr already did this - it's not the future of blogging, it's the past.\n\nGhost isn't about sharing \"Fuck Yeah [<a href=\"http:\/\/fuckyeahdogs.tumblr.com\/\" onclick=\"javascript:_gaq.push(['_trackEvent','outbound-article','http:\/\/fuckyeahdogs.tumblr.com']);\">Dogs<\/a>\/<a href=\"http:\/\/fuckyeahsharks.tumblr.com\/\" onclick=\"javascript:_gaq.push(['_trackEvent','outbound-article','http:\/\/fuckyeahsharks.tumblr.com']);\" rel=\"lightbox\" class=\"cboxElement\">Sharks<\/a>\/<a href=\"http:\/\/fuckyeahgirlswithtattoos.tumblr.com\/\" onclick=\"javascript:_gaq.push(['_trackEvent','outbound-article','http:\/\/fuckyeahgirlswithtattoos.tumblr.com']);\">Girls with Tattoos<\/a>]\" - it's about publishing - which means writing - rather than mashing a few buttons to make sure that everyone can see and appreciate your latest funny picture\/status, which is surely the most funny picture\/status you've ever posted.\n\nTumblr, Pinterest and Facebook already have this locked down. It's not the future.\n<h5><em><strong>So... are you actually going to build this thing?<\/strong><\/em><\/h5>\n\nThe concept page was a way for me to test demand and interest. To see if anyone actually agreed with my frustrations and, more importantly, my solutions. I plucked a random figure of \"10,000 pageviews\" out of the air before I hit the publish button. If it got less than 10,000 pageviews, I would surrender to the fact that it would only ever be an idea. I've now exceeded that goal 9 times over, so yes, I'm looking at how Ghost can now be made into a reality.\n<h5><em><strong>How can I find out when it's done? \/\/ SHUT UP AND TAKE MY MONEY<\/strong><\/em><\/h5>\n\nOk, ok - there's a holding page up on <a href=\"http:\/\/tryghost.org\" onclick=\"javascript:_gaq.push(['_trackEvent','outbound-article','http:\/\/tryghost.org']);\">http:\/\/TryGhost.org<\/a> - put your email address in.\n<hr>\n<h3>How are you going to do this?<\/h3>\n\nThere's three main ways of going about this, each has merits as well as drawbacks.\n\n<em><strong>1.) Build it from scratch<\/strong><\/em> - Many people (particularly the Hacker News crowd) expressed the sentiment that there was little point in forking WordPress. When you're going to strip out so much, you get to a point where you might as well start from scratch anyway. Take away the crutches of being stuck with older technologies and put together something which is as sophisticated in code as it is in UI\/UX.\n<ul>\n<li><em>Pros:<\/em> The idea of something completely new is exciting, opportunity to build something very sophisticated, complete control over everything.<\/li>\n<li><em>Cons:<\/em> Lose the  WordPress ecosystem which includes millions of users and thousands of developers, potentially spend the next 6 months fighting over whether to use PHP\/RoR\/Django\/Python\/Node\/Whateverthefuck because everyone loves to evangelise the technology they know best.<\/li>\n<\/ul>\n\n<em><strong>2.) Fork WordPress<\/strong><\/em> - This was the original idea I put out. Take the WordPress codebase, as is, and modify it to turn it into something new. Initially the codebase is practically the same, which means developers already know it. Then it can change over time and evolve into its own thing.\n<ul>\n<li><em>Pros:<\/em> Easy start with existing codebase, potential to evolve, doesn't lose WordPress ecosystem initially.<\/li>\n<li><em>Cons:<\/em> Stuck with existing codebase - the good as well as the bad,  eventually needs to be rewritten completely, less control, loses the WordPress ecosystem after a while anyway, makes it complicated to transition from legacy code to new code.<\/li>\n<\/ul>\n\n<em><strong>3.) Make it a plugin\/extension<\/strong><\/em> - Lots of people asked why Ghost couldn't just be a WordPress plugin. It would certainly be the easiest route of the 3, it's possible to completely remove \/wp-admin\/ and replace with with \/ghost\/ ... but I feel like it kind of misses the point. This route bolts Ghost on, but it's still WordPress under the hood. From a UI\/UX standpoint it would function - but it wouldn't revolutionise anything else. It makes WordPress itself about blogging again, rather than creating something new.\n<ul>\n<li><em>Pros:<\/em> Very easy development, very easy deployment, keeps WordPress ecosystem forever, doesn't force anyone to change.<\/li>\n<li><em>Cons:<\/em> The least exciting (for me, personally), much less control, it would be much harder to maintain something like this on a non-profit basis - which loses a piece of what Ghost is about.<\/li>\n<\/ul>\n<h3>What's the answer?<\/h3>\n\nI've spoken to a lot of smart people over the last few days. The one thing that everyone seems to agree on is that a fork is the worst of both worlds. So the one thing that I suggested as a way of making this happen, is the least likely to work in reality. Remember the foam prototype metaphor earlier? Learning and iterating - that's what happening now.\n\nThat leaves a choice between WordPress plugin or fresh build. The answer? Both.\n\nA WordPress plugin will act as a proof of concept and a working prototype, initially, because it's easier to leverage the existing WordPress ecosystem to create it than to go into a cave for 6 months trying to build this amazing thing that everyone will have forgotten about.\n\nThe plugin will not be perfect. It will add the Ghost UI\/UX and as much functionality as we can cram into it. It will completely remove \/wp-admin\/ and replace it with \/ghost\/ - effectively using WordPress core as a basic foundation to build on top of. It will give people who don't want to switch away from WordPress access to the Ghost UX which they want to have, and it will give people who want the full Ghost platform a taste of what's to come.\n\nIt will allow us to develop and learn and iterate on the concept pretty rapidly, which has a great deal of value.\n\nThis is step one. Assuming the plugin is actually used by people - it would then justify exploring building the standalone version of Ghost from the ground up. The plugin would subsequently serve as a great marketing tool for the platform. Think of it as an upgrade path. But that's a long way away. Having the idea is the easy part. Making it happen is what counts.\n\nHappily - amongst the thousands of people talking about Ghost for the last few days - several have been talking about how they've already built some working prototypes of my mockups and turned them into WordPress plugins or just local development sites. These will likely go on to be the starting point of the first Ghost plugin.<\/p>\n\nThere's a lot to do, and I'm amazed by the number of people who have offered their help with this. In the next few days I'll be kicking off work on the plugin properly and start putting together a more organised structure which explains how you can get involved and contribute to the project if you're interested. So... watch this space - and thanks for all your support so far.\n\n<a href=\"http:\/\/twitter.com\/TryGhost\" onclick=\"javascript:_gaq.push(['_trackEvent','outbound-article','http:\/\/twitter.com']);\" class=\"twitter-follow-button\">Follow @TryGhost<\/a>"), html: "<p>Three days ago I released a <a title=\"Ghost\" href=\"http:\/\/john.onolan.org\/ghost\/\">concept page<\/a> for a lite version of WordPress that I've been thinking about for a long time, called Ghost. I think it's fair to say that I didn't quite anticipate how strong the reaction would be - and I've hardly had time to catch my breath in the last 72 hours.<\/p>\n<p>The response was overwhelming, and overwhelmingly positive. In the first 6 hours my site got 35,000 page views after hitting the number 1 slot on <a href=\"http:\/\/news.ycombinator.com\/item?id=4743245\" onclick=\"javascript:_gaq.push(['_trackEvent','outbound-article','http:\/\/news.ycombinator.com']);\">Hacker News<\/a>. As of right now, the traffic count is just over <a href=\"http:\/\/john.onolan.org\/wp-content\/uploads\/2012\/11\/Screen-Shot-2012-11-09-at-17.51.21.png\" rel=\"lightbox\" class=\"cboxElement\">91,000 page views<\/a> - and Ghost has been featured all over the place. Notable mentions so far include Christina Warren from Mashable, who <a href=\"http:\/\/christina.is\/\" onclick=\"javascript:_gaq.push(['_trackEvent','outbound-article','http:\/\/christina.is']);\">wrote about it<\/a>. Michael Carney from PandoDaily <a href=\"http:\/\/pandodaily.com\/2012\/11\/07\/wordpress-guru-designs-a-concept-blogging-platform-that-doesnt-suck-gets-rave-reviews\/\" onclick=\"javascript:_gaq.push(['_trackEvent','outbound-article','http:\/\/pandodaily.com']);\">interviewed me about it<\/a>. Someone even <a href=\"http:\/\/www.voicens.com\/web\/?p=4425\" onclick=\"javascript:_gaq.push(['_trackEvent','outbound-article','http:\/\/www.voicens.com']);\">wrote about it in Chinese<\/a>. That's pretty cool.\\n<p>The feedback has been amazing, and while it's impossible to reply to all of the messages individually, I'm getting to as many of them as I can and I want to thank each and every one of you who took the time to send me a message or share the concept because you liked it. Now that the initial storm has died down a bit, I wanted to take some time to answer some of the more common questions and talk about what's next.<\/p>\n<h2>FAQ - Continued...<\/h2>\n<p>The most common question, bizarrely:<\/p>\n<h5><em><strong>Oh my god, why is that whole page made of images? What's wrong with you? \/\/ I can't take you seriously \/\/ Don't you know anything about the web? \/\/ You are literally Satan re-incarnate.<\/strong><\/em><\/h5>\n<p>This was really the only negativity I got in response to the post, and it surprised me. I put together the concept page as... just that... a concept. It was a way for me to get the ideas out of my head and \"down on paper\" - or so to speak. I used photoshop as a <em>tool<\/em> to write down my idea with text and images. If I used a sketchbook as a <em>tool <\/em>to create images and handwritten notes, then uploaded scans of it, I doubt anyone would complain. The concept page was never supposed to be a finished product because I had no idea if there would be any interest in it. I had no motivation to waste hours coding a custom layout for something might only ever be read by a few people and then forgotten.<\/p>\n<p>Hardware manufacturers make hundreds of foam cutout prototypes of products before they build one with working buttons and screens. I'm aware of all the usability problems with a web page made of images, and equally, foam cutouts without buttons or screens aren't particularly user friendly either. They're not supposed to be.<\/p>\n<p>Let's move on.<\/p>\n<h5><em><strong>What? Why no comments? I need comments.<\/strong><\/em><\/h5>\n<p>Because comments add a layer of complexity that is beyond the core focus of this platform, which is publishing. Again, that's not to say you couldn't have any comments. This could easily be added with a dedicated plugin where you own the data or (as mentioned) there are third party providers such as Disqus, IntenseDebate, Livefyre and Facebook who all have great platforms. The point of this isn't to say \"you can't have comments\" - it's to say \"comments aren't on by default\". It's about simplicity, more than anything else.<\/p>\n<h5><em><strong>Yeah, but WordPress are already going to revise their dashboard, WordPress.com is experimenting with a potential simplified version... so why bother with this?<\/strong><\/em><\/h5>\n<p><a href=\"http:\/\/john.onolan.org\/wp-content\/uploads\/2012\/11\/newwp.png\" rel=\"lightbox[2102]\" title=\"newwp\" class=\"cboxElement\"><img class=\"alignnone size-large wp-image-2117\" title=\"newwp\" src=\"http:\/\/john.onolan.org\/wp-content\/uploads\/2012\/11\/newwp-550x210.png\" alt=\"\" width=\"550\" height=\"210\"><\/a><\/p>\n<p>Sorry, but Tumblr already did this - it's not the future of blogging, it's the past.<\/p>\n<p>Ghost isn't about sharing \"Fuck Yeah [<a href=\"http:\/\/fuckyeahdogs.tumblr.com\/\" onclick=\"javascript:_gaq.push(['_trackEvent','outbound-article','http:\/\/fuckyeahdogs.tumblr.com']);\">Dogs<\/a>\/<a href=\"http:\/\/fuckyeahsharks.tumblr.com\/\" onclick=\"javascript:_gaq.push(['_trackEvent','outbound-article','http:\/\/fuckyeahsharks.tumblr.com']);\" rel=\"lightbox\" class=\"cboxElement\">Sharks<\/a>\/<a href=\"http:\/\/fuckyeahgirlswithtattoos.tumblr.com\/\" onclick=\"javascript:_gaq.push(['_trackEvent','outbound-article','http:\/\/fuckyeahgirlswithtattoos.tumblr.com']);\">Girls with Tattoos<\/a>]\" - it's about publishing - which means writing - rather than mashing a few buttons to make sure that everyone can see and appreciate your latest funny picture\/status, which is surely the most funny picture\/status you've ever posted.<\/p>\n<p>Tumblr, Pinterest and Facebook already have this locked down. It's not the future.<\/p>\n<h5><em><strong>So... are you actually going to build this thing?<\/strong><\/em><\/h5>\n<p>The concept page was a way for me to test demand and interest. To see if anyone actually agreed with my frustrations and, more importantly, my solutions. I plucked a random figure of \"10,000 pageviews\" out of the air before I hit the publish button. If it got less than 10,000 pageviews, I would surrender to the fact that it would only ever be an idea. I've now exceeded that goal 9 times over, so yes, I'm looking at how Ghost can now be made into a reality.<\/p>\n<h5><em><strong>How can I find out when it's done? \/\/ SHUT UP AND TAKE MY MONEY<\/strong><\/em><\/h5>\n<p>Ok, ok - there's a holding page up on <a href=\"http:\/\/tryghost.org\" onclick=\"javascript:_gaq.push(['_trackEvent','outbound-article','http:\/\/tryghost.org']);\">http:\/\/TryGhost.org<\/a> - put your email address in.<\/p>\n<hr>\n<h3>How are you going to do this?<\/h3>\n<p>There's three main ways of going about this, each has merits as well as drawbacks.<\/p>\n<p><em><strong>1.) Build it from scratch<\/strong><\/em> - Many people (particularly the Hacker News crowd) expressed the sentiment that there was little point in forking WordPress. When you're going to strip out so much, you get to a point where you might as well start from scratch anyway. Take away the crutches of being stuck with older technologies and put together something which is as sophisticated in code as it is in UI\/UX.<\/p>\n<ul>\n<li><em>Pros:<\/em> The idea of something completely new is exciting, opportunity to build something very sophisticated, complete control over everything.<\/li>\n<li><em>Cons:<\/em> Lose the  WordPress ecosystem which includes millions of users and thousands of developers, potentially spend the next 6 months fighting over whether to use PHP\/RoR\/Django\/Python\/Node\/Whateverthefuck because everyone loves to evangelise the technology they know best.<\/li>\n<\/ul>\n<p><em><strong>2.) Fork WordPress<\/strong><\/em> - This was the original idea I put out. Take the WordPress codebase, as is, and modify it to turn it into something new. Initially the codebase is practically the same, which means developers already know it. Then it can change over time and evolve into its own thing.<\/p>\n<ul>\n<li><em>Pros:<\/em> Easy start with existing codebase, potential to evolve, doesn't lose WordPress ecosystem initially.<\/li>\n<li><em>Cons:<\/em> Stuck with existing codebase - the good as well as the bad,  eventually needs to be rewritten completely, less control, loses the WordPress ecosystem after a while anyway, makes it complicated to transition from legacy code to new code.<\/li>\n<\/ul>\n<p><em><strong>3.) Make it a plugin\/extension<\/strong><\/em> - Lots of people asked why Ghost couldn't just be a WordPress plugin. It would certainly be the easiest route of the 3, it's possible to completely remove \/wp-admin\/ and replace with with \/ghost\/ ... but I feel like it kind of misses the point. This route bolts Ghost on, but it's still WordPress under the hood. From a UI\/UX standpoint it would function - but it wouldn't revolutionise anything else. It makes WordPress itself about blogging again, rather than creating something new.<\/p>\n<ul>\n<li><em>Pros:<\/em> Very easy development, very easy deployment, keeps WordPress ecosystem forever, doesn't force anyone to change.<\/li>\n<li><em>Cons:<\/em> The least exciting (for me, personally), much less control, it would be much harder to maintain something like this on a non-profit basis - which loses a piece of what Ghost is about.<\/li>\n<\/ul>\n<h3>What's the answer?<\/h3>\n<p>I've spoken to a lot of smart people over the last few days. The one thing that everyone seems to agree on is that a fork is the worst of both worlds. So the one thing that I suggested as a way of making this happen, is the least likely to work in reality. Remember the foam prototype metaphor earlier? Learning and iterating - that's what happening now.<\/p>\n<p>That leaves a choice between WordPress plugin or fresh build. The answer? Both.<\/p>\n<p>A WordPress plugin will act as a proof of concept and a working prototype, initially, because it's easier to leverage the existing WordPress ecosystem to create it than to go into a cave for 6 months trying to build this amazing thing that everyone will have forgotten about.<\/p>\n<p>The plugin will not be perfect. It will add the Ghost UI\/UX and as much functionality as we can cram into it. It will completely remove \/wp-admin\/ and replace it with \/ghost\/ - effectively using WordPress core as a basic foundation to build on top of. It will give people who don't want to switch away from WordPress access to the Ghost UX which they want to have, and it will give people who want the full Ghost platform a taste of what's to come.<\/p>\n<p>It will allow us to develop and learn and iterate on the concept pretty rapidly, which has a great deal of value.<\/p>\n<p>This is step one. Assuming the plugin is actually used by people - it would then justify exploring building the standalone version of Ghost from the ground up. The plugin would subsequently serve as a great marketing tool for the platform. Think of it as an upgrade path. But that's a long way away. Having the idea is the easy part. Making it happen is what counts.<\/p>\n<p>Happily - amongst the thousands of people talking about Ghost for the last few days - several have been talking about how they've already built some working prototypes of my mockups and turned them into WordPress plugins or just local development sites. These will likely go on to be the starting point of the first Ghost plugin.<\/p>\n<p>There's a lot to do, and I'm amazed by the number of people who have offered their help with this. In the next few days I'll be kicking off work on the plugin properly and start putting together a more organised structure which explains how you can get involved and contribute to the project if you're interested. So... watch this space - and thanks for all your support so far.<\/p>\n<p><a href=\"http:\/\/twitter.com\/TryGhost\" onclick=\"javascript:_gaq.push(['_trackEvent','outbound-article','http:\/\/twitter.com']);\" class=\"twitter-follow-button\">Follow @TryGhost<\/a><\/p>", feature_image: 'ghostpost.jpg', status: status, @@ -634,7 +652,7 @@ DataGenerator.forModel = (function () { roles; posts = _.map(DataGenerator.Content.posts, function (post) { - return _.pick(post, 'title', 'markdown'); + return _.pick(post, 'title', 'mobiledoc'); }); tags = DataGenerator.Content.tags; diff --git a/core/test/utils/fixtures/filter-param/index.js b/core/test/utils/fixtures/filter-param/index.js index 4039340889..87c448778a 100644 --- a/core/test/utils/fixtures/filter-param/index.js +++ b/core/test/utils/fixtures/filter-param/index.js @@ -4,6 +4,7 @@ var _ = require('lodash'), ObjectId = require('bson-objectid'), db = require('../../../../server/data/db'), + markdownToMobiledoc = require('../../../utils/fixtures/data-generator').markdownToMobiledoc, data = {}; // Password = Sl1m3rson @@ -83,7 +84,7 @@ data.posts = [ id: ObjectId.generate(), title: 'First Post', slug: 'first-post', - markdown: 'Hello World!', + mobiledoc: markdownToMobiledoc('Hello World!'), featured: false, author_id: data.users[0].id, tags: [data.tags[0].id] @@ -92,7 +93,7 @@ data.posts = [ id: ObjectId.generate(), title: 'Second Post', slug: 'second-post', - markdown: 'Hello World!', + mobiledoc: markdownToMobiledoc('Hello World!'), featured: false, author_id: data.users[1].id, tags: [data.tags[1].id, data.tags[2].id, data.tags[3].id, data.tags[5].id] @@ -101,7 +102,7 @@ data.posts = [ id: ObjectId.generate(), title: 'Third Post', slug: 'third-post', - markdown: 'Hello World!', + mobiledoc: markdownToMobiledoc('Hello World!'), featured: false, author_id: data.users[0].id, tags: [data.tags[1].id] @@ -110,7 +111,7 @@ data.posts = [ id: ObjectId.generate(), title: 'Fourth Post', slug: 'fourth-post', - markdown: 'Hello World!', + mobiledoc: markdownToMobiledoc('Hello World!'), featured: false, author_id: data.users[0].id, tags: [data.tags[2].id] @@ -119,7 +120,7 @@ data.posts = [ id: ObjectId.generate(), title: 'Fifth Post', slug: 'fifth-post', - markdown: 'Hello World!', + mobiledoc: markdownToMobiledoc('Hello World!'), featured: true, author_id: data.users[1].id, tags: [data.tags[5].id] @@ -128,7 +129,7 @@ data.posts = [ id: ObjectId.generate(), title: 'Sixth Post', slug: 'sixth-post', - markdown: 'Hello World!', + mobiledoc: markdownToMobiledoc('Hello World!'), featured: false, author_id: data.users[1].id, feature_image: 'some/image/path.jpg', @@ -138,7 +139,7 @@ data.posts = [ id: ObjectId.generate(), title: 'Seventh Post', slug: 'seventh-post', - markdown: 'Hello World!', + mobiledoc: markdownToMobiledoc('Hello World!'), featured: false, author_id: data.users[0].id, feature_image: 'some/image/path.jpg', @@ -148,7 +149,7 @@ data.posts = [ id: ObjectId.generate(), title: 'Eighth Post', slug: 'eighth-post', - markdown: 'Hello World!', + mobiledoc: markdownToMobiledoc('Hello World!'), featured: true, author_id: data.users[0].id, tags: [data.tags[0].id, data.tags[2].id, data.tags[3].id] @@ -157,7 +158,7 @@ data.posts = [ id: ObjectId.generate(), title: 'Ninth Post', slug: 'ninth-post', - markdown: 'Hello World!', + mobiledoc: markdownToMobiledoc('Hello World!'), featured: false, author_id: data.users[0].id, tags: [data.tags[1].id, data.tags[3].id] @@ -166,7 +167,7 @@ data.posts = [ id: ObjectId.generate(), title: 'Tenth Post', slug: 'tenth-post', - markdown: 'Hello World!', + mobiledoc: markdownToMobiledoc('Hello World!'), featured: false, author_id: data.users[0].id, tags: [data.tags[2].id] @@ -175,7 +176,7 @@ data.posts = [ id: ObjectId.generate(), title: 'Eleventh Post', slug: 'eleventh-post', - markdown: 'Hello World!', + mobiledoc: markdownToMobiledoc('Hello World!'), featured: false, author_id: data.users[0].id, feature_image: 'some/image/path.jpg', @@ -185,7 +186,7 @@ data.posts = [ id: ObjectId.generate(), title: 'Twelfth Post', slug: 'twelfth-post', - markdown: 'Hello World!', + mobiledoc: markdownToMobiledoc('Hello World!'), featured: false, author_id: data.users[0].id, tags: [data.tags[3].id] @@ -194,7 +195,7 @@ data.posts = [ id: ObjectId.generate(), title: 'Thirteenth Post', slug: 'thirteenth-post', - markdown: 'Hello World!', + mobiledoc: markdownToMobiledoc('Hello World!'), featured: false, author_id: data.users[0].id, tags: [] @@ -203,7 +204,7 @@ data.posts = [ id: ObjectId.generate(), title: 'Fourteenth Post', slug: 'fourteenth-post', - markdown: 'Hello World!', + mobiledoc: markdownToMobiledoc('Hello World!'), featured: true, author_id: data.users[0].id, tags: [data.tags[3].id] @@ -212,7 +213,7 @@ data.posts = [ id: ObjectId.generate(), title: 'Fifteenth Post', slug: 'fifteenth-post', - markdown: 'Hello World! I am a featured page', + mobiledoc: markdownToMobiledoc('Hello World! I am a featured page'), featured: true, page: 1, author_id: data.users[0].id, @@ -222,7 +223,7 @@ data.posts = [ id: ObjectId.generate(), title: 'Sixteenth Post', slug: 'sixteenth-post', - markdown: 'Hello World!', + mobiledoc: markdownToMobiledoc('Hello World!'), featured: false, author_id: data.users[0].id, tags: [] @@ -231,7 +232,7 @@ data.posts = [ id: ObjectId.generate(), title: 'Seventeenth Post', slug: 'seventeenth-post', - markdown: 'Hello World!', + mobiledoc: markdownToMobiledoc('Hello World!'), featured: false, author_id: data.users[0].id, tags: [] @@ -240,7 +241,7 @@ data.posts = [ id: ObjectId.generate(), title: 'Eighteenth Post', slug: 'eighteenth-post', - markdown: 'Hello World!', + mobiledoc: markdownToMobiledoc('Hello World!'), featured: false, author_id: data.users[0].id, tags: [] @@ -249,7 +250,7 @@ data.posts = [ id: ObjectId.generate(), title: 'Nineteenth Post', slug: 'nineteenth-post', - markdown: 'Hello World!', + mobiledoc: markdownToMobiledoc('Hello World!'), featured: false, status: 'draft', author_id: data.users[0].id, @@ -259,7 +260,7 @@ data.posts = [ id: ObjectId.generate(), title: 'Twentieth Post', slug: 'twentieth-post', - markdown: 'Hello World!', + mobiledoc: markdownToMobiledoc('Hello World!'), featured: false, author_id: data.users[0].id, tags: [] @@ -268,7 +269,7 @@ data.posts = [ id: ObjectId.generate(), title: 'About Page', slug: 'about', - markdown: 'About Me!', + mobiledoc: markdownToMobiledoc('About Me!'), featured: false, page: 1, author_id: data.users[0].id,