Created
February 21, 2018 05:18
-
-
Save stelcheck/b463816e4092cde62934d7ea64d25ac9 to your computer and use it in GitHub Desktop.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
# rubocop:disable Style/CaseEquality | |
# rubocop:disable Style/MultilineTernaryOperator | |
# rubocop:disable Style/NestedTernaryOperator | |
module Fastlane | |
module Actions | |
class MattermostAction < Action | |
def self.is_supported?(platform) | |
true | |
end | |
# As there is a text limit in the notifications, we are | |
# usually interested in the last part of the message | |
# e.g. for tests | |
def self.trim_message(message) | |
# We want the last 7000 characters, instead of the first 7000, as the error is at the bottom | |
start_index = [message.length - 7000, 0].max | |
message = message[start_index..-1] | |
message | |
end | |
def self.run(options) | |
require 'slack-notifier' | |
options[:message] = self.trim_message(options[:message].to_s || '') | |
options[:message] = Slack::Notifier::Util::LinkFormatter.format(options[:message]) | |
channel = nil | |
if options[:channel].to_s.length > 0 | |
channel = options[:channel] | |
channel = ('#' + notifier.channel) unless ['#', '@'].include?(notifier.channel[0]) # send message to channel by default | |
end | |
notifier = Slack::Notifier.new(options[:mattermost_url]) do | |
defaults( | |
username: options[:overwrite_webhook_username_and_icon] ? nil : options[:username], | |
channel: channel | |
) | |
end | |
icon_url = options[:overwrite_webhook_username_and_icon] ? nil : options[:icon_url] | |
mattermost_attachment = generate_mattermost_attachments(options) | |
return [notifier, mattermost_attachment] if Helper.is_test? # tests will verify the mattermost attachments and other properties | |
results = notifier.ping '', | |
icon_url: icon_url, | |
attachments: [mattermost_attachment] | |
result = results.first | |
if !result.nil? && result.code.to_i == 200 | |
UI.success('Successfully sent Mattermost notification') | |
else | |
UI.verbose(result) | |
UI.user_error!("Error pushing Mattermost message, maybe the integration has no permission to post on this channel? Try removing the channel parameter in your Fastfile.") | |
end | |
end | |
def self.description | |
"Send a success/error message to your Mattermost channel" | |
end | |
def self.available_options | |
[ | |
FastlaneCore::ConfigItem.new(key: :pretext, | |
env_name: "MATTERMOST_PRETEXT", | |
description: "An optional line of text that will be shown above the attachment. This supports the standard Mattermost markup language", | |
optional: true), | |
FastlaneCore::ConfigItem.new(key: :message, | |
env_name: "MATTERMOST_MESSAGE", | |
description: "The message that should be displayed on Mattermost. This supports the standard Mattermost markup language", | |
optional: true), | |
FastlaneCore::ConfigItem.new(key: :channel, | |
env_name: "MATTERMOST_CHANNEL", | |
description: "channel or @username", | |
optional: true), | |
FastlaneCore::ConfigItem.new(key: :overwrite_webhook_username_and_icon, | |
env_name: "MATTERMOST_USE_WEBHOOK_CONFIGURED_USERNAME_AND_ICON", | |
description: "Use webook's default username and icon settings? (true/false)", | |
default_value: false, | |
is_string: false, | |
optional: true), | |
FastlaneCore::ConfigItem.new(key: :mattermost_url, | |
env_name: "MATTERMOST_URL", | |
sensitive: true, | |
description: "Create an Incoming WebHook for your Mattermost channel", | |
verify_block: proc do |value| | |
UI.user_error!("Invalid URL, must start with https:// or http://") unless value.start_with? "https://" or value.start_with? "http://" | |
end), | |
FastlaneCore::ConfigItem.new(key: :username, | |
env_name: "MATTERMOST_USERNAME", | |
description: "Overrides the webook's username property if overwrite_webhook_username_and_icon is false", | |
default_value: "fastlane", | |
is_string: true, | |
optional: true), | |
FastlaneCore::ConfigItem.new(key: :icon_url, | |
env_name: "MATTERMOST_ICON_URL", | |
description: "Overrides the webook's image property if overwrite_webhook_username_and_icon is false", | |
default_value: "https://s3-eu-west-1.amazonaws.com/fastlane.tools/fastlane.png", | |
is_string: true, | |
optional: true), | |
FastlaneCore::ConfigItem.new(key: :payload, | |
env_name: "MATTERMOST_PAYLOAD", | |
description: "Add additional information to this post. payload must be a hash containg any key with any value", | |
default_value: {}, | |
is_string: false), | |
FastlaneCore::ConfigItem.new(key: :default_payloads, | |
env_name: "MATTERMOST_DEFAULT_PAYLOADS", | |
description: "Remove some of the default payloads. More information about the available payloads on GitHub", | |
optional: true, | |
is_string: false), | |
FastlaneCore::ConfigItem.new(key: :attachment_properties, | |
env_name: "MATTERMOST_ATTACHMENT_PROPERTIES", | |
description: "Merge additional properties in the Mattermost attachment, see https://docs.mattermost.com/developer/message-attachments.html", | |
default_value: {}, | |
is_string: false), | |
FastlaneCore::ConfigItem.new(key: :success, | |
env_name: "MATTERMOST_SUCCESS", | |
description: "Was this build successful? (true/false)", | |
optional: true, | |
default_value: true, | |
is_string: false) | |
] | |
end | |
def self.author | |
"Mattermost base on the Slack action by KrauseFx" | |
end | |
def self.example_code | |
[ | |
'mattermost(message: "App successfully released!")', | |
'mattermost( | |
message: "App successfully released!", | |
channel: "channel", # Optional, by default will post to the default channel configured for the POST URL. | |
success: true, # Optional, defaults to true. | |
payload: { # Optional, lets you specify any number of your own Mattermost attachments. | |
"Build Date" => Time.new.to_s, | |
"Built by" => "Jenkins", | |
}, | |
default_payloads: [:git_branch, :git_author], # Optional, lets you specify a whitelist of default payloads to include. Pass an empty array to suppress all the default payloads. | |
# Don\'t add this key, or pass nil, if you want all the default payloads. The available default payloads are: `lane`, `test_result`, `git_branch`, `git_author`, `last_git_commit_message`. | |
attachment_properties: { # Optional, lets you specify any other properties available for attachments in the Mattermost API (see https://docs.mattermost.com/developer/message-attachments.html). | |
# This hash is deep merged with the existing properties set using the other properties above. This allows your own fields properties to be appended to the existing fields that were created using the `payload` property for instance. | |
thumb_url: "http://example.com/path/to/thumb.png", | |
fields: [{ | |
title: "My Field", | |
value: "My Value", | |
short: true | |
}] | |
} | |
)' | |
] | |
end | |
def self.category | |
:notifications | |
end | |
def self.details | |
"Create an Incoming WebHook and export this as `MATTERMOST_URL`. Can send a message to **channel** (by default), a direct message to **@username** or a message to a private group **group** with success (green) or failure (red) status." | |
end | |
##################################################### | |
# @!group Helper | |
##################################################### | |
def self.generate_mattermost_attachments(options) | |
color = (options[:success] ? 'good' : 'danger') | |
should_add_payload = ->(payload_name) { options[:default_payloads].nil? || options[:default_payloads].join(" ").include?(payload_name.to_s) } | |
mattermost_attachment = { | |
fallback: options[:message], | |
pretext: options[:pretext], | |
text: options[:message], | |
color: color, | |
fields: [] | |
} | |
# custom user payloads | |
mattermost_attachment[:fields] += options[:payload].map do |k, v| | |
{ | |
title: k.to_s, | |
value: Slack::Notifier::LinkFormatter.format(v.to_s), | |
short: false | |
} | |
end | |
# Add the lane to the Mattermost message | |
# This might be nil, if mattermost is called as "one-off" action | |
if should_add_payload[:lane] && Actions.lane_context[Actions::SharedValues::LANE_NAME] | |
mattermost_attachment[:fields] << { | |
title: 'Lane', | |
value: Actions.lane_context[Actions::SharedValues::LANE_NAME], | |
short: true | |
} | |
end | |
# test_result | |
if should_add_payload[:test_result] | |
mattermost_attachment[:fields] << { | |
title: 'Result', | |
value: (options[:success] ? 'Success' : 'Error'), | |
short: true | |
} | |
end | |
# git branch | |
if Actions.git_branch && should_add_payload[:git_branch] | |
mattermost_attachment[:fields] << { | |
title: 'Git Branch', | |
value: Actions.git_branch, | |
short: true | |
} | |
end | |
# git_author | |
if Actions.git_author_email && should_add_payload[:git_author] | |
if FastlaneCore::Env.truthy?('FASTLANE_MATTERMOST_HIDE_AUTHOR_ON_SUCCESS') && options[:success] | |
# We only show the git author if the build failed | |
else | |
mattermost_attachment[:fields] << { | |
title: 'Git Author', | |
value: Actions.git_author_email, | |
short: true | |
} | |
end | |
end | |
# last_git_commit | |
if Actions.last_git_commit_message && should_add_payload[:last_git_commit] | |
mattermost_attachment[:fields] << { | |
title: 'Git Commit', | |
value: Actions.last_git_commit_message, | |
short: false | |
} | |
end | |
# merge additional properties | |
deep_merge(mattermost_attachment, options[:attachment_properties]) | |
end | |
# Adapted from https://stackoverflow.com/a/30225093/158525 | |
def self.deep_merge(a, b) | |
merger = proc do |key, v1, v2| | |
Hash === v1 && Hash === v2 ? | |
v1.merge(v2, &merger) : Array === v1 && Array === v2 ? | |
v1 | v2 : [:undefined, nil, :nil].include?(v2) ? v1 : v2 | |
end | |
a.merge(b, &merger) | |
end | |
end | |
end | |
end | |
# rubocop:enable Style/CaseEquality | |
# rubocop:enable Style/MultilineTernaryOperator | |
# rubocop:enable Style/NestedTernaryOperator |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
Note: You will need to install
slack-notifier
as a dependency.