diff options
author | Jean-Philippe Lang <jp_lang@yahoo.fr> | 2012-10-25 19:44:04 +0000 |
---|---|---|
committer | Jean-Philippe Lang <jp_lang@yahoo.fr> | 2012-10-25 19:44:04 +0000 |
commit | 9e313087205721ed3c6239be7cde55598d29f9e6 (patch) | |
tree | c5ba4ab52453907afae728887b470f95008bbbdb | |
parent | b2e035fa96e4c2b06dcd559850d11987ba96ee1c (diff) | |
download | redmine-9e313087205721ed3c6239be7cde55598d29f9e6.tar.gz redmine-9e313087205721ed3c6239be7cde55598d29f9e6.zip |
REST API for retrieving wiki pages (#7082).
git-svn-id: svn+ssh://rubyforge.org/var/svn/redmine/trunk@10716 e93f8b46-1217-0410-a6f0-8f06a7374b81
-rw-r--r-- | app/controllers/wiki_controller.rb | 21 | ||||
-rw-r--r-- | app/models/wiki_page.rb | 2 | ||||
-rw-r--r-- | app/views/wiki/index.api.rsb | 13 | ||||
-rw-r--r-- | app/views/wiki/show.api.rsb | 11 | ||||
-rw-r--r-- | test/functional/wiki_controller_test.rb | 7 | ||||
-rw-r--r-- | test/integration/api_test/wiki_pages_test.rb | 85 | ||||
-rw-r--r-- | test/integration/routing/wiki_test.rb | 28 | ||||
-rw-r--r-- | test/unit/wiki_page_test.rb | 1 |
8 files changed, 161 insertions, 7 deletions
diff --git a/app/controllers/wiki_controller.rb b/app/controllers/wiki_controller.rb index 6e747ecda..e8f6ef35d 100644 --- a/app/controllers/wiki_controller.rb +++ b/app/controllers/wiki_controller.rb @@ -36,6 +36,7 @@ class WikiController < ApplicationController before_filter :find_wiki, :authorize before_filter :find_existing_or_new_page, :only => [:show, :edit, :update] before_filter :find_existing_page, :only => [:rename, :protect, :history, :diff, :annotate, :add_attachment, :destroy, :destroy_version] + accept_api_auth :index, :show helper :attachments include AttachmentsHelper @@ -45,7 +46,13 @@ class WikiController < ApplicationController # List of pages, sorted alphabetically and by parent (hierarchy) def index load_pages_for_index - @pages_by_parent_id = @pages.group_by(&:parent_id) + + respond_to do |format| + format.html { + @pages_by_parent_id = @pages.group_by(&:parent_id) + } + format.api + end end # List of page, by last update @@ -57,7 +64,7 @@ class WikiController < ApplicationController # display a page (in editing mode if it doesn't exist) def show if @page.new_record? - if User.current.allowed_to?(:edit_wiki_pages, @project) && editable? + if User.current.allowed_to?(:edit_wiki_pages, @project) && editable? && !api_request? edit render :action => 'edit' else @@ -66,8 +73,7 @@ class WikiController < ApplicationController return end if params[:version] && !User.current.allowed_to?(:view_wiki_edits, @project) - # Redirects user to the current version if he's not allowed to view previous versions - redirect_to :version => nil + deny_access return end @content = @page.content_for_version(params[:version]) @@ -89,7 +95,10 @@ class WikiController < ApplicationController @content.current_version? && Redmine::WikiFormatting.supports_section_edit? - render :action => 'show' + respond_to do |format| + format.html + format.api + end end # edit an existing page or a new one @@ -321,6 +330,6 @@ private end def load_pages_for_index - @pages = @wiki.pages.with_updated_on.all(:order => 'title', :include => {:wiki => :project}) + @pages = @wiki.pages.with_updated_on.order("#{WikiPage.table_name}.title").includes(:wiki => :project).includes(:parent).all end end diff --git a/app/models/wiki_page.rb b/app/models/wiki_page.rb index 37d6cf5a4..1eaf42ed3 100644 --- a/app/models/wiki_page.rb +++ b/app/models/wiki_page.rb @@ -50,7 +50,7 @@ class WikiPage < ActiveRecord::Base # eager load information about last updates, without loading text scope :with_updated_on, { - :select => "#{WikiPage.table_name}.*, #{WikiContent.table_name}.updated_on", + :select => "#{WikiPage.table_name}.*, #{WikiContent.table_name}.updated_on, #{WikiContent.table_name}.version", :joins => "LEFT JOIN #{WikiContent.table_name} ON #{WikiContent.table_name}.page_id = #{WikiPage.table_name}.id" } diff --git a/app/views/wiki/index.api.rsb b/app/views/wiki/index.api.rsb new file mode 100644 index 000000000..f1035c89b --- /dev/null +++ b/app/views/wiki/index.api.rsb @@ -0,0 +1,13 @@ +api.array :wiki_pages do + @pages.each do |page| + api.wiki_page do + api.title page.title + if page.parent + api.parent :title => page.parent.title + end + api.version page.version + api.created_on page.created_on + api.updated_on page.updated_on + end + end +end diff --git a/app/views/wiki/show.api.rsb b/app/views/wiki/show.api.rsb new file mode 100644 index 000000000..acf62bb94 --- /dev/null +++ b/app/views/wiki/show.api.rsb @@ -0,0 +1,11 @@ +api.wiki_page do + api.title @page.title + if @page.parent + api.parent :title => @page.parent.title + end + api.text @content.text + api.version @content.version + api.author(:id => @content.author_id, :name => @content.author.name) + api.created_on @page.created_on + api.updated_on @content.updated_on +end diff --git a/test/functional/wiki_controller_test.rb b/test/functional/wiki_controller_test.rb index 720a9b6f8..98fc20038 100644 --- a/test/functional/wiki_controller_test.rb +++ b/test/functional/wiki_controller_test.rb @@ -75,6 +75,13 @@ class WikiControllerTest < ActionController::TestCase assert_select 'a[href=?]', '/projects/ecookbook/wiki/CookBook_documentation', :text => /Current version/ end + def test_show_old_version_without_permission_should_be_denied + Role.anonymous.remove_permission! :view_wiki_edits + + get :show, :project_id => 'ecookbook', :id => 'CookBook_documentation', :version => '2' + assert_redirected_to '/login?back_url=http%3A%2F%2Ftest.host%2Fprojects%2Fecookbook%2Fwiki%2FCookBook_documentation%2F2' + end + def test_show_first_version get :show, :project_id => 'ecookbook', :id => 'CookBook_documentation', :version => '1' assert_response :success diff --git a/test/integration/api_test/wiki_pages_test.rb b/test/integration/api_test/wiki_pages_test.rb new file mode 100644 index 000000000..8d1ee326e --- /dev/null +++ b/test/integration/api_test/wiki_pages_test.rb @@ -0,0 +1,85 @@ +# Redmine - project management software +# Copyright (C) 2006-2012 Jean-Philippe Lang +# +# This program is free software; you can redistribute it and/or +# modify it under the terms of the GNU General Public License +# as published by the Free Software Foundation; either version 2 +# of the License, or (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program; if not, write to the Free Software +# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + +require File.expand_path('../../../test_helper', __FILE__) + +class ApiTest::WikiPagesTest < ActionController::IntegrationTest + fixtures :projects, :users, :roles, :members, :member_roles, + :enabled_modules, :wikis, :wiki_pages, :wiki_contents, + :wiki_content_versions, :attachments + + def setup + Setting.rest_api_enabled = '1' + end + + test "GET /projects/:project_id/wiki/index.xml should return wiki pages" do + get '/projects/ecookbook/wiki/index.xml' + assert_response :success + assert_equal 'application/xml', response.content_type + assert_select 'wiki_pages[type=array]' do + assert_select 'wiki_page', :count => Wiki.find(1).pages.count + assert_select 'wiki_page' do + assert_select 'title', :text => 'CookBook_documentation' + assert_select 'version', :text => '3' + assert_select 'created_on' + assert_select 'updated_on' + end + end + end + + test "GET /projects/:project_id/wiki/:title.xml should return wiki page" do + get '/projects/ecookbook/wiki/CookBook_documentation.xml' + assert_response :success + assert_equal 'application/xml', response.content_type + assert_select 'wiki_page' do + assert_select 'title', :text => 'CookBook_documentation' + assert_select 'version', :text => '3' + assert_select 'text' + assert_select 'author' + assert_select 'created_on' + assert_select 'updated_on' + end + end + + test "GET /projects/:project_id/wiki/:title.xml with unknown title and edit permission should respond with 404" do + get '/projects/ecookbook/wiki/Invalid_Page.xml', {}, credentials('jsmith') + assert_response 404 + assert_equal 'application/xml', response.content_type + end + + test "GET /projects/:project_id/wiki/:title/:version.xml should return wiki page version" do + get '/projects/ecookbook/wiki/CookBook_documentation/2.xml' + assert_response :success + assert_equal 'application/xml', response.content_type + assert_select 'wiki_page' do + assert_select 'title', :text => 'CookBook_documentation' + assert_select 'version', :text => '2' + assert_select 'text' + assert_select 'author' + assert_select 'created_on' + assert_select 'updated_on' + end + end + + test "GET /projects/:project_id/wiki/:title/:version.xml without permission should be denied" do + Role.anonymous.remove_permission! :view_wiki_edits + + get '/projects/ecookbook/wiki/CookBook_documentation/2.xml' + assert_response 401 + assert_equal 'application/xml', response.content_type + end +end diff --git a/test/integration/routing/wiki_test.rb b/test/integration/routing/wiki_test.rb index e7f6fa0f7..29b305cd6 100644 --- a/test/integration/routing/wiki_test.rb +++ b/test/integration/routing/wiki_test.rb @@ -34,6 +34,16 @@ class RoutingWikiTest < ActionController::IntegrationTest :id => 'lalala', :format => 'pdf' } ) assert_routing( + { :method => 'get', :path => "/projects/567/wiki/lalala.xml" }, + { :controller => 'wiki', :action => 'show', :project_id => '567', + :id => 'lalala', :format => 'xml' } + ) + assert_routing( + { :method => 'get', :path => "/projects/567/wiki/lalala.json" }, + { :controller => 'wiki', :action => 'show', :project_id => '567', + :id => 'lalala', :format => 'json' } + ) + assert_routing( { :method => 'get', :path => "/projects/1/wiki/CookBook_documentation/diff" }, { :controller => 'wiki', :action => 'diff', :project_id => '1', :id => 'CookBook_documentation' } @@ -44,6 +54,16 @@ class RoutingWikiTest < ActionController::IntegrationTest :id => 'CookBook_documentation', :version => '2' } ) assert_routing( + { :method => 'get', :path => "/projects/1/wiki/CookBook_documentation/2.xml" }, + { :controller => 'wiki', :action => 'show', :project_id => '1', + :id => 'CookBook_documentation', :version => '2', :format => 'xml' } + ) + assert_routing( + { :method => 'get', :path => "/projects/1/wiki/CookBook_documentation/2.json" }, + { :controller => 'wiki', :action => 'show', :project_id => '1', + :id => 'CookBook_documentation', :version => '2', :format => 'json' } + ) + assert_routing( { :method => 'get', :path => "/projects/1/wiki/CookBook_documentation/2/diff" }, { :controller => 'wiki', :action => 'diff', :project_id => '1', :id => 'CookBook_documentation', :version => '2' } @@ -72,6 +92,14 @@ class RoutingWikiTest < ActionController::IntegrationTest { :method => 'get', :path => "/projects/567/wiki/index" }, { :controller => 'wiki', :action => 'index', :project_id => '567' } ) + assert_routing( + { :method => 'get', :path => "/projects/567/wiki/index.xml" }, + { :controller => 'wiki', :action => 'index', :project_id => '567', :format => 'xml' } + ) + assert_routing( + { :method => 'get', :path => "/projects/567/wiki/index.json" }, + { :controller => 'wiki', :action => 'index', :project_id => '567', :format => 'json' } + ) end def test_wiki_resources diff --git a/test/unit/wiki_page_test.rb b/test/unit/wiki_page_test.rb index 8eb2af0ea..28e12e05e 100644 --- a/test/unit/wiki_page_test.rb +++ b/test/unit/wiki_page_test.rb @@ -130,6 +130,7 @@ class WikiPageTest < ActiveSupport::TestCase assert_not_nil page.read_attribute(:updated_on) assert_equal Time.gm(2007, 3, 6, 23, 10, 51), page.content.updated_on assert_equal page.content.updated_on, page.updated_on + assert_not_nil page.read_attribute(:version) end def test_descendants |