commit_reference_filter_spec.rb 7.0 KB
Newer Older
1 2
require 'spec_helper'

3
describe Banzai::Filter::CommitReferenceFilter do
4 5
  include FilterSpecHelper

6
  let(:project) { create(:project, :public, :repository) }
7 8 9 10 11 12 13 14 15
  let(:commit)  { project.commit }

  it 'requires project context' do
    expect { described_class.call('') }.to raise_error(ArgumentError, /:project/)
  end

  %w(pre code a style).each do |elem|
    it "ignores valid references contained inside '#{elem}' element" do
      exp = act = "<#{elem}>Commit #{commit.id}</#{elem}>"
16
      expect(reference_filter(act).to_html).to eq exp
17 18 19 20 21 22 23
    end
  end

  context 'internal reference' do
    let(:reference) { commit.id }

    # Let's test a variety of commit SHA sizes just to be paranoid
24
    [7, 8, 12, 18, 20, 32, 40].each do |size|
25
      it "links to a valid reference of #{size} characters" do
26
        doc = reference_filter("See #{reference[0...size]}")
27 28

        expect(doc.css('a').first.text).to eq commit.short_id
29
        expect(doc.css('a').first.attr('href'))
30
          .to eq urls.project_commit_url(project, reference)
31 32 33 34
      end
    end

    it 'always uses the short ID as the link text' do
35
      doc = reference_filter("See #{commit.id}")
36 37
      expect(doc.text).to eq "See #{commit.short_id}"

38
      doc = reference_filter("See #{commit.id[0...7]}")
39 40 41 42
      expect(doc.text).to eq "See #{commit.short_id}"
    end

    it 'links with adjacent text' do
43
      doc = reference_filter("See (#{reference}.)")
44

