SOLR-14889: improve templated variable escaping in ref-guide _config.yml

This commit is contained in:
Chris Hostetter 2020-09-29 11:09:15 -07:00
parent 8c7502dfeb
commit 52183dfbf6
2 changed files with 47 additions and 64 deletions

View File

@ -15,28 +15,6 @@
* limitations under the License. * limitations under the License.
*/ */
// TODO 1: the separation of sources between tools and refGuide is awkward; it'd be
// better to separate the refGuideTools as a plain Java module and then depend on
// it as a project dependency. This would enable this module to *not* be a java module at all
// and inherit from base, adding just refGuide-related tasks.
// OR (better) one could rewrite those tools in Groovy (or Kotlin) and use them directly, without
// an additional compilation phase.
// TODO 2: property expansion via ant properties is awkward in gradle. We can do cleaner than going
// through ant -- we could use gradle's expand when copying or at least use some more humane
// property names.
// TODO 3: currently buildscript dependencies are hardcoded (asciidoctor) because they can't be resolved
// using Palantir's plugin. This is another reason to switch to gradle-based tools -- then
// only the build script dependencies would be needed and asciidoctorj would be removed from version
// properties entirely (it is only used locally in this build file).
import java.time.*
import java.time.format.*
import java.nio.file.*
import org.asciidoctor.*
import groovy.json.StringEscapeUtils
buildscript { buildscript {
repositories { repositories {
mavenCentral() mavenCentral()
@ -50,7 +28,7 @@ buildscript {
plugins { plugins {
id 'java' id 'java'
id 'com.github.jruby-gradle.base' version '2.0.0-alpha.7' id 'com.github.jruby-gradle.base' version '2.0.0'
} }
// This project does not contribute anything to main dependencies. // This project does not contribute anything to main dependencies.
@ -146,13 +124,11 @@ ext {
htmlLuceneJavadocs = "https://lucene.apache.org/core/${solrGuideVersionPath}_0/" htmlLuceneJavadocs = "https://lucene.apache.org/core/${solrGuideVersionPath}_0/"
} }
// ivy* props will be set in setupLazyProps
// (because they need to be computed after evaluation is complete).
templateProps = [ templateProps = [
javadocLink: "https://docs.oracle.com/en/java/javase/11/docs/api/", javadocLink: "https://docs.oracle.com/en/java/javase/11/docs/api/",
solrGuideDraftStatus: propertyOrDefault("solrGuideDraft", "true").toBoolean() ? "DRAFT" : "", solrGuideDraftStatus: propertyOrDefault("solrGuideDraft", "true").toBoolean() ? "DRAFT" : "",
solrRootPath: StringEscapeUtils.escapeJava(project(':solr').projectDir.toString() + File.separator), solrRootPath: project(':solr').projectDir.toString() + File.separator,
solrDocsVersion: solrDocsVersion, solrDocsVersion: solrDocsVersion,
solrGuideVersionPath: solrGuideVersionPath, solrGuideVersionPath: solrGuideVersionPath,
@ -163,10 +139,19 @@ ext {
] ]
} }
task setupLazyProps { task prepareSources(type: Sync) {
dependsOn configurations.depVer
// If replaceable properties change, we have to rerun the task.
inputs.properties templateProps
final def escapedProps = [:] // must be final and only contents may change in doFirst, otherwise it's not picked up by expand.
doFirst { doFirst {
// Copy over template properties and add dependency versions resolved during execution phase.
final def props = templateProps.clone()
// These properties have to be resolved after the configuration phase is complete (palantir's constraint) // These properties have to be resolved after the configuration phase is complete (palantir's constraint)
// so we can't use them as input for caches. // so we can't use them as input for caches. But as this task depends on the configuration, it's used correctly
[ [
["ivyCommonsCodec", "commons-codec", "commons-codec"], ["ivyCommonsCodec", "commons-codec", "commons-codec"],
["ivyDropwizardMetrics", "io.dropwizard.metrics", "metrics-core"], ["ivyDropwizardMetrics", "io.dropwizard.metrics", "metrics-core"],
@ -175,19 +160,17 @@ task setupLazyProps {
["ivyTika", "org.apache.tika", "tika-core"], ["ivyTika", "org.apache.tika", "tika-core"],
["ivyZookeeper", "org.apache.zookeeper", "zookeeper"], ["ivyZookeeper", "org.apache.zookeeper", "zookeeper"],
].each { p, depGroup, depId -> ].each { p, depGroup, depId ->
templateProps[p] = getVersion(depGroup, depId, configurations.depVer) props[p] = getVersion(depGroup, depId, configurations.depVer)
} }
// Emit info about properties for clarity. // Emit info about properties for clarity.
logger.warn("Building ref guide with:\n" + templateProps.collect({ k, v -> " ${k} -> ${v}" }).join('\n')) logger.lifecycle('Building ref guide with:\n{}', props.collect({ k, v -> " ${k} -> ${v}" }).join('\n'))
// Escape all the properties, so they can be inserted into YAML templates.
props.each{ k, v ->
escapedProps[k] = v.replace("'","''")
}
} }
}
task prepareSources(type: Sync) {
dependsOn setupLazyProps
// If replaceable properties change, we have to rerun the task.
inputs.properties templateProps
from(file("src"), { from(file("src"), {
exclude '**/*.template' exclude '**/*.template'
@ -197,7 +180,7 @@ task prepareSources(type: Sync) {
include '**/*.template' include '**/*.template'
rename '(.+)\\.template', '$1' rename '(.+)\\.template', '$1'
filteringCharset = 'UTF-8' filteringCharset = 'UTF-8'
expand(templateProps) expand(escapedProps)
}) })
into buildContentDir into buildContentDir
@ -259,5 +242,5 @@ task buildSite(type: JavaExec) {
// Hook up custom tasks with standard tasks. // Hook up custom tasks with standard tasks.
check.dependsOn buildSite check.dependsOn buildSite
// Do not hook site building to assemble, at least for now. // Hook site building to assemble.
// assemble.dependsOn buildSite assemble.dependsOn buildSiteJekyll

View File

@ -7,7 +7,7 @@
# #
# Gems that are included for building the site. jekyll-asciidoc allows Jekyll to use Asciidoctor for variables and settings # Gems that are included for building the site. jekyll-asciidoc allows Jekyll to use Asciidoctor for variables and settings
gems: [jekyll-asciidoc] plugins: [jekyll-asciidoc]
destination: ../html-site destination: ../html-site
@ -68,21 +68,21 @@ asciidoc: {}
# NOTE: If you add any attributes here for use in adoc files, you almost certainly need to also add # NOTE: If you add any attributes here for use in adoc files, you almost certainly need to also add
# them to the <asciidoctor:convert/> ant task for precommit validation as well. # them to the <asciidoctor:convert/> ant task for precommit validation as well.
solr-attributes: &solr-attributes-ref solr-attributes: &solr-attributes-ref
solr-root-path: "${solrRootPath}" solr-root-path: '${solrRootPath}'
solr-guide-draft-status: "${solrGuideDraftStatus}" solr-guide-draft-status: '${solrGuideDraftStatus}'
solr-guide-version-path: "${solrGuideVersionPath}" solr-guide-version-path: '${solrGuideVersionPath}'
solr-docs-version: "${solrDocsVersion}" solr-docs-version: '${solrDocsVersion}'
java-javadocs: "${javadocLink}" java-javadocs: '${javadocLink}'
solr-javadocs: "${htmlSolrJavadocs}" solr-javadocs: '${htmlSolrJavadocs}'
lucene-javadocs: "${htmlLuceneJavadocs}" lucene-javadocs: '${htmlLuceneJavadocs}'
build-date: "${buildDate}" build-date: '${buildDate}'
build-year: "${buildYear}" build-year: '${buildYear}'
ivy-commons-codec-version: "${ivyCommonsCodec}" ivy-commons-codec-version: '${ivyCommonsCodec}'
ivy-dropwizard-version: "${ivyDropwizardMetrics}" ivy-dropwizard-version: '${ivyDropwizardMetrics}'
ivy-log4j-version: "${ivyLog4j}" ivy-log4j-version: '${ivyLog4j}'
ivy-opennlp-version: "${ivyOpennlpTools}" ivy-opennlp-version: '${ivyOpennlpTools}'
ivy-tika-version: "${ivyTika}" ivy-tika-version: '${ivyTika}'
ivy-zookeeper-version: "${ivyZookeeper}" ivy-zookeeper-version: '${ivyZookeeper}'
asciidoctor: asciidoctor:
safe: 0 safe: 0