From 345301284a9b47d98bf7af8c09d1c6301b9a2a81 Mon Sep 17 00:00:00 2001 From: Eric Davis Date: Sat, 5 Jun 2010 03:52:59 +0000 Subject: [PATCH] Added JSON support to the issues API. #1214 git-svn-id: svn+ssh://rubyforge.org/var/svn/redmine/trunk@3766 e93f8b46-1217-0410-a6f0-8f06a7374b81 --- app/controllers/application_controller.rb | 7 + app/controllers/issues_controller.rb | 9 +- test/integration/issues_api_test.rb | 158 +++++++++++++++++++++- 3 files changed, 170 insertions(+), 4 deletions(-) diff --git a/app/controllers/application_controller.rb b/app/controllers/application_controller.rb index 54339b4e..90912547 100644 --- a/app/controllers/application_controller.rb +++ b/app/controllers/application_controller.rb @@ -349,4 +349,11 @@ class ApplicationController < ActionController::Base render_error "An error occurred while executing the query and has been logged. Please report this error to your Redmine administrator." end + # Converts the errors on an ActiveRecord object into a common JSON format + def object_errors_to_json(object) + object.errors.collect do |attribute, error| + { attribute => error } + end.to_json + end + end diff --git a/app/controllers/issues_controller.rb b/app/controllers/issues_controller.rb index 6b42ce62..8b5d73fa 100644 --- a/app/controllers/issues_controller.rb +++ b/app/controllers/issues_controller.rb @@ -82,6 +82,7 @@ class IssuesController < ApplicationController respond_to do |format| format.html { render :template => 'issues/index.rhtml', :layout => !request.xhr? } format.xml { render :layout => false } + format.json { render :text => @issues.to_json, :layout => false } format.atom { render_feed(@issues, :title => "#{@project || Setting.app_title}: #{l(:label_issue_plural)}") } format.csv { send_data(issues_to_csv(@issues, @project), :type => 'text/csv; header=present', :filename => 'export.csv') } format.pdf { send_data(issues_to_pdf(@issues, @project, @query), :type => 'application/pdf', :filename => 'export.pdf') } @@ -122,6 +123,7 @@ class IssuesController < ApplicationController respond_to do |format| format.html { render :template => 'issues/show.rhtml' } format.xml { render :layout => false } + format.json { render :text => @issue.to_json, :layout => false } format.atom { render :action => 'changes', :layout => false, :content_type => 'application/atom+xml' } format.pdf { send_data(issue_to_pdf(@issue), :type => 'application/pdf', :filename => "#{@project.identifier}-#{@issue.id}.pdf") } end @@ -146,12 +148,14 @@ class IssuesController < ApplicationController { :action => 'show', :id => @issue }) } format.xml { render :action => 'show', :status => :created, :location => url_for(:controller => 'issues', :action => 'show', :id => @issue) } + format.json { render :text => @issue.to_json, :status => :created, :location => url_for(:controller => 'issues', :action => 'show'), :layout => false } end return else respond_to do |format| format.html { render :action => 'new' } format.xml { render(:xml => @issue.errors, :status => :unprocessable_entity); return } + format.json { render :text => object_errors_to_json(@issue), :status => :unprocessable_entity, :layout => false } end end end @@ -181,6 +185,7 @@ class IssuesController < ApplicationController respond_to do |format| format.html { redirect_back_or_default({:action => 'show', :id => @issue}) } format.xml { head :ok } + format.json { head :ok } end else render_attachment_warning_if_needed(@issue) @@ -190,6 +195,7 @@ class IssuesController < ApplicationController respond_to do |format| format.html { render :action => 'edit' } format.xml { render :xml => @issue.errors, :status => :unprocessable_entity } + format.json { render :text => object_errors_to_json(@issue), :status => :unprocessable_entity, :layout => false } end end end @@ -305,7 +311,7 @@ class IssuesController < ApplicationController TimeEntry.update_all("issue_id = #{reassign_to.id}", ['issue_id IN (?)', @issues]) end else - unless params[:format] == 'xml' + unless params[:format] == 'xml' || params[:format] == 'json' # display the destroy form if it's a user request return end @@ -315,6 +321,7 @@ class IssuesController < ApplicationController respond_to do |format| format.html { redirect_to :action => 'index', :project_id => @project } format.xml { head :ok } + format.json { head :ok } end end diff --git a/test/integration/issues_api_test.rb b/test/integration/issues_api_test.rb index ab148983..a1e2cb70 100644 --- a/test/integration/issues_api_test.rb +++ b/test/integration/issues_api_test.rb @@ -54,7 +54,20 @@ class IssuesApiTest < ActionController::IntegrationTest should_respond_with :success should_respond_with_content_type 'application/xml' end - + + context "/index.json" do + setup do + get '/issues.json' + end + + should_respond_with :success + should_respond_with_content_type 'application/json' + + should 'return a valid JSON string' do + assert ActiveSupport::JSON.decode(response.body) + end + end + context "/index.xml with filter" do setup do get '/issues.xml?status_id=5' @@ -69,6 +82,27 @@ class IssuesApiTest < ActionController::IntegrationTest end end + context "/index.json with filter" do + setup do + get '/issues.json?status_id=5' + end + + should_respond_with :success + should_respond_with_content_type 'application/json' + + should 'return a valid JSON string' do + assert ActiveSupport::JSON.decode(response.body) + end + + should "show only issues with the status_id" do + json = ActiveSupport::JSON.decode(response.body) + status_ids_used = json.collect {|j| j['status_id'] } + assert_equal 3, status_ids_used.length + assert status_ids_used.all? {|id| id == 5 } + end + + end + context "/issues/1.xml" do setup do get '/issues/1.xml' @@ -78,6 +112,19 @@ class IssuesApiTest < ActionController::IntegrationTest should_respond_with_content_type 'application/xml' end + context "/issues/1.json" do + setup do + get '/issues/1.json' + end + + should_respond_with :success + should_respond_with_content_type 'application/json' + + should 'return a valid JSON string' do + assert ActiveSupport::JSON.decode(response.body) + end + end + context "POST /issues.xml" do setup do @issue_count = Issue.count @@ -111,7 +158,42 @@ class IssuesApiTest < ActionController::IntegrationTest assert_tag :errors, :child => {:tag => 'error', :content => "Subject can't be blank"} end end - + + context "POST /issues.json" do + setup do + @issue_count = Issue.count + @attributes = {:project_id => 1, :subject => 'API test', :tracker_id => 2, :status_id => 3} + post '/issues.json', {:issue => @attributes}, :authorization => credentials('jsmith') + end + + should_respond_with :created + should_respond_with_content_type 'application/json' + + should "create an issue with the attributes" do + assert_equal Issue.count, @issue_count + 1 + + issue = Issue.first(:order => 'id DESC') + @attributes.each do |attribute, value| + assert_equal value, issue.send(attribute) + end + end + end + + context "POST /issues.json with failure" do + setup do + @attributes = {:project_id => 1} + post '/issues.json', {:issue => @attributes}, :authorization => credentials('jsmith') + end + + should_respond_with :unprocessable_entity + should_respond_with_content_type 'application/json' + + should "have an errors element" do + json = ActiveSupport::JSON.decode(response.body) + assert_equal "can't be blank", json.first['subject'] + end + end + context "PUT /issues/1.xml" do setup do @issue_count = Issue.count @@ -166,6 +248,61 @@ class IssuesApiTest < ActionController::IntegrationTest end end + context "PUT /issues/1.json" do + setup do + @issue_count = Issue.count + @journal_count = Journal.count + @attributes = {:subject => 'API update'} + + put '/issues/1.json', {:issue => @attributes}, :authorization => credentials('jsmith') + end + + should_respond_with :ok + should_respond_with_content_type 'application/json' + + should "not create a new issue" do + assert_equal Issue.count, @issue_count + end + + should "create a new journal" do + assert_equal Journal.count, @journal_count + 1 + end + + should "update the issue" do + issue = Issue.find(1) + @attributes.each do |attribute, value| + assert_equal value, issue.send(attribute) + end + end + + end + + context "PUT /issues/1.json with failed update" do + setup do + @attributes = {:subject => ''} + @issue_count = Issue.count + @journal_count = Journal.count + + put '/issues/1.json', {:issue => @attributes}, :authorization => credentials('jsmith') + end + + should_respond_with :unprocessable_entity + should_respond_with_content_type 'application/json' + + should "not create a new issue" do + assert_equal Issue.count, @issue_count + end + + should "not create a new journal" do + assert_equal Journal.count, @journal_count + end + + should "have an errors attribute" do + json = ActiveSupport::JSON.decode(response.body) + assert_equal "can't be blank", json.first['subject'] + end + end + context "DELETE /issues/1.xml" do setup do @issue_count = Issue.count @@ -180,7 +317,22 @@ class IssuesApiTest < ActionController::IntegrationTest assert_nil Issue.find_by_id(1) end end - + + context "DELETE /issues/1.json" do + setup do + @issue_count = Issue.count + delete '/issues/1.json', {}, :authorization => credentials('jsmith') + end + + should_respond_with :ok + should_respond_with_content_type 'application/json' + + should "delete the issue" do + assert_equal Issue.count, @issue_count -1 + assert_nil Issue.find_by_id(1) + end + end + def credentials(user, password=nil) ActionController::HttpAuthentication::Basic.encode_credentials(user, password || user) end