view src/main/kotlin/name/blackcap/passman/MergeSubcommand.kt @ 29:bf78f7f9dad3 default tip

Fix timestamp-matching bug.
author David Barts <n5jrn@me.com>
date Mon, 30 Dec 2024 17:10:11 -0800
parents ea65ab890f66
children
line wrap: on
line source

package name.blackcap.passman

import org.apache.commons.cli.*
import java.sql.ResultSet

class MergeSubcommand(): Subcommand() {
    private companion object {
        const val FORCE = "force"
        const val HELP = "help"
        const val VERBOSE = "verbose"
    }
    private lateinit var commandLine: CommandLine
    private lateinit var db: Database

    override fun run(args: Array<String>) {
        parseArguments(args)
        if (commandLine.hasOption(MergeSubcommand.HELP)) {
            return
        }
        db = Database.default
        doMerge()
    }

    private fun parseArguments(args: Array<String>) {
        val options = Options().apply {
            addOption("v", MergeSubcommand.VERBOSE, false, "Verbose mode, print what we are doing.")
            addOption("f", MergeSubcommand.FORCE, false, "Do not ask before overwriting.")
            addOption("h", MergeSubcommand.HELP, false, "Print this help message.")
        }
        try {
            commandLine = DefaultParser().parse(options, args)
        } catch (e: ParseException) {
            throw SubcommandException(message = e.message ?: "syntax error", status = 2, cause = e)
        }
        if (commandLine.hasOption(MergeSubcommand.HELP)) {
            HelpFormatter().printHelp("$SHORTNAME merge [options] other_database", options)
            return
        }
        if (commandLine.args.isEmpty()) {
            throw SubcommandException(message = "expecting other database name", status = 2)
        }
        if (commandLine.args.size > 1) {
            throw SubcommandException(message = "unexpected trailing arguments", status = 2)
        }
    }

    private fun doMerge() {
        val otherFile = commandLine.args[0]
        val otherDb = Database.open(
            fileName = otherFile,
            passwordPrompt = "Key for ${see(otherFile)}: ", create = false
        )
        otherDb.connection.prepareStatement("select name, username, password, notes, created, modified, accessed from passwords").use { stmt ->
            val results = stmt.executeQuery()
            while (results.next()) {
                val otherEntry = makeEntry(otherDb, results)
                vprint("read ${see(otherEntry.name)}…")
                val thisEntry = getEntry(db, otherEntry.name)
                if (thisEntry == null) {
                    vprintln(" missing, inserting it")
                    otherEntry.insert(db)
                } else {
                    doCompare(thisEntry, otherEntry)
                    thisEntry.password.clear()
                }
                otherEntry.password.clear()
            }
        }
    }

    private fun makeEntry(dbParam: Database, results: ResultSet) = Entry(
        name = results.getDecryptedString(1, dbParam.encryption)!!,
        username = results.getDecryptedString(2, dbParam.encryption)!!,
        password = results.getDecrypted(3, dbParam.encryption)!!,
        notes = results.getDecryptedString(4, dbParam.encryption),
        created = results.getDate(5),
        modified = results.getDate(6),
        accessed = results.getDate(7)
    )

    private fun getEntry(dbParam: Database, name: String): Entry? {
        dbParam.connection.prepareStatement("select name, username, password, notes, created, modified, accessed from passwords where id = ?").use { stmt ->
            stmt.setLong(1, dbParam.makeKey(name))
            val results = stmt.executeQuery()
            return if (results.next()) makeEntry(dbParam, results) else null
        }
    }

    private fun doCompare(thisEntry: Entry, otherEntry: Entry) {
        if (otherEntry.modifiedOrCreated.after(thisEntry.modifiedOrCreated) && okToChange(thisEntry, otherEntry)) {
            vprintln(" newer, updating it")
            db.connection.prepareStatement("update passwords set name = ?, username = ?, password = ?, notes = ?, modified = ? where id = ?").use {
                it.setEncryptedString(1, otherEntry.name, db.encryption)
                it.setEncryptedString(2, otherEntry.username, db.encryption)
                it.setEncrypted(3, otherEntry.password, db.encryption)
                it.setEncryptedString(4, otherEntry.notes, db.encryption)
                it.setLong(5, otherEntry.modifiedOrCreated.time)
                it.setLong(6, db.makeKey(thisEntry.name))
                it.executeUpdate()
            }
        } else {
            vprintln(" older or update denied, ignoring it")
        }
    }

    private fun okToChange(thisEntry: Entry, otherEntry: Entry): Boolean =
        commandLine.hasOption(FORCE) || askUserIfOkToOverwrite(thisEntry, otherEntry)

    private fun vprint(message: String) {
        if (commandLine.hasOption(VERBOSE)) {
            print(message)
        }
    }

    private fun vprintln(message: String) {
        if (commandLine.hasOption(VERBOSE)) {
            println(message)
        }
    }
}