b07ea3783d783d9501dbd997866bb4271ad1d5ed
[jalview.git] / build.gradle
1 import org.apache.tools.ant.filters.ReplaceTokens
2 //import org.apache.tools.ant.filters.ReplaceRegexp
3 import org.gradle.internal.os.OperatingSystem
4 import org.gradle.plugins.ide.eclipse.model.*
5
6
7 import groovy.transform.ExternalizeMethods
8
9 buildscript {
10   dependencies {
11     classpath 'org.openclover:clover:4.3.1'
12     classpath 'org.apache.commons:commons-compress:1.18'
13   }
14 }
15
16 plugins {
17   id 'java'
18   id 'application'
19   id 'eclipse'
20   id 'com.github.johnrengelman.shadow' version '4.0.3'
21   id 'com.install4j.gradle' version '7.0.9'
22 }
23
24 repositories {
25   jcenter()
26   mavenCentral()
27   mavenLocal()
28   flatDir {
29     dirs gradlePluginsDir
30   }
31 }
32
33 mainClassName = launcherClass
34 def cloverInstrDir = file("$buildDir/$cloverSourcesInstrDir")
35 def classes = "$jalviewDir/$classesDir"
36
37 if (clover.equals("true")) {
38   use_clover = true
39   classes = "$buildDir/$cloverClassesDir"
40 } else {
41   use_clover = false
42   classes = "$jalviewDir/$classesDir"
43 }
44
45 // configure classpath/args for j8/j11 compilation
46
47 def jalviewDirAbsolutePath = file(jalviewDir).getAbsolutePath()
48 def libDir
49 def libDistDir
50 def compile_source_compatibility
51 def compile_target_compatibility
52
53 ext {
54   // where the getdown channel will be built.
55   // TODO: consider allowing this expression to  be overridden by -P arg
56   getdownWebsiteDir = jalviewDir + '/' + getdown_website_dir + '/' + JAVA_VERSION
57   getdownAppDir = getdownWebsiteDir + '/' + getdown_app_dir
58   getdownJ11libDir = getdownWebsiteDir + '/' + getdown_j11lib_dir
59   getdownResourceDir = getdownWebsiteDir + '/' + getdown_resource_dir
60   getdownLauncher = jalviewDir + '/' + getdown_launcher
61   getdownFilesDir = jalviewDir + '/' + getdown_files_dir + '/' + JAVA_VERSION + '/'
62   getdown_app_base = getdown_channel_base+"/"+getdown_channel_name+"/"+JAVA_VERSION+"/"
63   modules_compileClasspath = fileTree(dir: "$jalviewDir/$j11modDir", include: ["*.jar"])
64   modules_runtimeClasspath = modules_compileClasspath
65   gitHash = ""
66   gitBranch = ""
67 }
68
69 def JAVA_INTEGER_VERSION
70 def additional_compiler_args = []
71 // these are getdown.txt properties defined dependent on the JAVA_VERSION
72 def getdown_alt_java_min_version
73 // this property is assigned below and expanded to multiple lines in the getdown task
74 def getdown_alt_multi_java_location
75 if (JAVA_VERSION.equals("1.8")) {
76   JAVA_INTEGER_VERSION = "8"
77   libDir = j11libDir
78   libDistDir = j8libDir
79   compile_source_compatibility = 1.8
80   compile_target_compatibility = 1.8
81   getdown_alt_java_min_version = getdown_alt_java8_min_version
82   getdown_alt_multi_java_location = getdown_alt_java8_txt_multi_java_location
83 } else if (JAVA_VERSION.equals("11")) {
84   JAVA_INTEGER_VERSION = "11"
85   libDir = j11libDir
86   libDistDir = j11libDir
87   compile_source_compatibility = 11
88   compile_target_compatibility = 11
89   getdown_alt_java_min_version = getdown_alt_java11_min_version
90   getdown_alt_multi_java_location = getdown_alt_java11_txt_multi_java_location
91   additional_compiler_args += [
92     '--module-path', ext.modules_compileClasspath.asPath,
93     '--add-modules', j11modules
94   ]
95 } else {
96   throw new GradleException("JAVA_VERSION=$JAVA_VERSION not currently supported by Jalview")
97 }
98
99 sourceSets {
100
101   main {
102     java {
103       srcDirs "$jalviewDir/$sourceDir"
104       outputDir = file("$classes")
105     }
106
107     resources {
108       srcDirs "$jalviewDir/$resourceDir"
109       srcDirs "$jalviewDir/$libDistDir"
110     }
111
112     jar.destinationDir = file("$jalviewDir/$packageDir")
113
114     compileClasspath = files(sourceSets.main.java.outputDir)
115     compileClasspath += fileTree(dir: "$jalviewDir/$libDir", include: ["*.jar"])
116
117     runtimeClasspath = compileClasspath
118   }
119   clover {
120     java {
121       srcDirs = [ cloverInstrDir ]
122       outputDir = file("${buildDir}/${cloverClassesDir}")
123     }
124
125     resources {
126       srcDirs = sourceSets.main.resources.srcDirs
127     }
128     compileClasspath = configurations.cloverRuntime + files( sourceSets.clover.java.outputDir )
129     compileClasspath += files(sourceSets.main.java.outputDir)
130     compileClasspath += sourceSets.main.compileClasspath
131     compileClasspath += fileTree(dir: "$jalviewDir/$utilsDir", include: ["**/*.jar"])
132     compileClasspath += fileTree(dir: "$jalviewDir/$libDir", include: ["*.jar"])
133
134     runtimeClasspath = compileClasspath
135   }
136
137   test {
138     java {
139       srcDirs "$jalviewDir/$testSourceDir"
140       outputDir = file("$jalviewDir/$testOutputDir")
141     }
142
143     resources {
144       srcDirs = sourceSets.main.resources.srcDirs
145     }
146
147     compileClasspath = files( sourceSets.test.java.outputDir )
148
149     if (use_clover) {
150       compileClasspath += sourceSets.clover.compileClasspath
151     } else {
152       compileClasspath += files(sourceSets.main.java.outputDir)
153     }
154     //compileClasspath += sourceSets.main.compileClasspath
155     //compileClasspath += files( sourceSets.main.resources.srcDirs)
156     compileClasspath += fileTree(dir: "$jalviewDir/$utilsDir", include: ["**/*.jar"])
157     compileClasspath += fileTree(dir: "$jalviewDir/$libDir", include: ["*.jar"])
158
159     runtimeClasspath = compileClasspath
160   }
161 }
162
163 // clover bits
164 dependencies {
165   if (use_clover) {
166     cloverCompile 'org.openclover:clover:4.3.1'
167     testCompile 'org.openclover:clover:4.3.1'
168   }
169 }
170
171 configurations {
172   cloverRuntime
173   cloverRuntime.extendsFrom cloverCompile
174 }
175
176 eclipse {
177   project {
178     name = "Jalview with gradle build"
179
180     natures 'org.eclipse.jdt.core.javanature',
181         'org.eclipse.jdt.groovy.core.groovyNature',
182         'org.eclipse.buildship.core.gradleprojectnature'
183
184     buildCommand 'org.eclipse.jdt.core.javabuilder'
185     buildCommand 'org.eclipse.buildship.core.gradleprojectbuilder'
186   }
187
188   classpath {
189     //defaultOutputDir = sourceSets.main.java.outputDir
190     def removeThese = []
191     configurations.each{ if (it.isCanBeResolved()) {
192         removeThese += it
193       }
194     }
195
196     minusConfigurations += removeThese
197     plusConfigurations = [ ]
198     file {
199
200       whenMerged { cp ->
201         def removeTheseToo = []
202         HashMap<String, Boolean> addedSrcPath = new HashMap<>();
203         cp.entries.each { entry ->
204           if (entry.kind == 'src') {
205             if (addedSrcPath.getAt(entry.path) || !(entry.path == "src" || entry.path == "test")) {
206               removeTheseToo += entry
207             } else {
208               addedSrcPath.putAt(entry.path, true)
209             }
210           }
211         }
212         cp.entries.removeAll(removeTheseToo)
213         
214         cp.entries += new Output("bin/main")
215         cp.entries += new Library(fileReference(helpParentDir))
216         cp.entries += new Library(fileReference(resourceDir))
217         
218         HashMap<String, Boolean> addedLibPath = new HashMap<>();
219         def allPaths = sourceSets.test.compileClasspath + sourceSets.main.compileClasspath
220         sourceSets.main.compileClasspath.each{
221           //if ((it.isDirectory() || ! it.exists()) && ! (it.equals(sourceSets.main.java.outputDir))) {
222           //no longer want to add outputDir as eclipse is using its own output dir in bin/main
223           if (it.isDirectory() || ! it.exists()) {
224             // don't add dirs to classpath
225             return
226           }
227           def itPath = it.toString()
228           if (itPath.startsWith(jalviewDirAbsolutePath+"/")) {
229             itPath = itPath.substring(jalviewDirAbsolutePath.length()+1)
230           }
231           if (addedLibPath.get(itPath)) {
232             //println("Not adding duplicate entry "+itPath)
233           } else {
234             //println("Adding entry "+itPath)
235             cp.entries += new Library(fileReference(itPath))
236             addedLibPath.put(itPath, true)
237           }
238         }
239
240         sourceSets.test.compileClasspath.each{
241           //if ((it.isDirectory() || ! it.exists()) && ! (it.equals(sourceSets.main.java.outputDir))) {
242           //no longer want to add outputDir as eclipse is using its own output dir in bin/main
243           if (it.isDirectory() || ! it.exists()) {
244             // don't add dirs to classpath
245             return false // groovy "break" in .each loop
246           }
247           def itPath = it.toString()
248           if (itPath.startsWith(jalviewDirAbsolutePath+"/")) {
249             itPath = itPath.substring(jalviewDirAbsolutePath.length()+1)
250           }
251           if (addedLibPath.get(itPath)) {
252             // don't duplicate
253           } else {
254             def lib = new Library(fileReference(itPath))
255             // this doesn't work... yet.  Adding test=true attribute using withXml below
256             //def attrs = new Node(null, 'attributes', ["test":"true"])
257             //lib.appendNode(attrs) //
258             cp.entries += lib
259             addedLibPath.put(itPath, true)
260           }
261         }
262       }  
263
264       // withXml changes ignored by buildship, these add the "test=true" attribute
265       withXml {
266         def node = it.asNode()
267         
268         def srcTestAttributes
269         node.children().each{ cpe ->
270           def attributes = cpe.attributes()
271           if (attributes.get("kind") == "src" && attributes.get("path") == "test") {
272             srcTestAttributes = cpe.find { a -> a.name() == "attributes" }
273             return
274           }
275         }
276         def addTestAttribute = true
277         srcTestAttributes.each{a ->
278           if (a.name() == "attribute" && a.attributes().getAt("name") == "test") {
279             addTestAttribute = false
280           }
281         }
282         if (addTestAttribute) {
283           srcTestAttributes.append(new Node(null, "attribute", [name:"test", value:"true"]))
284         }
285
286         node.children().each{ cpe ->
287           def attributes = cpe.attributes()
288           if (attributes.get("kind") == "lib" && attributes.get("path").startsWith("utils/")) {
289             cpe.appendNode('attributes')
290                 .appendNode('attribute', [name:"test", value:"true"])
291           }
292         }
293       } // withXML
294     } // file
295
296     containers 'org.eclipse.buildship.core.gradleclasspathcontainer'
297   } // classpath
298
299   jdt {
300     // for the IDE, use java 11 compatibility
301     sourceCompatibility = 11
302     targetCompatibility = 11
303     javaRuntimeName = "JavaSE-11"
304
305     file {
306       withProperties { props ->
307         def jalview_prefs = new Properties()
308         def ins = new FileInputStream(jalviewDirAbsolutePath+"/"+eclipse_extra_jdt_prefs_file)
309         jalview_prefs.load(ins)
310         ins.close()
311         jalview_prefs.forEach { t, v ->
312           if (props.getAt(t) == null) {
313             props.putAt(t, v)
314           }
315         }
316       }
317     }
318   }
319   
320   synchronizationTasks eclipseClasspath
321   autoBuildTasks eclipseClasspath
322
323
324 task cloverInstr() {
325   // only instrument source, we build test classes as normal
326   inputs.files files (sourceSets.main.allJava) // , fileTree(dir:"$jalviewDir/$testSourceDir", include: ["**/*.java"]))
327   outputs.dir cloverInstrDir
328
329   doFirst {
330     delete cloverInstrDir
331     def argsList = ["--initstring", "${buildDir}/clover/clover.db",
332       "-d", "${buildDir}/${cloverSourcesInstrDir}"]
333     argsList.addAll(inputs.files.files.collect({ file ->
334       file.absolutePath
335     }))
336     String[] args = argsList.toArray()
337     println("About to instrument "+args.length +" files")
338     com.atlassian.clover.CloverInstr.mainImpl(args)
339   }
340 }
341   
342
343 task cloverReport {
344   group = "Verification"
345   description = "Createst the Clover report"
346   inputs.dir "${buildDir}/clover"
347   outputs.dir "${reportsDir}/clover"
348   onlyIf {
349     file("${buildDir}/clover/clover.db").exists()
350   }
351   doFirst {
352     def argsList = ["--initstring", "${buildDir}/clover/clover.db",
353       "-o", "${reportsDir}/clover"]
354     String[] args = argsList.toArray()
355     com.atlassian.clover.reporters.html.HtmlReporter.runReport(args)
356
357     // and generate ${reportsDir}/clover/clover.xml
358     args = ["--initstring", "${buildDir}/clover/clover.db",
359       "-o", "${reportsDir}/clover/clover.xml"].toArray()
360     com.atlassian.clover.reporters.xml.XMLReporter.runReport(args)
361   }
362 }
363
364 // end clover bits
365
366
367 compileJava {
368
369   doFirst {
370     sourceCompatibility = compile_source_compatibility
371     targetCompatibility = compile_target_compatibility
372     options.compilerArgs = additional_compiler_args
373     print ("Setting target compatibility to "+targetCompatibility+"\n")
374   }
375
376 }
377
378 compileTestJava {
379   if (use_clover) {
380     dependsOn compileCloverJava
381     classpath += configurations.cloverRuntime
382   } else {
383     classpath += sourceSets.main.runtimeClasspath
384   }
385   doFirst {
386     sourceCompatibility = compile_source_compatibility
387     targetCompatibility = compile_target_compatibility
388     options.compilerArgs = additional_compiler_args
389     print ("Setting target compatibility to "+targetCompatibility+"\n")
390   }
391 }
392
393
394 compileCloverJava {
395
396   doFirst {
397     sourceCompatibility = compile_source_compatibility
398     targetCompatibility = compile_target_compatibility
399     options.compilerArgs += additional_compiler_args
400     print ("Setting target compatibility to "+targetCompatibility+"\n")
401   }
402   classpath += configurations.cloverRuntime
403 }
404
405 clean {
406   delete sourceSets.main.java.outputDir
407 }
408
409 cleanTest {
410   delete sourceSets.test.java.outputDir
411   delete cloverInstrDir
412 }
413
414 def getDate(format) {
415   def date = new Date()
416   //return date.format("dd MMMM yyyy")
417   return date.format(format)
418 }
419
420 task setGitHash(type: Exec) {
421   workingDir = jalviewDir
422   commandLine "git", "rev-parse", "--short", "HEAD"
423   standardOutput = new ByteArrayOutputStream()
424   project.ext.gitHash = {
425     return standardOutput.toString().trim()
426   }
427 }
428
429 task setGitBranch(type: Exec) {
430   workingDir = jalviewDir
431   commandLine "git", "rev-parse", "--abbrev-ref", "HEAD"
432   standardOutput = new ByteArrayOutputStream()
433   project.ext.gitBranch = {
434     return standardOutput.toString().trim()
435   }
436 }
437
438 task setGitVals {
439   dependsOn setGitHash
440   dependsOn setGitBranch
441 }
442
443 task createBuildProperties(type: WriteProperties) {
444   dependsOn setGitVals
445   inputs.dir("$jalviewDir/$sourceDir")
446   inputs.dir("$jalviewDir/$resourceDir")
447   outputFile "$classes/$buildPropertiesFile"
448   // taking time specific comment out to allow better incremental builds
449   //comment "--Jalview Build Details--\n"+getDate("yyyy-MM-dd HH:mm:ss")
450   comment "--Jalview Build Details--\n"+getDate("yyyy-MM-dd")
451   property "BUILD_DATE", getDate("dd MMMM yyyy")
452   property "VERSION", JALVIEW_VERSION
453   property "INSTALLATION", INSTALLATION+" git-commit:"+project.ext.gitHash+" ["+project.ext.gitBranch+"]"
454   outputs.file(outputFile)
455   outputs.dir("$classes")
456 }
457
458 task syncDocs(type: Sync) {
459   def syncDir = "$classes/$docDir"
460   from fileTree("$jalviewDir/$docDir")
461   into syncDir
462
463 }
464
465 def helpFile = "$classes/$helpDir/help.jhm"
466
467 task copyHelp(type: Copy) {
468   def inputDir = "$jalviewDir/$helpParentDir/$helpDir"
469   def outputDir = "$classes/$helpDir"
470   from(inputDir) {
471     exclude '**/*.gif'
472     exclude '**/*.jpg'
473     exclude '**/*.png'
474     filter(ReplaceTokens, beginToken: '$$', endToken: '$$', tokens: ['Version-Rel': JALVIEW_VERSION])
475   }
476   from(inputDir) {
477     include '**/*.gif'
478     include '**/*.jpg'
479     include '**/*.png'
480   }
481   into outputDir
482
483   inputs.dir(inputDir)
484   outputs.files(helpFile)
485   outputs.dir(outputDir)
486 }
487
488 task syncLib(type: Sync) {
489   def syncDir = "$classes/$libDistDir"
490   from fileTree("$jalviewDir/$libDistDir")
491   into syncDir
492 }
493
494 task syncResources(type: Sync) {
495   from "$jalviewDir/$resourceDir"
496   include "**/*.*"
497   exclude "install4j"
498   into "$classes"
499   preserve {
500     include "**"
501   }
502 }
503
504 task prepare {
505   dependsOn syncResources
506   dependsOn syncDocs
507   dependsOn copyHelp
508 }
509
510
511 //testReportDirName = "test-reports" // note that test workingDir will be $jalviewDir
512 test {
513   dependsOn prepare
514   dependsOn compileJava
515   if (use_clover) {
516     dependsOn cloverInstr
517   }
518
519   print("Running tests " + (use_clover?"WITH":"WITHOUT") + " clover [clover="+use_clover+"]\n")
520
521   useTestNG() {
522     includeGroups testngGroups
523     preserveOrder true
524     useDefaultListeners=true
525   }
526
527   workingDir = jalviewDir
528   //systemProperties 'clover.jar' System.properties.clover.jar
529   sourceCompatibility = compile_source_compatibility
530   targetCompatibility = compile_target_compatibility
531   jvmArgs += additional_compiler_args
532   print ("Setting target compatibility to "+targetCompatibility+"\n")
533 }
534
535 task buildIndices(type: JavaExec) {
536   dependsOn copyHelp
537   classpath = sourceSets.main.compileClasspath
538   main = "com.sun.java.help.search.Indexer"
539   workingDir = "$classes/$helpDir"
540   def argDir = "html"
541   args = [ argDir ]
542   inputs.dir("$workingDir/$argDir")
543
544   outputs.dir("$classes/doc")
545   outputs.dir("$classes/help")
546   outputs.file("$workingDir/JavaHelpSearch/DOCS")
547   outputs.file("$workingDir/JavaHelpSearch/DOCS.TAB")
548   outputs.file("$workingDir/JavaHelpSearch/OFFSETS")
549   outputs.file("$workingDir/JavaHelpSearch/POSITIONS")
550   outputs.file("$workingDir/JavaHelpSearch/SCHEMA")
551   outputs.file("$workingDir/JavaHelpSearch/TMAP")
552 }
553
554 task compileLinkCheck(type: JavaCompile) {
555   options.fork = true
556   classpath = files("$jalviewDir/$utilsDir")
557   destinationDir = file("$jalviewDir/$utilsDir")
558   source = fileTree(dir: "$jalviewDir/$utilsDir", include: ["HelpLinksChecker.java", "BufferedLineReader.java"])
559
560   inputs.file("$jalviewDir/$utilsDir/HelpLinksChecker.java")
561   inputs.file("$jalviewDir/$utilsDir/HelpLinksChecker.java")
562   outputs.file("$jalviewDir/$utilsDir/HelpLinksChecker.class")
563   outputs.file("$jalviewDir/$utilsDir/BufferedLineReader.class")
564 }
565
566 def helplinkscheckeroutputfile = file("$jalviewDir/$utilsDir/HelpLinksChecker.out")
567 task linkCheck(type: JavaExec) {
568   dependsOn prepare, compileLinkCheck
569   classpath = files("$jalviewDir/$utilsDir")
570   main = "HelpLinksChecker"
571   workingDir = jalviewDir
572   def help = "$classes/$helpDir"
573   args = [ "$classes/$helpDir", "-nointernet" ]
574
575   doFirst {
576     helplinkscheckeroutputfile.createNewFile()
577     standardOutput new FileOutputStream(helplinkscheckeroutputfile, false)
578   }
579
580   outputs.file(helplinkscheckeroutputfile)
581 }
582
583 task cleanPackageDir(type: Delete) {
584   delete fileTree("$jalviewDir/$packageDir").include("*.jar")
585 }
586
587 jar {
588   dependsOn linkCheck
589   dependsOn buildIndices
590   dependsOn createBuildProperties
591
592   manifest {
593     attributes "Main-Class": mainClass,
594     "Permissions": "all-permissions",
595     "Application-Name": "Jalview Desktop",
596     "Codebase": application_codebase
597   }
598
599   destinationDir = file("$jalviewDir/$packageDir")
600   archiveName = rootProject.name+".jar"
601
602   exclude "cache*/**"
603   exclude "*.jar"
604   exclude "*.jar.*"
605   exclude "**/*.jar"
606   exclude "**/*.jar.*"
607
608   inputs.dir("$classes")
609   outputs.file("$jalviewDir/$packageDir/$archiveName")
610 }
611
612 task copyJars(type: Copy) {
613   from fileTree("$classes").include("**/*.jar").include("*.jar").files
614   into "$jalviewDir/$packageDir"
615 }
616
617 // doing a Sync instead of Copy as Copy doesn't deal with "outputs" very well
618 task syncJars(type: Sync) {
619   from fileTree("$jalviewDir/$libDistDir").include("**/*.jar").include("*.jar").files
620   into "$jalviewDir/$packageDir"
621   preserve {
622     include jar.archiveName
623   }
624 }
625
626 task makeDist {
627   group = "build"
628   description = "Put all required libraries in dist"
629   // order of "cleanPackageDir", "copyJars", "jar" important!
630   jar.mustRunAfter cleanPackageDir
631   syncJars.mustRunAfter cleanPackageDir
632   dependsOn cleanPackageDir
633   dependsOn syncJars
634   dependsOn jar
635   outputs.dir("$jalviewDir/$packageDir")
636 }
637
638 task cleanDist {
639   dependsOn cleanPackageDir
640   dependsOn cleanTest
641   dependsOn clean
642 }
643
644 shadowJar {
645   dependsOn makeDist
646   from ("$jalviewDir/$libDistDir") {
647     include("*.jar")
648   }
649   mainClassName = shadowJarMainClass
650   mergeServiceFiles()
651   classifier = "all-"+JAVA_VERSION
652   minimize()
653 }
654
655 task getdownWebsite() {
656   group = "distribution"
657   description = "Create the getdown minimal app folder, and website folder for this version of jalview. Website folder also used for offline app installer"
658   dependsOn makeDist
659   def getdownWebsiteResourceFilenames = []
660   def getdownTextString = ""
661   def getdownResourceDir = project.ext.getdownResourceDir
662   def getdownAppDir = project.ext.getdownAppDir
663   def getdownResourceFilenames = []
664   doFirst {
665     // go through properties looking for getdown_txt_...
666     def props = project.properties.sort { it.key }
667     props.put("getdown_txt_java_min_version", getdown_alt_java_min_version)
668     props.put("getdown_txt_multi_java_location", getdown_alt_multi_java_location)
669
670     if (getdown_local == "true") {
671       getdown_app_base = file(getdownWebsiteDir).toURI().toString()
672     }
673     props.put("getdown_txt_appbase", getdown_app_base)
674     props.each{ prop, val ->
675       if (prop.startsWith("getdown_txt_") && val != null) {
676         if (prop.startsWith("getdown_txt_multi_")) {
677           def key = prop.substring(18)
678           val.split(",").each{ v ->
679             def line = key + " = " + v + "\n"
680             getdownTextString += line
681           }
682         } else {
683           // file values rationalised
684           if (val.indexOf('/') > -1) {
685             def r = null
686             if (val.indexOf('/') == 0) {
687               // absolute path
688               r = file(val)
689             } else if (val.indexOf('/') > 0) {
690               // relative path (relative to jalviewDir)
691               r = file( jalviewDir + '/' + val )
692             }
693             if (r.exists()) {
694               val = getdown_resource_dir + '/' + r.getName()
695               getdownWebsiteResourceFilenames += val
696               getdownResourceFilenames += r.getPath()
697             }
698           }
699           def line = prop.substring(12) + " = " + val + "\n"
700           getdownTextString += line
701         }
702       }
703     }
704
705     getdownWebsiteResourceFilenames.each{ filename ->
706       getdownTextString += "resource = "+filename+"\n"
707     }
708     getdownResourceFilenames.each{ filename ->
709       copy {
710         from filename
711         into project.ext.getdownResourceDir
712       }
713     }
714
715     def codeFiles = []
716     makeDist.outputs.files.each{ f ->
717       if (f.isDirectory()) {
718         def files = fileTree(dir: f, include: ["*"]).getFiles()
719         codeFiles += files
720       } else if (f.exists()) {
721         codeFiles += f
722       }
723     }
724     codeFiles.sort().each{f ->
725       def line = "code = " + getdown_app_dir + '/' + f.getName() + "\n"
726       getdownTextString += line
727       copy {
728         from f.getPath()
729         into project.ext.getdownAppDir
730       }
731     }
732
733     // NOT USING MODULES YET, EVERYTHING SHOULD BE IN dist
734     /*
735      if (JAVA_VERSION.equals("11")) {
736      def j11libFiles = fileTree(dir: "$jalviewDir/$j11libDir", include: ["*.jar"]).getFiles()
737      j11libFiles.sort().each{f ->
738      def line = "code = " + getdown_j11lib_dir + '/' + f.getName() + "\n"
739      getdownTextString += line
740      copy {
741      from f.getPath()
742      into project.ext.getdownJ11libDir
743      }
744      }
745      }
746      */
747
748     // getdown-launcher.jar should not be in main application class path so the main application can move it when updated.  Listed as a resource so it gets updated.
749     //getdownTextString += "class = " + file(getdownLauncher).getName() + "\n"
750     getdownTextString += "resource = " + file(getdownLauncher).getName() + "\n"
751     getdownTextString += "class = " + mainClass + "\n"
752
753     def getdown_txt = file(project.ext.getdownWebsiteDir + "/getdown.txt")
754     getdown_txt.write(getdownTextString)
755
756     copy {
757       from getdown_txt
758       into project.ext.getdownFilesDir
759     }
760
761     copy {
762       from getdownLauncher
763       into project.ext.getdownFilesDir
764     }
765
766     copy {
767       from getdownLauncher
768       into project.ext.getdownWebsiteDir
769     }
770
771     copy {
772       from jalviewDir + '/' + project.getProperty('getdown_txt_ui.background_image')
773       from jalviewDir + '/' + project.getProperty('getdown_txt_ui.error_background')
774       from jalviewDir + '/' + project.getProperty('getdown_txt_ui.progress_image')
775       from jalviewDir + '/' + project.getProperty('getdown_txt_ui.icon')
776       from jalviewDir + '/' + project.getProperty('getdown_txt_ui.mac_dock_icon')
777       into project.ext.getdownFilesDir + '/' + getdown_resource_dir
778     }
779   }
780
781   inputs.dir(jalviewDir + '/' + packageDir)
782   outputs.dir(project.ext.getdownWebsiteDir)
783   outputs.dir(project.ext.getdownFilesDir)
784 }
785
786 task getdownDigest(type: JavaExec) {
787   group = "distribution"
788   description = "Digest the getdown website folder"
789   dependsOn getdownWebsite
790   classpath = files(jalviewDir + '/' + getdown_core, jalviewDir+'/'+getdown_launcher)
791   main = "com.threerings.getdown.tools.Digester"
792   args project.ext.getdownWebsiteDir
793   inputs.dir(project.ext.getdownWebsiteDir)
794   outputs.file(project.ext.getdownWebsiteDir + '/' + "digest2.txt")
795 }
796
797 task getdown() {
798   group = "distribution"
799   description = "Create the minimal and full getdown app folder for installers and website and create digest file"
800   dependsOn getdownDigest
801 }
802
803 clean {
804   delete project.ext.getdownWebsiteDir
805   delete project.ext.getdownFilesDir
806 }
807
808 install4j {
809   def install4jHomeDir = "/opt/install4j"
810   def hostname = "hostname".execute().text.trim()
811   if (hostname.equals("jv-bamboo")) {
812     install4jHomeDir = System.getProperty("user.home")+"/buildtools/install4j"
813   } else if (OperatingSystem.current().isMacOsX()) {
814     install4jHomeDir = '/Applications/install4j.app/Contents/Resources/app'
815     if (! file(install4jHomeDir).exists()) {
816       install4jHomeDir = System.getProperty("user.home")+install4jHomeDir
817     }
818   } else if (OperatingSystem.current().isLinux()) {
819     install4jHomeDir = System.getProperty("user.home")+"/buildtools/install4j"
820   }
821   installDir = file(install4jHomeDir)
822   mediaTypes = Arrays.asList(install4jMediaTypes.split(","))
823   if (install4jFaster.equals("true")) {
824     faster = true
825   }
826 }
827
828 def install4jConf
829 def macosJavaVMDir
830 def macosJavaVMTgz
831 def windowsJavaVMDir
832 def windowsJavaVMTgz
833 def install4jDir = "$jalviewDir/$install4jResourceDir"
834 def install4jConfFile = "jalview-installers-java"+JAVA_VERSION+".install4j"
835 install4jConf = "$install4jDir/$install4jConfFile"
836
837 task copyInstall4jTemplate(type: Copy) {
838   macosJavaVMDir = System.env.HOME+"/buildtools/jre/openjdk-java_vm/getdown/macos-jre"+JAVA_VERSION+"/jre"
839   macosJavaVMTgz = System.env.HOME+"/buildtools/jre/openjdk-java_vm/install4j/tgz/macos-jre"+JAVA_VERSION+".tar.gz"
840   windowsJavaVMDir = System.env.HOME+"/buildtools/jre/openjdk-java_vm/getdown/windows-jre"+JAVA_VERSION+"/jre"
841   windowsJavaVMTgz = System.env.HOME+"/buildtools/jre/openjdk-java_vm/install4j/tgz/windows-jre"+JAVA_VERSION+".tar.gz"
842   from (install4jDir) {
843     include install4jTemplate
844     rename (install4jTemplate, install4jConfFile)
845     filter(ReplaceTokens, beginToken: '', endToken: '', tokens: ['9999999999': JAVA_VERSION])
846     filter(ReplaceTokens, beginToken: '$$', endToken: '$$',
847       tokens: [
848         'JAVA_VERSION': JAVA_VERSION,
849         'JAVA_INTEGER_VERSION': JAVA_INTEGER_VERSION,
850         'VERSION': JALVIEW_VERSION,
851         'MACOS_JAVA_VM_DIR': macosJavaVMDir,
852         'MACOS_JAVA_VM_TGZ': macosJavaVMTgz,
853         'WINDOWS_JAVA_VM_DIR': windowsJavaVMDir,
854         'WINDOWS_JAVA_VM_TGZ': windowsJavaVMTgz,
855         'INSTALL4JINFOPLISTFILEASSOCIATIONS': install4jInfoPlistFileAssociations,
856         'COPYRIGHT_MESSAGE': install4jCopyrightMessage,
857         'MACOS_BUNDLE_ID': install4jMacOSBundleId
858       ]
859     )
860     if (OSX_KEYPASS=="") {
861       filter(ReplaceTokens, beginToken: 'codeSigning macEnabled="', endToken: '"', tokens: ['true':'codeSigning macEnabled="false"'])
862       filter(ReplaceTokens, beginToken: 'runPostProcessor="true" ',endToken: 'Processor', tokens: ['post':'runPostProcessor="false" postProcessor'])
863     }
864   }
865   into install4jDir
866   outputs.files(install4jConf)
867
868   doLast {
869     // include file associations in installer
870     def installerFileAssociationsXml = file("$install4jDir/$install4jInstallerFileAssociations").text
871     ant.replaceregexp(
872       byline: false,
873       flags: "s",
874       match: '<action name="EXTENSIONS_REPLACED_BY_GRADLE".*?</action>',
875       replace: installerFileAssociationsXml,
876       file: install4jConf
877     )
878     /*
879     // include uninstaller applescript app files in dmg
880     def installerDMGUninstallerXml = file("$install4jDir/$install4jDMGUninstallerAppFiles").text
881     ant.replaceregexp(
882       byline: false,
883       flags: "s",
884       match: '<file name="UNINSTALL_OLD_JALVIEW_APP_REPLACED_IN_GRADLE" file=.*?>',
885       replace: installerDMGUninstallerXml,
886       file: install4jConf
887     )
888     */
889   }
890 }
891
892 task installers(type: com.install4j.gradle.Install4jTask) {
893   group = "distribution"
894   description = "Create the install4j installers"
895   dependsOn getdown
896   dependsOn copyInstall4jTemplate
897   projectFile = file(install4jConf)
898   println("Using projectFile "+projectFile)
899   variables = [majorVersion: version.substring(2, 11), build: 001, OSX_KEYSTORE: OSX_KEYSTORE, JSIGN_SH: JSIGN_SH]
900   destination = "$jalviewDir/$install4jBuildDir/$JAVA_VERSION"
901   buildSelected = true
902
903   if (OSX_KEYPASS) {
904     macKeystorePassword=OSX_KEYPASS
905     
906   }
907   
908   inputs.dir(project.ext.getdownWebsiteDir)
909   inputs.file(install4jConf)
910   inputs.dir(macosJavaVMDir)
911   inputs.dir(windowsJavaVMDir)
912   outputs.dir("$jalviewDir/$install4jBuildDir/$JAVA_VERSION")
913 }
914
915 clean {
916   delete install4jConf
917 }