2015-10-29 14:40:19 -04:00
|
|
|
/*
|
|
|
|
* Licensed to Elasticsearch under one or more contributor
|
|
|
|
* license agreements. See the NOTICE file distributed with
|
|
|
|
* this work for additional information regarding copyright
|
|
|
|
* ownership. Elasticsearch licenses this file to you 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.
|
|
|
|
*/
|
|
|
|
|
2017-01-17 16:38:32 -05:00
|
|
|
import java.nio.file.Path
|
2017-03-29 15:11:38 -04:00
|
|
|
import java.util.regex.Matcher
|
2016-04-15 11:13:12 -04:00
|
|
|
import org.eclipse.jgit.lib.Repository
|
|
|
|
import org.eclipse.jgit.lib.RepositoryBuilder
|
2015-10-31 02:39:35 -04:00
|
|
|
import org.gradle.plugins.ide.eclipse.model.SourceFolder
|
2016-04-28 10:02:11 -04:00
|
|
|
import org.apache.tools.ant.taskdefs.condition.Os
|
2017-03-29 15:11:38 -04:00
|
|
|
import org.elasticsearch.gradle.VersionProperties
|
2017-05-18 04:14:24 -04:00
|
|
|
import org.elasticsearch.gradle.Version
|
2015-10-29 14:40:19 -04:00
|
|
|
|
|
|
|
// common maven publishing configuration
|
|
|
|
subprojects {
|
2016-03-17 15:21:31 -04:00
|
|
|
group = 'org.elasticsearch'
|
2017-03-29 15:11:38 -04:00
|
|
|
version = VersionProperties.elasticsearch
|
2016-05-13 19:32:35 -04:00
|
|
|
description = "Elasticsearch subproject ${project.path}"
|
2016-11-24 03:03:43 -05:00
|
|
|
}
|
|
|
|
|
2017-01-17 16:38:32 -05:00
|
|
|
Path rootPath = rootDir.toPath()
|
2016-11-24 03:03:43 -05:00
|
|
|
// setup pom license info, but only for artifacts that are part of elasticsearch
|
2017-01-17 16:38:32 -05:00
|
|
|
configure(subprojects.findAll { it.projectDir.toPath().startsWith(rootPath) }) {
|
2016-03-17 15:21:31 -04:00
|
|
|
|
2016-05-05 20:53:01 -04:00
|
|
|
// we only use maven publish to add tasks for pom generation
|
|
|
|
plugins.withType(MavenPublishPlugin).whenPluginAdded {
|
|
|
|
publishing {
|
|
|
|
publications {
|
|
|
|
// add license information to generated poms
|
|
|
|
all {
|
|
|
|
pom.withXml { XmlProvider xml ->
|
|
|
|
Node node = xml.asNode()
|
|
|
|
node.appendNode('inceptionYear', '2009')
|
|
|
|
|
|
|
|
Node license = node.appendNode('licenses').appendNode('license')
|
|
|
|
license.appendNode('name', 'The Apache Software License, Version 2.0')
|
|
|
|
license.appendNode('url', 'http://www.apache.org/licenses/LICENSE-2.0.txt')
|
|
|
|
license.appendNode('distribution', 'repo')
|
2016-05-13 16:57:08 -04:00
|
|
|
|
|
|
|
Node developer = node.appendNode('developers').appendNode('developer')
|
|
|
|
developer.appendNode('name', 'Elastic')
|
|
|
|
developer.appendNode('url', 'http://www.elastic.co')
|
2016-05-05 20:53:01 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-10-29 14:40:19 -04:00
|
|
|
}
|
|
|
|
|
2017-05-17 15:58:37 -04:00
|
|
|
// introspect all versions of ES that may be tested agains for backwards compatibility
|
2017-05-18 04:14:24 -04:00
|
|
|
Version currentVersion = Version.fromString(VersionProperties.elasticsearch.minus('-SNAPSHOT'))
|
|
|
|
int prevMajor = currentVersion.major - 1
|
2017-03-29 15:11:38 -04:00
|
|
|
File versionFile = file('core/src/main/java/org/elasticsearch/Version.java')
|
|
|
|
List<String> versionLines = versionFile.readLines('UTF-8')
|
2017-05-18 04:14:24 -04:00
|
|
|
List<Version> versions = []
|
2017-05-17 15:58:37 -04:00
|
|
|
// keep track of the previous major version's last minor, so we know where wire compat begins
|
|
|
|
int prevMinorIndex = -1 // index in the versions list of the last minor from the prev major
|
|
|
|
int lastPrevMinor = -1 // the minor version number from the prev major we most recently seen
|
2017-03-29 15:11:38 -04:00
|
|
|
for (String line : versionLines) {
|
2017-05-17 15:58:37 -04:00
|
|
|
Matcher match = line =~ /\W+public static final Version V_(\d+)_(\d+)_(\d+)(_UNRELEASED)? .*/
|
2017-03-29 15:11:38 -04:00
|
|
|
if (match.matches()) {
|
2017-05-17 15:58:37 -04:00
|
|
|
int major = Integer.parseInt(match.group(1))
|
|
|
|
int minor = Integer.parseInt(match.group(2))
|
|
|
|
int bugfix = Integer.parseInt(match.group(3))
|
2017-05-18 04:14:24 -04:00
|
|
|
Version foundVersion = new Version(major, minor, bugfix, false)
|
|
|
|
if (currentVersion != foundVersion) {
|
|
|
|
versions.add(foundVersion)
|
2017-05-17 15:58:37 -04:00
|
|
|
}
|
|
|
|
if (major == prevMajor && minor > lastPrevMinor) {
|
|
|
|
prevMinorIndex = versions.size() - 1
|
|
|
|
lastPrevMinor = minor
|
|
|
|
}
|
2017-03-29 15:11:38 -04:00
|
|
|
}
|
|
|
|
}
|
2017-05-18 04:14:24 -04:00
|
|
|
if (versions.toSorted { it.id } != versions) {
|
|
|
|
println "Versions: ${versions}"
|
|
|
|
throw new GradleException("Versions.java contains out of order version constants")
|
2017-05-17 15:58:37 -04:00
|
|
|
}
|
2017-05-18 04:14:24 -04:00
|
|
|
if (currentVersion.bugfix == 0) {
|
2017-05-17 15:58:37 -04:00
|
|
|
// If on a release branch, after the initial release of that branch, the bugfix version will
|
|
|
|
// be bumped, and will be != 0. On master and N.x branches, we want to test against the
|
|
|
|
// unreleased version of closest branch. So for those cases, the version includes -SNAPSHOT,
|
2017-05-18 04:14:24 -04:00
|
|
|
// and the bwc-zip distribution will checkout and build that version.
|
|
|
|
Version last = versions[-1]
|
|
|
|
versions[-1] = new Version(last.major, last.minor, last.bugfix, true)
|
2017-05-17 15:58:37 -04:00
|
|
|
}
|
2017-03-29 15:11:38 -04:00
|
|
|
|
|
|
|
// injecting groovy property variables into all projects
|
2015-10-29 14:40:19 -04:00
|
|
|
allprojects {
|
|
|
|
project.ext {
|
2016-02-04 18:44:30 -05:00
|
|
|
// for ide hacks...
|
2015-11-05 01:28:57 -05:00
|
|
|
isEclipse = System.getProperty("eclipse.launcher") != null || gradle.startParameter.taskNames.contains('eclipse') || gradle.startParameter.taskNames.contains('cleanEclipse')
|
2016-02-04 18:44:30 -05:00
|
|
|
isIdea = System.getProperty("idea.active") != null || gradle.startParameter.taskNames.contains('idea') || gradle.startParameter.taskNames.contains('cleanIdea')
|
2017-03-29 15:11:38 -04:00
|
|
|
// for backcompat testing
|
2017-05-17 15:58:37 -04:00
|
|
|
indexCompatVersions = versions
|
|
|
|
wireCompatVersions = versions.subList(prevMinorIndex, versions.size())
|
2015-10-29 14:40:19 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
subprojects {
|
2015-11-15 04:43:13 -05:00
|
|
|
project.afterEvaluate {
|
|
|
|
// include license and notice in jars
|
2015-10-29 14:40:19 -04:00
|
|
|
tasks.withType(Jar) {
|
|
|
|
into('META-INF') {
|
|
|
|
from project.rootProject.rootDir
|
2015-11-25 19:21:38 -05:00
|
|
|
include 'LICENSE.txt'
|
|
|
|
include 'NOTICE.txt'
|
2015-10-29 14:40:19 -04:00
|
|
|
}
|
|
|
|
}
|
2015-11-15 04:43:13 -05:00
|
|
|
// ignore missing javadocs
|
|
|
|
tasks.withType(Javadoc) { Javadoc javadoc ->
|
2015-11-19 14:06:47 -05:00
|
|
|
// the -quiet here is because of a bug in gradle, in that adding a string option
|
|
|
|
// by itself is not added to the options. By adding quiet, both this option and
|
|
|
|
// the "value" -quiet is added, separated by a space. This is ok since the javadoc
|
|
|
|
// command already adds -quiet, so we are just duplicating it
|
2015-11-20 15:18:30 -05:00
|
|
|
// see https://discuss.gradle.org/t/add-custom-javadoc-option-that-does-not-take-an-argument/5959
|
2015-11-24 18:04:40 -05:00
|
|
|
javadoc.options.encoding='UTF8'
|
2015-11-15 04:43:13 -05:00
|
|
|
javadoc.options.addStringOption('Xdoclint:all,-missing', '-quiet')
|
2016-05-23 15:28:54 -04:00
|
|
|
/*
|
|
|
|
TODO: building javadocs with java 9 b118 is currently broken with weird errors, so
|
|
|
|
for now this is commented out...try again with the next ea build...
|
|
|
|
javadoc.executable = new File(project.javaHome, 'bin/javadoc')
|
|
|
|
if (project.javaVersion == JavaVersion.VERSION_1_9) {
|
|
|
|
// TODO: remove this hack! gradle should be passing this...
|
|
|
|
javadoc.options.addStringOption('source', '8')
|
|
|
|
}*/
|
2015-11-15 04:43:13 -05:00
|
|
|
}
|
2015-10-29 14:40:19 -04:00
|
|
|
}
|
|
|
|
|
2015-11-25 19:21:38 -05:00
|
|
|
/* Sets up the dependencies that we build as part of this project but
|
|
|
|
register as thought they were external to resolve internally. We register
|
|
|
|
them as external dependencies so the build plugin that we use can be used
|
|
|
|
to build elasticsearch plugins outside of the elasticsearch source tree. */
|
2015-11-22 04:12:17 -05:00
|
|
|
ext.projectSubstitutions = [
|
2016-06-03 12:48:45 -04:00
|
|
|
"org.elasticsearch.gradle:build-tools:${version}": ':build-tools',
|
2015-11-20 14:58:50 -05:00
|
|
|
"org.elasticsearch:rest-api-spec:${version}": ':rest-api-spec',
|
|
|
|
"org.elasticsearch:elasticsearch:${version}": ':core',
|
2016-06-22 13:40:01 -04:00
|
|
|
"org.elasticsearch.client:rest:${version}": ':client:rest',
|
|
|
|
"org.elasticsearch.client:sniffer:${version}": ':client:sniffer',
|
|
|
|
"org.elasticsearch.client:test:${version}": ':client:test',
|
2016-07-26 11:54:46 -04:00
|
|
|
"org.elasticsearch.client:transport:${version}": ':client:transport',
|
2015-12-17 19:57:39 -05:00
|
|
|
"org.elasticsearch.test:framework:${version}": ':test:framework',
|
2015-12-03 17:52:51 -05:00
|
|
|
"org.elasticsearch.distribution.integ-test-zip:elasticsearch:${version}": ':distribution:integ-test-zip',
|
2015-11-20 14:58:50 -05:00
|
|
|
"org.elasticsearch.distribution.zip:elasticsearch:${version}": ':distribution:zip',
|
2015-11-25 19:21:38 -05:00
|
|
|
"org.elasticsearch.distribution.tar:elasticsearch:${version}": ':distribution:tar',
|
|
|
|
"org.elasticsearch.distribution.rpm:elasticsearch:${version}": ':distribution:rpm',
|
|
|
|
"org.elasticsearch.distribution.deb:elasticsearch:${version}": ':distribution:deb',
|
2016-02-16 18:32:21 -05:00
|
|
|
"org.elasticsearch.test:logger-usage:${version}": ':test:logger-usage',
|
2016-07-15 17:34:21 -04:00
|
|
|
// for transport client
|
2016-07-22 22:26:35 -04:00
|
|
|
"org.elasticsearch.plugin:transport-netty4-client:${version}": ':modules:transport-netty4',
|
2016-07-15 17:34:21 -04:00
|
|
|
"org.elasticsearch.plugin:reindex-client:${version}": ':modules:reindex',
|
|
|
|
"org.elasticsearch.plugin:lang-mustache-client:${version}": ':modules:lang-mustache',
|
2017-05-12 09:58:06 -04:00
|
|
|
"org.elasticsearch.plugin:parent-join-client:${version}": ':modules:parent-join',
|
2016-07-15 17:34:21 -04:00
|
|
|
"org.elasticsearch.plugin:percolator-client:${version}": ':modules:percolator',
|
2015-11-22 04:12:17 -05:00
|
|
|
]
|
2017-05-18 04:14:24 -04:00
|
|
|
if (wireCompatVersions[-1].snapshot) {
|
|
|
|
// if the most previous version is a snapshot, we need to connect that version to the
|
|
|
|
// bwc-zip project which will checkout and build that snapshot version
|
|
|
|
ext.projectSubstitutions["org.elasticsearch.distribution.zip:elasticsearch:${wireCompatVersions[-1]}"] = ':distribution:bwc-zip'
|
|
|
|
}
|
2017-03-24 01:32:13 -04:00
|
|
|
project.afterEvaluate {
|
|
|
|
configurations.all {
|
|
|
|
resolutionStrategy.dependencySubstitution { DependencySubstitutions subs ->
|
|
|
|
projectSubstitutions.each { k,v ->
|
|
|
|
subs.substitute(subs.module(k)).with(subs.project(v))
|
|
|
|
}
|
2015-11-22 04:12:17 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure similar tasks in dependent projects run first. The projectsEvaluated here is
|
|
|
|
// important because, while dependencies.all will pickup future dependencies,
|
|
|
|
// it is not necessarily true that the task exists in both projects at the time
|
|
|
|
// the dependency is added.
|
|
|
|
gradle.projectsEvaluated {
|
|
|
|
allprojects {
|
2015-12-17 19:57:39 -05:00
|
|
|
if (project.path == ':test:framework') {
|
|
|
|
// :test:framework:test cannot run before and after :core:test
|
2015-11-22 04:12:17 -05:00
|
|
|
return
|
|
|
|
}
|
|
|
|
configurations.all {
|
|
|
|
dependencies.all { Dependency dep ->
|
|
|
|
Project upstreamProject = null
|
|
|
|
if (dep instanceof ProjectDependency) {
|
|
|
|
upstreamProject = dep.dependencyProject
|
|
|
|
} else {
|
|
|
|
// gradle doesn't apply substitutions until resolve time, so they won't
|
|
|
|
// show up as a ProjectDependency above
|
|
|
|
String substitution = projectSubstitutions.get("${dep.group}:${dep.name}:${dep.version}")
|
|
|
|
if (substitution != null) {
|
|
|
|
upstreamProject = findProject(substitution)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (upstreamProject != null) {
|
|
|
|
if (project.path == upstreamProject.path) {
|
|
|
|
// TODO: distribution integ tests depend on themselves (!), fix that
|
|
|
|
return
|
|
|
|
}
|
|
|
|
for (String taskName : ['test', 'integTest']) {
|
|
|
|
Task task = project.tasks.findByName(taskName)
|
|
|
|
Task upstreamTask = upstreamProject.tasks.findByName(taskName)
|
|
|
|
if (task != null && upstreamTask != null) {
|
|
|
|
task.mustRunAfter(upstreamTask)
|
|
|
|
}
|
|
|
|
}
|
2015-10-29 14:40:19 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-10-31 00:32:16 -04:00
|
|
|
// intellij configuration
|
2015-10-29 14:40:19 -04:00
|
|
|
allprojects {
|
|
|
|
apply plugin: 'idea'
|
2015-12-15 11:32:21 -05:00
|
|
|
|
2016-02-04 18:44:30 -05:00
|
|
|
if (isIdea) {
|
|
|
|
project.buildDir = file('build-idea')
|
|
|
|
}
|
2015-12-15 11:32:21 -05:00
|
|
|
idea {
|
|
|
|
module {
|
|
|
|
inheritOutputDirs = false
|
2016-02-04 18:44:30 -05:00
|
|
|
outputDir = file('build-idea/classes/main')
|
|
|
|
testOutputDir = file('build-idea/classes/test')
|
2015-12-15 11:32:21 -05:00
|
|
|
|
2016-04-14 17:54:52 -04:00
|
|
|
// also ignore other possible build dirs
|
|
|
|
excludeDirs += file('build')
|
|
|
|
excludeDirs += file('build-eclipse')
|
|
|
|
|
2015-12-15 11:32:21 -05:00
|
|
|
iml {
|
|
|
|
// fix so that Gradle idea plugin properly generates support for resource folders
|
|
|
|
// see also https://issues.gradle.org/browse/GRADLE-2975
|
|
|
|
withXml {
|
|
|
|
it.asNode().component.content.sourceFolder.findAll { it.@url == 'file://$MODULE_DIR$/src/main/resources' }.each {
|
|
|
|
it.attributes().remove('isTestSource')
|
|
|
|
it.attributes().put('type', 'java-resource')
|
|
|
|
}
|
|
|
|
it.asNode().component.content.sourceFolder.findAll { it.@url == 'file://$MODULE_DIR$/src/test/resources' }.each {
|
|
|
|
it.attributes().remove('isTestSource')
|
|
|
|
it.attributes().put('type', 'java-test-resource')
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-12-22 12:24:13 -05:00
|
|
|
|
|
|
|
task cleanIdeaBuildDir(type: Delete) {
|
|
|
|
delete 'build-idea'
|
|
|
|
}
|
|
|
|
cleanIdeaBuildDir.setGroup("ide")
|
|
|
|
cleanIdeaBuildDir.setDescription("Deletes the IDEA build directory.")
|
|
|
|
|
|
|
|
tasks.cleanIdea.dependsOn(cleanIdeaBuildDir)
|
2015-10-31 00:32:16 -04:00
|
|
|
}
|
|
|
|
|
2015-11-20 14:58:50 -05:00
|
|
|
idea {
|
|
|
|
project {
|
|
|
|
vcs = 'Git'
|
2015-10-31 00:32:16 -04:00
|
|
|
}
|
|
|
|
}
|
2015-11-22 19:07:39 -05:00
|
|
|
// Make sure gradle idea was run before running anything in intellij (including import).
|
|
|
|
File ideaMarker = new File(projectDir, '.local-idea-is-configured')
|
|
|
|
tasks.idea.doLast {
|
|
|
|
ideaMarker.setText('', 'UTF-8')
|
|
|
|
}
|
|
|
|
if (System.getProperty('idea.active') != null && ideaMarker.exists() == false) {
|
|
|
|
throw new GradleException('You must run gradle idea from the root of elasticsearch before importing into IntelliJ')
|
|
|
|
}
|
2015-10-31 00:32:16 -04:00
|
|
|
|
2015-10-31 02:39:35 -04:00
|
|
|
// eclipse configuration
|
2015-10-31 00:32:16 -04:00
|
|
|
allprojects {
|
2015-10-29 14:40:19 -04:00
|
|
|
apply plugin: 'eclipse'
|
2016-01-28 17:40:29 -05:00
|
|
|
// Name all the non-root projects after their path so that paths get grouped together when imported into eclipse.
|
|
|
|
if (path != ':') {
|
|
|
|
eclipse.project.name = path
|
2016-04-28 10:08:21 -04:00
|
|
|
if (Os.isFamily(Os.FAMILY_WINDOWS)) {
|
2016-04-28 10:02:11 -04:00
|
|
|
eclipse.project.name = eclipse.project.name.replace(':', '_')
|
|
|
|
}
|
2016-01-28 17:40:29 -05:00
|
|
|
}
|
2015-10-29 14:40:19 -04:00
|
|
|
|
2015-10-31 02:39:35 -04:00
|
|
|
plugins.withType(JavaBasePlugin) {
|
2016-02-04 18:44:30 -05:00
|
|
|
File eclipseBuild = project.file('build-eclipse')
|
|
|
|
eclipse.classpath.defaultOutputDir = eclipseBuild
|
|
|
|
if (isEclipse) {
|
|
|
|
// set this so generated dirs will be relative to eclipse build
|
|
|
|
project.buildDir = eclipseBuild
|
|
|
|
}
|
2015-10-31 02:39:35 -04:00
|
|
|
eclipse.classpath.file.whenMerged { classpath ->
|
|
|
|
// give each source folder a unique corresponding output folder
|
|
|
|
int i = 0;
|
|
|
|
classpath.entries.findAll { it instanceof SourceFolder }.each { folder ->
|
|
|
|
i++;
|
|
|
|
// this is *NOT* a path or a file.
|
2016-02-04 18:44:30 -05:00
|
|
|
folder.output = "build-eclipse/" + i
|
2015-10-31 02:39:35 -04:00
|
|
|
}
|
2015-10-29 14:40:19 -04:00
|
|
|
}
|
|
|
|
}
|
2015-11-02 10:44:51 -05:00
|
|
|
task copyEclipseSettings(type: Copy) {
|
|
|
|
// TODO: "package this up" for external builds
|
|
|
|
from new File(project.rootDir, 'buildSrc/src/main/resources/eclipse.settings')
|
|
|
|
into '.settings'
|
|
|
|
}
|
2015-11-02 09:39:14 -05:00
|
|
|
// otherwise .settings is not nuked entirely
|
2016-05-31 17:20:21 -04:00
|
|
|
task wipeEclipseSettings(type: Delete) {
|
2015-11-11 22:26:55 -05:00
|
|
|
delete '.settings'
|
|
|
|
}
|
2016-05-31 17:20:21 -04:00
|
|
|
tasks.cleanEclipse.dependsOn(wipeEclipseSettings)
|
2015-10-30 23:00:05 -04:00
|
|
|
// otherwise the eclipse merging is *super confusing*
|
2015-11-11 22:26:55 -05:00
|
|
|
tasks.eclipse.dependsOn(cleanEclipse, copyEclipseSettings)
|
2015-10-29 14:40:19 -04:00
|
|
|
}
|
|
|
|
|
2015-11-17 18:37:26 -05:00
|
|
|
// we need to add the same --debug-jvm option as
|
|
|
|
// the real RunTask has, so we can pass it through
|
|
|
|
class Run extends DefaultTask {
|
|
|
|
boolean debug = false
|
|
|
|
|
|
|
|
@org.gradle.api.internal.tasks.options.Option(
|
|
|
|
option = "debug-jvm",
|
|
|
|
description = "Enable debugging configuration, to allow attaching a debugger to elasticsearch."
|
|
|
|
)
|
|
|
|
public void setDebug(boolean enabled) {
|
|
|
|
project.project(':distribution').run.clusterConfig.debug = enabled
|
2015-11-25 19:21:38 -05:00
|
|
|
}
|
2015-11-17 18:37:26 -05:00
|
|
|
}
|
|
|
|
task run(type: Run) {
|
2015-11-16 19:02:44 -05:00
|
|
|
dependsOn ':distribution:run'
|
|
|
|
description = 'Runs elasticsearch in the foreground'
|
|
|
|
group = 'Verification'
|
2015-11-19 01:22:47 -05:00
|
|
|
impliesSubProjects = true
|
2015-11-16 19:02:44 -05:00
|
|
|
}
|