require 'spec_helper' describe Banzai::Filter::MilestoneReferenceFilter do include FilterSpecHelper let(:group) { create(:group, :public) } let(:project) { create(:project, :public, group: group) } it 'requires project context' do expect { described_class.call('') }.to raise_error(ArgumentError, /:project/) end shared_examples 'reference parsing' do %w(pre code a style).each do |elem| it "ignores valid references contained inside '#{elem}' element" do exp = act = "<#{elem}>milestone #{reference}" expect(reference_filter(act).to_html).to eq exp end end it 'includes default classes' do doc = reference_filter("Milestone #{reference}") expect(doc.css('a').first.attr('class')).to eq 'gfm gfm-milestone has-tooltip' end it 'includes a data-project attribute' do doc = reference_filter("Milestone #{reference}") link = doc.css('a').first expect(link).to have_attribute('data-project') expect(link.attr('data-project')).to eq project.id.to_s end it 'includes a data-milestone attribute' do doc = reference_filter("See #{reference}") link = doc.css('a').first expect(link).to have_attribute('data-milestone') expect(link.attr('data-milestone')).to eq milestone.id.to_s end it 'supports an :only_path context' do doc = reference_filter("Milestone #{reference}", only_path: true) link = doc.css('a').first.attr('href') expect(link).not_to match %r(https?://) expect(link).to eq urls.milestone_path(milestone) end end shared_examples 'Integer-based references' do it 'links to a valid reference' do doc = reference_filter("See #{reference}") expect(doc.css('a').first.attr('href')).to eq urls.milestone_url(milestone) end it 'links with adjacent text' do doc = reference_filter("Milestone (#{reference}.)") expect(doc.to_html).to match(%r(\(#{milestone.name}\.\))) end it 'ignores invalid milestone IIDs' do exp = act = "Milestone #{invalidate_reference(reference)}" expect(reference_filter(act).to_html).to eq exp end end shared_examples 'String-based single-word references' do let(:reference) { "#{Milestone.reference_prefix}#{milestone.name}" } before do milestone.update!(name: 'gfm') end it 'links to a valid reference' do doc = reference_filter("See #{reference}") expect(doc.css('a').first.attr('href')).to eq urls.milestone_url(milestone) expect(doc.text).to eq 'See gfm' end it 'links with adjacent text' do doc = reference_filter("Milestone (#{reference}.)") expect(doc.to_html).to match(%r(\(#{milestone.name}\.\))) end it 'ignores invalid milestone names' do exp = act = "Milestone #{Milestone.reference_prefix}#{milestone.name.reverse}" expect(reference_filter(act).to_html).to eq exp end end shared_examples 'String-based multi-word references in quotes' do let(:reference) { milestone.to_reference(format: :name) } before do milestone.update!(name: 'gfm references') end it 'links to a valid reference' do doc = reference_filter("See #{reference}") expect(doc.css('a').first.attr('href')).to eq urls.milestone_url(milestone) expect(doc.text).to eq 'See gfm references' end it 'links with adjacent text' do doc = reference_filter("Milestone (#{reference}.)") expect(doc.to_html).to match(%r(\(#{milestone.name}\.\))) end it 'ignores invalid milestone names' do exp = act = %(Milestone #{Milestone.reference_prefix}"#{milestone.name.reverse}") expect(reference_filter(act).to_html).to eq exp end end shared_examples 'referencing a milestone in a link href' do let(:unquoted_reference) { "#{Milestone.reference_prefix}#{milestone.name}" } let(:link_reference) { %Q{Milestone} } before do milestone.update!(name: 'gfm') end it 'links to a valid reference' do doc = reference_filter("See #{link_reference}") expect(doc.css('a').first.attr('href')).to eq urls.milestone_url(milestone) end it 'links with adjacent text' do doc = reference_filter("Milestone (#{link_reference}.)") expect(doc.to_html).to match(%r(\(Milestone\.\))) end it 'includes a data-project attribute' do doc = reference_filter("Milestone #{link_reference}") link = doc.css('a').first expect(link).to have_attribute('data-project') expect(link.attr('data-project')).to eq project.id.to_s end it 'includes a data-milestone attribute' do doc = reference_filter("See #{link_reference}") link = doc.css('a').first expect(link).to have_attribute('data-milestone') expect(link.attr('data-milestone')).to eq milestone.id.to_s end end shared_examples 'linking to a milestone as the entire link' do let(:unquoted_reference) { "#{Milestone.reference_prefix}#{milestone.name}" } let(:link) { urls.milestone_url(milestone) } let(:link_reference) { %Q{#{link}} } it 'replaces the link text with the milestone reference' do doc = reference_filter("See #{link}") expect(doc.css('a').first.text).to eq(unquoted_reference) end it 'includes a data-project attribute' do doc = reference_filter("Milestone #{link_reference}") link = doc.css('a').first expect(link).to have_attribute('data-project') expect(link.attr('data-project')).to eq project.id.to_s end it 'includes a data-milestone attribute' do doc = reference_filter("See #{link_reference}") link = doc.css('a').first expect(link).to have_attribute('data-milestone') expect(link.attr('data-milestone')).to eq milestone.id.to_s end end shared_examples 'cross-project / cross-namespace complete reference' do let(:namespace) { create(:namespace) } let(:another_project) { create(:project, :public, namespace: namespace) } let(:milestone) { create(:milestone, project: another_project) } let(:reference) { "#{another_project.full_path}%#{milestone.iid}" } let!(:result) { reference_filter("See #{reference}") } it 'points to referenced project milestone page' do expect(result.css('a').first.attr('href')).to eq urls .project_milestone_url(another_project, milestone) end it 'link has valid text' do doc = reference_filter("See (#{reference}.)") expect(doc.css('a').first.text) .to eq("#{milestone.name} in #{another_project.full_path}") end it 'has valid text' do doc = reference_filter("See (#{reference}.)") expect(doc.text) .to eq("See (#{milestone.name} in #{another_project.full_path}.)") end it 'escapes the name attribute' do allow_any_instance_of(Milestone).to receive(:title).and_return(%{">whateverwhateverwhatever