Commit dfaa527d authored by Michael Kozono's avatar Michael Kozono

Merge branch 'email_metadata' into 'master'

Add meta fields to email receiver error

See merge request gitlab-org/gitlab!65001
parents b7eea1bb b6e33c8c
...@@ -135,6 +135,7 @@ The following metrics are available: ...@@ -135,6 +135,7 @@ The following metrics are available:
| `gitlab_spamcheck_request_duration_seconds` | Histogram | 13.12 | The duration for requests between Rails and the anti-spam engine | | | `gitlab_spamcheck_request_duration_seconds` | Histogram | 13.12 | The duration for requests between Rails and the anti-spam engine | |
| `service_desk_thank_you_email` | Counter | 14.0 | Total number of email responses to new service desk emails | | | `service_desk_thank_you_email` | Counter | 14.0 | Total number of email responses to new service desk emails | |
| `service_desk_new_note_email` | Counter | 14.0 | Total number of email notifications on new service desk comment | | | `service_desk_new_note_email` | Counter | 14.0 | Total number of email notifications on new service desk comment | |
| `email_receiver_error` | Counter | 14.1 | Total number of errors when processing incoming emails | |
## Metrics controlled by a feature flag ## Metrics controlled by a feature flag
......
...@@ -4,14 +4,6 @@ module Gitlab ...@@ -4,14 +4,6 @@ module Gitlab
module Email module Email
module Handler module Handler
module ReplyProcessing module ReplyProcessing
private
attr_reader :project_id, :project_slug, :project_path, :incoming_email_token
def author
raise NotImplementedError
end
# rubocop:disable Gitlab/ModuleWithInstanceVariables # rubocop:disable Gitlab/ModuleWithInstanceVariables
def project def project
return @project if instance_variable_defined?(:@project) return @project if instance_variable_defined?(:@project)
...@@ -27,6 +19,14 @@ module Gitlab ...@@ -27,6 +19,14 @@ module Gitlab
end end
# rubocop:enable Gitlab/ModuleWithInstanceVariables # rubocop:enable Gitlab/ModuleWithInstanceVariables
private
attr_reader :project_id, :project_slug, :project_path, :incoming_email_token
def author
raise NotImplementedError
end
def message def message
@message ||= process_message @message ||= process_message
end end
......
...@@ -46,10 +46,6 @@ module Gitlab ...@@ -46,10 +46,6 @@ module Gitlab
:receive_email_service_desk :receive_email_service_desk
end end
private
attr_reader :project_id, :project_path, :service_desk_key
def project def project
strong_memoize(:project) do strong_memoize(:project) do
@project = service_desk_key ? project_from_key : super @project = service_desk_key ? project_from_key : super
...@@ -58,6 +54,10 @@ module Gitlab ...@@ -58,6 +54,10 @@ module Gitlab
end end
end end
private
attr_reader :project_id, :project_path, :service_desk_key
def project_from_key def project_from_key
return unless match = service_desk_key.match(PROJECT_KEY_PATTERN) return unless match = service_desk_key.match(PROJECT_KEY_PATTERN)
......
...@@ -22,6 +22,9 @@ module Gitlab ...@@ -22,6 +22,9 @@ module Gitlab
handler.execute.tap do handler.execute.tap do
Gitlab::Metrics::BackgroundTransaction.current&.add_event(handler.metrics_event, handler.metrics_params) Gitlab::Metrics::BackgroundTransaction.current&.add_event(handler.metrics_event, handler.metrics_params)
end end
rescue StandardError => e
Gitlab::Metrics::BackgroundTransaction.current&.add_event('email_receiver_error', error: e.class.name)
raise e
end end
def mail_metadata def mail_metadata
...@@ -33,7 +36,11 @@ module Gitlab ...@@ -33,7 +36,11 @@ module Gitlab
references: Array(mail.references), references: Array(mail.references),
delivered_to: delivered_to.map(&:value), delivered_to: delivered_to.map(&:value),
envelope_to: envelope_to.map(&:value), envelope_to: envelope_to.map(&:value),
x_envelope_to: x_envelope_to.map(&:value) x_envelope_to: x_envelope_to.map(&:value),
meta: {
client_id: "email/#{mail.from.first}",
project: handler&.project&.full_path
}
} }
end end
......
...@@ -5,106 +5,125 @@ require 'spec_helper' ...@@ -5,106 +5,125 @@ require 'spec_helper'
RSpec.describe Gitlab::Email::Receiver do RSpec.describe Gitlab::Email::Receiver do
include_context :email_shared_context include_context :email_shared_context
shared_examples 'correctly finds the mail key and adds metric event' do let(:metric_transaction) { instance_double(Gitlab::Metrics::WebTransaction) }
let(:metric_transaction) { double('Gitlab::Metrics::WebTransaction') }
specify :aggregate_failures do shared_examples 'successful receive' do
let_it_be(:project) { create(:project) }
let(:handler) { double(:handler, project: project, execute: true, metrics_event: nil, metrics_params: nil) }
it 'correctly finds the mail key' do
expect(Gitlab::Email::Handler).to receive(:for).with(an_instance_of(Mail::Message), 'gitlabhq/gitlabhq+auth_token').and_return(handler) expect(Gitlab::Email::Handler).to receive(:for).with(an_instance_of(Mail::Message), 'gitlabhq/gitlabhq+auth_token').and_return(handler)
receiver.execute
end
it 'adds metric event' do
allow(receiver).to receive(:handler).and_return(handler)
expect(::Gitlab::Metrics::BackgroundTransaction).to receive(:current).and_return(metric_transaction) expect(::Gitlab::Metrics::BackgroundTransaction).to receive(:current).and_return(metric_transaction)
expect(metric_transaction).to receive(:add_event).with(handler.metrics_event, handler.metrics_params) expect(metric_transaction).to receive(:add_event).with(handler.metrics_event, handler.metrics_params)
receiver.execute receiver.execute
end end
it 'returns valid metadata' do
allow(receiver).to receive(:handler).and_return(handler)
metadata = receiver.mail_metadata
expect(metadata.keys).to match_array(%i(mail_uid from_address to_address mail_key references delivered_to envelope_to x_envelope_to meta))
expect(metadata[:meta]).to include(client_id: 'email/jake@example.com', project: project.full_path)
expect(metadata[meta_key]).to eq(meta_value)
end
end end
context 'when the email contains a valid email address in a header' do context 'when the email contains a valid email address in a header' do
let(:handler) { double(:handler) }
let(:metadata) { receiver.mail_metadata }
before do before do
allow(handler).to receive(:execute)
allow(handler).to receive(:metrics_params)
allow(handler).to receive(:metrics_event)
stub_incoming_email_setting(enabled: true, address: "incoming+%{key}@appmail.example.com") stub_incoming_email_setting(enabled: true, address: "incoming+%{key}@appmail.example.com")
expect(receiver.mail_metadata.keys).to match_array(%i(mail_uid from_address to_address mail_key references delivered_to envelope_to x_envelope_to))
end end
context 'when in a Delivered-To header' do context 'when in a Delivered-To header' do
let(:email_raw) { fixture_file('emails/forwarded_new_issue.eml') } let(:email_raw) { fixture_file('emails/forwarded_new_issue.eml') }
let(:meta_key) { :delivered_to }
let(:meta_value) { ["incoming+gitlabhq/gitlabhq+auth_token@appmail.example.com", "support@example.com"] }
it_behaves_like 'correctly finds the mail key and adds metric event' it_behaves_like 'successful receive'
it 'parses the metadata' do
expect(metadata[:delivered_to]). to eq(["incoming+gitlabhq/gitlabhq+auth_token@appmail.example.com", "support@example.com"])
end
end end
context 'when in an Envelope-To header' do context 'when in an Envelope-To header' do
let(:email_raw) { fixture_file('emails/envelope_to_header.eml') } let(:email_raw) { fixture_file('emails/envelope_to_header.eml') }
let(:meta_key) { :envelope_to }
let(:meta_value) { ["incoming+gitlabhq/gitlabhq+auth_token@appmail.example.com"] }
it_behaves_like 'correctly finds the mail key and adds metric event' it_behaves_like 'successful receive'
it 'parses the metadata' do
expect(metadata[:envelope_to]). to eq(["incoming+gitlabhq/gitlabhq+auth_token@appmail.example.com"])
end
end end
context 'when in an X-Envelope-To header' do context 'when in an X-Envelope-To header' do
let(:email_raw) { fixture_file('emails/x_envelope_to_header.eml') } let(:email_raw) { fixture_file('emails/x_envelope_to_header.eml') }
let(:meta_key) { :x_envelope_to }
let(:meta_value) { ["incoming+gitlabhq/gitlabhq+auth_token@appmail.example.com"] }
it_behaves_like 'correctly finds the mail key and adds metric event' it_behaves_like 'successful receive'
it 'parses the metadata' do
expect(metadata[:x_envelope_to]). to eq(["incoming+gitlabhq/gitlabhq+auth_token@appmail.example.com"])
end
end end
context 'when enclosed with angle brackets in an Envelope-To header' do context 'when enclosed with angle brackets in an Envelope-To header' do
let(:email_raw) { fixture_file('emails/envelope_to_header_with_angle_brackets.eml') } let(:email_raw) { fixture_file('emails/envelope_to_header_with_angle_brackets.eml') }
let(:meta_key) { :envelope_to }
let(:meta_value) { ["<incoming+gitlabhq/gitlabhq+auth_token@appmail.example.com>"] }
it_behaves_like 'correctly finds the mail key and adds metric event' it_behaves_like 'successful receive'
end end
end end
context "when we cannot find a capable handler" do shared_examples 'failed receive' do
let(:email_raw) { fixture_file('emails/valid_reply.eml').gsub(mail_key, "!!!") } it 'adds metric event' do
expect(::Gitlab::Metrics::BackgroundTransaction).to receive(:current).and_return(metric_transaction)
expect(metric_transaction).to receive(:add_event).with('email_receiver_error', { error: expected_error.name })
it "raises an UnknownIncomingEmail error" do expect { receiver.execute }.to raise_error(expected_error)
expect { receiver.execute }.to raise_error(Gitlab::Email::UnknownIncomingEmail)
end end
end end
context "when the email is blank" do context 'when we cannot find a capable handler' do
let(:email_raw) { "" } let(:email_raw) { fixture_file('emails/valid_reply.eml').gsub(mail_key, '!!!') }
let(:expected_error) { Gitlab::Email::UnknownIncomingEmail }
it "raises an EmptyEmailError" do it_behaves_like 'failed receive'
expect { receiver.execute }.to raise_error(Gitlab::Email::EmptyEmailError)
end
end end
context "when the email was auto generated with Auto-Submitted header" do context 'when the email is blank' do
let(:email_raw) { fixture_file("emails/auto_submitted.eml") } let(:email_raw) { '' }
let(:expected_error) { Gitlab::Email::EmptyEmailError }
it "raises an AutoGeneratedEmailError" do it_behaves_like 'failed receive'
expect { receiver.execute }.to raise_error(Gitlab::Email::AutoGeneratedEmailError)
end
end end
context "when the email was auto generated with X-Autoreply header" do context 'when the email was auto generated with Auto-Submitted header' do
let(:email_raw) { fixture_file("emails/auto_reply.eml") } let(:email_raw) { fixture_file('emails/auto_submitted.eml') }
let(:expected_error) { Gitlab::Email::AutoGeneratedEmailError }
it "raises an AutoGeneratedEmailError" do it_behaves_like 'failed receive'
expect { receiver.execute }.to raise_error(Gitlab::Email::AutoGeneratedEmailError)
end
end end
it "requires all handlers to have a unique metric_event" do context 'when the email was auto generated with X-Autoreply header' do
let(:email_raw) { fixture_file('emails/auto_reply.eml') }
let(:expected_error) { Gitlab::Email::AutoGeneratedEmailError }
it_behaves_like 'failed receive'
end
it 'requires all handlers to have a unique metric_event' do
events = Gitlab::Email::Handler.handlers.map do |handler| events = Gitlab::Email::Handler.handlers.map do |handler|
handler.new(Mail::Message.new, 'gitlabhq/gitlabhq+auth_token').metrics_event handler.new(Mail::Message.new, 'gitlabhq/gitlabhq+auth_token').metrics_event
end end
expect(events.uniq.count).to eq events.count expect(events.uniq.count).to eq events.count
end end
it 'requires all handlers to respond to #project' do
Gitlab::Email::Handler.load_handlers.each do |handler|
expect { handler.new(nil, nil).project }.not_to raise_error
end
end
end end
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment