plugins { id "com.android.library" apply false // Necessary for Android plugin to find its classes id "com.google.osdetector" apply false id "me.champeau.gradle.japicmp" apply false id "net.ltgt.errorprone" apply false } import net.ltgt.gradle.errorprone.CheckSeverity subprojects { apply plugin: "checkstyle" apply plugin: "maven" apply plugin: "idea" apply plugin: "signing" apply plugin: "jacoco" apply plugin: "com.google.osdetector" apply plugin: "net.ltgt.errorprone" group = "io.grpc" version = "1.29.0-SNAPSHOT" // CURRENT_GRPC_VERSION repositories { maven { // The google mirror is less flaky than mavenCentral() url "https://maven-central.storage-download.googleapis.com/repos/central/data/" } mavenCentral() mavenLocal() } tasks.withType(JavaCompile) { it.options.compilerArgs += [ "-Xlint:all", "-Xlint:-options", "-Xlint:-path", "-Xlint:-try" ] it.options.encoding = "UTF-8" if (rootProject.hasProperty('failOnWarnings') && rootProject.failOnWarnings.toBoolean()) { it.options.compilerArgs += ["-Werror"] } } ext { def exeSuffix = osdetector.os == 'windows' ? ".exe" : "" protocPluginBaseName = 'protoc-gen-grpc-java' javaPluginPath = "$rootDir/compiler/build/exe/java_plugin/$protocPluginBaseName$exeSuffix" nettyVersion = '4.1.45.Final' guavaVersion = '28.1-android' googleauthVersion = '0.20.0' protobufVersion = '3.11.0' protocVersion = protobufVersion opencensusVersion = '0.24.0' configureProtoCompilation = { String generatedSourcePath = "${projectDir}/src/generated" project.protobuf { protoc { if (project.hasProperty('protoc')) { path = project.protoc } else { artifact = "com.google.protobuf:protoc:${protocVersion}" } } generateProtoTasks { all().each { task -> // Recompile protos when build.gradle has been changed, because // it's possible the version of protoc has been changed. task.inputs.file "${rootProject.projectDir}/build.gradle" } } } if (rootProject.childProjects.containsKey('grpc-compiler')) { // Only when the codegen is built along with the project, will we be able to run // the grpc code generator. project.protobuf { plugins { grpc { path = javaPluginPath } } generateProtoTasks { all().each { task -> project["syncGeneratedSources${task.sourceSet.name}"].dependsOn task task.dependsOn ':grpc-compiler:java_pluginExecutable' // Recompile protos when the codegen has been changed task.inputs.file javaPluginPath task.plugins { grpc { option 'noversion' } } } } } task syncGeneratedSources { } sourceSets.all { sourceSet -> task "syncGeneratedSources${sourceSet.name}"(type: Sync) { from "$buildDir/generated/source/proto/${sourceSet.name}/grpc" into "$generatedSourcePath/${sourceSet.name}/grpc" } syncGeneratedSources.dependsOn "syncGeneratedSources${sourceSet.name}" } // Re-sync as part of a normal build, to avoid forgetting to run the sync assemble.dependsOn syncGeneratedSources } else { // Otherwise, we just use the checked-in generated code. project.sourceSets { main { java { srcDir "${generatedSourcePath}/main/grpc" } } test { java { srcDir "${generatedSourcePath}/test/grpc" } } } } tasks.withType(JavaCompile) { it.options.errorprone.excludedPaths = ".*/src/generated/[^/]+/java/.*" + "|.*/build/generated/source/proto/[^/]+/java/.*" } } libraries = [ android_annotations: "com.google.android:annotations:4.1.1.4", animalsniffer_annotations: "org.codehaus.mojo:animal-sniffer-annotations:1.18", errorprone: "com.google.errorprone:error_prone_annotations:2.3.4", cronet_api: 'org.chromium.net:cronet-api:76.3809.111', cronet_embedded: 'org.chromium.net:cronet-embedded:66.3359.158', gson: "com.google.code.gson:gson:2.8.6", guava: "com.google.guava:guava:${guavaVersion}", hpack: 'com.twitter:hpack:0.10.1', javax_annotation: 'javax.annotation:javax.annotation-api:1.2', jsr305: 'com.google.code.findbugs:jsr305:3.0.2', google_api_protos: 'com.google.api.grpc:proto-google-common-protos:1.17.0', google_auth_credentials: "com.google.auth:google-auth-library-credentials:${googleauthVersion}", google_auth_oauth2_http: "com.google.auth:google-auth-library-oauth2-http:${googleauthVersion}", okhttp: 'com.squareup.okhttp:okhttp:2.7.4', okio: 'com.squareup.okio:okio:1.13.0', opencensus_api: "io.opencensus:opencensus-api:${opencensusVersion}", opencensus_contrib_grpc_metrics: "io.opencensus:opencensus-contrib-grpc-metrics:${opencensusVersion}", opencensus_impl: "io.opencensus:opencensus-impl:${opencensusVersion}", opencensus_impl_lite: "io.opencensus:opencensus-impl-lite:${opencensusVersion}", instrumentation_api: 'com.google.instrumentation:instrumentation-api:0.4.3', perfmark: 'io.perfmark:perfmark-api:0.19.0', pgv: 'io.envoyproxy.protoc-gen-validate:pgv-java-stub:0.2.0', protobuf: "com.google.protobuf:protobuf-java:${protobufVersion}", protobuf_lite: "com.google.protobuf:protobuf-javalite:${protobufVersion}", protobuf_util: "com.google.protobuf:protobuf-java-util:${protobufVersion}", lang: "org.apache.commons:commons-lang3:3.5", netty: "io.netty:netty-codec-http2:[${nettyVersion}]", netty_epoll: "io.netty:netty-transport-native-epoll:${nettyVersion}:linux-x86_64", netty_proxy_handler: "io.netty:netty-handler-proxy:${nettyVersion}", // Keep the following references of tcnative version in sync whenever it's updated // SECURITY.md (multiple occurrences) // examples/example-tls/build.gradle // examples/example-tls/pom.xml netty_tcnative: 'io.netty:netty-tcnative-boringssl-static:2.0.28.Final', conscrypt: 'org.conscrypt:conscrypt-openjdk-uber:2.2.1', re2j: 'com.google.re2j:re2j:1.2', // Test dependencies. junit: 'junit:junit:4.12', mockito: 'org.mockito:mockito-core:2.28.2', truth: 'com.google.truth:truth:1.0', guava_testlib: "com.google.guava:guava-testlib:${guavaVersion}", robolectric: "org.robolectric:robolectric:4.3.1", // Benchmark dependencies hdrhistogram: 'org.hdrhistogram:HdrHistogram:2.1.10', math: 'org.apache.commons:commons-math3:3.6', // Jetty ALPN dependencies jetty_alpn_agent: 'org.mortbay.jetty.alpn:jetty-alpn-agent:2.0.9' ] } // Define a separate configuration for managing the dependency on Jetty ALPN agent. configurations { compile { // Detect Maven Enforcer's dependencyConvergence failures. We only // care for artifacts used as libraries by others. if (!(project.name in [ 'grpc-benchmarks', 'grpc-interop-testing', 'grpc-gae-interop-testing-jdk8', ])) { resolutionStrategy.failOnVersionConflict() } } } // Disable JavaDoc doclint on Java 8. It's annoying. if (JavaVersion.current().isJava8Compatible()) { allprojects { tasks.withType(Javadoc) { options.addStringOption('Xdoclint:none', '-quiet') } } } // For jdk10 we must explicitly choose between html4 and html5, otherwise we get a warning if (JavaVersion.current().isJava10Compatible()) { allprojects { tasks.withType(Javadoc) { options.addBooleanOption('html4', true) } } } jacoco { toolVersion = "0.8.2" } checkstyle { configDir = file("$rootDir/buildscripts") toolVersion = "6.17" ignoreFailures = false if (rootProject.hasProperty("checkstyle.ignoreFailures")) { ignoreFailures = rootProject.properties["checkstyle.ignoreFailures"].toBoolean() } } if (rootProject.properties.get('errorProne', true)) { dependencies { errorprone 'com.google.errorprone:error_prone_core:2.3.4' errorproneJavac 'com.google.errorprone:javac:9+181-r4173-1' } } else { // Disable Error Prone allprojects { afterEvaluate { project -> project.tasks.withType(JavaCompile) { options.errorprone.enabled = false } } } } plugins.withId("java") { sourceCompatibility = 1.7 targetCompatibility = 1.7 dependencies { testCompile libraries.junit, libraries.mockito, libraries.truth } compileTestJava { // serialVersionUID is basically guaranteed to be useless in our tests options.compilerArgs += [ "-Xlint:-serial" ] } jar.manifest { attributes('Implementation-Title': name, 'Implementation-Version': version, 'Built-By': System.getProperty('user.name'), 'Built-JDK': System.getProperty('java.version'), 'Source-Compatibility': sourceCompatibility, 'Target-Compatibility': targetCompatibility) } javadoc.options { encoding = 'UTF-8' use = true links 'https://docs.oracle.com/javase/8/docs/api/' source = "8" } checkstyleMain { source = fileTree(dir: "$projectDir/src/main", include: "**/*.java") } checkstyleTest { source = fileTree(dir: "$projectDir/src/test", include: "**/*.java") } // At a test failure, log the stack trace to the console so that we don't // have to open the HTML in a browser. test { testLogging { exceptionFormat = 'full' showExceptions true showCauses true showStackTraces true } maxHeapSize = '1500m' } if (rootProject.properties.get('errorProne', true)) { dependencies { annotationProcessor 'com.google.guava:guava-beta-checker:1.0' } } compileJava { // This project targets Java 7 (no method references) options.errorprone.check("UnnecessaryAnonymousClass", CheckSeverity.OFF) // This project targets Java 7 (no time.Duration class) options.errorprone.check("PreferJavaTimeOverload", CheckSeverity.OFF) } compileTestJava { // LinkedList doesn't hurt much in tests and has lots of usages options.errorprone.check("JdkObsolete", CheckSeverity.OFF) options.errorprone.check("UnnecessaryAnonymousClass", CheckSeverity.OFF) options.errorprone.check("PreferJavaTimeOverload", CheckSeverity.OFF) } } plugins.withId("me.champeau.gradle.jmh") { dependencies { jmh 'org.openjdk.jmh:jmh-core:1.19', 'org.openjdk.jmh:jmh-generator-bytecode:1.19' } // invoke jmh on a single benchmark class like so: // ./gradlew -PjmhIncludeSingleClass=StatsTraceContextBenchmark clean :grpc-core:jmh jmh { warmupIterations = 10 iterations = 10 fork = 1 // None of our benchmarks need the tests, and we have pseudo-circular // dependencies that break when including them. (context's testCompile // depends on core; core's testCompile depends on testing) includeTests = false if (project.hasProperty('jmhIncludeSingleClass')) { include = [ project.property('jmhIncludeSingleClass') ] } } } plugins.withId("maven-publish") { publishing { publications { // do not use mavenJava, as java plugin will modify it via "magic" maven(MavenPublication) { pom { name = project.group + ":" + project.name url = 'https://github.com/grpc/grpc-java' afterEvaluate { // description is not available until evaluated. description = project.description } scm { connection = 'scm:git:https://github.com/grpc/grpc-java.git' developerConnection = 'scm:git:git@github.com:grpc/grpc-java.git' url = 'https://github.com/grpc/grpc-java' } licenses { license { name = 'Apache 2.0' url = 'https://opensource.org/licenses/Apache-2.0' } } developers { developer { id = "grpc.io" name = "gRPC Contributors" email = "grpc-io@googlegroups.com" url = "https://grpc.io/" organization = "gRPC Authors" organizationUrl = "https://www.google.com" } } withXml { if (!(project.name in [ "grpc-stub", "grpc-protobuf", "grpc-protobuf-lite", ])) { asNode().dependencies.'*'.findAll() { dep -> dep.artifactId.text() in ['grpc-api', 'grpc-core'] }.each() { core -> core.version*.value = "[" + core.version.text() + "]" } } } } } } repositories { maven { if (rootProject.hasProperty('repositoryDir')) { url = new File(rootProject.repositoryDir).toURI() } else { String stagingUrl if (rootProject.hasProperty('repositoryId')) { stagingUrl = 'https://oss.sonatype.org/service/local/staging/deployByRepositoryId/' + rootProject.repositoryId } else { stagingUrl = 'https://oss.sonatype.org/service/local/staging/deploy/maven2/' } credentials { if (rootProject.hasProperty('ossrhUsername') && rootProject.hasProperty('ossrhPassword')) { username = rootProject.ossrhUsername password = rootProject.ossrhPassword } } def releaseUrl = stagingUrl def snapshotUrl = 'https://oss.sonatype.org/content/repositories/snapshots/' url = version.endsWith('SNAPSHOT') ? snapshotUrl : releaseUrl } } } } signing { required false sign publishing.publications.maven } plugins.withId("java") { task javadocJar(type: Jar) { classifier = 'javadoc' from javadoc } task sourcesJar(type: Jar) { classifier = 'sources' from sourceSets.main.allSource } publishing { publications { maven { if (project.name != 'grpc-netty-shaded') { from components.java } artifact javadocJar artifact sourcesJar } } } } } // Run with: ./gradlew japicmp --continue plugins.withId("me.champeau.gradle.japicmp") { def baselineGrpcVersion = '1.6.1' // Get the baseline version's jar for this subproject File baselineArtifact = null // Use a detached configuration, otherwise the current version's jar will take precedence // over the baseline jar. // A necessary hack, the intuitive thing does NOT work: // https://discuss.gradle.org/t/is-the-default-configuration-leaking-into-independent-configurations/2088/6 def oldGroup = project.group try { project.group = 'virtual_group_for_japicmp' String depModule = "io.grpc:${project.name}:${baselineGrpcVersion}@jar" String depJar = "${project.name}-${baselineGrpcVersion}.jar" Configuration configuration = configurations.detachedConfiguration( dependencies.create(depModule) ) baselineArtifact = files(configuration.files).filter { it.name.equals(depJar) }.singleFile } finally { project.group = oldGroup } // Add a japicmp task that compares the current .jar with baseline .jar task japicmp(type: me.champeau.gradle.japicmp.JapicmpTask, dependsOn: jar) { oldClasspath = files(baselineArtifact) newClasspath = files(jar.archivePath) onlyBinaryIncompatibleModified = false // Be quiet about things that did not change onlyModified = true // This task should fail if there are incompatible changes failOnModification = true ignoreMissingClasses = true htmlOutputFile = file("$buildDir/reports/japi.html") packageExcludes = ['io.grpc.internal'] // Also break on source incompatible changes, not just binary. // Eg adding abstract method to public class. // TODO(zpencer): enable after japicmp-gradle-plugin/pull/14 // breakOnSourceIncompatibility = true // Ignore any classes or methods marked @ExperimentalApi // TODO(zpencer): enable after japicmp-gradle-plugin/pull/15 // annotationExcludes = ['@io.grpc.ExperimentalApi'] } } }