summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorJean-Philippe Lang <jp_lang@yahoo.fr>2012-10-25 19:44:04 +0000
committerJean-Philippe Lang <jp_lang@yahoo.fr>2012-10-25 19:44:04 +0000
commit9e313087205721ed3c6239be7cde55598d29f9e6 (patch)
treec5ba4ab52453907afae728887b470f95008bbbdb
parentb2e035fa96e4c2b06dcd559850d11987ba96ee1c (diff)
downloadredmine-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.rb21
-rw-r--r--app/models/wiki_page.rb2
-rw-r--r--app/views/wiki/index.api.rsb13
-rw-r--r--app/views/wiki/show.api.rsb11
-rw-r--r--test/functional/wiki_controller_test.rb7
-rw-r--r--test/integration/api_test/wiki_pages_test.rb85
-rw-r--r--test/integration/routing/wiki_test.rb28
-rw-r--r--test/unit/wiki_page_test.rb1
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