All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.gradle.performance.results.SlackReporter.groovy Maven / Gradle / Ivy

There is a newer version: 8.11.1
Show newest version
/*
 * Copyright 2017 the original author or authors.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package org.gradle.performance.results

import com.google.common.base.Charsets
import org.apache.http.HttpStatus
import org.apache.http.client.methods.HttpPost
import org.apache.http.entity.StringEntity
import org.apache.http.impl.client.CloseableHttpClient
import org.apache.http.impl.client.HttpClients
import org.mortbay.util.ajax.JSON
import org.slf4j.Logger
import org.slf4j.LoggerFactory

class SlackReporter implements DataReporter {
    private static final Logger LOGGER = LoggerFactory.getLogger(SlackReporter)
    private static final String[] NOTIFY_BRANCHES = ["master", "release"]
    private static final String SLACK_WEBHOOK_URL_ENV = "SLACK_WEBHOOK_URL"
    private static final String SLACK_PERFORMANCE_REPORT_CHANNEL_ENV = "SLACK_PERFORMANCE_REPORT_CHANNEL"
    private static final String BUILD__URL_ENV = "BUILD_URL"
    private static final String[] EXCLAMATIONS = ["Hurrah!", "Whoopee!", "Excellent!", "Yippee!", "Splendid!", "OMG!", "Amazing!", "Fantastic!", "Awesome!", "Brilliant!"]

    private final URI webhook
    private final String slackPerformanceReportChannel
    private final CloseableHttpClient httpClient
    private final String buildUrl
    private final DataReporter delegate

    SlackReporter(URI webhook, String slackPerformanceReportChannel, DataReporter delegate) {
        this.webhook = webhook
        this.slackPerformanceReportChannel = slackPerformanceReportChannel
        this.delegate = delegate
        this.httpClient = HttpClients.createDefault()
        this.buildUrl = System.getenv(BUILD__URL_ENV)
    }

    @Override
    void report(CrossVersionPerformanceResults results) {
        delegate.report(results)

        if (!NOTIFY_BRANCHES.contains(results.vcsBranch)) {
            return
        }

        def significantlyFasterThanBaselines = results.baselineVersions.every { it.significantlySlowerThan(results.current) }
        if (!significantlyFasterThanBaselines) {
            return
        }

        def exclamation = EXCLAMATIONS[new Random().nextInt(EXCLAMATIONS.length)]
        def changes = ""
        def message = "$exclamation Looks like *${results.testId}* is now faster on `${results.vcsBranch}` after $changes."
        if (buildUrl) {
            message += " (<$buildUrl|Go to build>)"
        }

        def stats = results.baselineVersions.collect { baseline ->
            baseline.getSpeedStatsAgainst(results.displayName, results.current).trim()
        }

        def json = JSON.toString(
            text: message,
            username: "Performance tests",
            icon_emoji: ":dash:",
            channel: slackPerformanceReportChannel,
            attachments: stats.collect {
                [
                    text: "```\n$it\n```",
                    mrkdwn_in: ["text"]
                ]
            }
        )

        def post = new HttpPost(webhook)
        post.setHeader("Accept", "application/json")
        post.setHeader("Content-type", "application/json")
        post.setEntity(new StringEntity(json, Charsets.UTF_8))

        def response = httpClient.execute(post)
        try {
            if (response.statusLine.statusCode != HttpStatus.SC_OK) {
                LOGGER.warn("Could not notify Slack: {}", response.statusLine)
            }
        } finally {
            response.close()
        }
    }

    static DataReporter wrap(DataReporter reporter) {
        def slackWebhookUrl = System.getenv(SLACK_WEBHOOK_URL_ENV)
        def slackPerformanceReportChannel = System.getenv(SLACK_PERFORMANCE_REPORT_CHANNEL_ENV)
        if (slackWebhookUrl) {
            new SlackReporter(URI.create(slackWebhookUrl), slackPerformanceReportChannel, reporter)
        } else {
            return reporter
        }
    }

    @Override
    void close() {
        try {
            httpClient.close()
        } finally {
            delegate.close()
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy