pull in jasmine-ajax, start testing our ajax callbacks with meaningful responses
This commit is contained in:
parent
a5cbf59c6a
commit
2b31999f28
5 changed files with 271 additions and 14 deletions
|
|
@ -128,6 +128,14 @@ describe CommentsController do
|
||||||
@message = bob.post(:status_message, :text => "hey", :to => aspect_to_post.id)
|
@message = bob.post(:status_message, :text => "hey", :to => aspect_to_post.id)
|
||||||
@comments = [alice, bob, eve].map{ |u| u.comment("hey", :post => @message) }
|
@comments = [alice, bob, eve].map{ |u| u.comment("hey", :post => @message) }
|
||||||
end
|
end
|
||||||
|
|
||||||
|
it 'generates a jasmine fixture', :fixture => true do
|
||||||
|
2.times { alice.comment("hey", :post => @message) }
|
||||||
|
get :index, :post_id => @message.id
|
||||||
|
|
||||||
|
save_fixture(response.body, "ajax_comments_on_post")
|
||||||
|
end
|
||||||
|
|
||||||
it 'works for mobile' do
|
it 'works for mobile' do
|
||||||
get :index, :post_id => @message.id, :format => 'mobile'
|
get :index, :post_id => @message.id, :format => 'mobile'
|
||||||
response.should be_success
|
response.should be_success
|
||||||
|
|
|
||||||
|
|
@ -83,6 +83,13 @@ describe LikesController do
|
||||||
@message = alice.post(:status_message, :text => "hey", :to => @alices_aspect.id)
|
@message = alice.post(:status_message, :text => "hey", :to => @alices_aspect.id)
|
||||||
@message = alice.comment( "hey", :post => @message) if class_const == Comment
|
@message = alice.comment( "hey", :post => @message) if class_const == Comment
|
||||||
end
|
end
|
||||||
|
|
||||||
|
it 'generates a jasmine fixture', :fixture => true do
|
||||||
|
get :index, id_field => @message.id
|
||||||
|
|
||||||
|
save_fixture(response.body, "ajax_likes_on_#{class_const.to_s.underscore}")
|
||||||
|
end
|
||||||
|
|
||||||
it 'returns a 404 for a post not visible to the user' do
|
it 'returns a 404 for a post not visible to the user' do
|
||||||
sign_in eve
|
sign_in eve
|
||||||
get :index, id_field => @message.id
|
get :index, id_field => @message.id
|
||||||
|
|
|
||||||
207
spec/javascripts/helpers/mock-ajax.js
Normal file
207
spec/javascripts/helpers/mock-ajax.js
Normal file
|
|
@ -0,0 +1,207 @@
|
||||||
|
/*
|
||||||
|
Jasmine-Ajax : a set of helpers for testing AJAX requests under the Jasmine
|
||||||
|
BDD framework for JavaScript.
|
||||||
|
|
||||||
|
Supports both Prototype.js and jQuery.
|
||||||
|
|
||||||
|
http://github.com/pivotal/jasmine-ajax
|
||||||
|
|
||||||
|
Jasmine Home page: http://pivotal.github.com/jasmine
|
||||||
|
|
||||||
|
Copyright (c) 2008-2010 Pivotal Labs
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining
|
||||||
|
a copy of this software and associated documentation files (the
|
||||||
|
"Software"), to deal in the Software without restriction, including
|
||||||
|
without limitation the rights to use, copy, modify, merge, publish,
|
||||||
|
distribute, sublicense, and/or sell copies of the Software, and to
|
||||||
|
permit persons to whom the Software is furnished to do so, subject to
|
||||||
|
the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be
|
||||||
|
included in all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||||
|
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||||||
|
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||||
|
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||||||
|
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
// Jasmine-Ajax interface
|
||||||
|
var ajaxRequests = [];
|
||||||
|
|
||||||
|
function mostRecentAjaxRequest() {
|
||||||
|
if (ajaxRequests.length > 0) {
|
||||||
|
return ajaxRequests[ajaxRequests.length - 1];
|
||||||
|
} else {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function clearAjaxRequests() {
|
||||||
|
ajaxRequests = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fake XHR for mocking Ajax Requests & Responses
|
||||||
|
function FakeXMLHttpRequest() {
|
||||||
|
var extend = Object.extend || $.extend;
|
||||||
|
extend(this, {
|
||||||
|
requestHeaders: {},
|
||||||
|
|
||||||
|
open: function() {
|
||||||
|
this.method = arguments[0];
|
||||||
|
this.url = arguments[1];
|
||||||
|
this.readyState = 1;
|
||||||
|
},
|
||||||
|
|
||||||
|
setRequestHeader: function(header, value) {
|
||||||
|
this.requestHeaders[header] = value;
|
||||||
|
},
|
||||||
|
|
||||||
|
abort: function() {
|
||||||
|
this.readyState = 0;
|
||||||
|
},
|
||||||
|
|
||||||
|
readyState: 0,
|
||||||
|
|
||||||
|
onreadystatechange: function(isTimeout) {
|
||||||
|
},
|
||||||
|
|
||||||
|
status: null,
|
||||||
|
|
||||||
|
send: function(data) {
|
||||||
|
this.params = data;
|
||||||
|
this.readyState = 2;
|
||||||
|
},
|
||||||
|
|
||||||
|
getResponseHeader: function(name) {
|
||||||
|
return this.responseHeaders[name];
|
||||||
|
},
|
||||||
|
|
||||||
|
getAllResponseHeaders: function() {
|
||||||
|
var responseHeaders = [];
|
||||||
|
for (var i in this.responseHeaders) {
|
||||||
|
if (this.responseHeaders.hasOwnProperty(i)) {
|
||||||
|
responseHeaders.push(i + ': ' + this.responseHeaders[i]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return responseHeaders.join('\r\n');
|
||||||
|
},
|
||||||
|
|
||||||
|
responseText: null,
|
||||||
|
|
||||||
|
response: function(response) {
|
||||||
|
this.status = response.status;
|
||||||
|
this.responseText = response.responseText || "";
|
||||||
|
this.readyState = 4;
|
||||||
|
this.responseHeaders = response.responseHeaders ||
|
||||||
|
{"Content-type": response.contentType || "application/json" };
|
||||||
|
// uncomment for jquery 1.3.x support
|
||||||
|
// jasmine.Clock.tick(20);
|
||||||
|
|
||||||
|
this.onreadystatechange();
|
||||||
|
},
|
||||||
|
responseTimeout: function() {
|
||||||
|
this.readyState = 4;
|
||||||
|
jasmine.Clock.tick(jQuery.ajaxSettings.timeout || 30000);
|
||||||
|
this.onreadystatechange('timeout');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
jasmine.Ajax = {
|
||||||
|
|
||||||
|
isInstalled: function() {
|
||||||
|
return jasmine.Ajax.installed == true;
|
||||||
|
},
|
||||||
|
|
||||||
|
assertInstalled: function() {
|
||||||
|
if (!jasmine.Ajax.isInstalled()) {
|
||||||
|
throw new Error("Mock ajax is not installed, use jasmine.Ajax.useMock()")
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
useMock: function() {
|
||||||
|
if (!jasmine.Ajax.isInstalled()) {
|
||||||
|
var spec = jasmine.getEnv().currentSpec;
|
||||||
|
spec.after(jasmine.Ajax.uninstallMock);
|
||||||
|
|
||||||
|
jasmine.Ajax.installMock();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
installMock: function() {
|
||||||
|
if (typeof jQuery != 'undefined') {
|
||||||
|
jasmine.Ajax.installJquery();
|
||||||
|
} else if (typeof Prototype != 'undefined') {
|
||||||
|
jasmine.Ajax.installPrototype();
|
||||||
|
} else {
|
||||||
|
throw new Error("jasmine.Ajax currently only supports jQuery and Prototype");
|
||||||
|
}
|
||||||
|
jasmine.Ajax.installed = true;
|
||||||
|
},
|
||||||
|
|
||||||
|
installJquery: function() {
|
||||||
|
jasmine.Ajax.mode = 'jQuery';
|
||||||
|
jasmine.Ajax.real = jQuery.ajaxSettings.xhr;
|
||||||
|
jQuery.ajaxSettings.xhr = jasmine.Ajax.jQueryMock;
|
||||||
|
|
||||||
|
},
|
||||||
|
|
||||||
|
installPrototype: function() {
|
||||||
|
jasmine.Ajax.mode = 'Prototype';
|
||||||
|
jasmine.Ajax.real = Ajax.getTransport;
|
||||||
|
|
||||||
|
Ajax.getTransport = jasmine.Ajax.prototypeMock;
|
||||||
|
},
|
||||||
|
|
||||||
|
uninstallMock: function() {
|
||||||
|
jasmine.Ajax.assertInstalled();
|
||||||
|
if (jasmine.Ajax.mode == 'jQuery') {
|
||||||
|
jQuery.ajaxSettings.xhr = jasmine.Ajax.real;
|
||||||
|
} else if (jasmine.Ajax.mode == 'Prototype') {
|
||||||
|
Ajax.getTransport = jasmine.Ajax.real;
|
||||||
|
}
|
||||||
|
jasmine.Ajax.reset();
|
||||||
|
},
|
||||||
|
|
||||||
|
reset: function() {
|
||||||
|
jasmine.Ajax.installed = false;
|
||||||
|
jasmine.Ajax.mode = null;
|
||||||
|
jasmine.Ajax.real = null;
|
||||||
|
},
|
||||||
|
|
||||||
|
jQueryMock: function() {
|
||||||
|
var newXhr = new FakeXMLHttpRequest();
|
||||||
|
ajaxRequests.push(newXhr);
|
||||||
|
return newXhr;
|
||||||
|
},
|
||||||
|
|
||||||
|
prototypeMock: function() {
|
||||||
|
return new FakeXMLHttpRequest();
|
||||||
|
},
|
||||||
|
|
||||||
|
installed: false,
|
||||||
|
mode: null
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// Jasmine-Ajax Glue code for Prototype.js
|
||||||
|
if (typeof Prototype != 'undefined' && Ajax && Ajax.Request) {
|
||||||
|
Ajax.Request.prototype.originalRequest = Ajax.Request.prototype.request;
|
||||||
|
Ajax.Request.prototype.request = function(url) {
|
||||||
|
this.originalRequest(url);
|
||||||
|
ajaxRequests.push(this);
|
||||||
|
};
|
||||||
|
|
||||||
|
Ajax.Request.prototype.response = function(responseOptions) {
|
||||||
|
return this.transport.response(responseOptions);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
@ -1,7 +1,11 @@
|
||||||
describe("Diaspora.Widgets.CommentStream", function() {
|
describe("Diaspora.Widgets.CommentStream", function() {
|
||||||
var commentStream;
|
var commentStream;
|
||||||
beforeEach(function() {
|
beforeEach(function() {
|
||||||
|
spec.readFixture("ajax_comments_on_post");
|
||||||
|
|
||||||
jasmine.Clock.useMock();
|
jasmine.Clock.useMock();
|
||||||
|
jasmine.Ajax.useMock();
|
||||||
|
|
||||||
spec.loadFixture("aspects_index_with_posts");
|
spec.loadFixture("aspects_index_with_posts");
|
||||||
Diaspora.I18n.locale = { };
|
Diaspora.I18n.locale = { };
|
||||||
|
|
||||||
|
|
@ -13,7 +17,21 @@ describe("Diaspora.Widgets.CommentStream", function() {
|
||||||
|
|
||||||
describe("toggling comments", function() {
|
describe("toggling comments", function() {
|
||||||
it("toggles class hidden on the comments ul", function () {
|
it("toggles class hidden on the comments ul", function () {
|
||||||
spyOn($, "ajax");
|
spyOn($, "ajax").andCallThrough();
|
||||||
|
|
||||||
|
expect($("ul.comments:first")).not.toHaveClass("hidden");
|
||||||
|
|
||||||
|
commentStream.showComments($.Event());
|
||||||
|
|
||||||
|
mostRecentAjaxRequest().response({
|
||||||
|
responseHeaders: {
|
||||||
|
"Content-type": "text/html"
|
||||||
|
},
|
||||||
|
responseText: spec.readFixture("ajax_comments_on_post"),
|
||||||
|
status: 200
|
||||||
|
});
|
||||||
|
|
||||||
|
jasmine.Clock.tick(200);
|
||||||
|
|
||||||
expect($("ul.comments:first")).not.toHaveClass("hidden");
|
expect($("ul.comments:first")).not.toHaveClass("hidden");
|
||||||
|
|
||||||
|
|
@ -22,16 +40,10 @@ describe("Diaspora.Widgets.CommentStream", function() {
|
||||||
jasmine.Clock.tick(200);
|
jasmine.Clock.tick(200);
|
||||||
|
|
||||||
expect($("ul.comments:first")).toHaveClass("hidden");
|
expect($("ul.comments:first")).toHaveClass("hidden");
|
||||||
|
|
||||||
commentStream.showComments($.Event());
|
|
||||||
|
|
||||||
$.ajax.mostRecentCall.args[0].success("comment response");
|
|
||||||
jasmine.Clock.tick(200);
|
|
||||||
|
|
||||||
expect($("ul.comments:first")).not.toHaveClass("hidden");
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it("changes the text on the show comments link", function() {
|
it("changes the text on the show comments link", function() {
|
||||||
|
spyOn($, "ajax").andCallThrough();
|
||||||
Diaspora.I18n.loadLocale({'comments' : {
|
Diaspora.I18n.loadLocale({'comments' : {
|
||||||
'show': 'show comments translation',
|
'show': 'show comments translation',
|
||||||
'hide': 'hide comments translation'
|
'hide': 'hide comments translation'
|
||||||
|
|
@ -41,6 +53,14 @@ describe("Diaspora.Widgets.CommentStream", function() {
|
||||||
|
|
||||||
commentStream.showComments($.Event());
|
commentStream.showComments($.Event());
|
||||||
|
|
||||||
|
mostRecentAjaxRequest().response({
|
||||||
|
responseHeaders: {
|
||||||
|
"Content-type": "text/html"
|
||||||
|
},
|
||||||
|
responseText: spec.readFixture("ajax_comments_on_post"),
|
||||||
|
status: 200
|
||||||
|
});
|
||||||
|
|
||||||
jasmine.Clock.tick(200);
|
jasmine.Clock.tick(200);
|
||||||
|
|
||||||
expect(link.text()).toEqual("hide comments translation");
|
expect(link.text()).toEqual("hide comments translation");
|
||||||
|
|
@ -53,13 +73,20 @@ describe("Diaspora.Widgets.CommentStream", function() {
|
||||||
});
|
});
|
||||||
|
|
||||||
it("only requests the comments when the loaded class is not present", function() {
|
it("only requests the comments when the loaded class is not present", function() {
|
||||||
spyOn($, "ajax");
|
spyOn($, "ajax").andCallThrough();
|
||||||
|
|
||||||
expect(commentStream.commentsList).not.toHaveClass("loaded");
|
expect(commentStream.commentsList).not.toHaveClass("loaded");
|
||||||
|
|
||||||
commentStream.showComments($.Event());
|
commentStream.showComments($.Event());
|
||||||
|
|
||||||
$.ajax.mostRecentCall.args[0].success("comment response");
|
mostRecentAjaxRequest().response({
|
||||||
|
responseHeaders: {
|
||||||
|
"Content-type": "text/html"
|
||||||
|
},
|
||||||
|
responseText: spec.readFixture("ajax_comments_on_post"),
|
||||||
|
status: 200
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
expect($.ajax.callCount).toEqual(1);
|
expect($.ajax.callCount).toEqual(1);
|
||||||
expect(commentStream.commentsList).toHaveClass("loaded");
|
expect(commentStream.commentsList).toHaveClass("loaded");
|
||||||
|
|
|
||||||
|
|
@ -1,8 +1,10 @@
|
||||||
describe("Diaspora.Widgets.Likes", function() {
|
describe("Diaspora.Widgets.Likes", function() {
|
||||||
var likes;
|
var likes;
|
||||||
beforeEach(function() {
|
beforeEach(function() {
|
||||||
|
spec.readFixture("ajax_likes_on_post");
|
||||||
spec.loadFixture("aspects_index_with_a_post_with_likes");
|
spec.loadFixture("aspects_index_with_a_post_with_likes");
|
||||||
likes = Diaspora.BaseWidget.instantiate("Likes", $(".stream_element .likes_container"));
|
likes = Diaspora.BaseWidget.instantiate("Likes", $(".stream_element .likes_container"));
|
||||||
|
jasmine.Ajax.useMock();
|
||||||
$.fx.off = true;
|
$.fx.off = true;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
@ -42,13 +44,19 @@ describe("Diaspora.Widgets.Likes", function() {
|
||||||
});
|
});
|
||||||
|
|
||||||
it("makes the likes list's html the response of the request", function() {
|
it("makes the likes list's html the response of the request", function() {
|
||||||
spyOn($, "ajax");
|
spyOn($, "ajax").andCallThrough();
|
||||||
|
|
||||||
likes.expandLikes($.Event());
|
likes.expandLikes($.Event());
|
||||||
|
|
||||||
$.ajax.mostRecentCall.args[0].success("some html response");
|
mostRecentAjaxRequest().response({
|
||||||
|
responseHeaders: {
|
||||||
|
"Content-type": "text/html"
|
||||||
|
},
|
||||||
|
responseText: spec.readFixture("ajax_likes_on_post"),
|
||||||
|
status: 200
|
||||||
|
});
|
||||||
|
|
||||||
expect($(".stream_element .likes_list").html()).toEqual("some html response");
|
expect($(".stream_element .likes_list").html()).toEqual(spec.readFixture("ajax_likes_on_post"));
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
||||||
Loading…
Reference in a new issue