import java.nio.charset.Charset
import java.util.function.Function

/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF 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.
 */

// This adds javacc generation support.

configure(rootProject) {
  configurations {
    javacc
  }

  dependencies {
    javacc "net.java.dev.javacc:javacc:${scriptDepVersions['javacc']}"
  }

  task javacc() {
    description "Regenerate sources for corresponding javacc grammar files."
    group "generation"

    dependsOn allprojects.collect { prj -> prj.tasks.withType(JavaCCTask) }
  }
}

def commonCleanups = { FileTree generatedFiles ->
  // This is a minor typo in a comment that nonetheless people have hand-corrected in the past.
  generatedFiles.matching({ include "CharStream.java" }).each {file ->
    modifyFile(file, { text ->
      return text.replace(
          "implemetation",
          "implementation");
    })
  }

  generatedFiles.each {file ->
    modifyFile(file, { text ->
      // Normalize EOLs and tabs (EOLs are a side-effect of modifyFile).
      text = text.replace("\t", "    ");
      text = text.replaceAll("JavaCC - OriginalChecksum=[^*]+", "(filtered)")
      text = text.replace("StringBuffer", "StringBuilder")
      return text
    })
  }

  generatedFiles.matching({ include "*TokenManager.java" }).each { file ->
    modifyFile(file, { text ->
      // Remove redundant imports.
      text = text.replaceAll(
          /(?m)^import .+/,
          "")
      // Add CharStream imports.
      text = text.replaceAll(
          /package (.+)/,
          '''
          package $1
          import org.apache.lucene.queryparser.charstream.CharStream; 
          '''.trim())
      // Eliminates redundant cast message.
      text = text.replace(
          "int hiByte = (int)(curChar >> 8);",
          "int hiByte = curChar >> 8;")
      // Access to forbidden APIs.
      text = text.replace(
          "public  java.io.PrintStream debugStream = System.out;",
          "// (debugStream omitted).")
      text = text.replace(
          "public  void setDebugStream(java.io.PrintStream ds) { debugStream = ds; }",
          "// (setDebugStream omitted).")
      text = text.replace(
          "public class QueryParserTokenManager ",
          '@SuppressWarnings("unused") public class QueryParserTokenManager ')
      text = text.replace(
          "public class StandardSyntaxParserTokenManager ",
          '@SuppressWarnings("unused") public class StandardSyntaxParserTokenManager ')
      return text
    })
  }
}

configure(project(":lucene:queryparser")) {
  task javaccParserClassicInternal(type: JavaCCTask) {
    description "Regenerate classic query parser from lucene/queryparser/classic/QueryParser.jj"
    group "generation"

    javaccFile = file('src/java/org/apache/lucene/queryparser/classic/QueryParser.jj')

    afterGenerate << commonCleanups
    afterGenerate << { FileTree generatedFiles ->
      generatedFiles.matching { include "QueryParser.java" }.each { file ->
        modifyFile(file, { text ->
          text = text.replace(
              "public QueryParser(CharStream ",
              "protected QueryParser(CharStream ")
          text = text.replace(
              "public QueryParser(QueryParserTokenManager ",
              "protected QueryParser(QueryParserTokenManager ")
          text = text.replace(
              "new java.util.ArrayList<int[]>",
              "new java.util.ArrayList<>")
          text = text.replace(
              "final private LookaheadSuccess jj_ls =",
              "static final private LookaheadSuccess jj_ls =")
          text = text.replace(
              "public class QueryParser ",
              '@SuppressWarnings({"unused","null"}) public class QueryParser ')
          text = text.replace(
              "final public Query TopLevelQuery(",
              "@Override final public Query TopLevelQuery(")
          text = text.replace(
              "public void ReInit(CharStream ",
              "@Override public void ReInit(CharStream ")
          return text
        })
      }
    }
  }

  task javaccParserSurroundInternal(type: JavaCCTask) {
    description "Regenerate surround query parser from lucene/queryparser/surround/parser/QueryParser.jj"
    group "generation"

    javaccFile = file('src/java/org/apache/lucene/queryparser/surround/parser/QueryParser.jj')

    afterGenerate << commonCleanups
    afterGenerate << { FileTree generatedFiles ->
      generatedFiles.matching { include "QueryParser.java" }.each { file ->
        modifyFile(file, { text ->
          text = text.replace(
              "import org.apache.lucene.analysis.TokenStream;",
              "")
          text = text.replace(
              "new java.util.ArrayList<int[]>",
              "new java.util.ArrayList<>")
          text = text.replace(
              "public class QueryParser ",
              '@SuppressWarnings({"unused","null"}) public class QueryParser ')
          return text
        })
      }
    }
  }

  task javaccParserFlexibleInternal(type: JavaCCTask) {
    description "Regenerate flexible query parser from queryparser/flexible/standard/parser/StandardSyntaxParser.jj"
    group "generation"

    javaccFile = file('src/java/org/apache/lucene/queryparser/flexible/standard/parser/StandardSyntaxParser.jj')

    afterGenerate << commonCleanups
    afterGenerate << { FileTree generatedFiles ->
      generatedFiles.matching { include "ParseException.java" }.each { file ->
        modifyFile(file, { text ->
          // Modify constructor.
          text = text.replace(
              "class ParseException extends Exception",
              "class ParseException extends QueryNodeParseException")

          // Modify imports.
          text = text.replace(
              "package org.apache.lucene.queryparser.flexible.standard.parser;", '''\
            package org.apache.lucene.queryparser.flexible.standard.parser;
  
            import org.apache.lucene.queryparser.flexible.messages.*;
            import org.apache.lucene.queryparser.flexible.core.*;
            import org.apache.lucene.queryparser.flexible.core.messages.*;
            ''')

          // Modify constructors and code bits
          text = text.replaceAll(
              /(?s)[ ]*public ParseException\(Token currentTokenVal[^}]+[}]/, '''\
            public ParseException(Token currentTokenVal,
              int[][] expectedTokenSequencesVal, String[] tokenImageVal) 
            {
              super(new MessageImpl(QueryParserMessages.INVALID_SYNTAX, initialise(
              currentTokenVal, expectedTokenSequencesVal, tokenImageVal)));
              this.currentToken = currentTokenVal;
              this.expectedTokenSequences = expectedTokenSequencesVal;
              this.tokenImage = tokenImageVal;
            }
            ''')

          text = text.replaceAll(
              /(?s)[ ]*public ParseException\(String message\)[^}]+[}]/, '''\
            public ParseException(Message message) 
            {
              super(message);
            }
            ''')

          text = text.replaceAll(
              /(?s)[ ]*public ParseException\(\)[^}]+[}]/, '''\
            public ParseException() 
            {
              super(new MessageImpl(QueryParserMessages.INVALID_SYNTAX, "Error"));
            }
            ''')
          return text
        })
      }

      generatedFiles.matching { include "StandardSyntaxParser.java" }.each { file ->
        modifyFile(file, { text ->
          // Remove redundant cast
          text = text.replace(
              "new java.util.ArrayList<int[]>",
              "new java.util.ArrayList<>")
          text = text.replace(
              "new ArrayList<QueryNode>()",
              "new ArrayList<>()")
          text = text.replace(
              "Collections.<QueryNode> singletonList",
              "Collections.singletonList")
          text = text.replace(
              "public class StandardSyntaxParser ",
              '@SuppressWarnings({"unused","null"}) public class StandardSyntaxParser ')
          return text
        })
      }
    }
  }

  task javacc() {
    description "Regenerate query parsers (javacc syntax definitions)."
    group "generation"

    dependsOn wrapWithPersistentChecksums(javaccParserClassicInternal, [ andThenTasks: ["spotlessJava", "spotlessJavaApply"] ]),
        wrapWithPersistentChecksums(javaccParserSurroundInternal, [ andThenTasks: ["spotlessJava", "spotlessJavaApply"] ]),
        wrapWithPersistentChecksums(javaccParserFlexibleInternal, [ andThenTasks: ["spotlessJava", "spotlessJavaApply"] ])
  }

  regenerate.dependsOn javacc
}

