Gradle WAR plugin: как найти место назначения для копирования файлов



Я работал над тем, чтобы Gradle генерировал определение артефакта войны IntelliJ как часть задачи "идея" согласно этому сообщению в блоге. До сих пор это работало довольно успешно, хотя мой код довольно грязный, поэтому я пытаюсь его очистить.
Насколько я понимаю, военная задача должна содержать всю информацию, необходимую для создания необходимых тегов в IntelliJ. Например, учитывая объект Gradle Project, скажем gradleProject, я могу сделать что-то вроде:



def artifactManager = project.component.find { it.@name == 'ArtifactManager' } as Node
if (artifactManager) {
Node artifact = artifactManager.artifact.find { it.@type == 'exploded-war' }
if (artifact) {
artifactManager.remove(artifact)
}
} else {
artifactManager = project.appendNode('component', [name: 'ArtifactManager']);
}
def builder = new NodeBuilder();
def artifact = builder.artifact(type: 'exploded-war', name: "${gradleProject.name} exploded war") {
'output-path'("${gradleProject.buildDir}/${gradleProject.name}-exploded.war")
root(id: 'root') {
element(id: 'directory', name: 'WEB-INF') {
// copy web.xml
element(id: 'file-copy', path: gradleProject.war.webXml)
Set<String> excludeJars = [] as Set
element(id: 'directory', name: 'classes') {
// copy classes
element(id: 'module-output', name: gradleProject.name)
// copy depending projects module output (classes)
gradleProject.configurations.runtime.allDependencies.each {
if (it.hasProperty('dependencyProject')) { // TODO: make it functional?!?
if (it.dependencyProject.hasProperty('jar'))
excludeJars.add(it.dependencyProject.jar.archiveName)
element(id: 'module-output', name: it.dependencyProject.name)
}
}
}
// copy dependency jars excluding possible existing module jars (not sure it's necessary)
element(id: 'directory', name: 'lib') {
// TODO: this should handle ALL kinds of dependencies
gradleProject.configurations.runtime.each {
if (!excludeJars.contains(it.name))
element(id: 'file-copy', path: it)
}
}
}
}
}
artifactManager.append artifact


Чтобы изменить или создать свой интеллект .файл ipr.
Теперь я добавил некоторые файлы к моему артефакту войны в скрипте Gradle, как в:



war{
from project(':rest').file('resource/some.properties')
from project(':some_project').fileTree('some_folder')
metaInf {
from parent.file('build/config/website/' + environment + '/context.xml')
}
}


И я хотел бы добавить немного Заводного кода, чтобы забрать эти файлы и создать правильные XML-теги, чтобы IntelliJ скопировал их в нужных местах.
Какой самый простой способ сделать это? Я попробовал играть со следующим:



gradleProject.war.source.each { // ... create tags etc.  }


Но список исходных файлов включает в себя все файлы, независимо от их назначения (все библиотеки, web.в XML, классы и т. д.).

Как могу ли я достичь того, что пытаюсь сделать?

741   1  

1 ответ:

Я, наконец, понял это с помощью Gradle docs и немного проб и ошибок. Я получил желаемый результат, используя следующий код:

def artifactManager = project.component.find { it.@name == 'ArtifactManager' } as Node
if (artifactManager) {
  Node artifact = artifactManager.artifact.find { it.@type == 'exploded-war' }
  if (artifact) {
    artifactManager.remove(artifact)
  }
} else {
  artifactManager = project.appendNode('component', [name: 'ArtifactManager']);
}
def builder = new NodeBuilder();
def artifact = builder.artifact(type: 'exploded-war', name: "${gradleProject.name} exploded war") {
  'output-path'("${gradleProject.buildDir}/${gradleProject.name}-exploded.war")
  root(id: 'root') {
    println "Adding files to root"
    gradleProject.war.rootSpec.children.each {
      if (it.destPath.pathString.equals("")) {
        add(builder, it, "")
      }
    }
    println "Adding files to META-INF"
    element(id: 'directory', name: 'META-INF') {
      gradleProject.war.rootSpec.children.each {
        if (it.destPath.pathString.equals("META-INF")) {
          add(builder, it, "META-INF")
        }
      }
    }
    println "Adding files to WEB-INF"
    element(id: 'directory', name: 'WEB-INF') {
      element(id: 'file-copy', path: gradleProject.war.webXml)
      gradleProject.war.rootSpec.children.each {
        if (it.destPath.pathString.equals("WEB-INF")) {
          add(builder, it, "WEB-INF")
        }
      }
      Set<String> excludeJars = [] as Set
      element(id: 'directory', name: 'classes') {
        element(id: 'module-output', name: gradleProject.name)
        gradleProject.configurations.runtime.allDependencies.each {
          if (it.hasProperty('dependencyProject')) { 
            if (it.dependencyProject.hasProperty('jar'))
              excludeJars.add(it.dependencyProject.jar.archiveName)
            element(id: 'module-output', name: it.dependencyProject.name)
          }
        }
      }
      element(id: 'directory', name: 'lib') {
        gradleProject.configurations.runtime.each {
          if (!excludeJars.contains(it.name))
            element(id: 'file-copy', path: it)
        }
      }
    }
  }
}
artifactManager.append artifact

С функцией add, определенной как:

def add(NodeBuilder builder, Object s, String dest, String prefix = "") {
  println(prefix + "current destination: [" + dest + "]")
  if (s instanceof File) {
    println(prefix + "-- file " + s.toString())
    builder.element(id: 'file-copy', path: s)
  } else if (s instanceof FileTree) {
    println(prefix + "-- folder " + s.dir.toString())
    builder.element(id: 'dir-copy', path: s.dir)
  } else if (s instanceof CopySpecInternal) {
    print(prefix + "-- copy spec " + s.toString() + " with destination path: [" + s.destPath.pathString + "]; ")
    if (dest.equals(s.destPath.pathString)) {
      println("destination matches current folder")
      s.sourcePaths.each { path ->
          add(builder, path, dest, prefix + "  ")
      }
      s.children.each { child -> add(builder, child, dest, prefix + "  ") }
    } else {
      println("")
      if (s.destPath.segments != null && s.destPath.segments.length > 0 && s.destPath.segments[0].equals(dest)) {
        String relDest = new RelativePath(false, Arrays.copyOfRange(s.destPath.segments, 1, s.destPath.segments.length)).pathString
        builder.element(id: 'directory', name: relDest) {
          s.sourcePaths.each { path ->
            add(builder, path, relDest, prefix + "  ")
          }
          s.children.each { child -> add(builder, child, relDest, prefix + "  ") }
        }
      } else {
        println(prefix + "!! destination path is not relative to [" + dest + "]: [" + s.destPath.pathString + "]")
      }
    }
  } else println(prefix + "?? unknown " + s.toString())
}

Я в основном реверс-проектировал свойства задачи war и копался в дереве файлов для копирования. Насколько я могу судить, эта задача не предоставляет интерфейс для выполнения подобных вещей, поэтому мне пришлось прибегнуть к "халтурному" решению, которое не является моим любимым. Я был бы счастлив, если бы кто-нибудь мог чип с каким-то более чистым кодом.

Comments

    Ничего не найдено.