45 46 47 48 49 50 51
      expect(doc.to_html).to match(/\(<a.+>#{commit.short_id}<\/a>\.\)/)
    end

    it 'ignores invalid commit IDs' do
      invalid = invalidate_reference(reference)
      exp = act = "See #{invalid}"

52
      expect(reference_filter(act).to_html).to eq exp
53 54 55
    end

    it 'includes a title attribute' do
56
      doc = reference_filter("See #{reference}")
57
      expect(doc.css('a').first.attr('title')).to eq commit.title
58 59 60 61 62
    end

    it 'escapes the title attribute' do
      allow_any_instance_of(Commit).to receive(:title).and_return(%{"></a>whatever<a title="})

63
      doc = reference_filter("See #{reference}")
64 65 66 67
      expect(doc.text).to eq "See #{commit.short_id}"
    end

    it 'includes default classes' do
68
      doc = reference_filter("See #{reference}")
69
      expect(doc.css('a').first.attr('class')).to eq 'gfm gfm-commit has-tooltip'
70 71 72
    end

    it 'includes a data-project attribute' do
73
      doc = reference_filter("See #{reference}")
74 75 76 77 78 79 80
      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-commit attribute' do
81
      doc = reference_filter("See #{reference}")
82 83 84 85 86 87 88
      link = doc.css('a').first

      expect(link).to have_attribute('data-commit')
      expect(link.attr('data-commit')).to eq commit.id
    end

    it 'supports an :only_path context' do
89
      doc = reference_filter("See #{reference}", only_path: true)
90 91 92
      link = doc.css('a').first.attr('href')

      expect(link).not_to match %r(https?://)
93
      expect(link).to eq urls.project_commit_url(project, reference, only_path: true)
94
    end
M
micael.bergeron 已提交
95 96 97 98 99 100 101 102 103 104 105 106

    context "in merge request context" do
      let(:noteable) { create(:merge_request, target_project: project, source_project: project) }
      let(:commit) { noteable.commits.first }

      it 'handles merge request contextual commit references' do
        url = urls.diffs_project_merge_request_url(project, noteable, commit_id: commit.id)
        doc = reference_filter("See #{reference}", noteable: noteable)

        expect(doc.css('a').first[:href]).to eq(url)
      end
    end
107 108
  end

109 110
  context 'cross-project / cross-namespace complete reference' do
    let(:namespace) { create(:namespace) }
111
    let(:project2)  { create(:project, :public, :repository, namespace: namespace) }
112
    let(:commit)    { project2.commit }
113
    let(:reference) { "#{project2.full_path}@#{commit.short_id}" }
114

115 116
    it 'link has valid text' do
      doc = reference_filter("See (#{reference}.)")
117

118
      expect(doc.css('a').first.text).to eql("#{project2.full_path}@#{commit.short_id}")
119 120
    end

121 122 123
    it 'has valid text' do
      doc = reference_filter("See (#{reference}.)")

124
      expect(doc.text).to eql("See (#{project2.full_path}@#{commit.short_id}.)")
125 126 127 128 129 130 131 132 133 134 135
    end

    it 'ignores invalid commit IDs on the referenced project' do
      exp = act = "Committed #{invalidate_reference(reference)}"

      expect(reference_filter(act).to_html).to eq exp
    end
  end

  context 'cross-project / same-namespace complete reference' do
    let(:namespace) { create(:namespace) }
136
    let(:project)   { create(:project, namespace: namespace) }
137
    let(:project2)  { create(:project, :public, :repository, namespace: namespace) }
138
    let(:commit)    { project2.commit }
139
    let(:reference) { "#{project2.full_path}@#{commit.short_id}" }
140 141 142

    it 'link has valid text' do
      doc = reference_filter("See (#{reference}.)")
143

144 145 146 147 148 149 150
      expect(doc.css('a').first.text).to eql("#{project2.path}@#{commit.short_id}")
    end

    it 'has valid text' do
      doc = reference_filter("See (#{reference}.)")

      expect(doc.text).to eql("See (#{project2.path}@#{commit.short_id}.)")
151 152 153 154
    end

    it 'ignores invalid commit IDs on the referenced project' do
      exp = act = "Committed #{invalidate_reference(reference)}"
155 156 157 158 159 160 161

      expect(reference_filter(act).to_html).to eq exp
    end
  end

  context 'cross-project shorthand reference' do
    let(:namespace) { create(:namespace) }
162
    let(:project)   { create(:project, namespace: namespace) }
163
    let(:project2)  { create(:project, :public, :repository, namespace: namespace) }
164
    let(:commit)    { project2.commit }
165
    let(:reference) { "#{project2.full_path}@#{commit.short_id}" }
166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181

    it 'link has valid text' do
      doc = reference_filter("See (#{reference}.)")

      expect(doc.css('a').first.text).to eql("#{project2.path}@#{commit.short_id}")
    end

    it 'has valid text' do
      doc = reference_filter("See (#{reference}.)")

      expect(doc.text).to eql("See (#{project2.path}@#{commit.short_id}.)")
    end

    it 'ignores invalid commit IDs on the referenced project' do
      exp = act = "Committed #{invalidate_reference(reference)}"

182 183 184 185 186
      expect(reference_filter(act).to_html).to eq exp
    end
  end

  context 'cross-project URL reference' do
187
    let(:namespace) { create(:namespace) }
188
    let(:project2)  { create(:project, :public, :repository, namespace: namespace) }
189
    let(:commit)    { project2.commit }
190
    let(:reference) { urls.project_commit_url(project2, commit.id) }
191 192 193 194

    it 'links to a valid reference' do
      doc = reference_filter("See #{reference}")

195
      expect(doc.css('a').first.attr('href'))
196
        .to eq urls.project_commit_url(project2, commit.id)
197 198 199 200 201 202 203 204 205 206 207
    end

    it 'links with adjacent text' do
      doc = reference_filter("Fixed (#{reference}.)")

      expect(doc.to_html).to match(/\(<a.+>#{commit.reference_link_text(project)}<\/a>\.\)/)
    end

    it 'ignores invalid commit IDs on the referenced project' do
      act = "Committed #{invalidate_reference(reference)}"
      expect(reference_filter(act).to_html).to match(/<a.+>#{Regexp.escape(invalidate_reference(reference))}<\/a>/)
208 209 210
    end
  end
end