require "test_helper" module Api class MessagesControllerTest < ActionDispatch::IntegrationTest ## # test all routes which lead to this controller def test_routes assert_routing( { :path => "/api/0.6/user/messages/inbox", :method => :get }, { :controller => "api/messages", :action => "inbox" } ) assert_routing( { :path => "/api/0.6/user/messages/inbox.xml", :method => :get }, { :controller => "api/messages", :action => "inbox", :format => "xml" } ) assert_routing( { :path => "/api/0.6/user/messages/inbox.json", :method => :get }, { :controller => "api/messages", :action => "inbox", :format => "json" } ) assert_routing( { :path => "/api/0.6/user/messages/outbox", :method => :get }, { :controller => "api/messages", :action => "outbox" } ) assert_routing( { :path => "/api/0.6/user/messages/outbox.xml", :method => :get }, { :controller => "api/messages", :action => "outbox", :format => "xml" } ) assert_routing( { :path => "/api/0.6/user/messages/outbox.json", :method => :get }, { :controller => "api/messages", :action => "outbox", :format => "json" } ) assert_routing( { :path => "/api/0.6/user/messages/1", :method => :get }, { :controller => "api/messages", :action => "show", :id => "1" } ) assert_routing( { :path => "/api/0.6/user/messages/1.xml", :method => :get }, { :controller => "api/messages", :action => "show", :id => "1", :format => "xml" } ) assert_routing( { :path => "/api/0.6/user/messages/1.json", :method => :get }, { :controller => "api/messages", :action => "show", :id => "1", :format => "json" } ) assert_routing( { :path => "/api/0.6/user/messages", :method => :post }, { :controller => "api/messages", :action => "create" } ) assert_routing( { :path => "/api/0.6/user/messages/1", :method => :post }, { :controller => "api/messages", :action => "update", :id => "1" } ) assert_routing( { :path => "/api/0.6/user/messages/1", :method => :delete }, { :controller => "api/messages", :action => "destroy", :id => "1" } ) end def test_create_success recipient = create(:user) sender = create(:user) sender_token = create(:oauth_access_token, :resource_owner_id => sender.id, :scopes => %w[send_messages consume_messages]) sender_auth = bearer_authorization_header(sender_token.token) msg = build(:message) assert_difference "Message.count", 1 do assert_difference "ActionMailer::Base.deliveries.size", 1 do perform_enqueued_jobs do post api_messages_path, :params => { :title => msg.title, :recipient_id => recipient.id, :body => msg.body, :format => "json" }, :headers => sender_auth assert_response :success end end end assert_equal "application/json", response.media_type js = ActiveSupport::JSON.decode(@response.body) jsm = js["message"] assert_not_nil jsm assert_not_nil jsm["id"] assert_equal sender.id, jsm["from_user_id"] assert_equal sender.display_name, jsm["from_display_name"] assert_equal recipient.id, jsm["to_user_id"] assert_equal recipient.display_name, jsm["to_display_name"] assert_equal msg.title, jsm["title"] assert_not_nil jsm["sent_on"] assert_equal !msg.from_user_visible, jsm["deleted"] assert_not jsm.key?("message_read") assert_equal "markdown", jsm["body_format"] assert_equal msg.body, jsm["body"] end def test_create_fail recipient = create(:user) sender = create(:user) sender_token = create(:oauth_access_token, :resource_owner_id => sender.id, :scopes => %w[send_messages consume_messages]) sender_auth = bearer_authorization_header(sender_token.token) assert_no_difference "Message.count" do assert_no_difference "ActionMailer::Base.deliveries.size" do perform_enqueued_jobs do post api_messages_path, :params => { :title => "Title", :recipient_id => recipient.id, :body => "body" } end end end assert_response :unauthorized assert_no_difference "Message.count" do assert_no_difference "ActionMailer::Base.deliveries.size" do perform_enqueued_jobs do post api_messages_path, :params => { :recipient_id => recipient.id, :body => "body" }, :headers => sender_auth end end end assert_response :bad_request assert_no_difference "Message.count" do assert_no_difference "ActionMailer::Base.deliveries.size" do perform_enqueued_jobs do post api_messages_path, :params => { :title => "Title", :body => "body" }, :headers => sender_auth end end end assert_response :bad_request assert_no_difference "Message.count" do assert_no_difference "ActionMailer::Base.deliveries.size" do perform_enqueued_jobs do post api_messages_path, :params => { :title => "Title", :recipient_id => recipient.id }, :headers => sender_auth end end end assert_response :bad_request end def test_show recipient = create(:user) sender = create(:user) user3 = create(:user) sender_token = create(:oauth_access_token, :resource_owner_id => sender.id, :scopes => %w[consume_messages]) sender_auth = bearer_authorization_header(sender_token.token) recipient_token = create(:oauth_access_token, :resource_owner_id => recipient.id, :scopes => %w[consume_messages]) recipient_auth = bearer_authorization_header(recipient_token.token) user3_token = create(:oauth_access_token, :resource_owner_id => user3.id, :scopes => %w[send_messages consume_messages]) user3_auth = bearer_authorization_header(user3_token.token) msg = create(:message, :unread, :sender => sender, :recipient => recipient) # fail if not authorized get api_message_path(:id => msg.id) assert_response :unauthorized # only recipient and sender can read the message get api_message_path(:id => msg.id), :headers => user3_auth assert_response :forbidden # message does not exist get api_message_path(:id => 99999), :headers => user3_auth assert_response :not_found # verify xml output get api_message_path(:id => msg.id), :headers => recipient_auth assert_equal "application/xml", response.media_type assert_select "message", :count => 1 do assert_select "[id='#{msg.id}']" assert_select "[from_user_id='#{sender.id}']" assert_select "[from_display_name='#{sender.display_name}']" assert_select "[to_user_id='#{recipient.id}']" assert_select "[to_display_name='#{recipient.display_name}']" assert_select "[sent_on]" assert_select "[deleted='#{!msg.to_user_visible}']" assert_select "[message_read='#{msg.message_read}']" assert_select "[body_format='markdown']" assert_select "title", msg.title assert_select "body", msg.body end # verify json output get api_message_path(:id => msg.id, :format => "json"), :headers => recipient_auth assert_equal "application/json", response.media_type js = ActiveSupport::JSON.decode(@response.body) jsm = js["message"] assert_not_nil jsm assert_equal msg.id, jsm["id"] assert_equal sender.id, jsm["from_user_id"] assert_equal sender.display_name, jsm["from_display_name"] assert_equal recipient.id, jsm["to_user_id"] assert_equal recipient.display_name, jsm["to_display_name"] assert_equal msg.title, jsm["title"] assert_not_nil jsm["sent_on"] assert_equal msg.message_read, jsm["message_read"] assert_equal !msg.to_user_visible, jsm["deleted"] assert_equal "markdown", jsm["body_format"] assert_equal msg.body, jsm["body"] get api_message_path(:id => msg.id), :headers => sender_auth assert_equal "application/xml", response.media_type assert_select "message", :count => 1 do assert_select "[id='#{msg.id}']" assert_select "[from_user_id='#{sender.id}']" assert_select "[from_display_name='#{sender.display_name}']" assert_select "[to_user_id='#{recipient.id}']" assert_select "[to_display_name='#{recipient.display_name}']" assert_select "[sent_on]" assert_select "[deleted='#{!msg.from_user_visible}']" assert_select "[message_read='#{msg.message_read}']", 0 assert_select "[body_format='markdown']" assert_select "title", msg.title assert_select "body", msg.body end # verify json output get api_message_path(:id => msg.id, :format => "json"), :headers => sender_auth assert_equal "application/json", response.media_type js = ActiveSupport::JSON.decode(@response.body) jsm = js["message"] assert_not_nil jsm assert_equal msg.id, jsm["id"] assert_equal sender.id, jsm["from_user_id"] assert_equal sender.display_name, jsm["from_display_name"] assert_equal recipient.id, jsm["to_user_id"] assert_equal recipient.display_name, jsm["to_display_name"] assert_equal msg.title, jsm["title"] assert_not_nil jsm["sent_on"] assert_equal !msg.from_user_visible, jsm["deleted"] assert_not jsm.key?("message_read") assert_equal "markdown", jsm["body_format"] assert_equal msg.body, jsm["body"] end def test_update_status recipient = create(:user) sender = create(:user) user3 = create(:user) recipient_token = create(:oauth_access_token, :resource_owner_id => recipient.id, :scopes => %w[consume_messages]) recipient_auth = bearer_authorization_header(recipient_token.token) user3_token = create(:oauth_access_token, :resource_owner_id => user3.id, :scopes => %w[send_messages consume_messages]) user3_auth = bearer_authorization_header(user3_token.token) msg = create(:message, :unread, :sender => sender, :recipient => recipient) # attempt to mark message as read by recipient, not authenticated post api_message_path(:id => msg.id), :params => { :read_status => true } assert_response :unauthorized # attempt to mark message as read by recipient, not allowed post api_message_path(:id => msg.id), :params => { :read_status => true }, :headers => user3_auth assert_response :forbidden # missing parameter post api_message_path(:id => msg.id), :headers => recipient_auth assert_response :bad_request # wrong type of parameter post api_message_path(:id => msg.id), :params => { :read_status => "not a boolean" }, :headers => recipient_auth assert_response :bad_request # mark message as read by recipient post api_message_path(:id => msg.id, :format => "json"), :params => { :read_status => true }, :headers => recipient_auth assert_response :success assert_equal "application/json", response.media_type js = ActiveSupport::JSON.decode(@response.body) jsm = js["message"] assert_not_nil jsm assert_equal msg.id, jsm["id"] assert_equal sender.id, jsm["from_user_id"] assert_equal sender.display_name, jsm["from_display_name"] assert_equal recipient.id, jsm["to_user_id"] assert_equal recipient.display_name, jsm["to_display_name"] assert_equal msg.title, jsm["title"] assert_not_nil jsm["sent_on"] assert jsm["message_read"] assert_equal !msg.to_user_visible, jsm["deleted"] assert_equal "markdown", jsm["body_format"] assert_equal msg.body, jsm["body"] # mark message as unread by recipient post api_message_path(:id => msg.id, :format => "json"), :params => { :read_status => false }, :headers => recipient_auth assert_response :success assert_equal "application/json", response.media_type js = ActiveSupport::JSON.decode(@response.body) jsm = js["message"] assert_not_nil jsm assert_equal msg.id, jsm["id"] assert_equal sender.id, jsm["from_user_id"] assert_equal sender.display_name, jsm["from_display_name"] assert_equal recipient.id, jsm["to_user_id"] assert_equal recipient.display_name, jsm["to_display_name"] assert_equal msg.title, jsm["title"] assert_not_nil jsm["sent_on"] assert_not jsm["message_read"] assert_equal !msg.to_user_visible, jsm["deleted"] assert_equal "markdown", jsm["body_format"] assert_equal msg.body, jsm["body"] end def test_delete recipient = create(:user) recipient_token = create(:oauth_access_token, :resource_owner_id => recipient.id, :scopes => %w[consume_messages]) recipient_auth = bearer_authorization_header(recipient_token.token) sender = create(:user) sender_token = create(:oauth_access_token, :resource_owner_id => sender.id, :scopes => %w[send_messages consume_messages]) sender_auth = bearer_authorization_header(sender_token.token) user3 = create(:user) user3_token = create(:oauth_access_token, :resource_owner_id => user3.id, :scopes => %w[send_messages consume_messages]) user3_auth = bearer_authorization_header(user3_token.token) msg = create(:message, :read, :sender => sender, :recipient => recipient) # attempt to delete message, not authenticated delete api_message_path(:id => msg.id) assert_response :unauthorized # attempt to delete message, by user3 delete api_message_path(:id => msg.id), :headers => user3_auth assert_response :forbidden # delete message by recipient delete api_message_path(:id => msg.id, :format => "json"), :headers => recipient_auth assert_response :success assert_equal "application/json", response.media_type js = ActiveSupport::JSON.decode(@response.body) jsm = js["message"] assert_not_nil jsm assert_equal msg.id, jsm["id"] assert_equal sender.id, jsm["from_user_id"] assert_equal sender.display_name, jsm["from_display_name"] assert_equal recipient.id, jsm["to_user_id"] assert_equal recipient.display_name, jsm["to_display_name"] assert_equal msg.title, jsm["title"] assert_not_nil jsm["sent_on"] assert_equal msg.message_read, jsm["message_read"] assert jsm["deleted"] assert_equal "markdown", jsm["body_format"] assert_equal msg.body, jsm["body"] # delete message by sender delete api_message_path(:id => msg.id, :format => "json"), :headers => sender_auth assert_response :success assert_equal "application/json", response.media_type js = ActiveSupport::JSON.decode(@response.body) jsm = js["message"] assert_not_nil jsm assert_equal msg.id, jsm["id"] assert_equal sender.id, jsm["from_user_id"] assert_equal sender.display_name, jsm["from_display_name"] assert_equal recipient.id, jsm["to_user_id"] assert_equal recipient.display_name, jsm["to_display_name"] assert_equal msg.title, jsm["title"] assert_not_nil jsm["sent_on"] assert jsm["deleted"] assert_not jsm.key?("message_read") assert_equal "markdown", jsm["body_format"] assert_equal msg.body, jsm["body"] end def test_list_messages user1 = create(:user) user1_token = create(:oauth_access_token, :resource_owner_id => user1.id, :scopes => %w[send_messages consume_messages]) user1_auth = bearer_authorization_header(user1_token.token) user2 = create(:user) user2_token = create(:oauth_access_token, :resource_owner_id => user2.id, :scopes => %w[send_messages consume_messages]) user2_auth = bearer_authorization_header(user2_token.token) user3 = create(:user) user3_token = create(:oauth_access_token, :resource_owner_id => user3.id, :scopes => %w[send_messages consume_messages]) user3_auth = bearer_authorization_header(user3_token.token) # create some messages between users # user | inbox | outbox # 1 | 0 | 3 # 2 | 2 | 1 # 3 | 2 | 0 create(:message, :unread, :sender => user1, :recipient => user2) create(:message, :unread, :sender => user1, :recipient => user2) create(:message, :unread, :sender => user1, :recipient => user3) create(:message, :unread, :sender => user2, :recipient => user3) # only authorized users get inbox_api_messages_path assert_response :unauthorized get outbox_api_messages_path assert_response :unauthorized # no messages in user1.inbox get inbox_api_messages_path, :headers => user1_auth assert_response :success assert_equal "application/xml", response.media_type assert_select "message", :count => 0 # 3 messages in user1.outbox get outbox_api_messages_path, :headers => user1_auth assert_response :success assert_equal "application/xml", response.media_type assert_select "message", :count => 3 do assert_select "[from_user_id='#{user1.id}']" assert_select "[from_display_name='#{user1.display_name}']" assert_select "[to_user_id]" assert_select "[to_display_name]" assert_select "[sent_on]" assert_select "[message_read]", 0 assert_select "[deleted='false']" assert_select "[body_format]" assert_select "body", false assert_select "title" end # 2 messages in user2.inbox get inbox_api_messages_path, :headers => user2_auth assert_response :success assert_equal "application/xml", response.media_type assert_select "message", :count => 2 do assert_select "[from_user_id]" assert_select "[from_display_name]" assert_select "[to_user_id='#{user2.id}']" assert_select "[to_display_name='#{user2.display_name}']" assert_select "[sent_on]" assert_select "[message_read='false']" assert_select "[deleted='false']" assert_select "[body_format]" assert_select "body", false assert_select "title" end # 1 message in user2.outbox get outbox_api_messages_path, :headers => user2_auth assert_response :success assert_equal "application/xml", response.media_type assert_select "message", :count => 1 do assert_select "[from_user_id='#{user2.id}']" assert_select "[from_display_name='#{user2.display_name}']" assert_select "[to_user_id]" assert_select "[to_display_name]" assert_select "[sent_on]" assert_select "[deleted='false']" assert_select "[message_read]", 0 assert_select "[body_format]" assert_select "body", false assert_select "title" end # 2 messages in user3.inbox get inbox_api_messages_path, :headers => user3_auth assert_response :success assert_equal "application/xml", response.media_type assert_select "message", :count => 2 do assert_select "[from_user_id]" assert_select "[from_display_name]" assert_select "[to_user_id='#{user3.id}']" assert_select "[to_display_name='#{user3.display_name}']" assert_select "[sent_on]" assert_select "[message_read='false']" assert_select "[deleted='false']" assert_select "[body_format]" assert_select "body", false assert_select "title" end # 0 messages in user3.outbox get outbox_api_messages_path, :headers => user3_auth assert_response :success assert_equal "application/xml", response.media_type assert_select "message", :count => 0 end def test_paged_list_messages_asc recipient = create(:user) recipient_token = create(:oauth_access_token, :resource_owner_id => recipient.id, :scopes => %w[consume_messages]) recipient_auth = bearer_authorization_header(recipient_token.token) sender = create(:user) create_list(:message, 100, :unread, :sender => sender, :recipient => recipient) msgs_read = {} params = { :order => "oldest", :limit => 20 } 10.times do get inbox_api_messages_path(:format => "json"), :params => params, :headers => recipient_auth assert_response :success assert_equal "application/json", response.media_type js = ActiveSupport::JSON.decode(@response.body) jsm = js["messages"] assert_operator jsm.count, :<=, 20 break if jsm.nil? || jsm.count.zero? assert_operator(jsm[0]["id"], :>=, params[:from_id]) unless params[:from_id].nil? # ensure ascending order (0..jsm.count - 1).each do |i| assert_operator(jsm[i]["id"], :<, jsm[i + 1]["id"]) unless i == jsm.count - 1 msgs_read[jsm[i]["id"]] = jsm[i] end params[:from_id] = jsm[jsm.count - 1]["id"] end assert_equal 100, msgs_read.count end def test_paged_list_messages_desc recipient = create(:user) recipient_token = create(:oauth_access_token, :resource_owner_id => recipient.id, :scopes => %w[consume_messages]) recipient_auth = bearer_authorization_header(recipient_token.token) sender = create(:user) create_list(:message, 100, :unread, :sender => sender, :recipient => recipient) real_max_id = -1 msgs_read = {} params = { :order => "newest", :limit => 20 } 10.times do get inbox_api_messages_path(:format => "json"), :params => params, :headers => recipient_auth assert_response :success assert_equal "application/json", response.media_type js = ActiveSupport::JSON.decode(@response.body) jsm = js["messages"] assert_operator jsm.count, :<=, 20 break if jsm.nil? || jsm.count.zero? if params[:from_id].nil? real_max_id = jsm[0]["id"] else assert_operator jsm[0]["id"], :<=, params[:from_id] end # ensure descending order (0..jsm.count - 1).each do |i| assert_operator(jsm[i]["id"], :>, jsm[i + 1]["id"]) unless i == jsm.count - 1 msgs_read[jsm[i]["id"]] = jsm[i] end params[:from_id] = jsm[jsm.count - 1]["id"] end assert_equal 100, msgs_read.count assert_not_equal(-1, real_max_id) # invoke without min_id/max_id parameters, verify that we get the last batch get inbox_api_messages_path(:format => "json"), :params => { :limit => 20 }, :headers => recipient_auth assert_response :success assert_equal "application/json", response.media_type js = ActiveSupport::JSON.decode(@response.body) jsm = js["messages"] assert_not_nil jsm assert_equal real_max_id, jsm[0]["id"] end end end