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

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

/*
 * Copyright (c) 2024. Bernard Bou.
 */
package org.oewntk.sql.out

import org.oewntk.model.Key
import org.oewntk.model.Lex
import java.io.PrintStream

/**
 * Insert printers
 */
object Printers {

    const val WITH_COMMENT: Boolean = true

    // from maps

    /**
     * Print inserts for objects mapped by nids
     *
     * @param ps          print stream
     * @param table       table name
     * @param columns     column names
     * @param objectToNID objects-to-nid map
     * @param toRow       stringifier of objects
     * @param T           type of objects
     */
    fun  printInsert(
        ps: PrintStream,
        table: String,
        columns: String,
        objectToNID: Map,
        toRow: (T) -> String,
    ) {
        if (objectToNID.isEmpty()) {
            ps.print("-- NONE")
        } else {
            ps.print("INSERT INTO $table ($columns) VALUES")
            objectToNID.keys
                .withIndex()
                .forEach { (index, key) ->
                    val row = toRow.invoke(key)
                    if (index > 0) {
                        ps.print(',')
                    }
                    val nid = NIDMaps.lookup(objectToNID, key)
                    ps.print("\n($nid,$row)")
                }
            ps.println(";")
        }
    }

    /**
     * Print inserts for collection
     *
     * @param ps            print stream
     * @param table         table name
     * @param columns       column names
     * @param objects       collection of objects
     * @param toId          id extractor
     * @param objectIdToNID id-to-nid map
     * @param toRow         stringifier of objects
     * @param T             type of objects
     */
    fun  printInsert(
        ps: PrintStream,
        table: String,
        columns: String,
        objects: Collection,
        toId: (T) -> String,
        objectIdToNID: Map,
        toRow: (T) -> String,
    ) {
        if (objects.isEmpty()) {
            ps.print("-- NONE")
        } else {
            ps.print("INSERT INTO $table ($columns) VALUES")
            objects
                .asSequence()
                .map { it to NIDMaps.lookup(objectIdToNID, toId.invoke(it)) }
                .toList()
                .sortedBy { it.second }
                .withIndex()
                .forEach { (index, valueWithNID) ->
                    if (index > 0) {
                        ps.print(',')
                    }
                    val row = toRow.invoke(valueWithNID.first)
                    ps.print("\n(${valueWithNID.second},$row)")
                }
            ps.println(";")
        }
    }

    /**
     * Print inserts for collection with comments
     *
     * @param ps                   print stream
     * @param table                table name
     * @param columns              column names
     * @param objects              collection of objects
     * @param toId                 id extractor
     * @param objectIdToNID        id-to-nid map
     * @param toRowWithComment     double stringifier of objects, two strings are produced: [0] insert values , [1] comment
     * @param T                    type of objects
     */
    fun  printInsertWithComment(
        ps: PrintStream,
        table: String,
        columns: String,
        objects: Collection,
        toId: (T) -> String,
        objectIdToNID: Map,
        toRowWithComment: (T) -> Pair,
    ) {
        if (objects.isEmpty()) {
            ps.print("-- NONE")
        } else {
            ps.print("INSERT INTO $table ($columns) VALUES")
            objects
                .asSequence()
                .map { it to NIDMaps.lookup(objectIdToNID, toId.invoke(it)) }
                .toList()
                .sortedBy { it.second }
                .withIndex()
                .forEach { (index, valueWithNID) ->
                    if (index > 0) {
                        ps.print(',')
                    }
                    val rowWithComment = toRowWithComment.invoke(valueWithNID.first)
                    ps.print("\n(${valueWithNID.second},${rowWithComment.first}) /* ${rowWithComment.second} */")
                }
            ps.println(";")
        }
    }

    // lexes

