Welcome to mirror list, hosted at ThFree Co, Russian Federation.

create_issue_handler_spec.rb « handler « email « gitlab « lib « spec - gitlab.com/gitlab-org/gitlab-foss.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
blob: 10098a66ae993c4ac99f85c79cfd20aa8a679c2a (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
# frozen_string_literal: true

require 'spec_helper'

RSpec.describe Gitlab::Email::Handler::CreateIssueHandler do
  include_context :email_shared_context
  let!(:user) do
    create(
      :user,
      email: 'jake@adventuretime.ooo',
      incoming_email_token: 'auth_token'
    )
  end

  let!(:project)  { create(:project, :public, namespace: namespace, path: 'gitlabhq') }
  let(:namespace) { create(:namespace, path: 'gitlabhq') }
  let(:email_raw) { email_fixture('emails/valid_new_issue.eml') }

  it_behaves_like :reply_processing_shared_examples

  before do
    stub_incoming_email_setting(enabled: true, address: "incoming+%{key}@appmail.adventuretime.ooo")
    stub_config_setting(host: 'localhost')
  end

  context "when email key" do
    let(:mail) { Mail::Message.new(email_raw) }

    it "matches the new format" do
      handler = described_class.new(mail, "gitlabhq-gitlabhq-#{project.project_id}-#{user.incoming_email_token}-issue")

      expect(handler.instance_variable_get(:@project_id)).to eq project.project_id
      expect(handler.instance_variable_get(:@project_slug)).to eq project.full_path_slug
      expect(handler.instance_variable_get(:@incoming_email_token)).to eq user.incoming_email_token
      expect(handler.can_handle?).to be_truthy
    end

    it "matches the legacy format" do
      handler = described_class.new(mail, "h5bp/html5-boilerplate+#{user.incoming_email_token}")

      expect(handler.instance_variable_get(:@project_path)).to eq 'h5bp/html5-boilerplate'
      expect(handler.instance_variable_get(:@incoming_email_token)).to eq user.incoming_email_token
      expect(handler.can_handle?).to be_truthy
    end

    it "doesn't match either format" do
      handler = described_class.new(mail, "h5bp-html5-boilerplate+something+invalid")

      expect(handler.can_handle?).to be_falsey
    end
  end

  context "when everything is fine" do
    shared_examples "a new issue" do
      it "creates a new issue" do
        setup_attachment

        expect { receiver.execute }.to change { project.issues.count }.by(1)
        issue = project.issues.last

        expect(issue.author).to eq(user)
        expect(issue.title).to eq('New Issue by email')
        expect(issue.description).to include('reply by email')
        expect(issue.description).to include(markdown)
      end
    end

    it_behaves_like "a new issue"

    context "creates a new issue with legacy email address" do
      let(:email_raw) { fixture_file('emails/valid_new_issue_legacy.eml') }

      it_behaves_like "a new issue"
    end

    context "when the reply is blank" do
      let(:email_raw) { email_fixture("emails/valid_new_issue_empty.eml") }

      it "creates a new issue" do
        expect { receiver.execute }.to change { project.issues.count }.by(1)
        issue = project.issues.last

        expect(issue.author).to eq(user)
        expect(issue.title).to eq('New Issue by email')
        expect(issue.description).to eq('')
      end
    end

    context "when there are quotes in email" do
      let(:email_raw) { email_fixture("emails/valid_new_issue_with_quote.eml") }

      it "creates a new issue" do
        expect { receiver.execute }.to change { project.issues.count }.by(1)
        issue = project.issues.last

        expect(issue.author).to eq(user)
        expect(issue.title).to eq('New Issue by email')
        expect(issue.description).to include('reply by email')
        expect(issue.description).to include('> this is a quote')
      end
    end
  end

  context 'when all lines of email are quotes' do
    let(:email_raw) { email_fixture('emails/valid_new_issue_with_only_quotes.eml') }

    it 'creates email with correct body' do
      receiver.execute

      issue = Issue.last
      expect(issue.description).to include('This email has been forwarded without new content.')
    end
  end

  context "something is wrong" do
    context "when the issue could not be saved" do
      before do
        allow_any_instance_of(Issue).to receive(:persisted?).and_return(false)
        allow_any_instance_of(Issue).to receive(:ensure_metrics).and_return(nil)
      end

      it "raises an InvalidIssueError" do
        expect { receiver.execute }.to raise_error(Gitlab::Email::InvalidIssueError)
      end
    end

    context "when we can't find the incoming_email_token" do
      let(:email_raw) { email_fixture("emails/wrong_issue_incoming_email_token.eml") }

      it "raises an UserNotFoundError" do
        expect { receiver.execute }.to raise_error(Gitlab::Email::UserNotFoundError)
      end
    end

    context "when project is private" do
      let(:project) { create(:project, :private, namespace: namespace) }

      it "raises a ProjectNotFound if the user is not a member" do
        expect { receiver.execute }.to raise_error(Gitlab::Email::ProjectNotFound)
      end
    end

    context 'when project ID is invalid' do
      it 'raises a ProjectNotFound' do
        handler = described_class.new(email_raw, "gitlabhq-gitlabhq-#{Gitlab::Database::MAX_INT_VALUE}-#{user.incoming_email_token}-issue")

        expect { handler.execute }.to raise_error(Gitlab::Email::ProjectNotFound)
      end
    end

    context 'rate limiting' do
      let(:rate_limited_service_feature_enabled) { nil }

      before do
        stub_feature_flags(rate_limited_service_issues_create: rate_limited_service_feature_enabled)
      end

      context 'when :rate_limited_service Feature is disabled' do
        let(:rate_limited_service_feature_enabled) { false }

        it 'does not attempt to throttle' do
          expect(::Gitlab::ApplicationRateLimiter).not_to receive(:throttled?)

          setup_attachment
          receiver.execute
        end
      end

      context 'when :rate_limited_service Feature is enabled' do
        let(:rate_limited_service_feature_enabled) { true }

        it 'raises a RateLimitedService::RateLimitedError' do
          allow(::Gitlab::ApplicationRateLimiter).to receive(:throttled?).and_return(true)

          setup_attachment
          expect { receiver.execute }.to raise_error(RateLimitedService::RateLimitedError, _('This endpoint has been requested too many times. Try again later.'))
        end
      end
    end
  end

  def email_fixture(path)
    fixture_file(path).gsub('project_id', project.project_id.to_s)
  end
end