|
|
|
@ -134,6 +134,7 @@ describe ApplicationHelper do |
|
|
|
|
|
|
|
|
|
describe ".textilizable" do |
|
|
|
|
let(:project) { FactoryGirl.create :valid_project } |
|
|
|
|
let(:identifier) { project.identifier } |
|
|
|
|
let(:project_member) { FactoryGirl.create :user, |
|
|
|
|
:member_in_project => project, |
|
|
|
|
:member_through_role => FactoryGirl.create(:role, |
|
|
|
@ -145,15 +146,267 @@ describe ApplicationHelper do |
|
|
|
|
:tracker => project.trackers.first } |
|
|
|
|
|
|
|
|
|
before do |
|
|
|
|
project.reload |
|
|
|
|
project.wiki.start_page = "CookBook documentation" |
|
|
|
|
project.wiki.save! |
|
|
|
|
FactoryGirl.create :wiki_page_with_content, :wiki => project.wiki, :title => "CookBook_documentation" |
|
|
|
|
@project = project |
|
|
|
|
|
|
|
|
|
User.stubs(:current).returns(project_member) |
|
|
|
|
|
|
|
|
|
Setting.enabled_scm = Setting.enabled_scm << "Filesystem" unless Setting.enabled_scm.include? "Filesystem" |
|
|
|
|
end |
|
|
|
|
|
|
|
|
|
after do |
|
|
|
|
User.unstub(:current) |
|
|
|
|
|
|
|
|
|
Setting.enabled_scm.delete "Filesystem" |
|
|
|
|
end |
|
|
|
|
|
|
|
|
|
context "Changeset links" do |
|
|
|
|
let(:repository) { FactoryGirl.create :repository, :project => project } |
|
|
|
|
let(:changeset1) { FactoryGirl.create :changeset, |
|
|
|
|
:repository => repository, |
|
|
|
|
:comments => 'My very first commit' } |
|
|
|
|
let(:changeset2) { FactoryGirl.create :changeset, |
|
|
|
|
:repository => repository, |
|
|
|
|
:comments => 'This commit fixes #1, #2 and references #1 & #3' } |
|
|
|
|
let(:changeset_link) { link_to("r#{changeset1.revision}", |
|
|
|
|
{:controller => 'repositories', :action => 'revision', :id => identifier, :rev => changeset1.revision}, |
|
|
|
|
:class => 'changeset', :title => 'My very first commit') } |
|
|
|
|
let(:changeset_link2) { link_to("r#{changeset2.revision}", |
|
|
|
|
{:controller => 'repositories', :action => 'revision', :id => identifier, :rev => changeset2.revision}, |
|
|
|
|
:class => 'changeset', :title => 'This commit fixes #1, #2 and references #1 & #3') } |
|
|
|
|
|
|
|
|
|
before do |
|
|
|
|
project.repository = repository |
|
|
|
|
end |
|
|
|
|
|
|
|
|
|
context "Single link" do |
|
|
|
|
subject { textilizable("r#{changeset1.revision}") } |
|
|
|
|
|
|
|
|
|
it { should eq("<p>#{changeset_link}</p>") } |
|
|
|
|
end |
|
|
|
|
|
|
|
|
|
context "Single link with dot" do |
|
|
|
|
subject { textilizable("r#{changeset1.revision}.") } |
|
|
|
|
|
|
|
|
|
it { should eq("<p>#{changeset_link}.</p>") } |
|
|
|
|
end |
|
|
|
|
|
|
|
|
|
context "Two links comma separated" do |
|
|
|
|
subject { textilizable("r#{changeset1.revision}, r#{changeset2.revision}") } |
|
|
|
|
|
|
|
|
|
it { should eq("<p>#{changeset_link}, #{changeset_link2}</p>") } |
|
|
|
|
end |
|
|
|
|
|
|
|
|
|
context "Single link comma separated without a space" do |
|
|
|
|
subject { textilizable("r#{changeset1.revision},r#{changeset2.revision}") } |
|
|
|
|
|
|
|
|
|
it { should eq("<p>#{changeset_link},#{changeset_link2}</p>") } |
|
|
|
|
end |
|
|
|
|
|
|
|
|
|
context "Escaping" do |
|
|
|
|
subject { textilizable("!r#{changeset1.id}") } |
|
|
|
|
|
|
|
|
|
it { should eq("<p>r#{changeset1.id}</p>") } |
|
|
|
|
end |
|
|
|
|
end |
|
|
|
|
|
|
|
|
|
context "Document links" do |
|
|
|
|
let(:document) { FactoryGirl.create :document, |
|
|
|
|
:title => 'Test document', |
|
|
|
|
:project => project } |
|
|
|
|
let(:document_link) { link_to('Test document', |
|
|
|
|
{:controller => 'documents', :action => 'show', :id => document.id}, |
|
|
|
|
:class => 'document') } |
|
|
|
|
|
|
|
|
|
context "Plain link" do |
|
|
|
|
subject { textilizable("document##{document.id}") } |
|
|
|
|
|
|
|
|
|
it { should eq("<p>#{document_link}</p>") } |
|
|
|
|
end |
|
|
|
|
|
|
|
|
|
context "Link with document name" do |
|
|
|
|
subject { textilizable("document##{document.id}") } |
|
|
|
|
|
|
|
|
|
it { should eq("<p>#{document_link}</p>") } |
|
|
|
|
end |
|
|
|
|
|
|
|
|
|
context "Escaping plain link" do |
|
|
|
|
subject { textilizable("!document##{document.id}") } |
|
|
|
|
|
|
|
|
|
it { should eq("<p>document##{document.id}</p>") } |
|
|
|
|
end |
|
|
|
|
|
|
|
|
|
context "Escaping link with document name" do |
|
|
|
|
subject { textilizable('!document:"Test document"') } |
|
|
|
|
|
|
|
|
|
it { should eq('<p>document:"Test document"</p>') } |
|
|
|
|
end |
|
|
|
|
end |
|
|
|
|
|
|
|
|
|
context "Version link" do |
|
|
|
|
let(:version) { FactoryGirl.create :version, |
|
|
|
|
:name => '1.0', |
|
|
|
|
:project => project } |
|
|
|
|
let(:version_link) { link_to('1.0', |
|
|
|
|
{:controller => 'versions', :action => 'show', :id => version.id}, |
|
|
|
|
:class => 'version') } |
|
|
|
|
|
|
|
|
|
context "Link with version id" do |
|
|
|
|
subject { textilizable("version##{version.id}") } |
|
|
|
|
|
|
|
|
|
it { should eq("<p>#{version_link}</p>") } |
|
|
|
|
end |
|
|
|
|
|
|
|
|
|
context "Link with version" do |
|
|
|
|
subject { textilizable("version:1.0") } |
|
|
|
|
it { should eq("<p>#{version_link}</p>") } |
|
|
|
|
end |
|
|
|
|
|
|
|
|
|
context "Link with quoted version" do |
|
|
|
|
subject { textilizable('version:"1.0"') } |
|
|
|
|
|
|
|
|
|
it { should eq("<p>#{version_link}</p>") } |
|
|
|
|
end |
|
|
|
|
|
|
|
|
|
context "Escaping link with version id" do |
|
|
|
|
subject { textilizable("!version##{version.id}") } |
|
|
|
|
|
|
|
|
|
it { should eq("<p>version##{version.id}</p>") } |
|
|
|
|
end |
|
|
|
|
|
|
|
|
|
context "Escaping link with version" do |
|
|
|
|
subject { textilizable("!version:1.0") } |
|
|
|
|
|
|
|
|
|
it { should eq("<p>version:1.0</p>") } |
|
|
|
|
end |
|
|
|
|
|
|
|
|
|
context "Escaping link with quoted version" do |
|
|
|
|
subject { textilizable('!version:"1.0"') } |
|
|
|
|
|
|
|
|
|
it { should eq('<p>version:"1.0"</p>') } |
|
|
|
|
end |
|
|
|
|
end |
|
|
|
|
|
|
|
|
|
context "Message links" do |
|
|
|
|
let(:board) { FactoryGirl.create :board, :project => project } |
|
|
|
|
let(:message1) { FactoryGirl.create :message, :board => board } |
|
|
|
|
let(:message2) { FactoryGirl.create :message, |
|
|
|
|
:board => board, |
|
|
|
|
:parent => message1 } |
|
|
|
|
|
|
|
|
|
before do |
|
|
|
|
message1.reload |
|
|
|
|
@message_url = {:controller => 'messages', :action => 'show', :board_id => board.id, :id => message1.id} |
|
|
|
|
end |
|
|
|
|
|
|
|
|
|
context "Plain message" do |
|
|
|
|
subject { textilizable("message##{message1.id}") } |
|
|
|
|
|
|
|
|
|
it { should eq("<p>#{link_to(message1.subject, @message_url, :class => 'message')}</p>") } |
|
|
|
|
end |
|
|
|
|
|
|
|
|
|
context "Message with parent" do |
|
|
|
|
subject { textilizable("message##{message2.id}") } |
|
|
|
|
|
|
|
|
|
it { should eq("<p>#{link_to(message2.subject, @message_url.merge(:anchor => "message-#{message2.id}", :r => message2.id), :class => 'message')}</p>") } |
|
|
|
|
end |
|
|
|
|
end |
|
|
|
|
|
|
|
|
|
context "Issue links" do |
|
|
|
|
let(:issue_link) { link_to("##{issue.id}", |
|
|
|
|
work_package_path(issue), |
|
|
|
|
:class => 'issue work_package status-3 priority-1 created-by-me', :title => "#{issue.subject} (#{issue.status})") } |
|
|
|
|
|
|
|
|
|
context "Plain issue link" do |
|
|
|
|
subject { textilizable("##{issue.id}, [##{issue.id}], (##{issue.id}) and ##{issue.id}.") } |
|
|
|
|
|
|
|
|
|
it { should eq("<p>#{issue_link}, [#{issue_link}], (#{issue_link}) and #{issue_link}.</p>") } |
|
|
|
|
end |
|
|
|
|
|
|
|
|
|
context "Plain issue link to non-existing element" do |
|
|
|
|
subject { textilizable('#0123456789') } |
|
|
|
|
|
|
|
|
|
it { should eq('<p>#0123456789</p>') } |
|
|
|
|
end |
|
|
|
|
|
|
|
|
|
context "Escaping issue link" do |
|
|
|
|
subject { textilizable("!##{issue.id}.") } |
|
|
|
|
|
|
|
|
|
it { should eq("<p>##{issue.id}.</p>") } |
|
|
|
|
end |
|
|
|
|
end |
|
|
|
|
|
|
|
|
|
context "Project links" do |
|
|
|
|
let(:subproject) { FactoryGirl.create :valid_project, :parent => project } |
|
|
|
|
let(:project_url) { {:controller => 'projects', :action => 'show', :id => subproject.identifier} } |
|
|
|
|
|
|
|
|
|
context "Plain project link" do |
|
|
|
|
subject { textilizable("project##{subproject.id}") } |
|
|
|
|
|
|
|
|
|
it { should eq("<p>#{link_to(subproject.name, project_url, :class => 'project')}</p>") } |
|
|
|
|
end |
|
|
|
|
|
|
|
|
|
context "Plain project link via identifier" do |
|
|
|
|
subject { textilizable("project:#{subproject.identifier}") } |
|
|
|
|
|
|
|
|
|
it { should eq("<p>#{link_to(subproject.name, project_url, :class => 'project')}</p>") } |
|
|
|
|
end |
|
|
|
|
|
|
|
|
|
context "Plain project link via name" do |
|
|
|
|
subject { textilizable("project:\"#{subproject.name}\"") } |
|
|
|
|
|
|
|
|
|
it { should eq("<p>#{link_to(subproject.name, project_url, :class => 'project')}</p>") } |
|
|
|
|
end |
|
|
|
|
end |
|
|
|
|
|
|
|
|
|
context "Url links" do |
|
|
|
|
subject { textilizable("http://foo.bar/FAQ#3") } |
|
|
|
|
|
|
|
|
|
it { should eq('<p><a class="external" href="http://foo.bar/FAQ#3">http://foo.bar/FAQ#3</a></p>') } |
|
|
|
|
end |
|
|
|
|
|
|
|
|
|
context "Redmine links" do |
|
|
|
|
let(:repository) { FactoryGirl.create :repository, :project => project } |
|
|
|
|
let(:source_url) { {:controller => 'repositories', :action => 'entry', :id => identifier, :path => ['some', 'file']} } |
|
|
|
|
let(:source_url_with_ext) { {:controller => 'repositories', :action => 'entry', :id => identifier, :path => ['some', 'file.ext']} } |
|
|
|
|
|
|
|
|
|
before do |
|
|
|
|
project.repository = repository |
|
|
|
|
|
|
|
|
|
@to_test = { |
|
|
|
|
# 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.merge(:rev => 52), :class => 'source'), |
|
|
|
|
'source:/some/file.ext@52' => link_to('source:/some/file.ext@52', source_url_with_ext.merge(:rev => 52), :class => 'source'), |
|
|
|
|
'source:/some/file#L110' => link_to('source:/some/file#L110', source_url.merge(:anchor => 'L110'), :class => 'source'), |
|
|
|
|
'source:/some/file.ext#L110' => link_to('source:/some/file.ext#L110', source_url_with_ext.merge(:anchor => 'L110'), :class => 'source'), |
|
|
|
|
'source:/some/file@52#L110' => link_to('source:/some/file@52#L110', source_url.merge(:rev => 52, :anchor => 'L110'), :class => 'source'), |
|
|
|
|
'export:/some/file' => link_to('export:/some/file', source_url.merge(:format => 'raw'), :class => 'source download'), |
|
|
|
|
# escaping |
|
|
|
|
'!source:/some/file' => 'source:/some/file', |
|
|
|
|
# invalid expressions |
|
|
|
|
'source:' => 'source:', |
|
|
|
|
} |
|
|
|
|
end |
|
|
|
|
|
|
|
|
|
it "" do |
|
|
|
|
@to_test.each do |text, result| |
|
|
|
|
textilizable(text).should eql("<p>#{result}</p>") |
|
|
|
|
end |
|
|
|
|
end |
|
|
|
|
end |
|
|
|
|
|
|
|
|
|
context "Pre content should not parse wiki and redmine links" do |
|
|
|
|
let(:wiki) { FactoryGirl.create :wiki, |
|
|
|
|
:start_page => "CookBook documentation", |
|
|
|
|
:project => project } |
|
|
|
|
let(:wiki_page) { FactoryGirl.create :wiki_page_with_content, |
|
|
|
|
:wiki => wiki, |
|
|
|
|
:title => "CookBook_documentation" } |
|
|
|
|
let(:raw) { <<-RAW |
|
|
|
|
[[CookBook documentation]] |
|
|
|
|
|
|
|
|
@ -165,7 +418,7 @@ describe ApplicationHelper do |
|
|
|
|
##{issue.id} |
|
|
|
|
</pre> |
|
|
|
|
RAW |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
let(:expected) { <<-EXPECTED |
|
|
|
|
<p><a href="/projects/#{project.identifier}/wiki/CookBook_documentation" class="wiki-page">CookBook documentation</a></p> |
|
|
|
@ -176,7 +429,13 @@ RAW |
|
|
|
|
##{issue.id} |
|
|
|
|
</pre> |
|
|
|
|
EXPECTED |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
before do |
|
|
|
|
project.wiki = wiki |
|
|
|
|
wiki.pages << wiki_page |
|
|
|
|
end |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
subject { textilizable(raw).gsub(%r{[\r\n\t]}, '')} |
|
|
|
|
|
|
|
|
|