All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.oewntk.sql.out.SchemaGenerator.kt Maven / Gradle / Ivy

There is a newer version: 2.3.1
Show newest version
/*
 * Copyright (c) 2024. Bernard Bou.
 */

package org.oewntk.sql.out

import java.io.*
import java.net.URISyntaxException
import java.nio.file.Paths
import java.util.*
import java.util.function.BiConsumer
import java.util.jar.JarFile
import kotlin.system.exitProcess

/**
 * Main class that generates the SQL schema by instantiating templates
 *
 * @author Bernard Bou
 * @see "https://sqlunet.sourceforge.net/schema.html"
 */
class SchemaGenerator(
    private val variables: Variables,
) {

    /**
     * Generate schema
     *
     * @param module      module
     * @param output      output
     * @param inputSubdir input subdir (from sqltemplates/)
     * @param inputs      input files (or null if all)
     * @throws IOException io exception
     */
    @Throws(IOException::class)
    fun generate(
        module: String,
        output: String,
        inputSubdir: String,
        inputs: Array,
    ) {
        var outputFileOrDir: File? = null

        // Output
        if ("-" != output) {
            // output is not console
            if (output.endsWith(".sql")) {
                // output is plain sql file
                System.err.println("Output to file $output")
                outputFileOrDir = File(output)
                if (outputFileOrDir.exists()) {
                    System.err.println("Overwrite " + outputFileOrDir.absolutePath)
                    exitProcess(1)
                }
                outputFileOrDir.createNewFile()
            } else {
                // multiple outputs as per inputs
                outputFileOrDir = File(output)
                if (!outputFileOrDir.exists()) {
                    // System.err.println("Output to new dir " + arg1);
                    outputFileOrDir.mkdirs()
                }
                System.err.println(outputFileOrDir.absolutePath)
            }
        }

        // Input
        // Single output if console or file
        if (outputFileOrDir == null || outputFileOrDir.isFile) {
            if (outputFileOrDir == null) System.out else PrintStream(outputFileOrDir)
                .use { ps ->
                    processTemplates(module, inputSubdir, inputs) { inStream: InputStream, _ ->
                        try {
                            variables.varSubstitutionInIS(inStream, ps, useBackticks = true, compress = true)
                        } catch (e: IOException) {
                            e.printStackTrace()
                        }
                    }
                }
        } else if (outputFileOrDir.isDirectory) {
            val dir: File = outputFileOrDir
            processTemplates(module, inputSubdir, inputs) { inStream: InputStream, name: String ->
                System.err.println(name)
                val output2 = File(dir, name)
                try {
                    PrintStream(output2)
                        .use { ps ->
                            variables.varSubstitutionInIS(inStream, ps, useBackticks = true, compress = true)
                        }
                } catch (e: IOException) {
                    e.printStackTrace()
                }
            }
        } else {
            System.err.println("Internal error")
        }
    }

    /**
     * Process input template files
     *
     * @param module   module
     * @param path     path of inputs
     * @param inputs   inputs
     * @param consumer string consumer
     * @throws IOException io exception
     */
    @Throws(IOException::class)
    private fun processTemplates(
        module: String,
        path: String,
        inputs: Array,
        consumer: BiConsumer,
    ) {
        // external resources
        if (inputs.isNotEmpty()) {
            for (input in inputs) {
                val file = File(path, input)
                val fileName = Paths.get(input).fileName.toString()
                FileInputStream(file)
                    .use { fis ->
                        consumer.accept(fis, fileName)
                    }
            }
            return
        }

        // internal resources
        val jarFile = File(javaClass.protectionDomain.codeSource.location.path)
        if (jarFile.isFile) {
            // Run with JAR file
            val prefix = "$module/sqltemplates/$path/"
            JarFile(jarFile)
                .use { jar ->
                    val entries = jar.entries() //gives ALL entries in jar
                    while (entries.hasMoreElements()) {
                        val entry = entries.nextElement()
                        if (entry.isDirectory) {
                            continue
                        }

                        val name = entry.name
                        //filter according to the path
                        if (name.startsWith(prefix)) {
                            val fileName = Paths.get(name).fileName.toString()
                            jar.getInputStream(entry)
                                .use { inStream ->
                                    consumer.accept(inStream, fileName)
                                }
                        }
                    }
                }
        } else {
            // Run with IDE
            val url = SchemaGenerator::class.java.getResource("/$module/sqltemplates/$path")
            if (url != null) {
                try {
                    val dir = File(url.toURI())
                    val files = dir.listFiles()
                    if (files != null) {
                        for (file in files) {
                            System.err.println(file.name)
                            FileInputStream(file)
                                .use { fis ->
                                    consumer.accept(fis, file.name)
                                }
                        }
                    }
                } catch (ex: URISyntaxException) {
                    // never happens
                }
            }
        }
    }

    companion object {

        /**
         * Main entry point
         *
         * @param commandLineArgs command-line arguments
         * @throws IOException io exception
         */
        @Throws(IOException::class)
        @JvmStatic
        fun main(commandLineArgs: Array) {
            var args = commandLineArgs
            var compat = false
            if ("-compat" == args[0]) {
                compat = true
                args = args.copyOfRange(1, args.size)
            }

            val module = args[0]
            val output = args[1]
            val inputSubdir = args[2]
            val inputs = args.copyOfRange(3, args.size)

            val path = "$module/"
            val bundle = ResourceBundle.getBundle(path + (if (compat) "NamesCompat" else "Names"))
            val variables = Variables(bundle)
            SchemaGenerator(variables).generate(module, output, inputSubdir, inputs)
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy