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
This commit is contained in:
parent
f484fe8556
commit
345301284a
|
@ -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
|
||||
|
|
|
@ -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
|
||||
|
||||
|
|
|
@ -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
|
||||
|
|
Loading…
Reference in New Issue