    /**
     * Print inserts from lexes
     *
     * @param ps          print stream
     * @param table       table name
     * @param columns     column names
     * @param lexes       lexes
     * @param lexKeyToNID lex_key-to-nid map
     * @param toRow       stringifier of lexes
     */
    fun printInsert(
        ps: PrintStream,
        table: String,
        columns: String,
        lexes: Collection,
        lexKeyToNID: Map,
        toRow: (Lex) -> String,
    ) {
        if (lexes.isEmpty()) {
            ps.print("-- NONE")
        } else {
            ps.print("INSERT INTO $table ($columns) VALUES")
            lexes
                .asSequence()
                .map { it to NIDMaps.lookup(lexKeyToNID, Key.KeyLCP.of_t(it)) }
                .toList()
                .sortedBy { it.second }
                .withIndex()
                .forEach { (index, lexWithNID) ->
                    if (index > 0) {
                        ps.print(',')
                    }
                    val row = toRow.invoke(lexWithNID.first)
                    ps.print("\n(${lexWithNID.second},$row)")
                }
            ps.println(";")
        }
    }

    /**
     * Print inserts from lexes with comments
     *
     * @param ps                   print stream
     * @param table                table name
     * @param columns              column names
     * @param lexes                lexes
     * @param lexKeyToNID          lex_key-to-nid map
     * @param toRowWithComment     double stringifier of lexes, two strings are produced: [0] insert values , [1] comment
     */
    fun printInsertWithComment(
        ps: PrintStream,
        table: String,
        columns: String,
        lexes: Collection,
        lexKeyToNID: Map,
        toRowWithComment: (Lex) -> Pair,
    ) {
        if (lexes.isEmpty()) {
            ps.print("-- NONE")
        } else {
            ps.print("INSERT INTO $table ($columns) VALUES")
            lexes
                .asSequence()
                .map { it to NIDMaps.lookup(lexKeyToNID, Key.KeyLCP.of_t(it)) }
                .toList()
                .sortedBy { it.second }
                .withIndex()
                .forEach { (index, lexWithNID) ->
                    if (index > 0) {
                        ps.print(',')
                    }
                    val rowWithComment = toRowWithComment.invoke(lexWithNID.first)
                    ps.print("\n(${lexWithNID.second},${rowWithComment.first}) /* ${rowWithComment.second} */")
                }
            ps.println(";")
        }
    }

    // from sequences

    /**
     * Print inserts from sequence
     *
     * @param ps         print stream
     * @param table      table name
     * @param columns    column names
     * @param seq        sequence of objects
     * @param toRow      stringifier of objects
     * @param withNumber whether to number objects
     * @param T          type of objects in sequence
     */
    fun  printInsert(
        ps: PrintStream,
        table: String,
        columns: String,
        seq: Sequence,
        toRow: (T) -> String,
        withNumber: Boolean,
    ) {
        seq
            .withIndex()
            .forEach { (index, thing) ->
                if (index == 0) {
                    ps.print("INSERT INTO $table ($columns) VALUES")
                } else {
                    ps.print(',')
                }
                val row = toRow.invoke(thing)
                if (withNumber) {
                    ps.print("\n(${index + 1},$row)")
                } else {
                    ps.print("\n($row)")
                }
            }
        ps.println(";")
    }

    /**
     * Print inserts from sequence with comments
     *
     * @param ps                  print stream
     * @param table               table name
     * @param columns             column names
     * @param seq                 sequence of objects
     * @param toRowWithComment    double stringifier of objects, two strings are produced: [0] insert values , [1] comment
     * @param withNumber          whether to number objects
     * @param T                   type of objects in sequence
     */
    fun  printInsertWithComment(
        ps: PrintStream,
        table: String,
        columns: String,
        seq: Sequence,
        toRowWithComment: (T) -> Pair,
        withNumber: Boolean,
    ) {
        seq
            .withIndex()
            .forEach { (index, thing) ->
                if (index == 0) {
                    ps.print("INSERT INTO $table ($columns) VALUES")
                } else {
                    ps.print(',')
                }
                val rowAndComment = toRowWithComment.invoke(thing)
                if (withNumber) {
                    ps.print("\n(${index + 1},${rowAndComment.first}) /* ${rowAndComment.second} */")
                } else {
                    ps.print("\n(${rowAndComment.first}) /* ${rowAndComment.second} */")
                }
            }
        ps.println(";")
    }

