view src/main/kotlin/name/blackcap/passman/PasswordSubcommand.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 java.nio.file.Files
import java.nio.file.Path
import java.nio.file.StandardCopyOption

class PasswordSubcommand : Subcommand() {
    override fun run(args: Array<String>) {
        // Parse arguments
        if (args.size > 0 && (args[0] == "-h" || args[0].startsWith("--h"))) {
            println("usage: passman password")
            return
        }
        if (!args.isEmpty()) {
            throw SubcommandException(message = "unexpected arguments", status = 2)
        }

        // Open databases
        println("Changing database encryption key...")
        val oldPath = Path.of(DB_FILE)
        val oldDb = Database.default
        val newPath = Path.of(NEW_DB_FILE)
        if (Files.exists(newPath)) {
            println("WARNING: deleting ${see(NEW_DB_FILE)}")
            Files.delete(newPath)
        }
        val newDb = Database.open(fileName = NEW_DB_FILE, passwordPrompt = "New database key: ", create = true)

        // Copy old to new
        println("WARNING: do not interrupt this process or data may be lost!")
        copyRecords(oldDb, newDb)

        // Wrap up. XXX - this closes Database.default, so this subcommand may
        // only be invoked directly from the shell prompt, never in interactive
        // mode.
        oldDb.connection.close()
        newDb.connection.close()
        Files.move(newPath, oldPath, StandardCopyOption.REPLACE_EXISTING)
        println("Database key changed.")
    }

    private fun copyRecords(oldDb: Database, newDb: Database) {
        oldDb.connection.prepareStatement("select name, username, password, notes, created, modified, accessed from passwords").use { old ->
            val results = old.executeQuery()
            while (results.next()) {
                newDb.connection.prepareStatement("insert into passwords (id, name, username, password, notes, created, modified, accessed) values (?, ?, ?, ?, ?, ?, ?, ?)").use { new ->
                    // id and name
                    val name = results.getDecryptedString(1, oldDb.encryption)!!
                    new.setLong(1, newDb.makeKey(name))
                    new.setEncryptedString(2, name, newDb.encryption)
                    // username
                    new.setEncryptedString(3,
                        results.getDecryptedString(2, oldDb.encryption)!!, newDb.encryption)
                    // password
                    val password = results.getDecrypted(3, oldDb.encryption)!!
                    new.setEncrypted(4, password, newDb.encryption)
                    password.clear()
                    // notes
                    new.setEncryptedString(5,
                        results.getDecryptedString(4, oldDb.encryption), newDb.encryption)
                    // created, modified, accessed
                    new.setDateOrNull(6, results.getLong(5))
                    new.setDateOrNull(7, results.getLong(6))
                    new.setDateOrNull(8, results.getLong(7))
                    new.executeUpdate()
                }
            }
        }
    }
}