123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100 |
- # Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- require 'rails_helper'
- RSpec.describe 'Webhook > Mattermost', integration: true, performs_jobs: true, required_envs: %w[MATTERMOST_URL MATTERMOST_USER MATTERMOST_PASSWORD MATTERMOST_CHANNEL], time_zone: 'Europe/London' do # rubocop:disable RSpec/DescribeClass
- # Shared/persistent variables
- mattermost_hook_initialized = false
- mattermost_access_token = ''
- mattermost_zammad_channel = nil
- zammad_webhook = nil
- let(:zammad_base_url) { "#{Capybara.app_host}:#{Capybara.current_session.server.port}" }
- let(:mattermost_url) { ENV['MATTERMOST_URL'] }
- let(:mattermost_api_url) { "#{mattermost_url}/api/v4" }
- let(:mattermost_auth_header) { { Authorization: "Bearer #{mattermost_access_token}" } }
- let(:mattermost_default_headers) { { headers: mattermost_auth_header, json: true } }
- let(:mattermost_endpoints) do
- {
- auth: "#{mattermost_api_url}/users/login",
- channel_list: "#{mattermost_api_url}/channels",
- incoming_hooks: "#{mattermost_api_url}/hooks/incoming",
- hooks: "#{mattermost_url}/hooks",
- }
- end
- let(:mattermost_payloads) do
- {
- auth: { login_id: ENV['MATTERMOST_USER'], password: ENV['MATTERMOST_PASSWORD'] },
- incoming_hooks: { display_name: 'Zammad', description: 'Incoming webhook for Zammad' },
- }
- end
- before do
- next if mattermost_hook_initialized
- # Get auth token.
- auth_response = UserAgent.post(mattermost_endpoints[:auth], mattermost_payloads[:auth], { json: true })
- raise 'Authentication failed' if !auth_response.success?
- mattermost_access_token = auth_response.header['token']
- raise 'No access_token found' if mattermost_access_token.blank?
- # Get channel id.
- channel_response = UserAgent.get(mattermost_endpoints[:channel_list], {}, mattermost_default_headers)
- raise 'No channel found' if !channel_response.success? || channel_response.data.blank?
- mattermost_zammad_channel = channel_response.data.find { |channel| channel['name'].eql?(ENV['MATTERMOST_CHANNEL']) }
- raise 'No channel found' if mattermost_zammad_channel.nil?
- # Create incoming webhook.
- incoming_webhook_response = UserAgent.post(mattermost_endpoints[:incoming_hooks], mattermost_payloads[:incoming_hooks].merge(channel_id: mattermost_zammad_channel['id']), mattermost_default_headers)
- raise 'No incoming webhook found' if !incoming_webhook_response.success?
- webhook_id = incoming_webhook_response.data['id']
- raise 'No incoming webhook found' if webhook_id.blank?
- zammad_webhook = create(
- :mattermost_webhook,
- endpoint: "#{mattermost_endpoints[:hooks]}/#{webhook_id}",
- preferences: {
- pre_defined_webhook: {
- messaging_username: Faker::Internet.unique.username,
- messaging_channel: mattermost_zammad_channel['name'],
- messaging_icon_url: Faker::Internet.unique.url,
- },
- }
- )
- mattermost_hook_initialized = true
- end
- context 'when a trigger for ticket create is used' do
- let(:condition) { { 'ticket.action' => { 'operator' => 'is', 'value' => 'create' } } }
- let(:perform) { { 'notification.webhook' => { 'webhook_id' => zammad_webhook.id.to_s } } }
- let(:trigger) { create(:trigger, activator: 'action', condition: condition, perform: perform) }
- let(:message) { "Test for Mattermost (#{SecureRandom.uuid})" }
- before do
- trigger
- end
- it 'creates a post in the related Mattermost channel', :aggregate_failures do
- create(:ticket, group: Group.first, title: message)
- perform_enqueued_jobs commit_transaction: true
- posts_response = UserAgent.get(
- "#{mattermost_api_url}/channels/#{mattermost_zammad_channel['id']}/posts",
- {},
- mattermost_default_headers
- )
- last_post_id = posts_response.data['order'].first
- expect(last_post_id).not_to be_nil
- expect(posts_response.data.dig('posts', last_post_id, 'message')).to eq("# #{message}")
- end
- end
- end
|