# encoding: utf-8
#
# 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 ApplicationHelperTest < ActionView::TestCase
include ERB::Util
fixtures :projects, :roles, :enabled_modules, :users,
:repositories, :changesets,
:trackers, :issue_statuses, :issues, :versions, :documents,
:wikis, :wiki_pages, :wiki_contents,
:boards, :messages, :news,
:attachments, :enumerations
def setup
super
set_tmp_attachments_directory
end
context "#link_to_if_authorized" do
context "authorized user" do
should "be tested"
end
context "unauthorized user" do
should "be tested"
end
should "allow using the :controller and :action for the target link" do
User.current = User.find_by_login('admin')
@project = Issue.first.project # Used by helper
response = link_to_if_authorized("By controller/action",
{:controller => 'issues', :action => 'edit', :id => Issue.first.id})
assert_match /href/, response
end
end
def test_auto_links
to_test = {
'http://foo.bar' => 'http://foo.bar',
'http://foo.bar/~user' => 'http://foo.bar/~user',
'http://foo.bar.' => 'http://foo.bar.',
'https://foo.bar.' => 'https://foo.bar.',
'This is a link: http://foo.bar.' => 'This is a link: http://foo.bar.',
'A link (eg. http://foo.bar).' => 'A link (eg. http://foo.bar).',
'http://foo.bar/foo.bar#foo.bar.' => 'http://foo.bar/foo.bar#foo.bar.',
'http://www.foo.bar/Test_(foobar)' => 'http://www.foo.bar/Test_(foobar)',
'(see inline link : http://www.foo.bar/Test_(foobar))' => '(see inline link : http://www.foo.bar/Test_(foobar))',
'(see inline link : http://www.foo.bar/Test)' => '(see inline link : http://www.foo.bar/Test)',
'(see inline link : http://www.foo.bar/Test).' => '(see inline link : http://www.foo.bar/Test).',
'(see "inline link":http://www.foo.bar/Test_(foobar))' => '(see inline link)',
'(see "inline link":http://www.foo.bar/Test)' => '(see inline link)',
'(see "inline link":http://www.foo.bar/Test).' => '(see inline link).',
'www.foo.bar' => 'www.foo.bar',
'http://foo.bar/page?p=1&t=z&s=' => 'http://foo.bar/page?p=1&t=z&s=',
'http://foo.bar/page#125' => 'http://foo.bar/page#125',
'http://foo@www.bar.com' => 'http://foo@www.bar.com',
'http://foo:bar@www.bar.com' => 'http://foo:bar@www.bar.com',
'ftp://foo.bar' => 'ftp://foo.bar',
'ftps://foo.bar' => 'ftps://foo.bar',
'sftp://foo.bar' => 'sftp://foo.bar',
# two exclamation marks
'http://example.net/path!602815048C7B5C20!302.html' => 'http://example.net/path!602815048C7B5C20!302.html',
# escaping
'http://foo"bar' => 'http://foo"bar',
# wrap in angle brackets
' #{result} #{result} #{result} #{result} #{result} #{result}
Another paragraph",
# no multiline link text
"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
and another on a second line\":test",
# mailto link
"\"system administrator\":mailto:sysadmin@example.com?subject=redmine%20permissions" => "system administrator",
# two exclamation marks
'"a link":http://example.net/path!602815048C7B5C20!302.html' => 'a link',
# escaping
'"test":http://foo"bar' => 'test',
}
to_test.each { |text, result| assert_equal "
#{result}
", textilizable(text) } end if 'ruby'.respond_to?(:encoding) def test_textile_external_links_with_non_ascii_characters to_test = { 'This is a "link":http://foo.bar/тест' => 'This is a link' } to_test.each { |text, result| assert_equal "#{result}
", textilizable(text) } end else puts 'Skipping test_textile_external_links_with_non_ascii_characters, unsupported ruby version' end def test_redmine_links issue_link = link_to('#3', {:controller => 'issues', :action => 'show', :id => 3}, :class => 'issue status-1 priority-4 priority-lowest overdue', :title => 'Error 281 when updating a recipe (New)') note_link = link_to('#3', {:controller => 'issues', :action => 'show', :id => 3, :anchor => 'note-14'}, :class => 'issue status-1 priority-4 priority-lowest overdue', :title => 'Error 281 when updating a recipe (New)') changeset_link = link_to('r1', {:controller => 'repositories', :action => 'revision', :id => 'ecookbook', :rev => 1}, :class => 'changeset', :title => 'My very first commit') changeset_link2 = link_to('r2', {:controller => 'repositories', :action => 'revision', :id => 'ecookbook', :rev => 2}, :class => 'changeset', :title => 'This commit fixes #1, #2 and references #1 & #3') document_link = link_to('Test document', {:controller => 'documents', :action => 'show', :id => 1}, :class => 'document') version_link = link_to('1.0', {:controller => 'versions', :action => 'show', :id => 2}, :class => 'version') board_url = {:controller => 'boards', :action => 'show', :id => 2, :project_id => 'ecookbook'} message_url = {:controller => 'messages', :action => 'show', :board_id => 1, :id => 4} news_url = {:controller => 'news', :action => 'show', :id => 1} project_url = {:controller => 'projects', :action => 'show', :id => 'subproject1'} source_url = '/projects/ecookbook/repository/entry/some/file' source_url_with_rev = '/projects/ecookbook/repository/revisions/52/entry/some/file' source_url_with_ext = '/projects/ecookbook/repository/entry/some/file.ext' source_url_with_rev_and_ext = '/projects/ecookbook/repository/revisions/52/entry/some/file.ext' export_url = '/projects/ecookbook/repository/raw/some/file' export_url_with_rev = '/projects/ecookbook/repository/revisions/52/raw/some/file' export_url_with_ext = '/projects/ecookbook/repository/raw/some/file.ext' export_url_with_rev_and_ext = '/projects/ecookbook/repository/revisions/52/raw/some/file.ext' to_test = { # tickets '#3, [#3], (#3) and #3.' => "#{issue_link}, [#{issue_link}], (#{issue_link}) and #{issue_link}.", # ticket notes '#3-14' => note_link, '#3#note-14' => note_link, # should not ignore leading zero '#03' => '#03', # changesets 'r1' => changeset_link, 'r1.' => "#{changeset_link}.", 'r1, r2' => "#{changeset_link}, #{changeset_link2}", 'r1,r2' => "#{changeset_link},#{changeset_link2}", # documents 'document#1' => document_link, 'document:"Test document"' => document_link, # versions 'version#2' => version_link, 'version:1.0' => version_link, 'version:"1.0"' => version_link, # source 'source:some/file' => link_to('source:some/file', source_url, :class => 'source'), 'source:/some/file' => link_to('source:/some/file', source_url, :class => 'source'), 'source:/some/file.' => link_to('source:/some/file', source_url, :class => 'source') + ".", 'source:/some/file.ext.' => link_to('source:/some/file.ext', source_url_with_ext, :class => 'source') + ".", 'source:/some/file. ' => link_to('source:/some/file', source_url, :class => 'source') + ".", 'source:/some/file.ext. ' => link_to('source:/some/file.ext', source_url_with_ext, :class => 'source') + ".", 'source:/some/file, ' => link_to('source:/some/file', source_url, :class => 'source') + ",", 'source:/some/file@52' => link_to('source:/some/file@52', source_url_with_rev, :class => 'source'), 'source:/some/file.ext@52' => link_to('source:/some/file.ext@52', source_url_with_rev_and_ext, :class => 'source'), 'source:/some/file#L110' => link_to('source:/some/file#L110', source_url + "#L110", :class => 'source'), 'source:/some/file.ext#L110' => link_to('source:/some/file.ext#L110', source_url_with_ext + "#L110", :class => 'source'), 'source:/some/file@52#L110' => link_to('source:/some/file@52#L110', source_url_with_rev + "#L110", :class => 'source'), # export 'export:/some/file' => link_to('export:/some/file', export_url, :class => 'source download'), 'export:/some/file.ext' => link_to('export:/some/file.ext', export_url_with_ext, :class => 'source download'), 'export:/some/file@52' => link_to('export:/some/file@52', export_url_with_rev, :class => 'source download'), 'export:/some/file.ext@52' => link_to('export:/some/file.ext@52', export_url_with_rev_and_ext, :class => 'source download'), # forum 'forum#2' => link_to('Discussion', board_url, :class => 'board'), 'forum:Discussion' => link_to('Discussion', board_url, :class => 'board'), # message 'message#4' => link_to('Post 2', message_url, :class => 'message'), 'message#5' => link_to('RE: post 2', message_url.merge(:anchor => 'message-5', :r => 5), :class => 'message'), # news 'news#1' => link_to('eCookbook first release !', news_url, :class => 'news'), 'news:"eCookbook first release !"' => link_to('eCookbook first release !', news_url, :class => 'news'), # project 'project#3' => link_to('eCookbook Subproject 1', project_url, :class => 'project'), 'project:subproject1' => link_to('eCookbook Subproject 1', project_url, :class => 'project'), 'project:"eCookbook subProject 1"' => link_to('eCookbook Subproject 1', project_url, :class => 'project'), # not found '#0123456789' => '#0123456789', # invalid expressions 'source:' => 'source:', # url hash "http://foo.bar/FAQ#3" => 'http://foo.bar/FAQ#3', } @project = Project.find(1) to_test.each { |text, result| assert_equal "#{result}
", textilizable(text), "#{text} failed" } end def test_escaped_redmine_links_should_not_be_parsed to_test = [ '#3.', '#3-14.', '#3#-note14.', 'r1', 'document#1', 'document:"Test document"', 'version#2', 'version:1.0', 'version:"1.0"', 'source:/some/file' ] @project = Project.find(1) to_test.each { |text| assert_equal "#{text}
", textilizable("!" + text), "#{text} failed" } end def test_cross_project_redmine_links source_link = link_to('ecookbook:source:/some/file', {:controller => 'repositories', :action => 'entry', :id => 'ecookbook', :path => ['some', 'file']}, :class => 'source') changeset_link = link_to('ecookbook:r2', {:controller => 'repositories', :action => 'revision', :id => 'ecookbook', :rev => 2}, :class => 'changeset', :title => 'This commit fixes #1, #2 and references #1 & #3') to_test = { # documents 'document:"Test document"' => 'document:"Test document"', 'ecookbook:document:"Test document"' => 'Test document', 'invalid:document:"Test document"' => 'invalid:document:"Test document"', # versions 'version:"1.0"' => 'version:"1.0"', 'ecookbook:version:"1.0"' => '1.0', 'invalid:version:"1.0"' => 'invalid:version:"1.0"', # changeset 'r2' => 'r2', 'ecookbook:r2' => changeset_link, 'invalid:r2' => 'invalid:r2', # source 'source:/some/file' => 'source:/some/file', 'ecookbook:source:/some/file' => source_link, 'invalid:source:/some/file' => 'invalid:source:/some/file', } @project = Project.find(3) to_test.each { |text, result| assert_equal "#{result}
", textilizable(text), "#{text} failed" } end def test_multiple_repositories_redmine_links svn = Repository::Subversion.create!(:project_id => 1, :identifier => 'svn1', :url => 'file:///foo/hg') Changeset.create!(:repository => svn, :committed_on => Time.now, :revision => '123') hg = Repository::Mercurial.create!(:project_id => 1, :identifier => 'hg1', :url => '/foo/hg') Changeset.create!(:repository => hg, :committed_on => Time.now, :revision => '123', :scmid => 'abcd') changeset_link = link_to('r2', {:controller => 'repositories', :action => 'revision', :id => 'ecookbook', :rev => 2}, :class => 'changeset', :title => 'This commit fixes #1, #2 and references #1 & #3') svn_changeset_link = link_to('svn1|r123', {:controller => 'repositories', :action => 'revision', :id => 'ecookbook', :repository_id => 'svn1', :rev => 123}, :class => 'changeset', :title => '') hg_changeset_link = link_to('hg1|abcd', {:controller => 'repositories', :action => 'revision', :id => 'ecookbook', :repository_id => 'hg1', :rev => 'abcd'}, :class => 'changeset', :title => '') source_link = link_to('source:some/file', {:controller => 'repositories', :action => 'entry', :id => 'ecookbook', :path => ['some', 'file']}, :class => 'source') hg_source_link = link_to('source:hg1|some/file', {:controller => 'repositories', :action => 'entry', :id => 'ecookbook', :repository_id => 'hg1', :path => ['some', 'file']}, :class => 'source') to_test = { 'r2' => changeset_link, 'svn1|r123' => svn_changeset_link, 'invalid|r123' => 'invalid|r123', 'commit:hg1|abcd' => hg_changeset_link, 'commit:invalid|abcd' => 'commit:invalid|abcd', # source 'source:some/file' => source_link, 'source:hg1|some/file' => hg_source_link, 'source:invalid|some/file' => 'source:invalid|some/file', } @project = Project.find(1) to_test.each { |text, result| assert_equal "#{result}
", textilizable(text), "#{text} failed" } end def test_cross_project_multiple_repositories_redmine_links svn = Repository::Subversion.create!(:project_id => 1, :identifier => 'svn1', :url => 'file:///foo/hg') Changeset.create!(:repository => svn, :committed_on => Time.now, :revision => '123') hg = Repository::Mercurial.create!(:project_id => 1, :identifier => 'hg1', :url => '/foo/hg') Changeset.create!(:repository => hg, :committed_on => Time.now, :revision => '123', :scmid => 'abcd') changeset_link = link_to('ecookbook:r2', {:controller => 'repositories', :action => 'revision', :id => 'ecookbook', :rev => 2}, :class => 'changeset', :title => 'This commit fixes #1, #2 and references #1 & #3') svn_changeset_link = link_to('ecookbook:svn1|r123', {:controller => 'repositories', :action => 'revision', :id => 'ecookbook', :repository_id => 'svn1', :rev => 123}, :class => 'changeset', :title => '') hg_changeset_link = link_to('ecookbook:hg1|abcd', {:controller => 'repositories', :action => 'revision', :id => 'ecookbook', :repository_id => 'hg1', :rev => 'abcd'}, :class => 'changeset', :title => '') source_link = link_to('ecookbook:source:some/file', {:controller => 'repositories', :action => 'entry', :id => 'ecookbook', :path => ['some', 'file']}, :class => 'source') hg_source_link = link_to('ecookbook:source:hg1|some/file', {:controller => 'repositories', :action => 'entry', :id => 'ecookbook', :repository_id => 'hg1', :path => ['some', 'file']}, :class => 'source') to_test = { 'ecookbook:r2' => changeset_link, 'ecookbook:svn1|r123' => svn_changeset_link, 'ecookbook:invalid|r123' => 'ecookbook:invalid|r123', 'ecookbook:commit:hg1|abcd' => hg_changeset_link, 'ecookbook:commit:invalid|abcd' => 'ecookbook:commit:invalid|abcd', 'invalid:commit:invalid|abcd' => 'invalid:commit:invalid|abcd', # source 'ecookbook:source:some/file' => source_link, 'ecookbook:source:hg1|some/file' => hg_source_link, 'ecookbook:source:invalid|some/file' => 'ecookbook:source:invalid|some/file', 'invalid:source:invalid|some/file' => 'invalid:source:invalid|some/file', } @project = Project.find(3) to_test.each { |text, result| assert_equal "#{result}
", textilizable(text), "#{text} failed" } end def test_redmine_links_git_commit changeset_link = link_to('abcd', { :controller => 'repositories', :action => 'revision', :id => 'subproject1', :rev => 'abcd', }, :class => 'changeset', :title => 'test commit') to_test = { 'commit:abcd' => changeset_link, } @project = Project.find(3) r = Repository::Git.create!(:project => @project, :url => '/tmp/test/git') assert r c = Changeset.new(:repository => r, :committed_on => Time.now, :revision => 'abcd', :scmid => 'abcd', :comments => 'test commit') assert( c.save ) to_test.each { |text, result| assert_equal "#{result}
", textilizable(text) } end # TODO: Bazaar commit id contains mail address, so it contains '@' and '_'. def test_redmine_links_darcs_commit changeset_link = link_to('20080308225258-98289-abcd456efg.gz', { :controller => 'repositories', :action => 'revision', :id => 'subproject1', :rev => '123', }, :class => 'changeset', :title => 'test commit') to_test = { 'commit:20080308225258-98289-abcd456efg.gz' => changeset_link, } @project = Project.find(3) r = Repository::Darcs.create!( :project => @project, :url => '/tmp/test/darcs', :log_encoding => 'UTF-8') assert r c = Changeset.new(:repository => r, :committed_on => Time.now, :revision => '123', :scmid => '20080308225258-98289-abcd456efg.gz', :comments => 'test commit') assert( c.save ) to_test.each { |text, result| assert_equal "#{result}
", textilizable(text) } end def test_redmine_links_mercurial_commit changeset_link_rev = link_to('r123', { :controller => 'repositories', :action => 'revision', :id => 'subproject1', :rev => '123' , }, :class => 'changeset', :title => 'test commit') changeset_link_commit = link_to('abcd', { :controller => 'repositories', :action => 'revision', :id => 'subproject1', :rev => 'abcd' , }, :class => 'changeset', :title => 'test commit') to_test = { 'r123' => changeset_link_rev, 'commit:abcd' => changeset_link_commit, } @project = Project.find(3) r = Repository::Mercurial.create!(:project => @project, :url => '/tmp/test') assert r c = Changeset.new(:repository => r, :committed_on => Time.now, :revision => '123', :scmid => 'abcd', :comments => 'test commit') assert( c.save ) to_test.each { |text, result| assert_equal "#{result}
", textilizable(text) } end def test_attachment_links attachment_link = link_to('error281.txt', {:controller => 'attachments', :action => 'download', :id => '1'}, :class => 'attachment') to_test = { 'attachment:error281.txt' => attachment_link } to_test.each { |text, result| assert_equal "#{result}
", textilizable(text, :attachments => Issue.find(3).attachments), "#{text} failed" } end def test_wiki_links to_test = { '[[CookBook documentation]]' => 'CookBook documentation', '[[Another page|Page]]' => 'Page', # title content should be formatted '[[Another page|With _styled_ *title*]]' => 'With styled title', '[[Another page|With title containing HTML entities & markups]]' => 'With title containing <strong>HTML entities & markups</strong>', # link with anchor '[[CookBook documentation#One-section]]' => 'CookBook documentation', '[[Another page#anchor|Page]]' => 'Page', # UTF8 anchor '[[Another_page#Тест|Тест]]' => %|Тест|, # page that doesn't exist '[[Unknown page]]' => 'Unknown page', '[[Unknown page|404]]' => '404', # link to another project wiki '[[onlinestore:]]' => 'onlinestore', '[[onlinestore:|Wiki]]' => 'Wiki', '[[onlinestore:Start page]]' => 'Start page', '[[onlinestore:Start page|Text]]' => 'Text', '[[onlinestore:Unknown page]]' => 'Unknown page', # striked through link '-[[Another page|Page]]-' => '#{result}
", textilizable(text) } end def test_wiki_links_within_local_file_generation_context to_test = { # link to a page '[[CookBook documentation]]' => 'CookBook documentation', '[[CookBook documentation|documentation]]' => 'documentation', '[[CookBook documentation#One-section]]' => 'CookBook documentation', '[[CookBook documentation#One-section|documentation]]' => 'documentation', # page that doesn't exist '[[Unknown page]]' => 'Unknown page', '[[Unknown page|404]]' => '404', '[[Unknown page#anchor]]' => 'Unknown page', '[[Unknown page#anchor|404]]' => '404', } @project = Project.find(1) to_test.each { |text, result| assert_equal "#{result}
", textilizable(text, :wiki_links => :local) } end def test_wiki_links_within_wiki_page_context page = WikiPage.find_by_title('Another_page' ) to_test = { # link to another page '[[CookBook documentation]]' => 'CookBook documentation', '[[CookBook documentation|documentation]]' => 'documentation', '[[CookBook documentation#One-section]]' => 'CookBook documentation', '[[CookBook documentation#One-section|documentation]]' => 'documentation', # link to the current page '[[Another page]]' => 'Another page', '[[Another page|Page]]' => 'Page', '[[Another page#anchor]]' => 'Another page', '[[Another page#anchor|Page]]' => 'Page', # page that doesn't exist '[[Unknown page]]' => 'Unknown page', '[[Unknown page|404]]' => '404', '[[Unknown page#anchor]]' => 'Unknown page', '[[Unknown page#anchor|404]]' => '404', } @project = Project.find(1) to_test.each { |text, result| assert_equal "#{result}
", textilizable(WikiContent.new( :text => text, :page => page ), :text) } end def test_wiki_links_anchor_option_should_prepend_page_title_to_href to_test = { # link to a page '[[CookBook documentation]]' => 'CookBook documentation', '[[CookBook documentation|documentation]]' => 'documentation', '[[CookBook documentation#One-section]]' => 'CookBook documentation', '[[CookBook documentation#One-section|documentation]]' => 'documentation', # page that doesn't exist '[[Unknown page]]' => 'Unknown page', '[[Unknown page|404]]' => '404', '[[Unknown page#anchor]]' => 'Unknown page', '[[Unknown page#anchor|404]]' => '404', } @project = Project.find(1) to_test.each { |text, result| assert_equal "#{result}
", textilizable(text, :wiki_links => :anchor) } end def test_html_tags to_test = { "<div>content</div>
", "<div class=\"bold\">content</div>
", "" => "<script>some script;</script>
", # do not escape pre/code tags "\nline 1\nline2" => "
\nline 1\nline2", "
\nline 1\nline2
" => "\nline 1\nline2
",
"" => "content
<div>content</div>", "HTML comment: " => "
HTML comment: <!-- no comments -->
", "