// We always regenerate, no need to declare outputs.
class JavaCCTask extends DefaultTask {
  @InputFile
  File javaccFile

  /**
   * Apply closures to all generated files before they're copied back
   * to mainline code.
   */
  // A subtle bug here is that this makes it not an input... should be a list of replacements instead?
  @Internal
  List<Closure<FileTree>> afterGenerate = new ArrayList<>()

  @OutputFiles
  List<File> getGeneratedSources() {
    // Return the list of generated files.
    def baseDir = javaccFile.parentFile
    def baseName = javaccFile.name.replace(".jj", "")

    return [
        project.file("${baseDir}/${baseName}.java"),
        project.file("${baseDir}/${baseName}Constants.java"),
        project.file("${baseDir}/${baseName}TokenManager.java"),
        project.file("${baseDir}/ParseException.java"),
        project.file("${baseDir}/Token.java"),
        project.file("${baseDir}/TokenMgrError.java")
    ]
  }

  JavaCCTask() {
    dependsOn(project.rootProject.configurations.javacc)
  }

  @TaskAction
  def generate() {
    if (!javaccFile || !javaccFile.exists()) {
      throw new GradleException("Input file does not exist: ${javaccFile}")
    }

    // Run javacc generation into temporary folder so that we know all the generated files
    // and can post-process them easily.
    def tempDir = this.getTemporaryDir()
    tempDir.mkdirs()
    project.delete project.fileTree(tempDir, { include: "**/*.java" })

    def targetDir = javaccFile.parentFile
    logger.lifecycle("Recompiling JavaCC: ${project.rootDir.relativePath(javaccFile)}")

    def output = new ByteArrayOutputStream()
    def result = project.javaexec {
      classpath {
        project.rootProject.configurations.javacc
      }

      ignoreExitValue = true
      standardOutput = output
      errorOutput = output

      main = "org.javacc.parser.Main"
      args += [
          "-OUTPUT_DIRECTORY=${tempDir}",
          javaccFile
      ]
    }

    // Unless we request verbose logging, don't emit javacc output.
    if (result.exitValue != 0) {
      throw new GradleException("JavaCC failed to compile ${javaccFile}, here is the compilation output:\n${output}")
    }

    // Make sure we don't have warnings.
    if (output.toString(Charset.defaultCharset()).contains("Warning:")) {
      throw new GradleException("JavaCC emitted warnings for ${javaccFile}, here is the compilation output:\n${output}")
    }

    // Apply any custom modifications.
    def generatedFiles = project.fileTree(tempDir)

    afterGenerate.each {closure ->
      closure.call(generatedFiles)
    }

    // Copy back to mainline sources.
    project.copy {
      from tempDir
      into targetDir

      // We don't need CharStream interface as we redirect to our own.
      exclude "CharStream.java"
    }
  }
}