Commit 23049485 authored by Sarah Yasonik's avatar Sarah Yasonik Committed by Dmytro Zaporozhets (DZ)

Add support for arrays to HTTP integration custom mappings

parent 9bceafeb
...@@ -631,7 +631,7 @@ Parsed field from an alert used for custom mappings. ...@@ -631,7 +631,7 @@ Parsed field from an alert used for custom mappings.
| Field | Type | Description | | Field | Type | Description |
| ----- | ---- | ----------- | | ----- | ---- | ----------- |
| `label` | [`String`](#string) | Human-readable label of the payload path. | | `label` | [`String`](#string) | Human-readable label of the payload path. |
| `path` | [`[String!]`](#string) | Path to value inside payload JSON. | | `path` | [`[PayloadAlertFieldPathSegment!]`](#payloadalertfieldpathsegment) | Path to value inside payload JSON. |
| `type` | [`AlertManagementPayloadAlertFieldType`](#alertmanagementpayloadalertfieldtype) | Type of the parsed value. | | `type` | [`AlertManagementPayloadAlertFieldType`](#alertmanagementpayloadalertfieldtype) | Type of the parsed value. |
### `AlertManagementPayloadAlertMappingField` ### `AlertManagementPayloadAlertMappingField`
...@@ -642,7 +642,7 @@ Parsed field (with its name) from an alert used for custom mappings. ...@@ -642,7 +642,7 @@ Parsed field (with its name) from an alert used for custom mappings.
| ----- | ---- | ----------- | | ----- | ---- | ----------- |
| `fieldName` | [`AlertManagementPayloadAlertFieldName`](#alertmanagementpayloadalertfieldname) | A GitLab alert field name. | | `fieldName` | [`AlertManagementPayloadAlertFieldName`](#alertmanagementpayloadalertfieldname) | A GitLab alert field name. |
| `label` | [`String`](#string) | Human-readable label of the payload path. | | `label` | [`String`](#string) | Human-readable label of the payload path. |
| `path` | [`[String!]`](#string) | Path to value inside payload JSON. | | `path` | [`[PayloadAlertFieldPathSegment!]`](#payloadalertfieldpathsegment) | Path to value inside payload JSON. |
| `type` | [`AlertManagementPayloadAlertFieldType`](#alertmanagementpayloadalertfieldtype) | Type of the parsed value. | | `type` | [`AlertManagementPayloadAlertFieldType`](#alertmanagementpayloadalertfieldtype) | Type of the parsed value. |
### `AlertManagementPrometheusIntegration` ### `AlertManagementPrometheusIntegration`
...@@ -8966,6 +8966,10 @@ A `PackagesPackageID` is a global ID. It is encoded as a string. ...@@ -8966,6 +8966,10 @@ A `PackagesPackageID` is a global ID. It is encoded as a string.
An example `PackagesPackageID` is: `"gid://gitlab/Packages::Package/1"`. An example `PackagesPackageID` is: `"gid://gitlab/Packages::Package/1"`.
### `PayloadAlertFieldPathSegment`
String or integer.
### `ProjectID` ### `ProjectID`
A `ProjectID` is a global ID. It is encoded as a string. A `ProjectID` is a global ID. It is encoded as a string.
......
...@@ -12,7 +12,7 @@ module Types ...@@ -12,7 +12,7 @@ module Types
description: 'A GitLab alert field name.' description: 'A GitLab alert field name.'
argument :path, argument :path,
[GraphQL::STRING_TYPE], [::Types::AlertManagement::PayloadAlertFieldPathSegmentType],
required: true, required: true,
description: 'Path to value inside payload JSON.' description: 'Path to value inside payload JSON.'
......
# frozen_string_literal: true
module Types
module AlertManagement
class PayloadAlertFieldPathSegmentType < BaseScalar
graphql_name 'PayloadAlertFieldPathSegment'
description 'String or integer.'
def self.coerce_input(value, ctx)
return value if value.is_a?(::Integer)
GraphQL::STRING_TYPE.coerce_input(value, ctx)
end
def self.coerce_result(value, ctx)
return value if value.is_a?(::Integer)
GraphQL::STRING_TYPE.coerce_result(value, ctx)
end
end
end
end
...@@ -9,7 +9,7 @@ module Types ...@@ -9,7 +9,7 @@ module Types
authorize :read_alert_management_alert authorize :read_alert_management_alert
field :path, field :path,
[GraphQL::STRING_TYPE], [Types::AlertManagement::PayloadAlertFieldPathSegmentType],
null: true, null: true,
description: 'Path to value inside payload JSON.' description: 'Path to value inside payload JSON.'
......
...@@ -14,7 +14,7 @@ module Types ...@@ -14,7 +14,7 @@ module Types
description: 'A GitLab alert field name.' description: 'A GitLab alert field name.'
field :path, field :path,
[GraphQL::STRING_TYPE], [::Types::AlertManagement::PayloadAlertFieldPathSegmentType],
null: true, null: true,
description: 'Path to value inside payload JSON.' description: 'Path to value inside payload JSON.'
......
...@@ -16,18 +16,22 @@ module AlertManagement ...@@ -16,18 +16,22 @@ module AlertManagement
validates :label, presence: true validates :label, presence: true
validates :type, inclusion: { in: SUPPORTED_TYPES } validates :type, inclusion: { in: SUPPORTED_TYPES }
validate :ensure_path_is_non_empty_list_of_strings validate :valid_path
private private
def ensure_path_is_non_empty_list_of_strings def valid_path
return if path_is_non_empty_list_of_strings? return if valid_path?
errors.add(:path, 'must be a list of strings') errors.add(:path, 'must be a list of strings or integers')
end end
def path_is_non_empty_list_of_strings? def valid_path?
path.is_a?(Array) && !path.empty? && path.all? { |segment| segment.is_a?(String) } path.is_a?(Array) && !path.empty? && valid_path_elements?(path)
end
def valid_path_elements?(path)
path.all? { |segment| segment.is_a?(String) || segment.is_a?(Integer) }
end end
end end
end end
---
title: Add support for array indexing to alert integration custom mappings
merge_request: 58610
author:
type: added
...@@ -21,18 +21,44 @@ module Gitlab ...@@ -21,18 +21,44 @@ module Gitlab
type = type_of(value) type = type_of(value)
return unless type return unless type
label = path.last.humanize
::AlertManagement::AlertPayloadField.new( ::AlertManagement::AlertPayloadField.new(
project: project, project: project,
path: path, path: path,
label: label, label: label_for(path),
type: type type: type
) )
end end
# Code duplication with Gitlab::InlineHash#merge_keys ahead! # Code duplication with Gitlab::InlineHash#merge_keys ahead!
# See https://gitlab.com/gitlab-org/gitlab/-/issues/299856 # See https://gitlab.com/gitlab-org/gitlab/-/issues/299856
#
# Determines the keys and indicies needed to identify a value
# in a hash with nested values.
#
# Example:
# {
# apple: [:a, :b],
# pickle: {
# dill: true
# },
# pear: [{ bosc: 5, bartlett: [1, [2]] }]
# }
#
# Becomes:
# [
# [[:apple], [:a, :b]],
# [[:apple, 0], :a],
# [[:apple, 1], :b],
# [[:pickle, :dill], true],
# [[:pear, 0, :bosc], 5]
# [[:pear], [{:bosc=>5, :bartlett=>[1, [2]]}]],
# [[:pear, 0, :bartlett], [1, [2]]],
# [[:pear, 0, :bartlett, 0], 1],
# [[:pear, 0, :bartlett, 1], [2]],
# [[:pear, 0, :bartlett, 1, 0], 2],
# ]
#
# @return Enumerator
def deep_traverse(hash) def deep_traverse(hash)
return to_enum(__method__, hash) unless block_given? return to_enum(__method__, hash) unless block_given?
...@@ -43,6 +69,12 @@ module Gitlab ...@@ -43,6 +69,12 @@ module Gitlab
if value.is_a?(Hash) if value.is_a?(Hash)
value.each { |k, v| pairs.unshift [key + [k], v] } value.each { |k, v| pairs.unshift [key + [k], v] }
elsif value.is_a?(Array)
yield key, value
value.each.with_index do |element, index|
pairs.unshift [key + [index], element]
end
else else
yield key, value yield key, value
end end
...@@ -59,6 +91,19 @@ module Gitlab ...@@ -59,6 +91,19 @@ module Gitlab
::AlertManagement::AlertPayloadField::STRING_TYPE ::AlertManagement::AlertPayloadField::STRING_TYPE
end end
end end
# EX) ['first', 'second'] => 'first/second'
# EX) ['first', 'second', 0, 1, 'third'] => 'first/second[0][1]/third'
#
# Assumes first element in path is a string (as we only
# expect a Hash as payload input)
def label_for(path)
path.reduce do |label, element|
next "#{label}[#{element}]" if element.is_a?(Integer)
"#{label}/#{element}"
end
end
end end
end end
end end
# frozen_string_literal: true
require 'spec_helper'
RSpec.describe GitlabSchema.types['PayloadAlertFieldPathSegment'] do
specify { expect(described_class.graphql_name).to eq('PayloadAlertFieldPathSegment') }
describe '.coerce_input' do
subject { described_class.coerce_isolated_input(input) }
context 'with string' do
let(:input) { 'string' }
it { is_expected.to eq input }
end
context 'with integer' do
let(:input) { 16 }
it { is_expected.to eq input }
end
context 'with non-string or integer' do
let(:input) { [1, 2, 3] }
it { is_expected.to eq nil }
end
end
describe '.coerce_result' do
subject { described_class.coerce_isolated_result(input) }
context 'with string' do
let(:input) { 'string' }
it { is_expected.to eq input }
end
context 'with integer' do
let(:input) { 16 }
it { is_expected.to eq input }
end
context 'with non-string or integer' do
let(:input) { [1, 2, 3] }
it { is_expected.to eq input.to_s }
end
end
end
...@@ -30,21 +30,33 @@ RSpec.describe Gitlab::AlertManagement::AlertPayloadFieldExtractor do ...@@ -30,21 +30,33 @@ RSpec.describe Gitlab::AlertManagement::AlertPayloadFieldExtractor do
discarded_null: nil, discarded_null: nil,
discarded_bool_true: true, discarded_bool_true: true,
discarded_bool_false: false, discarded_bool_false: false,
arr: %w[one two three] arr: [
{ key_a: 'string', key_b: ['nested_arr_value'] },
'non_hash_value',
[['array_a'], 'array_b']
]
) )
end end
it 'returns all the possible field combination and types suggestions' do it 'returns all the possible field combination and types suggestions' do
expect(fields).to contain_exactly( expect(fields).to contain_exactly(
a_field(['str'], 'Str', 'string'), a_field(['str'], 'str', 'string'),
a_field(%w(nested key), 'Key', 'string'), a_field(%w(nested key), 'nested/key', 'string'),
a_field(%w(nested deep key), 'Key', 'string'), a_field(%w(nested deep key), 'nested/deep/key', 'string'),
a_field(['time'], 'Time', 'datetime'), a_field(['time'], 'time', 'datetime'),
a_field(['time_iso_8601_and_rfc_3339'], 'Time iso 8601 and rfc 3339', 'datetime'), a_field(['time_iso_8601_and_rfc_3339'], 'time_iso_8601_and_rfc_3339', 'datetime'),
a_field(['time_iso_8601'], 'Time iso 8601', 'datetime'), a_field(['time_iso_8601'], 'time_iso_8601', 'datetime'),
a_field(['time_iso_8601_short'], 'Time iso 8601 short', 'datetime'), a_field(['time_iso_8601_short'], 'time_iso_8601_short', 'datetime'),
a_field(['time_rfc_3339'], 'Time rfc 3339', 'datetime'), a_field(['time_rfc_3339'], 'time_rfc_3339', 'datetime'),
a_field(['arr'], 'Arr', 'array') a_field(['arr'], 'arr', 'array'),
a_field(['arr', 0, 'key_a'], 'arr[0]/key_a', 'string'),
a_field(['arr', 0, 'key_b'], 'arr[0]/key_b', 'array'),
a_field(['arr', 0, 'key_b', 0], 'arr[0]/key_b[0]', 'string'),
a_field(['arr', 1], 'arr[1]', 'string'),
a_field(['arr', 2], 'arr[2]', 'array'),
a_field(['arr', 2, 0], 'arr[2][0]', 'array'),
a_field(['arr', 2, 1], 'arr[2][1]', 'string'),
a_field(['arr', 2, 0, 0], 'arr[2][0][0]', 'string')
) )
end end
end end
......
...@@ -16,7 +16,7 @@ RSpec.describe AlertManagement::AlertPayloadField do ...@@ -16,7 +16,7 @@ RSpec.describe AlertManagement::AlertPayloadField do
shared_examples 'has invalid path' do shared_examples 'has invalid path' do
it 'is invalid' do it 'is invalid' do
expect(alert_payload_field.valid?).to eq(false) expect(alert_payload_field.valid?).to eq(false)
expect(alert_payload_field.errors.full_messages).to eq(['Path must be a list of strings']) expect(alert_payload_field.errors.full_messages).to eq(['Path must be a list of strings or integers'])
end end
end end
...@@ -32,11 +32,17 @@ RSpec.describe AlertManagement::AlertPayloadField do ...@@ -32,11 +32,17 @@ RSpec.describe AlertManagement::AlertPayloadField do
it_behaves_like 'has invalid path' it_behaves_like 'has invalid path'
end end
context 'when path does not contain only strings' do context 'when path does not contain only strings or integers' do
let(:alert_payload_field) { build(:alert_management_alert_payload_field, path: ['title', 1]) } let(:alert_payload_field) { build(:alert_management_alert_payload_field, path: ['title', {}]) }
it_behaves_like 'has invalid path' it_behaves_like 'has invalid path'
end end
context 'when path contains only strings and integers' do
let(:alert_payload_field) { build(:alert_management_alert_payload_field, path: ['title', 1]) }
it { is_expected.to be_valid }
end
end end
end end
end end
...@@ -11,14 +11,17 @@ RSpec.describe 'Creating a new HTTP Integration' do ...@@ -11,14 +11,17 @@ RSpec.describe 'Creating a new HTTP Integration' do
let(:payload_example) do let(:payload_example) do
{ {
'alert' => { 'name' => 'Test alert' }, 'alert' => { 'name' => 'Test alert' },
'started_at' => Time.current.strftime('%d %B %Y, %-l:%M%p (%Z)') 'started_at' => Time.current.strftime('%d %B %Y, %-l:%M%p (%Z)'),
'tool' => %w[DataDog V1]
}.to_json }.to_json
end end
let(:payload_attribute_mappings) do let(:payload_attribute_mappings) do
[ [
{ fieldName: 'TITLE', path: %w[alert name], type: 'STRING' }, { fieldName: 'TITLE', path: %w[alert name], type: 'STRING' },
{ fieldName: 'START_TIME', path: %w[started_at], type: 'DATETIME', label: 'Start time' } { fieldName: 'START_TIME', path: %w[started_at], type: 'DATETIME', label: 'Start time' },
{ fieldName: 'MONITORING_TOOL', path: ['tool', 0], type: 'STRING', label: 'Tool[0]' },
{ fieldName: 'HOSTS', path: %w[tool], type: 'ARRAY', label: 'Tool' }
] ]
end end
...@@ -45,6 +48,13 @@ RSpec.describe 'Creating a new HTTP Integration' do ...@@ -45,6 +48,13 @@ RSpec.describe 'Creating a new HTTP Integration' do
token token
url url
apiUrl apiUrl
payloadExample
payloadAttributeMappings {
fieldName
path
label
type
}
} }
QL QL
end end
...@@ -61,9 +71,10 @@ RSpec.describe 'Creating a new HTTP Integration' do ...@@ -61,9 +71,10 @@ RSpec.describe 'Creating a new HTTP Integration' do
expect(response).to have_gitlab_http_status(:success) expect(response).to have_gitlab_http_status(:success)
expect(integration_response['id']).to eq(GitlabSchema.id_from_object(new_integration).to_s) expect(integration_response['id']).to eq(GitlabSchema.id_from_object(new_integration).to_s)
expect(new_integration.payload_example).to eq({}) expect(new_integration.payload_example).to eq({})
expect(new_integration.payload_attribute_mapping).to eq({}) expect(new_integration.payload_attribute_mapping).to eq({})
expect(integration_response['payloadExample']).to eq('{}')
expect(integration_response['payloadAttributeMappings']).to be_empty
end end
end end
...@@ -79,14 +90,26 @@ RSpec.describe 'Creating a new HTTP Integration' do ...@@ -79,14 +90,26 @@ RSpec.describe 'Creating a new HTTP Integration' do
post_graphql_mutation(mutation, current_user: current_user) post_graphql_mutation(mutation, current_user: current_user)
new_integration = ::AlertManagement::HttpIntegration.last! new_integration = ::AlertManagement::HttpIntegration.last!
integration_response = mutation_response['integration']
expect(new_integration.payload_example).to eq(Gitlab::Json.parse(payload_example)) expect(new_integration.payload_example).to eq(Gitlab::Json.parse(payload_example))
expect(new_integration.payload_attribute_mapping).to eq( expect(new_integration.payload_attribute_mapping).to eq(
{ {
'title' => { 'path' => %w[alert name], 'type' => 'string', 'label' => nil }, 'title' => { 'path' => %w[alert name], 'type' => 'string', 'label' => nil },
'start_time' => { 'path' => %w[started_at], 'type' => 'datetime', 'label' => 'Start time' } 'start_time' => { 'path' => %w[started_at], 'type' => 'datetime', 'label' => 'Start time' },
'monitoring_tool' => { 'path' => ['tool', 0], 'type' => 'string', 'label' => 'Tool[0]' },
'hosts' => { 'path' => %w[tool], 'type' => 'array', 'label' => 'Tool' }
} }
) )
expect(integration_response['payloadExample']).to eq(payload_example)
expect(integration_response['payloadAttributeMappings']).to eq(
[
{ 'fieldName' => 'TITLE', 'path' => %w[alert name], 'type' => 'STRING', 'label' => nil },
{ 'fieldName' => 'START_TIME', 'path' => %w[started_at], 'type' => 'DATETIME', 'label' => 'Start time' },
{ 'fieldName' => 'MONITORING_TOOL', 'path' => ['tool', 0], 'type' => 'STRING', 'label' => 'Tool[0]' },
{ 'fieldName' => 'HOSTS', 'path' => %w[tool], 'type' => 'ARRAY', 'label' => 'Tool' }
]
)
end end
[:project_path, :active, :name].each do |argument| [:project_path, :active, :name].each do |argument|
......
...@@ -43,6 +43,9 @@ RSpec.describe 'Updating an existing HTTP Integration' do ...@@ -43,6 +43,9 @@ RSpec.describe 'Updating an existing HTTP Integration' do
payloadExample payloadExample
payloadAttributeMappings { payloadAttributeMappings {
fieldName fieldName
path
label
type
} }
} }
QL QL
...@@ -55,9 +58,12 @@ RSpec.describe 'Updating an existing HTTP Integration' do ...@@ -55,9 +58,12 @@ RSpec.describe 'Updating an existing HTTP Integration' do
it 'updates integration without the custom mapping params', :aggregate_failures do it 'updates integration without the custom mapping params', :aggregate_failures do
post_graphql_mutation(mutation, current_user: current_user) post_graphql_mutation(mutation, current_user: current_user)
integration.reload
integration_response = mutation_response['integration'] integration_response = mutation_response['integration']
expect(response).to have_gitlab_http_status(:success) expect(response).to have_gitlab_http_status(:success)
expect(integration.payload_example).to eq({})
expect(integration.payload_attribute_mapping).to eq({})
expect(integration_response['payloadExample']).to eq('{}') expect(integration_response['payloadExample']).to eq('{}')
expect(integration_response['payloadAttributeMappings']).to be_empty expect(integration_response['payloadAttributeMappings']).to be_empty
end end
...@@ -76,8 +82,10 @@ RSpec.describe 'Updating an existing HTTP Integration' do ...@@ -76,8 +82,10 @@ RSpec.describe 'Updating an existing HTTP Integration' do
it 'updates the custom mapping params', :aggregate_failures do it 'updates the custom mapping params', :aggregate_failures do
post_graphql_mutation(mutation, current_user: current_user) post_graphql_mutation(mutation, current_user: current_user)
expect(response).to have_gitlab_http_status(:success)
integration.reload integration.reload
integration_response = mutation_response['integration']
expect(response).to have_gitlab_http_status(:success)
expect(integration.payload_example).to eq(Gitlab::Json.parse(payload_example)) expect(integration.payload_example).to eq(Gitlab::Json.parse(payload_example))
expect(integration.payload_attribute_mapping).to eq( expect(integration.payload_attribute_mapping).to eq(
'start_time' => { 'start_time' => {
...@@ -91,6 +99,13 @@ RSpec.describe 'Updating an existing HTTP Integration' do ...@@ -91,6 +99,13 @@ RSpec.describe 'Updating an existing HTTP Integration' do
'type' => 'string' 'type' => 'string'
} }
) )
expect(integration_response['payloadExample']).to eq(payload_example)
expect(integration_response['payloadAttributeMappings']).to eq(
[
{ 'fieldName' => 'TITLE', 'path' => %w[alert name], 'type' => 'STRING', 'label' => nil },
{ 'fieldName' => 'START_TIME', 'path' => %w[started_at], 'type' => 'DATETIME', 'label' => 'Start time' }
]
)
end end
context 'when the integration already has custom mapping params' do context 'when the integration already has custom mapping params' do
......
...@@ -129,12 +129,12 @@ RSpec.describe 'getting Alert Management HTTP Integrations' do ...@@ -129,12 +129,12 @@ RSpec.describe 'getting Alert Management HTTP Integrations' do
], ],
'payloadAlertFields' => [ 'payloadAlertFields' => [
{ {
'label' => 'Name', 'label' => 'alert/name',
'path' => %w(alert name), 'path' => %w(alert name),
'type' => 'STRING' 'type' => 'STRING'
}, },
{ {
'label' => 'Desc', 'label' => 'alert/desc',
'path' => %w(alert desc), 'path' => %w(alert desc),
'type' => 'STRING' 'type' => 'STRING'
} }
......
...@@ -55,10 +55,12 @@ RSpec.describe 'parse alert payload fields' do ...@@ -55,10 +55,12 @@ RSpec.describe 'parse alert payload fields' do
specify do specify do
expect(parsed_fields).to eq([ expect(parsed_fields).to eq([
{ 'path' => %w[title], 'label' => 'Title', 'type' => 'STRING' }, { 'path' => %w[title], 'label' => 'title', 'type' => 'STRING' },
{ 'path' => %w[started_at], 'label' => 'Started at', 'type' => 'DATETIME' }, { 'path' => %w[started_at], 'label' => 'started_at', 'type' => 'DATETIME' },
{ 'path' => %w[nested key], 'label' => 'Key', 'type' => 'STRING' }, { 'path' => %w[nested key], 'label' => 'nested/key', 'type' => 'STRING' },
{ 'path' => %w[arr], 'label' => 'Arr', 'type' => 'ARRAY' } { 'path' => %w[arr], 'label' => 'arr', 'type' => 'ARRAY' },
{ 'path' => ['arr', 1], 'label' => 'arr[1]', 'type' => 'STRING' },
{ 'path' => ['arr', 0], 'label' => 'arr[0]', 'type' => 'STRING' }
]) ])
end end
......
...@@ -38,7 +38,7 @@ RSpec.describe AlertManagement::ExtractAlertPayloadFieldsService do ...@@ -38,7 +38,7 @@ RSpec.describe AlertManagement::ExtractAlertPayloadFieldsService do
field = fields.first field = fields.first
expect(fields.count).to eq(1) expect(fields.count).to eq(1)
expect(field.label).to eq('Foo') expect(field.label).to eq('foo')
expect(field.type).to eq('string') expect(field.type).to eq('string')
expect(field.path).to eq(%w[foo]) expect(field.path).to eq(%w[foo])
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