mirror of
https://github.com/discourse/discourse.git
synced 2024-11-25 21:33:43 +08:00
More Qunit tests including a CLI runner
This commit is contained in:
parent
8e96299653
commit
60fce196c7
1
Gemfile
1
Gemfile
|
@ -118,6 +118,7 @@ group :test, :development do
|
|||
gem 'rspec-given'
|
||||
gem 'pry-rails'
|
||||
gem 'pry-nav'
|
||||
gem 'webrick'
|
||||
end
|
||||
|
||||
group :development do
|
||||
|
|
|
@ -472,6 +472,7 @@ GEM
|
|||
uglifier (2.0.1)
|
||||
execjs (>= 0.3.0)
|
||||
multi_json (~> 1.0, >= 1.0.2)
|
||||
webrick (1.3.1)
|
||||
|
||||
PLATFORMS
|
||||
ruby
|
||||
|
@ -572,3 +573,4 @@ DEPENDENCIES
|
|||
turbo-sprockets-rails3
|
||||
uglifier
|
||||
vestal_versions!
|
||||
webrick
|
||||
|
|
|
@ -9,10 +9,10 @@
|
|||
Discourse.SelectedPostsCount = Em.Mixin.create({
|
||||
|
||||
selectedPostsCount: function() {
|
||||
if (!this.get('selectedPosts')) return 0;
|
||||
|
||||
if (this.get('allPostsSelected')) return this.get('posts_count') || this.get('topic.posts_count');
|
||||
|
||||
if (!this.get('selectedPosts')) return 0;
|
||||
|
||||
return this.get('selectedPosts.length');
|
||||
}.property('selectedPosts.length', 'allPostsSelected')
|
||||
|
||||
|
|
|
@ -293,7 +293,7 @@ Discourse.Post.reopenClass({
|
|||
create: function(obj, topic) {
|
||||
var result = this._super(obj);
|
||||
this.createActionSummary(result);
|
||||
if (obj.reply_to_user) {
|
||||
if (obj && obj.reply_to_user) {
|
||||
result.set('reply_to_user', Discourse.User.create(obj.reply_to_user));
|
||||
}
|
||||
result.set('topic', topic);
|
||||
|
|
|
@ -6,9 +6,12 @@ class DiscourseIIFE < Sprockets::Processor
|
|||
path = context.pathname.to_s
|
||||
|
||||
# Only discourse or admin paths
|
||||
return data unless (path =~ /\/javascripts\/discourse/ || path =~ /\/javascripts\/admin/)
|
||||
return data unless (path =~ /\/javascripts\/discourse/ || path =~ /\/javascripts\/admin/ || path =~ /\/test\/javascripts/)
|
||||
|
||||
# Ugh, ignore translations
|
||||
# Ignore the js helper
|
||||
return data if (path =~ /test\_helper\.js/)
|
||||
|
||||
# Ignore translations
|
||||
return data if (path =~ /\/translations/)
|
||||
|
||||
# We don't add IIFEs to handlebars
|
||||
|
|
48
lib/tasks/qunit.rake
Normal file
48
lib/tasks/qunit.rake
Normal file
|
@ -0,0 +1,48 @@
|
|||
desc "Runs the qunit test suite"
|
||||
|
||||
task "qunit:test" => :environment do
|
||||
|
||||
require "rack"
|
||||
require "webrick"
|
||||
|
||||
unless %x{which phantomjs > /dev/null 2>&1}
|
||||
abort "PhantomJS is not installed. Download from http://phantomjs.org"
|
||||
end
|
||||
|
||||
port = ENV['TEST_SERVER_PORT'] || 60099
|
||||
server = Thread.new do
|
||||
Rack::Server.start(:config => "config.ru",
|
||||
:Logger => WEBrick::Log.new("/dev/null"),
|
||||
:AccessLog => [],
|
||||
:Port => port)
|
||||
end
|
||||
|
||||
begin
|
||||
success = true
|
||||
test_path = "#{Rails.root}/vendor/assets/javascripts"
|
||||
cmd = "phantomjs #{test_path}/run-qunit.js \"http://localhost:#{port}/qunit\""
|
||||
|
||||
rake_system(cmd)
|
||||
|
||||
# A bit of a hack until we can figure this out on Travis
|
||||
tries = 0
|
||||
while tries < 3 && $?.exitstatus === 124
|
||||
tries += 1
|
||||
puts "\nTimed Out. Trying again...\n"
|
||||
sh(cmd)
|
||||
end
|
||||
|
||||
success &&= $?.success?
|
||||
|
||||
ensure
|
||||
server.kill
|
||||
end
|
||||
|
||||
if success
|
||||
puts "\nTests Passed"
|
||||
else
|
||||
puts "\nTests Failed"
|
||||
exit(1)
|
||||
end
|
||||
|
||||
end
|
|
@ -1,87 +0,0 @@
|
|||
/*global waitsFor:true expect:true describe:true beforeEach:true it:true md5:true */
|
||||
|
||||
describe("Discourse.BBCode", function() {
|
||||
|
||||
var format = Discourse.BBCode.format;
|
||||
|
||||
describe("quoting", function() {
|
||||
|
||||
// Format text without an avatar lookup
|
||||
function formatQuote(text) {
|
||||
return format(text, {lookupAvatar: false});
|
||||
}
|
||||
|
||||
it("can quote", function() {
|
||||
expect(formatQuote("[quote=\"eviltrout, post:1, topic:1\"]abc[/quote]")).
|
||||
toBe("</p><aside class='quote' data-post=\"1\" data-topic=\"1\" >\n <div class='title'>\n " +
|
||||
"<div class='quote-controls'></div>\n \n eviltrout\n said:\n </div>\n <blockquote>abc</blockquote>\n</aside>\n<p>");
|
||||
});
|
||||
|
||||
it("can nest quotes", function() {
|
||||
expect(formatQuote("[quote=\"eviltrout, post:1, topic:1\"]abc[quote=\"eviltrout, post:2, topic:2\"]nested[/quote][/quote]")).
|
||||
toBe("</p><aside class='quote' data-post=\"1\" data-topic=\"1\" >\n <div class='title'>\n <div " +
|
||||
"class='quote-controls'></div>\n \n eviltrout\n said:\n </div>\n <blockquote>abc</p><aside " +
|
||||
"class='quote' data-post=\"2\" data-topic=\"2\" >\n <div class='title'>\n <div class='quote-" +
|
||||
"controls'></div>\n \n eviltrout\n said:\n </div>\n <blockquote>nested</blockquote>\n</aside>\n<p></blockquote>\n</aside>\n<p>");
|
||||
});
|
||||
|
||||
it("can handle more than one quote", function() {
|
||||
expect(formatQuote("before[quote=\"eviltrout, post:1, topic:1\"]first[/quote]middle[quote=\"eviltrout, post:2, topic:2\"]second[/quote]after")).
|
||||
toBe("before</p><aside class='quote' data-post=\"1\" data-topic=\"1\" >\n <div class='title'>\n <div class='quote-cont" +
|
||||
"rols'></div>\n \n eviltrout\n said:\n </div>\n <blockquote>first</blockquote>\n</aside>\n<p>middle</p><aside cla" +
|
||||
"ss='quote' data-post=\"2\" data-topic=\"2\" >\n <div class='title'>\n <div class='quote-controls'></div>\n \n " +
|
||||
"eviltrout\n said:\n </div>\n <blockquote>second</blockquote>\n</aside>\n<p>after");
|
||||
});
|
||||
|
||||
describe("extractQuotes", function() {
|
||||
|
||||
var extractQuotes = Discourse.BBCode.extractQuotes;
|
||||
|
||||
it("returns an object a template renderer", function() {
|
||||
var q = "[quote=\"eviltrout, post:1, topic:2\"]hello[/quote]";
|
||||
var result = extractQuotes(q + " world");
|
||||
|
||||
expect(result.text).toBe(md5(q) + "\n world");
|
||||
expect(result.template).not.toBe(null);
|
||||
});
|
||||
|
||||
});
|
||||
|
||||
describe("buildQuoteBBCode", function() {
|
||||
|
||||
var build = Discourse.BBCode.buildQuoteBBCode;
|
||||
|
||||
var post = Discourse.Post.create({
|
||||
cooked: "<p><b>lorem</b> ipsum</p>",
|
||||
username: "eviltrout",
|
||||
post_number: 1,
|
||||
topic_id: 2
|
||||
});
|
||||
|
||||
it("returns an empty string when contents is undefined", function() {
|
||||
expect(build(post, undefined)).toBe("");
|
||||
expect(build(post, null)).toBe("");
|
||||
expect(build(post, "")).toBe("");
|
||||
});
|
||||
|
||||
it("returns the quoted contents", function() {
|
||||
expect(build(post, "lorem")).toBe("[quote=\"eviltrout, post:1, topic:2\"]\nlorem\n[/quote]\n\n");
|
||||
});
|
||||
|
||||
it("trims white spaces before & after the quoted contents", function() {
|
||||
expect(build(post, " lorem ")).toBe("[quote=\"eviltrout, post:1, topic:2\"]\nlorem\n[/quote]\n\n");
|
||||
});
|
||||
|
||||
it("marks quotes as full when the quote is the full message", function() {
|
||||
expect(build(post, "lorem ipsum")).toBe("[quote=\"eviltrout, post:1, topic:2, full:true\"]\nlorem ipsum\n[/quote]\n\n");
|
||||
});
|
||||
|
||||
it("keeps BBCode formatting", function() {
|
||||
expect(build(post, "**lorem** ipsum")).toBe("[quote=\"eviltrout, post:1, topic:2, full:true\"]\n**lorem** ipsum\n[/quote]\n\n");
|
||||
});
|
||||
|
||||
});
|
||||
|
||||
});
|
||||
|
||||
});
|
|
@ -1,9 +1,9 @@
|
|||
/*global module:true test:true ok:true visit:true expect:true exists:true count:true equal:true */
|
||||
/*global module:true test:true ok:true visit:true expect:true exists:true count:true equal:true present:true md5:true */
|
||||
|
||||
module("Discourse.BBCode");
|
||||
|
||||
var format = function(input, expected, text) {
|
||||
equal(Discourse.BBCode.format(input), expected, text);
|
||||
equal(Discourse.BBCode.format(input, {lookupAvatar: false}), expected, text);
|
||||
}
|
||||
|
||||
test('basic bbcode', function() {
|
||||
|
@ -36,4 +36,74 @@ test('tags with arguments', function() {
|
|||
format("[email=eviltrout@mailinator.com]evil trout[/email]", "<a href=\"mailto:eviltrout@mailinator.com\">evil trout</a>", "supports [email] with a title");
|
||||
format("[u][i]abc[/i][/u]", "<span class='bbcode-u'><span class='bbcode-i'>abc</span></span>", "can nest tags");
|
||||
format("[b]first[/b] [b]second[/b]", "<span class='bbcode-b'>first</span> <span class='bbcode-b'>second</span>", "can bold two things on the same line");
|
||||
});
|
||||
});
|
||||
|
||||
|
||||
test("quotes", function() {
|
||||
|
||||
var post = Discourse.Post.create({
|
||||
cooked: "<p><b>lorem</b> ipsum</p>",
|
||||
username: "eviltrout",
|
||||
post_number: 1,
|
||||
topic_id: 2
|
||||
});
|
||||
|
||||
var formatQuote = function(val, expected, text) {
|
||||
equal(Discourse.BBCode.buildQuoteBBCode(post, val), expected, text);
|
||||
}
|
||||
|
||||
formatQuote(undefined, "", "empty string for undefined content");
|
||||
formatQuote(null, "", "empty string for null content");
|
||||
formatQuote("", "", "empty string for empty string content");
|
||||
|
||||
formatQuote("lorem", "[quote=\"eviltrout, post:1, topic:2\"]\nlorem\n[/quote]\n\n", "correctly formats quotes");
|
||||
|
||||
formatQuote(" lorem \t ",
|
||||
"[quote=\"eviltrout, post:1, topic:2\"]\nlorem\n[/quote]\n\n",
|
||||
"trims white spaces before & after the quoted contents");
|
||||
|
||||
formatQuote("lorem ipsum",
|
||||
"[quote=\"eviltrout, post:1, topic:2, full:true\"]\nlorem ipsum\n[/quote]\n\n",
|
||||
"marks quotes as full when the quote is the full message");
|
||||
|
||||
formatQuote("**lorem** ipsum",
|
||||
"[quote=\"eviltrout, post:1, topic:2, full:true\"]\n**lorem** ipsum\n[/quote]\n\n",
|
||||
"keeps BBCode formatting");
|
||||
|
||||
});
|
||||
|
||||
test("quote formatting", function() {
|
||||
|
||||
// TODO: This HTML matching is quite ugly.
|
||||
format("[quote=\"eviltrout, post:1, topic:1\"]abc[/quote]",
|
||||
"</p><aside class='quote' data-post=\"1\" data-topic=\"1\" >\n <div class='title'>\n " +
|
||||
"<div class='quote-controls'></div>\n \n eviltrout\n said:\n </div>\n <blockquote>abc</blockquote>\n</aside>\n<p>",
|
||||
"renders quotes properly");
|
||||
|
||||
format("[quote=\"eviltrout, post:1, topic:1\"]abc[quote=\"eviltrout, post:2, topic:2\"]nested[/quote][/quote]",
|
||||
"</p><aside class='quote' data-post=\"1\" data-topic=\"1\" >\n <div class='title'>\n <div " +
|
||||
"class='quote-controls'></div>\n \n eviltrout\n said:\n </div>\n <blockquote>abc</p><aside " +
|
||||
"class='quote' data-post=\"2\" data-topic=\"2\" >\n <div class='title'>\n <div class='quote-" +
|
||||
"controls'></div>\n \n eviltrout\n said:\n </div>\n <blockquote>nested</blockquote>\n</aside>\n<p></blockquote>\n</aside>\n<p>",
|
||||
"can nest quotes");
|
||||
|
||||
format("before[quote=\"eviltrout, post:1, topic:1\"]first[/quote]middle[quote=\"eviltrout, post:2, topic:2\"]second[/quote]after",
|
||||
"before</p><aside class='quote' data-post=\"1\" data-topic=\"1\" >\n <div class='title'>\n <div class='quote-cont" +
|
||||
"rols'></div>\n \n eviltrout\n said:\n </div>\n <blockquote>first</blockquote>\n</aside>\n<p>middle</p><aside cla" +
|
||||
"ss='quote' data-post=\"2\" data-topic=\"2\" >\n <div class='title'>\n <div class='quote-controls'></div>\n \n " +
|
||||
"eviltrout\n said:\n </div>\n <blockquote>second</blockquote>\n</aside>\n<p>after",
|
||||
"can handle more than one quote");
|
||||
|
||||
});
|
||||
|
||||
|
||||
test("extract quotes", function() {
|
||||
|
||||
var q = "[quote=\"eviltrout, post:1, topic:2\"]hello[/quote]";
|
||||
var result = Discourse.BBCode.extractQuotes(q + " world");
|
||||
|
||||
equal(result.text, md5(q) + "\n world");
|
||||
present(result.template);
|
||||
|
||||
});
|
||||
|
||||
|
|
|
@ -7,7 +7,7 @@ var testObj = Em.Object.createWithMixins(Discourse.Presence, {
|
|||
nonEmptyString: "Evil Trout",
|
||||
emptyArray: [],
|
||||
nonEmptyArray: [1, 2, 3],
|
||||
age: 34,
|
||||
age: 34
|
||||
});
|
||||
|
||||
test("present", function() {
|
||||
|
@ -18,7 +18,6 @@ test("present", function() {
|
|||
ok(testObj.present('age'), "integers are present");
|
||||
});
|
||||
|
||||
|
||||
test("blank", function() {
|
||||
ok(testObj.blank('emptyString'), "Empty strings are blank");
|
||||
ok(!testObj.blank('nonEmptyString'), "Non empty strings are not blank");
|
||||
|
|
34
test/javascripts/mixins/selected_posts_count_test.js
Normal file
34
test/javascripts/mixins/selected_posts_count_test.js
Normal file
|
@ -0,0 +1,34 @@
|
|||
/*global module:true test:true ok:true visit:true expect:true exists:true count:true equal:true */
|
||||
module("Discourse.SelectedPostsCount");
|
||||
|
||||
var buildTestObj = function(params, topicParams) {
|
||||
return Ember.Object.createWithMixins(Discourse.SelectedPostsCount, params || {});
|
||||
};
|
||||
|
||||
test("without selectedPosts", function () {
|
||||
var testObj = buildTestObj();
|
||||
|
||||
equal(testObj.get('selectedPostsCount'), 0, "No posts are selected without a selectedPosts property");
|
||||
|
||||
testObj.set('selectedPosts', []);
|
||||
equal(testObj.get('selectedPostsCount'), 0, "No posts are selected when selectedPosts is an empty array");
|
||||
});
|
||||
|
||||
test("with some selectedPosts", function() {
|
||||
var testObj = buildTestObj({ selectedPosts: [Discourse.Post.create()] });
|
||||
equal(testObj.get('selectedPostsCount'), 1, "It returns the amount of posts");
|
||||
});
|
||||
|
||||
test("when all posts are selected and there is a posts_count", function() {
|
||||
var testObj = buildTestObj({ allPostsSelected: true, posts_count: 1024 });
|
||||
equal(testObj.get('selectedPostsCount'), 1024, "It returns the posts_count");
|
||||
});
|
||||
|
||||
test("when all posts are selected and there is topic with a posts_count", function() {
|
||||
var testObj = buildTestObj({
|
||||
allPostsSelected: true,
|
||||
topic: Discourse.Topic.create({ posts_count: 3456 })
|
||||
});
|
||||
|
||||
equal(testObj.get('selectedPostsCount'), 3456, "It returns the topic's posts_count");
|
||||
});
|
122
vendor/assets/javascripts/run-qunit.js
vendored
Normal file
122
vendor/assets/javascripts/run-qunit.js
vendored
Normal file
|
@ -0,0 +1,122 @@
|
|||
// PhantomJS QUnit Test Runner
|
||||
|
||||
/*globals QUnit phantom*/
|
||||
|
||||
var args = phantom.args;
|
||||
if (args.length < 1 || args.length > 2) {
|
||||
console.log("Usage: " + phantom.scriptName + " <URL> <timeout>");
|
||||
phantom.exit(1);
|
||||
}
|
||||
|
||||
var fs = require('fs');
|
||||
function print(str) {
|
||||
fs.write('/dev/stdout', str, 'w');
|
||||
}
|
||||
|
||||
var page = require('webpage').create();
|
||||
|
||||
page.onConsoleMessage = function(msg) {
|
||||
if (msg.slice(0,8) === 'WARNING:') { return; }
|
||||
if (msg.slice(0,6) === 'DEBUG:') { return; }
|
||||
|
||||
// Hack to access the print method
|
||||
// If there's a better way to do this, please change
|
||||
if (msg.slice(0,6) === 'PRINT:') {
|
||||
print(msg.slice(7));
|
||||
return;
|
||||
}
|
||||
|
||||
console.log(msg);
|
||||
};
|
||||
|
||||
page.open(args[0], function(status) {
|
||||
if (status !== 'success') {
|
||||
console.error("Unable to access network");
|
||||
phantom.exit(1);
|
||||
} else {
|
||||
page.evaluate(logQUnit);
|
||||
|
||||
var timeout = parseInt(args[1] || 60000, 10);
|
||||
var start = Date.now();
|
||||
var interval = setInterval(function() {
|
||||
if (Date.now() > start + timeout) {
|
||||
console.error("Tests timed out");
|
||||
phantom.exit(124);
|
||||
} else {
|
||||
var qunitDone = page.evaluate(function() {
|
||||
return window.qunitDone;
|
||||
});
|
||||
|
||||
if (qunitDone) {
|
||||
clearInterval(interval);
|
||||
if (qunitDone.failed > 0) {
|
||||
phantom.exit(1);
|
||||
} else {
|
||||
phantom.exit();
|
||||
}
|
||||
}
|
||||
}
|
||||
}, 500);
|
||||
}
|
||||
});
|
||||
|
||||
function logQUnit() {
|
||||
var moduleErrors = [];
|
||||
var testErrors = [];
|
||||
var assertionErrors = [];
|
||||
|
||||
console.log("\nRunning: " + JSON.stringify(QUnit.urlParams) + "\n");
|
||||
|
||||
QUnit.moduleDone(function(context) {
|
||||
if (context.failed) {
|
||||
var msg = "Module Failed: " + context.name + "\n" + testErrors.join("\n");
|
||||
moduleErrors.push(msg);
|
||||
testErrors = [];
|
||||
}
|
||||
});
|
||||
|
||||
QUnit.testDone(function(context) {
|
||||
if (context.failed) {
|
||||
var msg = " Test Failed: " + context.name + assertionErrors.join(" ");
|
||||
testErrors.push(msg);
|
||||
assertionErrors = [];
|
||||
console.log('PRINT: F');
|
||||
} else {
|
||||
console.log('PRINT: .');
|
||||
}
|
||||
});
|
||||
|
||||
QUnit.log(function(context) {
|
||||
if (context.result) { return; }
|
||||
|
||||
var msg = "\n Assertion Failed:";
|
||||
if (context.message) {
|
||||
msg += " " + context.message;
|
||||
}
|
||||
|
||||
if (context.expected) {
|
||||
msg += "\n Expected: " + context.expected + ", Actual: " + context.actual;
|
||||
}
|
||||
|
||||
assertionErrors.push(msg);
|
||||
});
|
||||
|
||||
QUnit.done(function(context) {
|
||||
console.log('\n');
|
||||
|
||||
if (moduleErrors.length > 0) {
|
||||
for (var idx=0; idx<moduleErrors.length; idx++) {
|
||||
console.error(moduleErrors[idx]+"\n");
|
||||
}
|
||||
}
|
||||
|
||||
var stats = [
|
||||
"Time: " + context.runtime + "ms",
|
||||
"Total: " + context.total,
|
||||
"Passed: " + context.passed,
|
||||
"Failed: " + context.failed
|
||||
];
|
||||
console.log(stats.join(", "));
|
||||
window.qunitDone = context;
|
||||
});
|
||||
}
|
Loading…
Reference in New Issue
Block a user