##// END OF EJS Templates
Removed some shoulda context....
Jean-Philippe Lang -
r11634:474c01074638
parent child
Show More
@@ -1,165 +1,139
1 1 # Redmine - project management software
2 2 # Copyright (C) 2006-2013 Jean-Philippe Lang
3 3 #
4 4 # This program is free software; you can redistribute it and/or
5 5 # modify it under the terms of the GNU General Public License
6 6 # as published by the Free Software Foundation; either version 2
7 7 # of the License, or (at your option) any later version.
8 8 #
9 9 # This program is distributed in the hope that it will be useful,
10 10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
11 11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 12 # GNU General Public License for more details.
13 13 #
14 14 # You should have received a copy of the GNU General Public License
15 15 # along with this program; if not, write to the Free Software
16 16 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
17 17
18 18 require File.expand_path('../../test_helper', __FILE__)
19 19
20 20 class WelcomeControllerTest < ActionController::TestCase
21 21 fixtures :projects, :news, :users, :members
22 22
23 23 def setup
24 24 User.current = nil
25 25 end
26 26
27 27 def test_index
28 28 get :index
29 29 assert_response :success
30 30 assert_template 'index'
31 31 assert_not_nil assigns(:news)
32 32 assert_not_nil assigns(:projects)
33 33 assert !assigns(:projects).include?(Project.where(:is_public => false).first)
34 34 end
35 35
36 36 def test_browser_language
37 37 Setting.default_language = 'en'
38 38 @request.env['HTTP_ACCEPT_LANGUAGE'] = 'fr,fr-fr;q=0.8,en-us;q=0.5,en;q=0.3'
39 39 get :index
40 40 assert_equal :fr, @controller.current_language
41 41 end
42 42
43 43 def test_browser_language_alternate
44 44 Setting.default_language = 'en'
45 45 @request.env['HTTP_ACCEPT_LANGUAGE'] = 'zh-TW'
46 46 get :index
47 47 assert_equal :"zh-TW", @controller.current_language
48 48 end
49 49
50 50 def test_browser_language_alternate_not_valid
51 51 Setting.default_language = 'en'
52 52 @request.env['HTTP_ACCEPT_LANGUAGE'] = 'fr-CA'
53 53 get :index
54 54 assert_equal :fr, @controller.current_language
55 55 end
56 56
57 57 def test_robots
58 58 get :robots
59 59 assert_response :success
60 60 assert_equal 'text/plain', @response.content_type
61 61 assert @response.body.match(%r{^Disallow: /projects/ecookbook/issues\r?$})
62 62 end
63 63
64 64 def test_warn_on_leaving_unsaved_turn_on
65 65 user = User.find(2)
66 66 user.pref.warn_on_leaving_unsaved = '1'
67 67 user.pref.save!
68 68 @request.session[:user_id] = 2
69 69
70 70 get :index
71 71 assert_tag 'script',
72 72 :attributes => {:type => "text/javascript"},
73 73 :content => %r{warnLeavingUnsaved}
74 74 end
75 75
76 76 def test_warn_on_leaving_unsaved_turn_off
77 77 user = User.find(2)
78 78 user.pref.warn_on_leaving_unsaved = '0'
79 79 user.pref.save!
80 80 @request.session[:user_id] = 2
81 81
82 82 get :index
83 83 assert_no_tag 'script',
84 84 :attributes => {:type => "text/javascript"},
85 85 :content => %r{warnLeavingUnsaved}
86 86 end
87 87
88 88 def test_logout_link_should_post
89 89 @request.session[:user_id] = 2
90 90
91 91 get :index
92 92 assert_select 'a[href=/logout][data-method=post]', :text => 'Sign out'
93 93 end
94 94
95 95 def test_call_hook_mixed_in
96 96 assert @controller.respond_to?(:call_hook)
97 97 end
98 98
99 99 def test_project_jump_box_should_escape_names_once
100 100 Project.find(1).update_attribute :name, 'Foo & Bar'
101 101 @request.session[:user_id] = 2
102 102
103 103 get :index
104 104 assert_select "#header select" do
105 105 assert_select "option", :text => 'Foo &amp; Bar'
106 106 end
107 107 end
108 108
109 context "test_api_offset_and_limit" do
110 context "without params" do
111 should "return 0, 25" do
112 assert_equal [0, 25], @controller.api_offset_and_limit({})
113 end
114 end
115
116 context "with limit" do
117 should "return 0, limit" do
118 assert_equal [0, 30], @controller.api_offset_and_limit({:limit => 30})
119 end
120
121 should "not exceed 100" do
122 assert_equal [0, 100], @controller.api_offset_and_limit({:limit => 120})
123 end
109 def test_api_offset_and_limit_without_params
110 assert_equal [0, 25], @controller.api_offset_and_limit({})
111 end
124 112
125 should "not be negative" do
126 assert_equal [0, 25], @controller.api_offset_and_limit({:limit => -10})
127 end
128 end
113 def test_api_offset_and_limit_with_limit
114 assert_equal [0, 30], @controller.api_offset_and_limit({:limit => 30})
115 assert_equal [0, 100], @controller.api_offset_and_limit({:limit => 120})
116 assert_equal [0, 25], @controller.api_offset_and_limit({:limit => -10})
117 end
129 118
130 context "with offset" do
131 should "return offset, 25" do
132 assert_equal [10, 25], @controller.api_offset_and_limit({:offset => 10})
133 end
119 def test_api_offset_and_limit_with_offset
120 assert_equal [10, 25], @controller.api_offset_and_limit({:offset => 10})
121 assert_equal [0, 25], @controller.api_offset_and_limit({:offset => -10})
122 end
134 123
135 should "not be negative" do
136 assert_equal [0, 25], @controller.api_offset_and_limit({:offset => -10})
137 end
124 def test_api_offset_and_limit_with_offset_and_limit
125 assert_equal [10, 50], @controller.api_offset_and_limit({:offset => 10, :limit => 50})
126 end
138 127
139 context "and limit" do
140 should "return offset, limit" do
141 assert_equal [10, 50], @controller.api_offset_and_limit({:offset => 10, :limit => 50})
142 end
143 end
144 end
128 def test_api_offset_and_limit_with_page
129 assert_equal [0, 25], @controller.api_offset_and_limit({:page => 1})
130 assert_equal [50, 25], @controller.api_offset_and_limit({:page => 3})
131 assert_equal [0, 25], @controller.api_offset_and_limit({:page => 0})
132 assert_equal [0, 25], @controller.api_offset_and_limit({:page => -2})
133 end
145 134
146 context "with page" do
147 should "return offset, 25" do
148 assert_equal [0, 25], @controller.api_offset_and_limit({:page => 1})
149 assert_equal [50, 25], @controller.api_offset_and_limit({:page => 3})
150 end
151
152 should "not be negative" do
153 assert_equal [0, 25], @controller.api_offset_and_limit({:page => 0})
154 assert_equal [0, 25], @controller.api_offset_and_limit({:page => -2})
155 end
156
157 context "and limit" do
158 should "return offset, limit" do
159 assert_equal [0, 100], @controller.api_offset_and_limit({:page => 1, :limit => 100})
160 assert_equal [200, 100], @controller.api_offset_and_limit({:page => 3, :limit => 100})
161 end
162 end
163 end
135 def test_api_offset_and_limit_with_page_and_limit
136 assert_equal [0, 100], @controller.api_offset_and_limit({:page => 1, :limit => 100})
137 assert_equal [200, 100], @controller.api_offset_and_limit({:page => 3, :limit => 100})
164 138 end
165 139 end
@@ -1,1219 +1,1213
1 1 # encoding: utf-8
2 2 #
3 3 # Redmine - project management software
4 4 # Copyright (C) 2006-2013 Jean-Philippe Lang
5 5 #
6 6 # This program is free software; you can redistribute it and/or
7 7 # modify it under the terms of the GNU General Public License
8 8 # as published by the Free Software Foundation; either version 2
9 9 # of the License, or (at your option) any later version.
10 10 #
11 11 # This program is distributed in the hope that it will be useful,
12 12 # but WITHOUT ANY WARRANTY; without even the implied warranty of
13 13 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 14 # GNU General Public License for more details.
15 15 #
16 16 # You should have received a copy of the GNU General Public License
17 17 # along with this program; if not, write to the Free Software
18 18 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
19 19
20 20 require File.expand_path('../../../test_helper', __FILE__)
21 21
22 22 class ApplicationHelperTest < ActionView::TestCase
23 23 include ERB::Util
24 24 include Rails.application.routes.url_helpers
25 25
26 26 fixtures :projects, :roles, :enabled_modules, :users,
27 27 :repositories, :changesets,
28 28 :trackers, :issue_statuses, :issues, :versions, :documents,
29 29 :wikis, :wiki_pages, :wiki_contents,
30 30 :boards, :messages, :news,
31 31 :attachments, :enumerations
32 32
33 33 def setup
34 34 super
35 35 set_tmp_attachments_directory
36 36 end
37 37
38 context "#link_to_if_authorized" do
39 context "for authorized user" do
40 should "allow using the :controller and :action for the target link" do
41 User.current = User.find_by_login('admin')
42
43 @project = Issue.first.project # Used by helper
44 response = link_to_if_authorized('By controller/actionr',
45 {:controller => 'issues', :action => 'edit', :id => Issue.first.id})
46 assert_match /href/, response
47 end
48 end
38 test "#link_to_if_authorized for authorized user should allow using the :controller and :action for the target link" do
39 User.current = User.find_by_login('admin')
49 40
50 context "for unauthorized user" do
51 should "display nothing if user isn't authorized" do
52 User.current = User.find_by_login('dlopper')
53 @project = Project.find('private-child')
54 issue = @project.issues.first
55 assert !issue.visible?
56
57 response = link_to_if_authorized('Never displayed',
58 {:controller => 'issues', :action => 'show', :id => issue})
59 assert_nil response
60 end
61 end
41 @project = Issue.first.project # Used by helper
42 response = link_to_if_authorized('By controller/actionr',
43 {:controller => 'issues', :action => 'edit', :id => Issue.first.id})
44 assert_match /href/, response
45 end
46
47 test "#link_to_if_authorized for unauthorized user should display nothing if user isn't authorized" do
48 User.current = User.find_by_login('dlopper')
49 @project = Project.find('private-child')
50 issue = @project.issues.first
51 assert !issue.visible?
52
53 response = link_to_if_authorized('Never displayed',
54 {:controller => 'issues', :action => 'show', :id => issue})
55 assert_nil response
62 56 end
63 57
64 58 def test_auto_links
65 59 to_test = {
66 60 'http://foo.bar' => '<a class="external" href="http://foo.bar">http://foo.bar</a>',
67 61 'http://foo.bar/~user' => '<a class="external" href="http://foo.bar/~user">http://foo.bar/~user</a>',
68 62 'http://foo.bar.' => '<a class="external" href="http://foo.bar">http://foo.bar</a>.',
69 63 'https://foo.bar.' => '<a class="external" href="https://foo.bar">https://foo.bar</a>.',
70 64 'This is a link: http://foo.bar.' => 'This is a link: <a class="external" href="http://foo.bar">http://foo.bar</a>.',
71 65 'A link (eg. http://foo.bar).' => 'A link (eg. <a class="external" href="http://foo.bar">http://foo.bar</a>).',
72 66 'http://foo.bar/foo.bar#foo.bar.' => '<a class="external" href="http://foo.bar/foo.bar#foo.bar">http://foo.bar/foo.bar#foo.bar</a>.',
73 67 'http://www.foo.bar/Test_(foobar)' => '<a class="external" href="http://www.foo.bar/Test_(foobar)">http://www.foo.bar/Test_(foobar)</a>',
74 68 '(see inline link : http://www.foo.bar/Test_(foobar))' => '(see inline link : <a class="external" href="http://www.foo.bar/Test_(foobar)">http://www.foo.bar/Test_(foobar)</a>)',
75 69 '(see inline link : http://www.foo.bar/Test)' => '(see inline link : <a class="external" href="http://www.foo.bar/Test">http://www.foo.bar/Test</a>)',
76 70 '(see inline link : http://www.foo.bar/Test).' => '(see inline link : <a class="external" href="http://www.foo.bar/Test">http://www.foo.bar/Test</a>).',
77 71 '(see "inline link":http://www.foo.bar/Test_(foobar))' => '(see <a href="http://www.foo.bar/Test_(foobar)" class="external">inline link</a>)',
78 72 '(see "inline link":http://www.foo.bar/Test)' => '(see <a href="http://www.foo.bar/Test" class="external">inline link</a>)',
79 73 '(see "inline link":http://www.foo.bar/Test).' => '(see <a href="http://www.foo.bar/Test" class="external">inline link</a>).',
80 74 'www.foo.bar' => '<a class="external" href="http://www.foo.bar">www.foo.bar</a>',
81 75 'http://foo.bar/page?p=1&t=z&s=' => '<a class="external" href="http://foo.bar/page?p=1&#38;t=z&#38;s=">http://foo.bar/page?p=1&#38;t=z&#38;s=</a>',
82 76 'http://foo.bar/page#125' => '<a class="external" href="http://foo.bar/page#125">http://foo.bar/page#125</a>',
83 77 'http://foo@www.bar.com' => '<a class="external" href="http://foo@www.bar.com">http://foo@www.bar.com</a>',
84 78 'http://foo:bar@www.bar.com' => '<a class="external" href="http://foo:bar@www.bar.com">http://foo:bar@www.bar.com</a>',
85 79 'ftp://foo.bar' => '<a class="external" href="ftp://foo.bar">ftp://foo.bar</a>',
86 80 'ftps://foo.bar' => '<a class="external" href="ftps://foo.bar">ftps://foo.bar</a>',
87 81 'sftp://foo.bar' => '<a class="external" href="sftp://foo.bar">sftp://foo.bar</a>',
88 82 # two exclamation marks
89 83 'http://example.net/path!602815048C7B5C20!302.html' => '<a class="external" href="http://example.net/path!602815048C7B5C20!302.html">http://example.net/path!602815048C7B5C20!302.html</a>',
90 84 # escaping
91 85 'http://foo"bar' => '<a class="external" href="http://foo&quot;bar">http://foo&quot;bar</a>',
92 86 # wrap in angle brackets
93 87 '<http://foo.bar>' => '&lt;<a class="external" href="http://foo.bar">http://foo.bar</a>&gt;',
94 88 # invalid urls
95 89 'http://' => 'http://',
96 90 'www.' => 'www.',
97 91 'test-www.bar.com' => 'test-www.bar.com',
98 92 }
99 93 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text) }
100 94 end
101 95
102 96 if 'ruby'.respond_to?(:encoding)
103 97 def test_auto_links_with_non_ascii_characters
104 98 to_test = {
105 99 'http://foo.bar/тСст' => '<a class="external" href="http://foo.bar/тСст">http://foo.bar/тСст</a>'
106 100 }
107 101 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text) }
108 102 end
109 103 else
110 104 puts 'Skipping test_auto_links_with_non_ascii_characters, unsupported ruby version'
111 105 end
112 106
113 107 def test_auto_mailto
114 108 to_test = {
115 109 'test@foo.bar' => '<a class="email" href="mailto:test@foo.bar">test@foo.bar</a>',
116 110 'test@www.foo.bar' => '<a class="email" href="mailto:test@www.foo.bar">test@www.foo.bar</a>',
117 111 }
118 112 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text) }
119 113 end
120 114
121 115 def test_inline_images
122 116 to_test = {
123 117 '!http://foo.bar/image.jpg!' => '<img src="http://foo.bar/image.jpg" alt="" />',
124 118 'floating !>http://foo.bar/image.jpg!' => 'floating <div style="float:right"><img src="http://foo.bar/image.jpg" alt="" /></div>',
125 119 'with class !(some-class)http://foo.bar/image.jpg!' => 'with class <img src="http://foo.bar/image.jpg" class="some-class" alt="" />',
126 120 'with style !{width:100px;height:100px}http://foo.bar/image.jpg!' => 'with style <img src="http://foo.bar/image.jpg" style="width:100px;height:100px;" alt="" />',
127 121 'with title !http://foo.bar/image.jpg(This is a title)!' => 'with title <img src="http://foo.bar/image.jpg" title="This is a title" alt="This is a title" />',
128 122 'with title !http://foo.bar/image.jpg(This is a double-quoted "title")!' => 'with title <img src="http://foo.bar/image.jpg" title="This is a double-quoted &quot;title&quot;" alt="This is a double-quoted &quot;title&quot;" />',
129 123 }
130 124 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text) }
131 125 end
132 126
133 127 def test_inline_images_inside_tags
134 128 raw = <<-RAW
135 129 h1. !foo.png! Heading
136 130
137 131 Centered image:
138 132
139 133 p=. !bar.gif!
140 134 RAW
141 135
142 136 assert textilizable(raw).include?('<img src="foo.png" alt="" />')
143 137 assert textilizable(raw).include?('<img src="bar.gif" alt="" />')
144 138 end
145 139
146 140 def test_attached_images
147 141 to_test = {
148 142 'Inline image: !logo.gif!' => 'Inline image: <img src="/attachments/download/3/logo.gif" title="This is a logo" alt="This is a logo" />',
149 143 'Inline image: !logo.GIF!' => 'Inline image: <img src="/attachments/download/3/logo.gif" title="This is a logo" alt="This is a logo" />',
150 144 'No match: !ogo.gif!' => 'No match: <img src="ogo.gif" alt="" />',
151 145 'No match: !ogo.GIF!' => 'No match: <img src="ogo.GIF" alt="" />',
152 146 # link image
153 147 '!logo.gif!:http://foo.bar/' => '<a href="http://foo.bar/"><img src="/attachments/download/3/logo.gif" title="This is a logo" alt="This is a logo" /></a>',
154 148 }
155 149 attachments = Attachment.all
156 150 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text, :attachments => attachments) }
157 151 end
158 152
159 153 def test_attached_images_filename_extension
160 154 set_tmp_attachments_directory
161 155 a1 = Attachment.new(
162 156 :container => Issue.find(1),
163 157 :file => mock_file_with_options({:original_filename => "testtest.JPG"}),
164 158 :author => User.find(1))
165 159 assert a1.save
166 160 assert_equal "testtest.JPG", a1.filename
167 161 assert_equal "image/jpeg", a1.content_type
168 162 assert a1.image?
169 163
170 164 a2 = Attachment.new(
171 165 :container => Issue.find(1),
172 166 :file => mock_file_with_options({:original_filename => "testtest.jpeg"}),
173 167 :author => User.find(1))
174 168 assert a2.save
175 169 assert_equal "testtest.jpeg", a2.filename
176 170 assert_equal "image/jpeg", a2.content_type
177 171 assert a2.image?
178 172
179 173 a3 = Attachment.new(
180 174 :container => Issue.find(1),
181 175 :file => mock_file_with_options({:original_filename => "testtest.JPE"}),
182 176 :author => User.find(1))
183 177 assert a3.save
184 178 assert_equal "testtest.JPE", a3.filename
185 179 assert_equal "image/jpeg", a3.content_type
186 180 assert a3.image?
187 181
188 182 a4 = Attachment.new(
189 183 :container => Issue.find(1),
190 184 :file => mock_file_with_options({:original_filename => "Testtest.BMP"}),
191 185 :author => User.find(1))
192 186 assert a4.save
193 187 assert_equal "Testtest.BMP", a4.filename
194 188 assert_equal "image/x-ms-bmp", a4.content_type
195 189 assert a4.image?
196 190
197 191 to_test = {
198 192 'Inline image: !testtest.jpg!' =>
199 193 'Inline image: <img src="/attachments/download/' + a1.id.to_s + '/testtest.JPG" alt="" />',
200 194 'Inline image: !testtest.jpeg!' =>
201 195 'Inline image: <img src="/attachments/download/' + a2.id.to_s + '/testtest.jpeg" alt="" />',
202 196 'Inline image: !testtest.jpe!' =>
203 197 'Inline image: <img src="/attachments/download/' + a3.id.to_s + '/testtest.JPE" alt="" />',
204 198 'Inline image: !testtest.bmp!' =>
205 199 'Inline image: <img src="/attachments/download/' + a4.id.to_s + '/Testtest.BMP" alt="" />',
206 200 }
207 201
208 202 attachments = [a1, a2, a3, a4]
209 203 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text, :attachments => attachments) }
210 204 end
211 205
212 206 def test_attached_images_should_read_later
213 207 set_fixtures_attachments_directory
214 208 a1 = Attachment.find(16)
215 209 assert_equal "testfile.png", a1.filename
216 210 assert a1.readable?
217 211 assert (! a1.visible?(User.anonymous))
218 212 assert a1.visible?(User.find(2))
219 213 a2 = Attachment.find(17)
220 214 assert_equal "testfile.PNG", a2.filename
221 215 assert a2.readable?
222 216 assert (! a2.visible?(User.anonymous))
223 217 assert a2.visible?(User.find(2))
224 218 assert a1.created_on < a2.created_on
225 219
226 220 to_test = {
227 221 'Inline image: !testfile.png!' =>
228 222 'Inline image: <img src="/attachments/download/' + a2.id.to_s + '/testfile.PNG" alt="" />',
229 223 'Inline image: !Testfile.PNG!' =>
230 224 'Inline image: <img src="/attachments/download/' + a2.id.to_s + '/testfile.PNG" alt="" />',
231 225 }
232 226 attachments = [a1, a2]
233 227 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text, :attachments => attachments) }
234 228 set_tmp_attachments_directory
235 229 end
236 230
237 231 def test_textile_external_links
238 232 to_test = {
239 233 'This is a "link":http://foo.bar' => 'This is a <a href="http://foo.bar" class="external">link</a>',
240 234 'This is an intern "link":/foo/bar' => 'This is an intern <a href="/foo/bar">link</a>',
241 235 '"link (Link title)":http://foo.bar' => '<a href="http://foo.bar" title="Link title" class="external">link</a>',
242 236 '"link (Link title with "double-quotes")":http://foo.bar' => '<a href="http://foo.bar" title="Link title with &quot;double-quotes&quot;" class="external">link</a>',
243 237 "This is not a \"Link\":\n\nAnother paragraph" => "This is not a \"Link\":</p>\n\n\n\t<p>Another paragraph",
244 238 # no multiline link text
245 239 "This is a double quote \"on the first line\nand another on a second line\":test" => "This is a double quote \"on the first line<br />and another on a second line\":test",
246 240 # mailto link
247 241 "\"system administrator\":mailto:sysadmin@example.com?subject=redmine%20permissions" => "<a href=\"mailto:sysadmin@example.com?subject=redmine%20permissions\">system administrator</a>",
248 242 # two exclamation marks
249 243 '"a link":http://example.net/path!602815048C7B5C20!302.html' => '<a href="http://example.net/path!602815048C7B5C20!302.html" class="external">a link</a>',
250 244 # escaping
251 245 '"test":http://foo"bar' => '<a href="http://foo&quot;bar" class="external">test</a>',
252 246 }
253 247 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text) }
254 248 end
255 249
256 250 if 'ruby'.respond_to?(:encoding)
257 251 def test_textile_external_links_with_non_ascii_characters
258 252 to_test = {
259 253 'This is a "link":http://foo.bar/тСст' => 'This is a <a href="http://foo.bar/тСст" class="external">link</a>'
260 254 }
261 255 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text) }
262 256 end
263 257 else
264 258 puts 'Skipping test_textile_external_links_with_non_ascii_characters, unsupported ruby version'
265 259 end
266 260
267 261 def test_redmine_links
268 262 issue_link = link_to('#3', {:controller => 'issues', :action => 'show', :id => 3},
269 263 :class => Issue.find(3).css_classes, :title => 'Error 281 when updating a recipe (New)')
270 264 note_link = link_to('#3', {:controller => 'issues', :action => 'show', :id => 3, :anchor => 'note-14'},
271 265 :class => Issue.find(3).css_classes, :title => 'Error 281 when updating a recipe (New)')
272 266
273 267 revision_link = link_to('r1', {:controller => 'repositories', :action => 'revision', :id => 'ecookbook', :rev => 1},
274 268 :class => 'changeset', :title => 'My very first commit do not escaping #<>&')
275 269 revision_link2 = link_to('r2', {:controller => 'repositories', :action => 'revision', :id => 'ecookbook', :rev => 2},
276 270 :class => 'changeset', :title => 'This commit fixes #1, #2 and references #1 & #3')
277 271
278 272 changeset_link2 = link_to('691322a8eb01e11fd7',
279 273 {:controller => 'repositories', :action => 'revision', :id => 'ecookbook', :rev => 1},
280 274 :class => 'changeset', :title => 'My very first commit do not escaping #<>&')
281 275
282 276 document_link = link_to('Test document', {:controller => 'documents', :action => 'show', :id => 1},
283 277 :class => 'document')
284 278
285 279 version_link = link_to('1.0', {:controller => 'versions', :action => 'show', :id => 2},
286 280 :class => 'version')
287 281
288 282 board_url = {:controller => 'boards', :action => 'show', :id => 2, :project_id => 'ecookbook'}
289 283
290 284 message_url = {:controller => 'messages', :action => 'show', :board_id => 1, :id => 4}
291 285
292 286 news_url = {:controller => 'news', :action => 'show', :id => 1}
293 287
294 288 project_url = {:controller => 'projects', :action => 'show', :id => 'subproject1'}
295 289
296 290 source_url = '/projects/ecookbook/repository/entry/some/file'
297 291 source_url_with_rev = '/projects/ecookbook/repository/revisions/52/entry/some/file'
298 292 source_url_with_ext = '/projects/ecookbook/repository/entry/some/file.ext'
299 293 source_url_with_rev_and_ext = '/projects/ecookbook/repository/revisions/52/entry/some/file.ext'
300 294 source_url_with_branch = '/projects/ecookbook/repository/revisions/branch/entry/some/file'
301 295
302 296 export_url = '/projects/ecookbook/repository/raw/some/file'
303 297 export_url_with_rev = '/projects/ecookbook/repository/revisions/52/raw/some/file'
304 298 export_url_with_ext = '/projects/ecookbook/repository/raw/some/file.ext'
305 299 export_url_with_rev_and_ext = '/projects/ecookbook/repository/revisions/52/raw/some/file.ext'
306 300 export_url_with_branch = '/projects/ecookbook/repository/revisions/branch/raw/some/file'
307 301
308 302 to_test = {
309 303 # tickets
310 304 '#3, [#3], (#3) and #3.' => "#{issue_link}, [#{issue_link}], (#{issue_link}) and #{issue_link}.",
311 305 # ticket notes
312 306 '#3-14' => note_link,
313 307 '#3#note-14' => note_link,
314 308 # should not ignore leading zero
315 309 '#03' => '#03',
316 310 # changesets
317 311 'r1' => revision_link,
318 312 'r1.' => "#{revision_link}.",
319 313 'r1, r2' => "#{revision_link}, #{revision_link2}",
320 314 'r1,r2' => "#{revision_link},#{revision_link2}",
321 315 'commit:691322a8eb01e11fd7' => changeset_link2,
322 316 # documents
323 317 'document#1' => document_link,
324 318 'document:"Test document"' => document_link,
325 319 # versions
326 320 'version#2' => version_link,
327 321 'version:1.0' => version_link,
328 322 'version:"1.0"' => version_link,
329 323 # source
330 324 'source:some/file' => link_to('source:some/file', source_url, :class => 'source'),
331 325 'source:/some/file' => link_to('source:/some/file', source_url, :class => 'source'),
332 326 'source:/some/file.' => link_to('source:/some/file', source_url, :class => 'source') + ".",
333 327 'source:/some/file.ext.' => link_to('source:/some/file.ext', source_url_with_ext, :class => 'source') + ".",
334 328 'source:/some/file. ' => link_to('source:/some/file', source_url, :class => 'source') + ".",
335 329 'source:/some/file.ext. ' => link_to('source:/some/file.ext', source_url_with_ext, :class => 'source') + ".",
336 330 'source:/some/file, ' => link_to('source:/some/file', source_url, :class => 'source') + ",",
337 331 'source:/some/file@52' => link_to('source:/some/file@52', source_url_with_rev, :class => 'source'),
338 332 'source:/some/file@branch' => link_to('source:/some/file@branch', source_url_with_branch, :class => 'source'),
339 333 'source:/some/file.ext@52' => link_to('source:/some/file.ext@52', source_url_with_rev_and_ext, :class => 'source'),
340 334 'source:/some/file#L110' => link_to('source:/some/file#L110', source_url + "#L110", :class => 'source'),
341 335 'source:/some/file.ext#L110' => link_to('source:/some/file.ext#L110', source_url_with_ext + "#L110", :class => 'source'),
342 336 'source:/some/file@52#L110' => link_to('source:/some/file@52#L110', source_url_with_rev + "#L110", :class => 'source'),
343 337 # export
344 338 'export:/some/file' => link_to('export:/some/file', export_url, :class => 'source download'),
345 339 'export:/some/file.ext' => link_to('export:/some/file.ext', export_url_with_ext, :class => 'source download'),
346 340 'export:/some/file@52' => link_to('export:/some/file@52', export_url_with_rev, :class => 'source download'),
347 341 'export:/some/file.ext@52' => link_to('export:/some/file.ext@52', export_url_with_rev_and_ext, :class => 'source download'),
348 342 'export:/some/file@branch' => link_to('export:/some/file@branch', export_url_with_branch, :class => 'source download'),
349 343 # forum
350 344 'forum#2' => link_to('Discussion', board_url, :class => 'board'),
351 345 'forum:Discussion' => link_to('Discussion', board_url, :class => 'board'),
352 346 # message
353 347 'message#4' => link_to('Post 2', message_url, :class => 'message'),
354 348 'message#5' => link_to('RE: post 2', message_url.merge(:anchor => 'message-5', :r => 5), :class => 'message'),
355 349 # news
356 350 'news#1' => link_to('eCookbook first release !', news_url, :class => 'news'),
357 351 'news:"eCookbook first release !"' => link_to('eCookbook first release !', news_url, :class => 'news'),
358 352 # project
359 353 'project#3' => link_to('eCookbook Subproject 1', project_url, :class => 'project'),
360 354 'project:subproject1' => link_to('eCookbook Subproject 1', project_url, :class => 'project'),
361 355 'project:"eCookbook subProject 1"' => link_to('eCookbook Subproject 1', project_url, :class => 'project'),
362 356 # not found
363 357 '#0123456789' => '#0123456789',
364 358 # invalid expressions
365 359 'source:' => 'source:',
366 360 # url hash
367 361 "http://foo.bar/FAQ#3" => '<a class="external" href="http://foo.bar/FAQ#3">http://foo.bar/FAQ#3</a>',
368 362 }
369 363 @project = Project.find(1)
370 364 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text), "#{text} failed" }
371 365 end
372 366
373 367 def test_redmine_links_with_a_different_project_before_current_project
374 368 vp1 = Version.generate!(:project_id => 1, :name => '1.4.4')
375 369 vp3 = Version.generate!(:project_id => 3, :name => '1.4.4')
376 370
377 371 @project = Project.find(3)
378 372 assert_equal %(<p><a href="/versions/#{vp1.id}" class="version">1.4.4</a> <a href="/versions/#{vp3.id}" class="version">1.4.4</a></p>),
379 373 textilizable("ecookbook:version:1.4.4 version:1.4.4")
380 374 end
381 375
382 376 def test_escaped_redmine_links_should_not_be_parsed
383 377 to_test = [
384 378 '#3.',
385 379 '#3-14.',
386 380 '#3#-note14.',
387 381 'r1',
388 382 'document#1',
389 383 'document:"Test document"',
390 384 'version#2',
391 385 'version:1.0',
392 386 'version:"1.0"',
393 387 'source:/some/file'
394 388 ]
395 389 @project = Project.find(1)
396 390 to_test.each { |text| assert_equal "<p>#{text}</p>", textilizable("!" + text), "#{text} failed" }
397 391 end
398 392
399 393 def test_cross_project_redmine_links
400 394 source_link = link_to('ecookbook:source:/some/file', {:controller => 'repositories', :action => 'entry', :id => 'ecookbook', :path => ['some', 'file']},
401 395 :class => 'source')
402 396
403 397 changeset_link = link_to('ecookbook:r2', {:controller => 'repositories', :action => 'revision', :id => 'ecookbook', :rev => 2},
404 398 :class => 'changeset', :title => 'This commit fixes #1, #2 and references #1 & #3')
405 399
406 400 to_test = {
407 401 # documents
408 402 'document:"Test document"' => 'document:"Test document"',
409 403 'ecookbook:document:"Test document"' => '<a href="/documents/1" class="document">Test document</a>',
410 404 'invalid:document:"Test document"' => 'invalid:document:"Test document"',
411 405 # versions
412 406 'version:"1.0"' => 'version:"1.0"',
413 407 'ecookbook:version:"1.0"' => '<a href="/versions/2" class="version">1.0</a>',
414 408 'invalid:version:"1.0"' => 'invalid:version:"1.0"',
415 409 # changeset
416 410 'r2' => 'r2',
417 411 'ecookbook:r2' => changeset_link,
418 412 'invalid:r2' => 'invalid:r2',
419 413 # source
420 414 'source:/some/file' => 'source:/some/file',
421 415 'ecookbook:source:/some/file' => source_link,
422 416 'invalid:source:/some/file' => 'invalid:source:/some/file',
423 417 }
424 418 @project = Project.find(3)
425 419 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text), "#{text} failed" }
426 420 end
427 421
428 422 def test_multiple_repositories_redmine_links
429 423 svn = Repository::Subversion.create!(:project_id => 1, :identifier => 'svn_repo-1', :url => 'file:///foo/hg')
430 424 Changeset.create!(:repository => svn, :committed_on => Time.now, :revision => '123')
431 425 hg = Repository::Mercurial.create!(:project_id => 1, :identifier => 'hg1', :url => '/foo/hg')
432 426 Changeset.create!(:repository => hg, :committed_on => Time.now, :revision => '123', :scmid => 'abcd')
433 427
434 428 changeset_link = link_to('r2', {:controller => 'repositories', :action => 'revision', :id => 'ecookbook', :rev => 2},
435 429 :class => 'changeset', :title => 'This commit fixes #1, #2 and references #1 & #3')
436 430 svn_changeset_link = link_to('svn_repo-1|r123', {:controller => 'repositories', :action => 'revision', :id => 'ecookbook', :repository_id => 'svn_repo-1', :rev => 123},
437 431 :class => 'changeset', :title => '')
438 432 hg_changeset_link = link_to('hg1|abcd', {:controller => 'repositories', :action => 'revision', :id => 'ecookbook', :repository_id => 'hg1', :rev => 'abcd'},
439 433 :class => 'changeset', :title => '')
440 434
441 435 source_link = link_to('source:some/file', {:controller => 'repositories', :action => 'entry', :id => 'ecookbook', :path => ['some', 'file']}, :class => 'source')
442 436 hg_source_link = link_to('source:hg1|some/file', {:controller => 'repositories', :action => 'entry', :id => 'ecookbook', :repository_id => 'hg1', :path => ['some', 'file']}, :class => 'source')
443 437
444 438 to_test = {
445 439 'r2' => changeset_link,
446 440 'svn_repo-1|r123' => svn_changeset_link,
447 441 'invalid|r123' => 'invalid|r123',
448 442 'commit:hg1|abcd' => hg_changeset_link,
449 443 'commit:invalid|abcd' => 'commit:invalid|abcd',
450 444 # source
451 445 'source:some/file' => source_link,
452 446 'source:hg1|some/file' => hg_source_link,
453 447 'source:invalid|some/file' => 'source:invalid|some/file',
454 448 }
455 449
456 450 @project = Project.find(1)
457 451 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text), "#{text} failed" }
458 452 end
459 453
460 454 def test_cross_project_multiple_repositories_redmine_links
461 455 svn = Repository::Subversion.create!(:project_id => 1, :identifier => 'svn1', :url => 'file:///foo/hg')
462 456 Changeset.create!(:repository => svn, :committed_on => Time.now, :revision => '123')
463 457 hg = Repository::Mercurial.create!(:project_id => 1, :identifier => 'hg1', :url => '/foo/hg')
464 458 Changeset.create!(:repository => hg, :committed_on => Time.now, :revision => '123', :scmid => 'abcd')
465 459
466 460 changeset_link = link_to('ecookbook:r2', {:controller => 'repositories', :action => 'revision', :id => 'ecookbook', :rev => 2},
467 461 :class => 'changeset', :title => 'This commit fixes #1, #2 and references #1 & #3')
468 462 svn_changeset_link = link_to('ecookbook:svn1|r123', {:controller => 'repositories', :action => 'revision', :id => 'ecookbook', :repository_id => 'svn1', :rev => 123},
469 463 :class => 'changeset', :title => '')
470 464 hg_changeset_link = link_to('ecookbook:hg1|abcd', {:controller => 'repositories', :action => 'revision', :id => 'ecookbook', :repository_id => 'hg1', :rev => 'abcd'},
471 465 :class => 'changeset', :title => '')
472 466
473 467 source_link = link_to('ecookbook:source:some/file', {:controller => 'repositories', :action => 'entry', :id => 'ecookbook', :path => ['some', 'file']}, :class => 'source')
474 468 hg_source_link = link_to('ecookbook:source:hg1|some/file', {:controller => 'repositories', :action => 'entry', :id => 'ecookbook', :repository_id => 'hg1', :path => ['some', 'file']}, :class => 'source')
475 469
476 470 to_test = {
477 471 'ecookbook:r2' => changeset_link,
478 472 'ecookbook:svn1|r123' => svn_changeset_link,
479 473 'ecookbook:invalid|r123' => 'ecookbook:invalid|r123',
480 474 'ecookbook:commit:hg1|abcd' => hg_changeset_link,
481 475 'ecookbook:commit:invalid|abcd' => 'ecookbook:commit:invalid|abcd',
482 476 'invalid:commit:invalid|abcd' => 'invalid:commit:invalid|abcd',
483 477 # source
484 478 'ecookbook:source:some/file' => source_link,
485 479 'ecookbook:source:hg1|some/file' => hg_source_link,
486 480 'ecookbook:source:invalid|some/file' => 'ecookbook:source:invalid|some/file',
487 481 'invalid:source:invalid|some/file' => 'invalid:source:invalid|some/file',
488 482 }
489 483
490 484 @project = Project.find(3)
491 485 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text), "#{text} failed" }
492 486 end
493 487
494 488 def test_redmine_links_git_commit
495 489 changeset_link = link_to('abcd',
496 490 {
497 491 :controller => 'repositories',
498 492 :action => 'revision',
499 493 :id => 'subproject1',
500 494 :rev => 'abcd',
501 495 },
502 496 :class => 'changeset', :title => 'test commit')
503 497 to_test = {
504 498 'commit:abcd' => changeset_link,
505 499 }
506 500 @project = Project.find(3)
507 501 r = Repository::Git.create!(:project => @project, :url => '/tmp/test/git')
508 502 assert r
509 503 c = Changeset.new(:repository => r,
510 504 :committed_on => Time.now,
511 505 :revision => 'abcd',
512 506 :scmid => 'abcd',
513 507 :comments => 'test commit')
514 508 assert( c.save )
515 509 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text) }
516 510 end
517 511
518 512 # TODO: Bazaar commit id contains mail address, so it contains '@' and '_'.
519 513 def test_redmine_links_darcs_commit
520 514 changeset_link = link_to('20080308225258-98289-abcd456efg.gz',
521 515 {
522 516 :controller => 'repositories',
523 517 :action => 'revision',
524 518 :id => 'subproject1',
525 519 :rev => '123',
526 520 },
527 521 :class => 'changeset', :title => 'test commit')
528 522 to_test = {
529 523 'commit:20080308225258-98289-abcd456efg.gz' => changeset_link,
530 524 }
531 525 @project = Project.find(3)
532 526 r = Repository::Darcs.create!(
533 527 :project => @project, :url => '/tmp/test/darcs',
534 528 :log_encoding => 'UTF-8')
535 529 assert r
536 530 c = Changeset.new(:repository => r,
537 531 :committed_on => Time.now,
538 532 :revision => '123',
539 533 :scmid => '20080308225258-98289-abcd456efg.gz',
540 534 :comments => 'test commit')
541 535 assert( c.save )
542 536 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text) }
543 537 end
544 538
545 539 def test_redmine_links_mercurial_commit
546 540 changeset_link_rev = link_to('r123',
547 541 {
548 542 :controller => 'repositories',
549 543 :action => 'revision',
550 544 :id => 'subproject1',
551 545 :rev => '123' ,
552 546 },
553 547 :class => 'changeset', :title => 'test commit')
554 548 changeset_link_commit = link_to('abcd',
555 549 {
556 550 :controller => 'repositories',
557 551 :action => 'revision',
558 552 :id => 'subproject1',
559 553 :rev => 'abcd' ,
560 554 },
561 555 :class => 'changeset', :title => 'test commit')
562 556 to_test = {
563 557 'r123' => changeset_link_rev,
564 558 'commit:abcd' => changeset_link_commit,
565 559 }
566 560 @project = Project.find(3)
567 561 r = Repository::Mercurial.create!(:project => @project, :url => '/tmp/test')
568 562 assert r
569 563 c = Changeset.new(:repository => r,
570 564 :committed_on => Time.now,
571 565 :revision => '123',
572 566 :scmid => 'abcd',
573 567 :comments => 'test commit')
574 568 assert( c.save )
575 569 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text) }
576 570 end
577 571
578 572 def test_attachment_links
579 573 to_test = {
580 574 'attachment:error281.txt' => '<a href="/attachments/download/1/error281.txt" class="attachment">error281.txt</a>'
581 575 }
582 576 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text, :attachments => Issue.find(3).attachments), "#{text} failed" }
583 577 end
584 578
585 579 def test_attachment_link_should_link_to_latest_attachment
586 580 set_tmp_attachments_directory
587 581 a1 = Attachment.generate!(:filename => "test.txt", :created_on => 1.hour.ago)
588 582 a2 = Attachment.generate!(:filename => "test.txt")
589 583
590 584 assert_equal %(<p><a href="/attachments/download/#{a2.id}/test.txt" class="attachment">test.txt</a></p>),
591 585 textilizable('attachment:test.txt', :attachments => [a1, a2])
592 586 end
593 587
594 588 def test_wiki_links
595 589 to_test = {
596 590 '[[CookBook documentation]]' => '<a href="/projects/ecookbook/wiki/CookBook_documentation" class="wiki-page">CookBook documentation</a>',
597 591 '[[Another page|Page]]' => '<a href="/projects/ecookbook/wiki/Another_page" class="wiki-page">Page</a>',
598 592 # title content should be formatted
599 593 '[[Another page|With _styled_ *title*]]' => '<a href="/projects/ecookbook/wiki/Another_page" class="wiki-page">With <em>styled</em> <strong>title</strong></a>',
600 594 '[[Another page|With title containing <strong>HTML entities &amp; markups</strong>]]' => '<a href="/projects/ecookbook/wiki/Another_page" class="wiki-page">With title containing &lt;strong&gt;HTML entities &amp; markups&lt;/strong&gt;</a>',
601 595 # link with anchor
602 596 '[[CookBook documentation#One-section]]' => '<a href="/projects/ecookbook/wiki/CookBook_documentation#One-section" class="wiki-page">CookBook documentation</a>',
603 597 '[[Another page#anchor|Page]]' => '<a href="/projects/ecookbook/wiki/Another_page#anchor" class="wiki-page">Page</a>',
604 598 # UTF8 anchor
605 599 '[[Another_page#ВСст|ВСст]]' => %|<a href="/projects/ecookbook/wiki/Another_page##{CGI.escape 'ВСст'}" class="wiki-page">ВСст</a>|,
606 600 # page that doesn't exist
607 601 '[[Unknown page]]' => '<a href="/projects/ecookbook/wiki/Unknown_page" class="wiki-page new">Unknown page</a>',
608 602 '[[Unknown page|404]]' => '<a href="/projects/ecookbook/wiki/Unknown_page" class="wiki-page new">404</a>',
609 603 # link to another project wiki
610 604 '[[onlinestore:]]' => '<a href="/projects/onlinestore/wiki" class="wiki-page">onlinestore</a>',
611 605 '[[onlinestore:|Wiki]]' => '<a href="/projects/onlinestore/wiki" class="wiki-page">Wiki</a>',
612 606 '[[onlinestore:Start page]]' => '<a href="/projects/onlinestore/wiki/Start_page" class="wiki-page">Start page</a>',
613 607 '[[onlinestore:Start page|Text]]' => '<a href="/projects/onlinestore/wiki/Start_page" class="wiki-page">Text</a>',
614 608 '[[onlinestore:Unknown page]]' => '<a href="/projects/onlinestore/wiki/Unknown_page" class="wiki-page new">Unknown page</a>',
615 609 # striked through link
616 610 '-[[Another page|Page]]-' => '<del><a href="/projects/ecookbook/wiki/Another_page" class="wiki-page">Page</a></del>',
617 611 '-[[Another page|Page]] link-' => '<del><a href="/projects/ecookbook/wiki/Another_page" class="wiki-page">Page</a> link</del>',
618 612 # escaping
619 613 '![[Another page|Page]]' => '[[Another page|Page]]',
620 614 # project does not exist
621 615 '[[unknowproject:Start]]' => '[[unknowproject:Start]]',
622 616 '[[unknowproject:Start|Page title]]' => '[[unknowproject:Start|Page title]]',
623 617 }
624 618
625 619 @project = Project.find(1)
626 620 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text) }
627 621 end
628 622
629 623 def test_wiki_links_within_local_file_generation_context
630 624
631 625 to_test = {
632 626 # link to a page
633 627 '[[CookBook documentation]]' => '<a href="CookBook_documentation.html" class="wiki-page">CookBook documentation</a>',
634 628 '[[CookBook documentation|documentation]]' => '<a href="CookBook_documentation.html" class="wiki-page">documentation</a>',
635 629 '[[CookBook documentation#One-section]]' => '<a href="CookBook_documentation.html#One-section" class="wiki-page">CookBook documentation</a>',
636 630 '[[CookBook documentation#One-section|documentation]]' => '<a href="CookBook_documentation.html#One-section" class="wiki-page">documentation</a>',
637 631 # page that doesn't exist
638 632 '[[Unknown page]]' => '<a href="Unknown_page.html" class="wiki-page new">Unknown page</a>',
639 633 '[[Unknown page|404]]' => '<a href="Unknown_page.html" class="wiki-page new">404</a>',
640 634 '[[Unknown page#anchor]]' => '<a href="Unknown_page.html#anchor" class="wiki-page new">Unknown page</a>',
641 635 '[[Unknown page#anchor|404]]' => '<a href="Unknown_page.html#anchor" class="wiki-page new">404</a>',
642 636 }
643 637
644 638 @project = Project.find(1)
645 639
646 640 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text, :wiki_links => :local) }
647 641 end
648 642
649 643 def test_wiki_links_within_wiki_page_context
650 644
651 645 page = WikiPage.find_by_title('Another_page' )
652 646
653 647 to_test = {
654 648 # link to another page
655 649 '[[CookBook documentation]]' => '<a href="/projects/ecookbook/wiki/CookBook_documentation" class="wiki-page">CookBook documentation</a>',
656 650 '[[CookBook documentation|documentation]]' => '<a href="/projects/ecookbook/wiki/CookBook_documentation" class="wiki-page">documentation</a>',
657 651 '[[CookBook documentation#One-section]]' => '<a href="/projects/ecookbook/wiki/CookBook_documentation#One-section" class="wiki-page">CookBook documentation</a>',
658 652 '[[CookBook documentation#One-section|documentation]]' => '<a href="/projects/ecookbook/wiki/CookBook_documentation#One-section" class="wiki-page">documentation</a>',
659 653 # link to the current page
660 654 '[[Another page]]' => '<a href="/projects/ecookbook/wiki/Another_page" class="wiki-page">Another page</a>',
661 655 '[[Another page|Page]]' => '<a href="/projects/ecookbook/wiki/Another_page" class="wiki-page">Page</a>',
662 656 '[[Another page#anchor]]' => '<a href="#anchor" class="wiki-page">Another page</a>',
663 657 '[[Another page#anchor|Page]]' => '<a href="#anchor" class="wiki-page">Page</a>',
664 658 # page that doesn't exist
665 659 '[[Unknown page]]' => '<a href="/projects/ecookbook/wiki/Unknown_page?parent=Another_page" class="wiki-page new">Unknown page</a>',
666 660 '[[Unknown page|404]]' => '<a href="/projects/ecookbook/wiki/Unknown_page?parent=Another_page" class="wiki-page new">404</a>',
667 661 '[[Unknown page#anchor]]' => '<a href="/projects/ecookbook/wiki/Unknown_page?parent=Another_page#anchor" class="wiki-page new">Unknown page</a>',
668 662 '[[Unknown page#anchor|404]]' => '<a href="/projects/ecookbook/wiki/Unknown_page?parent=Another_page#anchor" class="wiki-page new">404</a>',
669 663 }
670 664
671 665 @project = Project.find(1)
672 666
673 667 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(WikiContent.new( :text => text, :page => page ), :text) }
674 668 end
675 669
676 670 def test_wiki_links_anchor_option_should_prepend_page_title_to_href
677 671
678 672 to_test = {
679 673 # link to a page
680 674 '[[CookBook documentation]]' => '<a href="#CookBook_documentation" class="wiki-page">CookBook documentation</a>',
681 675 '[[CookBook documentation|documentation]]' => '<a href="#CookBook_documentation" class="wiki-page">documentation</a>',
682 676 '[[CookBook documentation#One-section]]' => '<a href="#CookBook_documentation_One-section" class="wiki-page">CookBook documentation</a>',
683 677 '[[CookBook documentation#One-section|documentation]]' => '<a href="#CookBook_documentation_One-section" class="wiki-page">documentation</a>',
684 678 # page that doesn't exist
685 679 '[[Unknown page]]' => '<a href="#Unknown_page" class="wiki-page new">Unknown page</a>',
686 680 '[[Unknown page|404]]' => '<a href="#Unknown_page" class="wiki-page new">404</a>',
687 681 '[[Unknown page#anchor]]' => '<a href="#Unknown_page_anchor" class="wiki-page new">Unknown page</a>',
688 682 '[[Unknown page#anchor|404]]' => '<a href="#Unknown_page_anchor" class="wiki-page new">404</a>',
689 683 }
690 684
691 685 @project = Project.find(1)
692 686
693 687 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text, :wiki_links => :anchor) }
694 688 end
695 689
696 690 def test_html_tags
697 691 to_test = {
698 692 "<div>content</div>" => "<p>&lt;div&gt;content&lt;/div&gt;</p>",
699 693 "<div class=\"bold\">content</div>" => "<p>&lt;div class=\"bold\"&gt;content&lt;/div&gt;</p>",
700 694 "<script>some script;</script>" => "<p>&lt;script&gt;some script;&lt;/script&gt;</p>",
701 695 # do not escape pre/code tags
702 696 "<pre>\nline 1\nline2</pre>" => "<pre>\nline 1\nline2</pre>",
703 697 "<pre><code>\nline 1\nline2</code></pre>" => "<pre><code>\nline 1\nline2</code></pre>",
704 698 "<pre><div>content</div></pre>" => "<pre>&lt;div&gt;content&lt;/div&gt;</pre>",
705 699 "HTML comment: <!-- no comments -->" => "<p>HTML comment: &lt;!-- no comments --&gt;</p>",
706 700 "<!-- opening comment" => "<p>&lt;!-- opening comment</p>",
707 701 # remove attributes except class
708 702 "<pre class='foo'>some text</pre>" => "<pre class='foo'>some text</pre>",
709 703 '<pre class="foo">some text</pre>' => '<pre class="foo">some text</pre>',
710 704 "<pre class='foo bar'>some text</pre>" => "<pre class='foo bar'>some text</pre>",
711 705 '<pre class="foo bar">some text</pre>' => '<pre class="foo bar">some text</pre>',
712 706 "<pre onmouseover='alert(1)'>some text</pre>" => "<pre>some text</pre>",
713 707 # xss
714 708 '<pre><code class=""onmouseover="alert(1)">text</code></pre>' => '<pre><code>text</code></pre>',
715 709 '<pre class=""onmouseover="alert(1)">text</pre>' => '<pre>text</pre>',
716 710 }
717 711 to_test.each { |text, result| assert_equal result, textilizable(text) }
718 712 end
719 713
720 714 def test_allowed_html_tags
721 715 to_test = {
722 716 "<pre>preformatted text</pre>" => "<pre>preformatted text</pre>",
723 717 "<notextile>no *textile* formatting</notextile>" => "no *textile* formatting",
724 718 "<notextile>this is <tag>a tag</tag></notextile>" => "this is &lt;tag&gt;a tag&lt;/tag&gt;"
725 719 }
726 720 to_test.each { |text, result| assert_equal result, textilizable(text) }
727 721 end
728 722
729 723 def test_pre_tags
730 724 raw = <<-RAW
731 725 Before
732 726
733 727 <pre>
734 728 <prepared-statement-cache-size>32</prepared-statement-cache-size>
735 729 </pre>
736 730
737 731 After
738 732 RAW
739 733
740 734 expected = <<-EXPECTED
741 735 <p>Before</p>
742 736 <pre>
743 737 &lt;prepared-statement-cache-size&gt;32&lt;/prepared-statement-cache-size&gt;
744 738 </pre>
745 739 <p>After</p>
746 740 EXPECTED
747 741
748 742 assert_equal expected.gsub(%r{[\r\n\t]}, ''), textilizable(raw).gsub(%r{[\r\n\t]}, '')
749 743 end
750 744
751 745 def test_pre_content_should_not_parse_wiki_and_redmine_links
752 746 raw = <<-RAW
753 747 [[CookBook documentation]]
754 748
755 749 #1
756 750
757 751 <pre>
758 752 [[CookBook documentation]]
759 753
760 754 #1
761 755 </pre>
762 756 RAW
763 757
764 758 expected = <<-EXPECTED
765 759 <p><a href="/projects/ecookbook/wiki/CookBook_documentation" class="wiki-page">CookBook documentation</a></p>
766 760 <p><a href="/issues/1" class="#{Issue.find(1).css_classes}" title="Can&#x27;t print recipes (New)">#1</a></p>
767 761 <pre>
768 762 [[CookBook documentation]]
769 763
770 764 #1
771 765 </pre>
772 766 EXPECTED
773 767
774 768 @project = Project.find(1)
775 769 assert_equal expected.gsub(%r{[\r\n\t]}, ''), textilizable(raw).gsub(%r{[\r\n\t]}, '')
776 770 end
777 771
778 772 def test_non_closing_pre_blocks_should_be_closed
779 773 raw = <<-RAW
780 774 <pre><code>
781 775 RAW
782 776
783 777 expected = <<-EXPECTED
784 778 <pre><code>
785 779 </code></pre>
786 780 EXPECTED
787 781
788 782 @project = Project.find(1)
789 783 assert_equal expected.gsub(%r{[\r\n\t]}, ''), textilizable(raw).gsub(%r{[\r\n\t]}, '')
790 784 end
791 785
792 786 def test_syntax_highlight
793 787 raw = <<-RAW
794 788 <pre><code class="ruby">
795 789 # Some ruby code here
796 790 </code></pre>
797 791 RAW
798 792
799 793 expected = <<-EXPECTED
800 794 <pre><code class="ruby syntaxhl"><span class=\"CodeRay\"><span class="comment"># Some ruby code here</span></span>
801 795 </code></pre>
802 796 EXPECTED
803 797
804 798 assert_equal expected.gsub(%r{[\r\n\t]}, ''), textilizable(raw).gsub(%r{[\r\n\t]}, '')
805 799 end
806 800
807 801 def test_to_path_param
808 802 assert_equal 'test1/test2', to_path_param('test1/test2')
809 803 assert_equal 'test1/test2', to_path_param('/test1/test2/')
810 804 assert_equal 'test1/test2', to_path_param('//test1/test2/')
811 805 assert_equal nil, to_path_param('/')
812 806 end
813 807
814 808 def test_wiki_links_in_tables
815 809 to_test = {"|[[Page|Link title]]|[[Other Page|Other title]]|\n|Cell 21|[[Last page]]|" =>
816 810 '<tr><td><a href="/projects/ecookbook/wiki/Page" class="wiki-page new">Link title</a></td>' +
817 811 '<td><a href="/projects/ecookbook/wiki/Other_Page" class="wiki-page new">Other title</a></td>' +
818 812 '</tr><tr><td>Cell 21</td><td><a href="/projects/ecookbook/wiki/Last_page" class="wiki-page new">Last page</a></td></tr>'
819 813 }
820 814 @project = Project.find(1)
821 815 to_test.each { |text, result| assert_equal "<table>#{result}</table>", textilizable(text).gsub(/[\t\n]/, '') }
822 816 end
823 817
824 818 def test_text_formatting
825 819 to_test = {'*_+bold, italic and underline+_*' => '<strong><em><ins>bold, italic and underline</ins></em></strong>',
826 820 '(_text within parentheses_)' => '(<em>text within parentheses</em>)',
827 821 'a *Humane Web* Text Generator' => 'a <strong>Humane Web</strong> Text Generator',
828 822 'a H *umane* W *eb* T *ext* G *enerator*' => 'a H <strong>umane</strong> W <strong>eb</strong> T <strong>ext</strong> G <strong>enerator</strong>',
829 823 'a *H* umane *W* eb *T* ext *G* enerator' => 'a <strong>H</strong> umane <strong>W</strong> eb <strong>T</strong> ext <strong>G</strong> enerator',
830 824 }
831 825 to_test.each { |text, result| assert_equal "<p>#{result}</p>", textilizable(text) }
832 826 end
833 827
834 828 def test_wiki_horizontal_rule
835 829 assert_equal '<hr />', textilizable('---')
836 830 assert_equal '<p>Dashes: ---</p>', textilizable('Dashes: ---')
837 831 end
838 832
839 833 def test_footnotes
840 834 raw = <<-RAW
841 835 This is some text[1].
842 836
843 837 fn1. This is the foot note
844 838 RAW
845 839
846 840 expected = <<-EXPECTED
847 841 <p>This is some text<sup><a href=\"#fn1\">1</a></sup>.</p>
848 842 <p id="fn1" class="footnote"><sup>1</sup> This is the foot note</p>
849 843 EXPECTED
850 844
851 845 assert_equal expected.gsub(%r{[\r\n\t]}, ''), textilizable(raw).gsub(%r{[\r\n\t]}, '')
852 846 end
853 847
854 848 def test_headings
855 849 raw = 'h1. Some heading'
856 850 expected = %|<a name="Some-heading"></a>\n<h1 >Some heading<a href="#Some-heading" class="wiki-anchor">&para;</a></h1>|
857 851
858 852 assert_equal expected, textilizable(raw)
859 853 end
860 854
861 855 def test_headings_with_special_chars
862 856 # This test makes sure that the generated anchor names match the expected
863 857 # ones even if the heading text contains unconventional characters
864 858 raw = 'h1. Some heading related to version 0.5'
865 859 anchor = sanitize_anchor_name("Some-heading-related-to-version-0.5")
866 860 expected = %|<a name="#{anchor}"></a>\n<h1 >Some heading related to version 0.5<a href="##{anchor}" class="wiki-anchor">&para;</a></h1>|
867 861
868 862 assert_equal expected, textilizable(raw)
869 863 end
870 864
871 865 def test_headings_in_wiki_single_page_export_should_be_prepended_with_page_title
872 866 page = WikiPage.new( :title => 'Page Title', :wiki_id => 1 )
873 867 content = WikiContent.new( :text => 'h1. Some heading', :page => page )
874 868
875 869 expected = %|<a name="Page_Title_Some-heading"></a>\n<h1 >Some heading<a href="#Page_Title_Some-heading" class="wiki-anchor">&para;</a></h1>|
876 870
877 871 assert_equal expected, textilizable(content, :text, :wiki_links => :anchor )
878 872 end
879 873
880 874 def test_table_of_content
881 875 raw = <<-RAW
882 876 {{toc}}
883 877
884 878 h1. Title
885 879
886 880 Lorem ipsum dolor sit amet, consectetuer adipiscing elit. Maecenas sed libero.
887 881
888 882 h2. Subtitle with a [[Wiki]] link
889 883
890 884 Nullam commodo metus accumsan nulla. Curabitur lobortis dui id dolor.
891 885
892 886 h2. Subtitle with [[Wiki|another Wiki]] link
893 887
894 888 h2. Subtitle with %{color:red}red text%
895 889
896 890 <pre>
897 891 some code
898 892 </pre>
899 893
900 894 h3. Subtitle with *some* _modifiers_
901 895
902 896 h3. Subtitle with @inline code@
903 897
904 898 h1. Another title
905 899
906 900 h3. An "Internet link":http://www.redmine.org/ inside subtitle
907 901
908 902 h2. "Project Name !/attachments/1234/logo_small.gif! !/attachments/5678/logo_2.png!":/projects/projectname/issues
909 903
910 904 RAW
911 905
912 906 expected = '<ul class="toc">' +
913 907 '<li><a href="#Title">Title</a>' +
914 908 '<ul>' +
915 909 '<li><a href="#Subtitle-with-a-Wiki-link">Subtitle with a Wiki link</a></li>' +
916 910 '<li><a href="#Subtitle-with-another-Wiki-link">Subtitle with another Wiki link</a></li>' +
917 911 '<li><a href="#Subtitle-with-red-text">Subtitle with red text</a>' +
918 912 '<ul>' +
919 913 '<li><a href="#Subtitle-with-some-modifiers">Subtitle with some modifiers</a></li>' +
920 914 '<li><a href="#Subtitle-with-inline-code">Subtitle with inline code</a></li>' +
921 915 '</ul>' +
922 916 '</li>' +
923 917 '</ul>' +
924 918 '</li>' +
925 919 '<li><a href="#Another-title">Another title</a>' +
926 920 '<ul>' +
927 921 '<li>' +
928 922 '<ul>' +
929 923 '<li><a href="#An-Internet-link-inside-subtitle">An Internet link inside subtitle</a></li>' +
930 924 '</ul>' +
931 925 '</li>' +
932 926 '<li><a href="#Project-Name">Project Name</a></li>' +
933 927 '</ul>' +
934 928 '</li>' +
935 929 '</ul>'
936 930
937 931 @project = Project.find(1)
938 932 assert textilizable(raw).gsub("\n", "").include?(expected)
939 933 end
940 934
941 935 def test_table_of_content_should_generate_unique_anchors
942 936 raw = <<-RAW
943 937 {{toc}}
944 938
945 939 h1. Title
946 940
947 941 h2. Subtitle
948 942
949 943 h2. Subtitle
950 944 RAW
951 945
952 946 expected = '<ul class="toc">' +
953 947 '<li><a href="#Title">Title</a>' +
954 948 '<ul>' +
955 949 '<li><a href="#Subtitle">Subtitle</a></li>' +
956 950 '<li><a href="#Subtitle-2">Subtitle</a></li>'
957 951 '</ul>'
958 952 '</li>' +
959 953 '</ul>'
960 954
961 955 @project = Project.find(1)
962 956 result = textilizable(raw).gsub("\n", "")
963 957 assert_include expected, result
964 958 assert_include '<a name="Subtitle">', result
965 959 assert_include '<a name="Subtitle-2">', result
966 960 end
967 961
968 962 def test_table_of_content_should_contain_included_page_headings
969 963 raw = <<-RAW
970 964 {{toc}}
971 965
972 966 h1. Included
973 967
974 968 {{include(Child_1)}}
975 969 RAW
976 970
977 971 expected = '<ul class="toc">' +
978 972 '<li><a href="#Included">Included</a></li>' +
979 973 '<li><a href="#Child-page-1">Child page 1</a></li>' +
980 974 '</ul>'
981 975
982 976 @project = Project.find(1)
983 977 assert textilizable(raw).gsub("\n", "").include?(expected)
984 978 end
985 979
986 980 def test_section_edit_links
987 981 raw = <<-RAW
988 982 h1. Title
989 983
990 984 Lorem ipsum dolor sit amet, consectetuer adipiscing elit. Maecenas sed libero.
991 985
992 986 h2. Subtitle with a [[Wiki]] link
993 987
994 988 h2. Subtitle with *some* _modifiers_
995 989
996 990 h2. Subtitle with @inline code@
997 991
998 992 <pre>
999 993 some code
1000 994
1001 995 h2. heading inside pre
1002 996
1003 997 <h2>html heading inside pre</h2>
1004 998 </pre>
1005 999
1006 1000 h2. Subtitle after pre tag
1007 1001 RAW
1008 1002
1009 1003 @project = Project.find(1)
1010 1004 set_language_if_valid 'en'
1011 1005 result = textilizable(raw, :edit_section_links => {:controller => 'wiki', :action => 'edit', :project_id => '1', :id => 'Test'}).gsub("\n", "")
1012 1006
1013 1007 # heading that contains inline code
1014 1008 assert_match Regexp.new('<div class="contextual" title="Edit this section">' +
1015 1009 '<a href="/projects/1/wiki/Test/edit\?section=4"><img alt="Edit" src="/images/edit.png(\?\d+)?" /></a></div>' +
1016 1010 '<a name="Subtitle-with-inline-code"></a>' +
1017 1011 '<h2 >Subtitle with <code>inline code</code><a href="#Subtitle-with-inline-code" class="wiki-anchor">&para;</a></h2>'),
1018 1012 result
1019 1013
1020 1014 # last heading
1021 1015 assert_match Regexp.new('<div class="contextual" title="Edit this section">' +
1022 1016 '<a href="/projects/1/wiki/Test/edit\?section=5"><img alt="Edit" src="/images/edit.png(\?\d+)?" /></a></div>' +
1023 1017 '<a name="Subtitle-after-pre-tag"></a>' +
1024 1018 '<h2 >Subtitle after pre tag<a href="#Subtitle-after-pre-tag" class="wiki-anchor">&para;</a></h2>'),
1025 1019 result
1026 1020 end
1027 1021
1028 1022 def test_default_formatter
1029 1023 with_settings :text_formatting => 'unknown' do
1030 1024 text = 'a *link*: http://www.example.net/'
1031 1025 assert_equal '<p>a *link*: <a class="external" href="http://www.example.net/">http://www.example.net/</a></p>', textilizable(text)
1032 1026 end
1033 1027 end
1034 1028
1035 1029 def test_due_date_distance_in_words
1036 1030 to_test = { Date.today => 'Due in 0 days',
1037 1031 Date.today + 1 => 'Due in 1 day',
1038 1032 Date.today + 100 => 'Due in about 3 months',
1039 1033 Date.today + 20000 => 'Due in over 54 years',
1040 1034 Date.today - 1 => '1 day late',
1041 1035 Date.today - 100 => 'about 3 months late',
1042 1036 Date.today - 20000 => 'over 54 years late',
1043 1037 }
1044 1038 ::I18n.locale = :en
1045 1039 to_test.each do |date, expected|
1046 1040 assert_equal expected, due_date_distance_in_words(date)
1047 1041 end
1048 1042 end
1049 1043
1050 1044 def test_avatar_enabled
1051 1045 with_settings :gravatar_enabled => '1' do
1052 1046 assert avatar(User.find_by_mail('jsmith@somenet.foo')).include?(Digest::MD5.hexdigest('jsmith@somenet.foo'))
1053 1047 assert avatar('jsmith <jsmith@somenet.foo>').include?(Digest::MD5.hexdigest('jsmith@somenet.foo'))
1054 1048 # Default size is 50
1055 1049 assert avatar('jsmith <jsmith@somenet.foo>').include?('size=50')
1056 1050 assert avatar('jsmith <jsmith@somenet.foo>', :size => 24).include?('size=24')
1057 1051 # Non-avatar options should be considered html options
1058 1052 assert avatar('jsmith <jsmith@somenet.foo>', :title => 'John Smith').include?('title="John Smith"')
1059 1053 # The default class of the img tag should be gravatar
1060 1054 assert avatar('jsmith <jsmith@somenet.foo>').include?('class="gravatar"')
1061 1055 assert !avatar('jsmith <jsmith@somenet.foo>', :class => 'picture').include?('class="gravatar"')
1062 1056 assert_nil avatar('jsmith')
1063 1057 assert_nil avatar(nil)
1064 1058 end
1065 1059 end
1066 1060
1067 1061 def test_avatar_disabled
1068 1062 with_settings :gravatar_enabled => '0' do
1069 1063 assert_equal '', avatar(User.find_by_mail('jsmith@somenet.foo'))
1070 1064 end
1071 1065 end
1072 1066
1073 1067 def test_link_to_user
1074 1068 user = User.find(2)
1075 1069 assert_equal '<a href="/users/2" class="user active">John Smith</a>', link_to_user(user)
1076 1070 end
1077 1071
1078 1072 def test_link_to_user_should_not_link_to_locked_user
1079 1073 with_current_user nil do
1080 1074 user = User.find(5)
1081 1075 assert user.locked?
1082 1076 assert_equal 'Dave2 Lopper2', link_to_user(user)
1083 1077 end
1084 1078 end
1085 1079
1086 1080 def test_link_to_user_should_link_to_locked_user_if_current_user_is_admin
1087 1081 with_current_user User.find(1) do
1088 1082 user = User.find(5)
1089 1083 assert user.locked?
1090 1084 assert_equal '<a href="/users/5" class="user locked">Dave2 Lopper2</a>', link_to_user(user)
1091 1085 end
1092 1086 end
1093 1087
1094 1088 def test_link_to_user_should_not_link_to_anonymous
1095 1089 user = User.anonymous
1096 1090 assert user.anonymous?
1097 1091 t = link_to_user(user)
1098 1092 assert_equal ::I18n.t(:label_user_anonymous), t
1099 1093 end
1100 1094
1101 1095 def test_link_to_attachment
1102 1096 a = Attachment.find(3)
1103 1097 assert_equal '<a href="/attachments/3/logo.gif">logo.gif</a>',
1104 1098 link_to_attachment(a)
1105 1099 assert_equal '<a href="/attachments/3/logo.gif">Text</a>',
1106 1100 link_to_attachment(a, :text => 'Text')
1107 1101 assert_equal '<a href="/attachments/3/logo.gif" class="foo">logo.gif</a>',
1108 1102 link_to_attachment(a, :class => 'foo')
1109 1103 assert_equal '<a href="/attachments/download/3/logo.gif">logo.gif</a>',
1110 1104 link_to_attachment(a, :download => true)
1111 1105 assert_equal '<a href="http://test.host/attachments/3/logo.gif">logo.gif</a>',
1112 1106 link_to_attachment(a, :only_path => false)
1113 1107 end
1114 1108
1115 1109 def test_thumbnail_tag
1116 1110 a = Attachment.find(3)
1117 1111 assert_equal '<a href="/attachments/3/logo.gif" title="logo.gif"><img alt="3" src="/attachments/thumbnail/3" /></a>',
1118 1112 thumbnail_tag(a)
1119 1113 end
1120 1114
1121 1115 def test_link_to_project
1122 1116 project = Project.find(1)
1123 1117 assert_equal %(<a href="/projects/ecookbook">eCookbook</a>),
1124 1118 link_to_project(project)
1125 1119 assert_equal %(<a href="/projects/ecookbook/settings">eCookbook</a>),
1126 1120 link_to_project(project, :action => 'settings')
1127 1121 assert_equal %(<a href="http://test.host/projects/ecookbook?jump=blah">eCookbook</a>),
1128 1122 link_to_project(project, {:only_path => false, :jump => 'blah'})
1129 1123 assert_equal %(<a href="/projects/ecookbook/settings" class="project">eCookbook</a>),
1130 1124 link_to_project(project, {:action => 'settings'}, :class => "project")
1131 1125 end
1132 1126
1133 1127 def test_link_to_project_settings
1134 1128 project = Project.find(1)
1135 1129 assert_equal '<a href="/projects/ecookbook/settings">eCookbook</a>', link_to_project_settings(project)
1136 1130
1137 1131 project.status = Project::STATUS_CLOSED
1138 1132 assert_equal '<a href="/projects/ecookbook">eCookbook</a>', link_to_project_settings(project)
1139 1133
1140 1134 project.status = Project::STATUS_ARCHIVED
1141 1135 assert_equal 'eCookbook', link_to_project_settings(project)
1142 1136 end
1143 1137
1144 1138 def test_link_to_legacy_project_with_numerical_identifier_should_use_id
1145 1139 # numeric identifier are no longer allowed
1146 1140 Project.update_all "identifier=25", "id=1"
1147 1141
1148 1142 assert_equal '<a href="/projects/1">eCookbook</a>',
1149 1143 link_to_project(Project.find(1))
1150 1144 end
1151 1145
1152 1146 def test_principals_options_for_select_with_users
1153 1147 User.current = nil
1154 1148 users = [User.find(2), User.find(4)]
1155 1149 assert_equal %(<option value="2">John Smith</option><option value="4">Robert Hill</option>),
1156 1150 principals_options_for_select(users)
1157 1151 end
1158 1152
1159 1153 def test_principals_options_for_select_with_selected
1160 1154 User.current = nil
1161 1155 users = [User.find(2), User.find(4)]
1162 1156 assert_equal %(<option value="2">John Smith</option><option value="4" selected="selected">Robert Hill</option>),
1163 1157 principals_options_for_select(users, User.find(4))
1164 1158 end
1165 1159
1166 1160 def test_principals_options_for_select_with_users_and_groups
1167 1161 User.current = nil
1168 1162 users = [User.find(2), Group.find(11), User.find(4), Group.find(10)]
1169 1163 assert_equal %(<option value="2">John Smith</option><option value="4">Robert Hill</option>) +
1170 1164 %(<optgroup label="Groups"><option value="10">A Team</option><option value="11">B Team</option></optgroup>),
1171 1165 principals_options_for_select(users)
1172 1166 end
1173 1167
1174 1168 def test_principals_options_for_select_with_empty_collection
1175 1169 assert_equal '', principals_options_for_select([])
1176 1170 end
1177 1171
1178 1172 def test_principals_options_for_select_should_include_me_option_when_current_user_is_in_collection
1179 1173 users = [User.find(2), User.find(4)]
1180 1174 User.current = User.find(4)
1181 1175 assert_include '<option value="4">&lt;&lt; me &gt;&gt;</option>', principals_options_for_select(users)
1182 1176 end
1183 1177
1184 1178 def test_stylesheet_link_tag_should_pick_the_default_stylesheet
1185 1179 assert_match 'href="/stylesheets/styles.css"', stylesheet_link_tag("styles")
1186 1180 end
1187 1181
1188 1182 def test_stylesheet_link_tag_for_plugin_should_pick_the_plugin_stylesheet
1189 1183 assert_match 'href="/plugin_assets/foo/stylesheets/styles.css"', stylesheet_link_tag("styles", :plugin => :foo)
1190 1184 end
1191 1185
1192 1186 def test_image_tag_should_pick_the_default_image
1193 1187 assert_match 'src="/images/image.png"', image_tag("image.png")
1194 1188 end
1195 1189
1196 1190 def test_image_tag_should_pick_the_theme_image_if_it_exists
1197 1191 theme = Redmine::Themes.themes.last
1198 1192 theme.images << 'image.png'
1199 1193
1200 1194 with_settings :ui_theme => theme.id do
1201 1195 assert_match %|src="/themes/#{theme.dir}/images/image.png"|, image_tag("image.png")
1202 1196 assert_match %|src="/images/other.png"|, image_tag("other.png")
1203 1197 end
1204 1198 ensure
1205 1199 theme.images.delete 'image.png'
1206 1200 end
1207 1201
1208 1202 def test_image_tag_sfor_plugin_should_pick_the_plugin_image
1209 1203 assert_match 'src="/plugin_assets/foo/images/image.png"', image_tag("image.png", :plugin => :foo)
1210 1204 end
1211 1205
1212 1206 def test_javascript_include_tag_should_pick_the_default_javascript
1213 1207 assert_match 'src="/javascripts/scripts.js"', javascript_include_tag("scripts")
1214 1208 end
1215 1209
1216 1210 def test_javascript_include_tag_for_plugin_should_pick_the_plugin_javascript
1217 1211 assert_match 'src="/plugin_assets/foo/javascripts/scripts.js"', javascript_include_tag("scripts", :plugin => :foo)
1218 1212 end
1219 1213 end
@@ -1,145 +1,129
1 1 # Redmine - project management software
2 2 # Copyright (C) 2006-2013 Jean-Philippe Lang
3 3 #
4 4 # This program is free software; you can redistribute it and/or
5 5 # modify it under the terms of the GNU General Public License
6 6 # as published by the Free Software Foundation; either version 2
7 7 # of the License, or (at your option) any later version.
8 8 #
9 9 # This program is distributed in the hope that it will be useful,
10 10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
11 11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 12 # GNU General Public License for more details.
13 13 #
14 14 # You should have received a copy of the GNU General Public License
15 15 # along with this program; if not, write to the Free Software
16 16 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
17 17
18 18 require File.expand_path('../../test_helper', __FILE__)
19 19
20 20 class RoleTest < ActiveSupport::TestCase
21 21 fixtures :roles, :workflows, :trackers
22 22
23 23 def test_sorted_scope
24 24 assert_equal Role.all.sort, Role.sorted.all
25 25 end
26 26
27 27 def test_givable_scope
28 28 assert_equal Role.all.reject(&:builtin?).sort, Role.givable.all
29 29 end
30 30
31 31 def test_builtin_scope
32 32 assert_equal Role.all.select(&:builtin?).sort, Role.builtin(true).all.sort
33 33 assert_equal Role.all.reject(&:builtin?).sort, Role.builtin(false).all.sort
34 34 end
35 35
36 36 def test_copy_from
37 37 role = Role.find(1)
38 38 copy = Role.new.copy_from(role)
39 39
40 40 assert_nil copy.id
41 41 assert_equal '', copy.name
42 42 assert_equal role.permissions, copy.permissions
43 43
44 44 copy.name = 'Copy'
45 45 assert copy.save
46 46 end
47 47
48 48 def test_copy_workflows
49 49 source = Role.find(1)
50 50 assert_equal 90, source.workflow_rules.size
51 51
52 52 target = Role.new(:name => 'Target')
53 53 assert target.save
54 54 target.workflow_rules.copy(source)
55 55 target.reload
56 56 assert_equal 90, target.workflow_rules.size
57 57 end
58 58
59 59 def test_permissions_should_be_unserialized_with_its_coder
60 60 Role::PermissionsAttributeCoder.expects(:load).once
61 61 Role.find(1).permissions
62 62 end
63 63
64 64 def test_add_permission
65 65 role = Role.find(1)
66 66 size = role.permissions.size
67 67 role.add_permission!("apermission", "anotherpermission")
68 68 role.reload
69 69 assert role.permissions.include?(:anotherpermission)
70 70 assert_equal size + 2, role.permissions.size
71 71 end
72 72
73 73 def test_remove_permission
74 74 role = Role.find(1)
75 75 size = role.permissions.size
76 76 perm = role.permissions[0..1]
77 77 role.remove_permission!(*perm)
78 78 role.reload
79 79 assert ! role.permissions.include?(perm[0])
80 80 assert_equal size - 2, role.permissions.size
81 81 end
82 82
83 83 def test_name
84 84 I18n.locale = 'fr'
85 85 assert_equal 'Manager', Role.find(1).name
86 86 assert_equal 'Anonyme', Role.anonymous.name
87 87 assert_equal 'Non membre', Role.non_member.name
88 88 end
89 89
90 90 def test_find_all_givable
91 91 assert_equal Role.all.reject(&:builtin?).sort, Role.find_all_givable
92 92 end
93 93
94 context "#anonymous" do
95 should "return the anonymous role" do
94 def test_anonymous_should_return_the_anonymous_role
95 assert_no_difference('Role.count') do
96 96 role = Role.anonymous
97 97 assert role.builtin?
98 98 assert_equal Role::BUILTIN_ANONYMOUS, role.builtin
99 99 end
100 end
101
102 def test_anonymous_with_a_missing_anonymous_role_should_return_the_anonymous_role
103 Role.where(:builtin => Role::BUILTIN_ANONYMOUS).delete_all
100 104
101 context "with a missing anonymous role" do
102 setup do
103 Role.delete_all("builtin = #{Role::BUILTIN_ANONYMOUS}")
104 end
105
106 should "create a new anonymous role" do
107 assert_difference('Role.count') do
108 Role.anonymous
109 end
110 end
111
112 should "return the anonymous role" do
113 role = Role.anonymous
114 assert role.builtin?
115 assert_equal Role::BUILTIN_ANONYMOUS, role.builtin
116 end
105 assert_difference('Role.count') do
106 role = Role.anonymous
107 assert role.builtin?
108 assert_equal Role::BUILTIN_ANONYMOUS, role.builtin
117 109 end
118 110 end
119 111
120 context "#non_member" do
121 should "return the non-member role" do
112 def test_non_member_should_return_the_non_member_role
113 assert_no_difference('Role.count') do
122 114 role = Role.non_member
123 115 assert role.builtin?
124 116 assert_equal Role::BUILTIN_NON_MEMBER, role.builtin
125 117 end
118 end
119
120 def test_non_member_with_a_missing_non_member_role_should_return_the_non_member_role
121 Role.where(:builtin => Role::BUILTIN_NON_MEMBER).delete_all
126 122
127 context "with a missing non-member role" do
128 setup do
129 Role.delete_all("builtin = #{Role::BUILTIN_NON_MEMBER}")
130 end
131
132 should "create a new non-member role" do
133 assert_difference('Role.count') do
134 Role.non_member
135 end
136 end
137
138 should "return the non-member role" do
139 role = Role.non_member
140 assert role.builtin?
141 assert_equal Role::BUILTIN_NON_MEMBER, role.builtin
142 end
123 assert_difference('Role.count') do
124 role = Role.non_member
125 assert role.builtin?
126 assert_equal Role::BUILTIN_NON_MEMBER, role.builtin
143 127 end
144 128 end
145 129 end
@@ -1,105 +1,101
1 1 # encoding: utf-8
2 2 #
3 3 # Redmine - project management software
4 4 # Copyright (C) 2006-2013 Jean-Philippe Lang
5 5 #
6 6 # This program is free software; you can redistribute it and/or
7 7 # modify it under the terms of the GNU General Public License
8 8 # as published by the Free Software Foundation; either version 2
9 9 # of the License, or (at your option) any later version.
10 10 #
11 11 # This program is distributed in the hope that it will be useful,
12 12 # but WITHOUT ANY WARRANTY; without even the implied warranty of
13 13 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 14 # GNU General Public License for more details.
15 15 #
16 16 # You should have received a copy of the GNU General Public License
17 17 # along with this program; if not, write to the Free Software
18 18 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
19 19
20 20 require File.expand_path('../../test_helper', __FILE__)
21 21
22 22 class WikiTest < ActiveSupport::TestCase
23 23 fixtures :projects, :wikis, :wiki_pages, :wiki_contents, :wiki_content_versions
24 24
25 25 def test_create
26 26 wiki = Wiki.new(:project => Project.find(2))
27 27 assert !wiki.save
28 28 assert_equal 1, wiki.errors.count
29 29
30 30 wiki.start_page = "Start page"
31 31 assert wiki.save
32 32 end
33 33
34 34 def test_update
35 35 @wiki = Wiki.find(1)
36 36 @wiki.start_page = "Another start page"
37 37 assert @wiki.save
38 38 @wiki.reload
39 39 assert_equal "Another start page", @wiki.start_page
40 40 end
41 41
42 42 def test_find_page_should_not_be_case_sensitive
43 43 wiki = Wiki.find(1)
44 44 page = WikiPage.find(2)
45 45
46 46 assert_equal page, wiki.find_page('Another_page')
47 47 assert_equal page, wiki.find_page('Another page')
48 48 assert_equal page, wiki.find_page('ANOTHER page')
49 49 end
50 50
51 51 def test_find_page_with_cyrillic_characters
52 52 wiki = Wiki.find(1)
53 53 page = WikiPage.find(10)
54 54 assert_equal page, wiki.find_page('Π­Ρ‚ΠΈΠΊΠ°_ΠΌΠ΅Π½Π΅Π΄ΠΆΠΌΠ΅Π½Ρ‚Π°')
55 55 end
56 56
57 57 def test_find_page_with_backslashes
58 58 wiki = Wiki.find(1)
59 59 page = WikiPage.create!(:wiki => wiki, :title => '2009\\02\\09')
60 60 assert_equal page, wiki.find_page('2009\\02\\09')
61 61 end
62 62
63 63 def test_find_page_without_redirect
64 64 wiki = Wiki.find(1)
65 65 page = wiki.find_page('Another_page')
66 66 assert_not_nil page
67 67 assert_equal 'Another_page', page.title
68 68 assert_equal false, wiki.page_found_with_redirect?
69 69 end
70 70
71 71 def test_find_page_with_redirect
72 72 wiki = Wiki.find(1)
73 73 WikiRedirect.create!(:wiki => wiki, :title => 'Old_title', :redirects_to => 'Another_page')
74 74 page = wiki.find_page('Old_title')
75 75 assert_not_nil page
76 76 assert_equal 'Another_page', page.title
77 77 assert_equal true, wiki.page_found_with_redirect?
78 78 end
79 79
80 80 def test_titleize
81 81 ja_test = "\xe3\x83\x86\xe3\x82\xb9\xe3\x83\x88"
82 82 ja_test.force_encoding('UTF-8') if ja_test.respond_to?(:force_encoding)
83 83 assert_equal 'Page_title_with_CAPITALES', Wiki.titleize('page title with CAPITALES')
84 84 assert_equal ja_test, Wiki.titleize(ja_test)
85 85 end
86 86
87 context "#sidebar" do
88 setup do
89 @wiki = Wiki.find(1)
90 end
91
92 should "return nil if undefined" do
93 assert_nil @wiki.sidebar
94 end
87 def test_sidebar_should_return_nil_if_undefined
88 @wiki = Wiki.find(1)
89 assert_nil @wiki.sidebar
90 end
95 91
96 should "return a WikiPage if defined" do
97 page = @wiki.pages.new(:title => 'Sidebar')
98 page.content = WikiContent.new(:text => 'Side bar content for test_show_with_sidebar')
99 page.save!
92 def test_sidebar_should_return_a_wiki_page_if_defined
93 @wiki = Wiki.find(1)
94 page = @wiki.pages.new(:title => 'Sidebar')
95 page.content = WikiContent.new(:text => 'Side bar content for test_show_with_sidebar')
96 page.save!
100 97
101 assert_kind_of WikiPage, @wiki.sidebar
102 assert_equal 'Sidebar', @wiki.sidebar.title
103 end
98 assert_kind_of WikiPage, @wiki.sidebar
99 assert_equal 'Sidebar', @wiki.sidebar.title
104 100 end
105 101 end
General Comments 0
You need to be logged in to leave comments. Login now