    /**
     * Print inserts from sequence
     *
     * @param ps         print sequence
     * @param table      table name
     * @param columns    column names
     * @param seq        sequence of objects
     * @param toRows     stringifier for multiple values
     * @param withNumber whether to number objects
     * @param T          type of objects in sequence
     */
    fun  printInserts(
        ps: PrintStream,
        table: String,
        columns: String,
        seq: Sequence,
        toRows: (T) -> List,
        withNumber: Boolean,
    ) {
        seq
            .flatMap(toRows)
            .withIndex()
            .forEach { (index, row) ->
                if (index == 0) {
                    ps.print("INSERT INTO $table ($columns) VALUES")
                } else {
                    ps.print(',')
                }
                if (withNumber) {
                    ps.print("\n(${index + 1},$row)")
                } else {
                    ps.print("\n($row)")
                }
            }
        ps.println(";")
    }

    /**
     * Print inserts from sequence with comments
     *
     * @param ps                    print stream
     * @param table                 table name
     * @param columns               column names
     * @param seq                   sequence of objects
     * @param toRowsWithComments double stringifier of objects, two strings are produced: [0] insert values , [1] comment
     * @param withNumber            whether to number objects
     * @param T                     type of objects in sequence
     */
    fun  printInsertsWithComment(
        ps: PrintStream,
        table: String,
        columns: String,
        seq: Sequence,
        toRowsWithComments: (T) -> Sequence>,
        withNumber: Boolean,
    ) {
        seq
            .flatMap(toRowsWithComments)
            .withIndex()
            .forEach { (index, rowAndComment) ->
                if (index == 0) {
                    ps.print("INSERT INTO $table ($columns) VALUES")
                } else {
                    ps.print(',')
                }
                if (withNumber) {
                    ps.print("\n(${index + 1},${rowAndComment.first}) /* ${rowAndComment.second} */")
                } else {
                    ps.print("\n(${rowAndComment.first}) /* ${rowAndComment.second} */")
                }
            }
        ps.println(";")
    }

    // from map

    /**
     * Print inserts, single value
     *
     * @param ps      print stream
     * @param table   table name
     * @param columns column names
     * @param format  value format
     * @param mapper  objects by id
     * @param T       type of objects
     */
    fun > printInsert(
        ps: PrintStream,
        table: String,
        columns: String,
        format: String,
        mapper: Map,
    ) {
        ps.print("INSERT INTO $table ($columns) VALUES")
        mapper.entries
            .asSequence()
            .sortedBy { it.value }
            .withIndex()
            .forEach { (index, entry) ->
                if (index > 0) {
                    ps.print(',')
                }
                val row = String.format(format, entry.value, Utils.escape(entry.key))
                ps.print("\n$row")
            }
        ps.println(";")
    }

    /**
     * Print inserts, 2 values
     *
     * @param ps      print stream
     * @param table   table name
     * @param columns column names
     * @param format  value format
     * @param mapper  object arrays by id
     * @param T       type of objects
     */
    fun > printInsert2(
        ps: PrintStream,
        table: String,
        columns: String,
        format: String,
        mapper: Map, T>,
    ) {
        ps.print("INSERT INTO $table ($columns) VALUES")
        mapper.entries
            .toList()
            .sortedBy { it.value }
            .withIndex()
            .forEach { (index, entry) ->
                if (index > 0) {
                    ps.print(',')
                }
                val keys = entry.key
                val row = String.format(format, entry.value, keys[0], keys[1])
                ps.print("\n$row")
            }
        ps.println(";")
    }

    /**
     * Print inserts, 3 values
     *
     * @param ps      print stream
     * @param table   table name
     * @param columns column names
     * @param format  value format
     * @param mapper  object arrays by id
     * @param T       type of objects
     */
    fun > printInsert3(
        ps: PrintStream,
        table: String,
        columns: String,
        format: String,
        mapper: Map, T>,
    ) {
        ps.print("INSERT INTO $table ($columns) VALUES")
        mapper.entries
            .toList()
            .sortedBy { it.value }
            .withIndex()
            .forEach { (index, entry) ->
                if (index > 0) {
                    ps.print(',')
                }
                val keys = entry.key
                val row = String.format(format, entry.value, keys[0], keys[1], keys[2])
                ps.print("\n$row")
            }
        ps.println(";")
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy