From 9cbeeaa6982d7a90c452043bb752c56554016e92 Mon Sep 17 00:00:00 2001 From: Alexey Alter-Pesotskiy Date: Mon, 9 Dec 2024 17:18:45 +0000 Subject: [PATCH] [CI] Implement SonarCloud --- .github/workflows/build-and-test.yml | 5 ++ .github/workflows/pr-checks.yml | 5 ++ build.gradle | 8 +- .../io/getstream/chat/android/Dependencies.kt | 2 + .../unittest/filter/UnitTestsFilter.kt | 8 +- .../command/unittest/model/TestType.kt | 1 + coverage.gradle | 88 +++++++++++++++++++ fastlane/Fastfile | 7 ++ sonar.gradle | 15 ++++ stream-chat-android-client/build.gradle | 1 - stream-chat-android-offline/build.gradle | 3 - 11 files changed, 133 insertions(+), 10 deletions(-) create mode 100644 coverage.gradle create mode 100644 sonar.gradle diff --git a/.github/workflows/build-and-test.yml b/.github/workflows/build-and-test.yml index 176a29478e6..29bb9122419 100644 --- a/.github/workflows/build-and-test.yml +++ b/.github/workflows/build-and-test.yml @@ -46,6 +46,11 @@ jobs: with: name: testDebugUnitTest path: ./**/build/reports/tests/testDebugUnitTest + - uses: ./.github/actions/setup-ruby + - name: Sonar + run: bundle exec fastlane run_sonar_analysis + env: + SONAR_TOKEN: ${{ secrets.SONAR_TOKEN }} size_check_xml: name: Size Check XML diff --git a/.github/workflows/pr-checks.yml b/.github/workflows/pr-checks.yml index 580ebf4e5e4..2026476863f 100644 --- a/.github/workflows/pr-checks.yml +++ b/.github/workflows/pr-checks.yml @@ -73,6 +73,11 @@ jobs: with: name: testDebugUnitTest path: ./**/build/reports/tests/testDebugUnitTest + - uses: ./.github/actions/setup-ruby + - name: Sonar + run: bundle exec fastlane run_sonar_analysis + env: + SONAR_TOKEN: ${{ secrets.SONAR_TOKEN }} rubocop: name: Rubocop diff --git a/build.gradle b/build.gradle index 5c1369038ff..81fdc1ce1c8 100644 --- a/build.gradle +++ b/build.gradle @@ -11,6 +11,7 @@ apply plugin: "com.github.ben-manes.versions" apply plugin: 'io.github.gradle-nexus.publish-plugin' apply plugin: 'binary-compatibility-validator' apply plugin: 'org.jetbrains.dokka' +apply from: "${rootDir}/sonar.gradle" apply from: "${rootDir}/scripts/sample-app-versioner.gradle" apply plugin: UnitTestsPlugin apply plugin: ReleasePlugin @@ -73,15 +74,16 @@ buildscript { classpath Dependencies.spotlessGradlePlugin classpath Dependencies.ksp classpath Dependencies.detektPlugin + classpath Dependencies.sonarQubePlugin } } subprojects { - if (it.name != 'stream-chat-android-docs' - && it.buildFile.exists()) { + if (it.name != 'stream-chat-android-docs' && it.buildFile.exists()) { apply from: "${rootDir}/spotless/spotless.gradle" } apply plugin: "io.gitlab.arturbosch.detekt" + apply from: "${rootDir}/coverage.gradle" } versionPrint { @@ -134,4 +136,4 @@ apiValidation { ] } -apply from: "${rootDir}/scripts/publish-root.gradle" +apply from: "${rootDir}/scripts/publish-root.gradle" \ No newline at end of file diff --git a/buildSrc/src/main/kotlin/io/getstream/chat/android/Dependencies.kt b/buildSrc/src/main/kotlin/io/getstream/chat/android/Dependencies.kt index 011b32d6f93..ac70b939fc1 100644 --- a/buildSrc/src/main/kotlin/io/getstream/chat/android/Dependencies.kt +++ b/buildSrc/src/main/kotlin/io/getstream/chat/android/Dependencies.kt @@ -79,6 +79,7 @@ object Versions { internal const val SHIMMER = "0.5.0" internal const val SHIMMER_COMPOSE = "1.3.1" internal const val SHOT = "6.1.0" + internal const val SONARQUBE_PLUGIN = "6.0.1.5171" internal const val SPOTLESS = "6.20.0" internal const val STREAM_LOG = "1.3.1" internal const val STREAM_PUSH = "1.1.9" @@ -252,6 +253,7 @@ object Dependencies { const val allureKotlinCommons = "io.qameta.allure:allure-kotlin-commons:${Versions.ALLURE_KOTLIN}" const val allureKotlinJunit = "io.qameta.allure:allure-kotlin-junit4:${Versions.ALLURE_KOTLIN}" const val allureKotlinAndroid = "io.qameta.allure:allure-kotlin-android:${Versions.ALLURE_KOTLIN}" + const val sonarQubePlugin = "org.sonarsource.scanner.gradle:sonarqube-gradle-plugin:${Versions.SONARQUBE_PLUGIN}" @JvmStatic fun isNonStable(version: String): Boolean = isStable(version).not() diff --git a/buildSrc/src/main/kotlin/io/getstream/chat/android/command/unittest/filter/UnitTestsFilter.kt b/buildSrc/src/main/kotlin/io/getstream/chat/android/command/unittest/filter/UnitTestsFilter.kt index dd90e2841fe..c788615a9ba 100644 --- a/buildSrc/src/main/kotlin/io/getstream/chat/android/command/unittest/filter/UnitTestsFilter.kt +++ b/buildSrc/src/main/kotlin/io/getstream/chat/android/command/unittest/filter/UnitTestsFilter.kt @@ -10,7 +10,9 @@ fun List.selectedUnitTestCommand(rootProject: Project): String { return filterUnitTestableModules(rootProject) .filter { (testableModule, _) -> modulesWithTest.contains(testableModule) } - .generateGradleCommand { (module, testType) -> "$module:${testType.testCommand}" } + .generateGradleCommand { (module, testType) -> + "$module:${testType.testCommand} $module:${TestType.JACOCO_TEST_COVERAGE.testCommand}" + } } private fun List.filterModulesWithTests(): List { @@ -24,7 +26,7 @@ private fun List.filterUnitTestableModules(rootProject: Project): List

project.hasUnitTest() && this.contains(project.name) } .map { project -> val testType = when { - project.tasks.any { task -> task.name == "testDebugUnitTest" } -> { + project.tasks.any { task -> task.name == TestType.ANDROID_LIBRARY_TEST.testCommand } -> { TestType.ANDROID_LIBRARY_TEST } @@ -38,5 +40,5 @@ private fun List.filterUnitTestableModules(rootProject: Project): List

- task.name == "testDebugUnitTest" || task.name == "test" + task.name == TestType.ANDROID_LIBRARY_TEST.testCommand || task.name == TestType.JAVA_LIBRARY_TEST.testCommand } diff --git a/buildSrc/src/main/kotlin/io/getstream/chat/android/command/unittest/model/TestType.kt b/buildSrc/src/main/kotlin/io/getstream/chat/android/command/unittest/model/TestType.kt index 66401036bc6..4430252e0cd 100644 --- a/buildSrc/src/main/kotlin/io/getstream/chat/android/command/unittest/model/TestType.kt +++ b/buildSrc/src/main/kotlin/io/getstream/chat/android/command/unittest/model/TestType.kt @@ -3,4 +3,5 @@ package io.getstream.chat.android.command.unittest.model enum class TestType(val testCommand: String) { JAVA_LIBRARY_TEST("test"), ANDROID_LIBRARY_TEST("testDebugUnitTest"), + JACOCO_TEST_COVERAGE("testCoverage"), } diff --git a/coverage.gradle b/coverage.gradle new file mode 100644 index 00000000000..c8ec9f6d877 --- /dev/null +++ b/coverage.gradle @@ -0,0 +1,88 @@ +apply plugin: 'jacoco' + +def isCore = (name == 'stream-chat-android-core') +def testTask = isCore ? "test" : "testDebugUnitTest" +def jacocoResults = "${buildDir}/reports/jacoco/report.xml" + +if (hasProperty('android')) { + android { + buildTypes { + debug { + testCoverageEnabled = true + enableUnitTestCoverage = true + enableAndroidTestCoverage true + } + } + } +} + +afterEvaluate { + tasks.withType(Test).configureEach { + jacoco.includeNoLocationClasses = true + jacoco.excludes = ['jdk.internal.*'] + } + + tasks.register("testCoverage", JacocoReport) { + dependsOn testTask + + reports { + html.required.set(false) + xml.required.set(true) + xml.outputLocation.set(file(jacocoResults)) + } + + executionData.setFrom(fileTree(dir: buildDir, includes: [ + "outputs/unit_test_code_coverage/debugUnitTest/testDebugUnitTest.exec", + "jacoco/test.exec" + ])) + + def excludeFilter = [ + '**/test/**', + '**/androidTest/**', + '**/R.class', + '**/R2.class', + '**/R$*.class', + '**/BuildConfig.*', + '**/Manifest*.*', + '**/*Test*.*' + ] + + def sources = isCore ? sourceSets.main.java.srcDirs : android.sourceSets.main.java.srcDirs + sourceDirectories.setFrom(files(sources)) + classDirectories.setFrom(files([ + fileTree( + dir: "${buildDir}/tmp/kotlin-classes/debug", + excludes: excludeFilter + ), + fileTree( + dir: "${buildDir}/classes/kotlin/main", + excludes: excludeFilter + ) + ])) + } +} + +def ignoredModules = [ + 'stream-chat-android-docs', + 'stream-chat-android-benchmark', + 'stream-chat-android-compose-sample', + 'stream-chat-android-e2e-test', + 'stream-chat-android-previewdata', + 'stream-chat-android-test', + 'stream-chat-android-ui-guides', + 'stream-chat-android-ui-components-sample', + 'stream-chat-android-pushprovider-firebase', + 'stream-chat-android-ai-assistant', + 'stream-chat-android-client-test', + 'stream-chat-android-ui-uitests' +] +if (!ignoredModules.contains(name)) { + apply plugin: "org.sonarqube" + + sonarqube { + properties { + property "sonar.junit.reportPaths", "${buildDir}/test-results/${testTask}" + property "sonar.coverage.jacoco.xmlReportPaths", jacocoResults + } + } +} diff --git a/fastlane/Fastfile b/fastlane/Fastfile index bc3cad6c907..6f9e22ec41e 100644 --- a/fastlane/Fastfile +++ b/fastlane/Fastfile @@ -34,6 +34,12 @@ lane :stop_mock_server do Net::HTTP.get_response(URI("http://localhost:#{mock_server_driver_port}/stop")) rescue nil end +lane :run_sonar_analysis do |options| + next unless is_check_required(sources: sources_matrix[:unit], force_check: @force_check) + + Dir.chdir('..') { sh('./gradlew sonar') } +end + lane :run_snapshot_test do |options| next unless is_check_required(sources: sources_matrix[:ui], force_check: @force_check) @@ -128,6 +134,7 @@ end private_lane :sources_matrix do { + unit: ['stream-chat-android-client/', 'stream-chat-android-client-test/', 'stream-chat-android-compose/', 'stream-chat-android-core/', 'stream-chat-android-markdown-transformer/', 'stream-chat-android-offline/', 'stream-chat-android-state/', 'stream-chat-android-ui-common/', 'stream-chat-android-ui-components/', 'stream-chat-android-ui-utils/'], ui: ['stream-chat-android-ui', '.github/workflows/snapshot-test'], e2e: ['buildSrc', 'stream-chat-android', '.github/workflows/e2e-test'], ruby: ['fastlane', 'Gemfile', 'Gemfile.lock'] diff --git a/sonar.gradle b/sonar.gradle new file mode 100644 index 00000000000..26954e647fc --- /dev/null +++ b/sonar.gradle @@ -0,0 +1,15 @@ +apply plugin: "org.sonarqube" + +sonarqube { + properties { + property("sonar.host.url", "https://sonarcloud.io") + property("sonar.token", "${System.getenv("SONAR_TOKEN")}") + property("sonar.organization", "getstream") + property("sonar.projectKey", "GetStream_stream-chat-android") + property("sonar.projectName", "stream-chat-android") + property "sonar.java.coveragePlugin", "jacoco" + property "sonar.sourceEncoding", "UTF-8" + property "sonar.java.binaries", "${rootDir}/**/build/tmp/kotlin-classes/debug,${rootDir}/**/build/classes/kotlin/main" + property "sonar.coverage.exclusions", "**/test/**,**/androidTest/**,**/R.class,**/R2.class,**/R\$*.class,**/BuildConfig.*,**/Manifest*.*,**/*Test*.*" + } +} diff --git a/stream-chat-android-client/build.gradle b/stream-chat-android-client/build.gradle index d802c186668..58d9d368eac 100644 --- a/stream-chat-android-client/build.gradle +++ b/stream-chat-android-client/build.gradle @@ -30,7 +30,6 @@ android { consumerProguardFiles 'consumer-proguard-rules.pro' } debug { - testCoverageEnabled false consumerProguardFiles 'consumer-proguard-rules.pro' } } diff --git a/stream-chat-android-offline/build.gradle b/stream-chat-android-offline/build.gradle index 9377cd70e6a..3491461ae7b 100644 --- a/stream-chat-android-offline/build.gradle +++ b/stream-chat-android-offline/build.gradle @@ -40,9 +40,6 @@ android { minifyEnabled false proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro' } - debug { - testCoverageEnabled false - } } }