view src/main/kotlin/name/blackcap/passman/ListSubcommand.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 69526ae8c8de
children
line wrap: on
line source

package name.blackcap.passman

import org.apache.commons.cli.*
import java.util.regex.PatternSyntaxException

class ListSubcommand(): Subcommand() {
    private companion object {
        const val CASE = "case"
        const val FIXED = "fixed"
        const val HELP = "help"
        const val LONG = "long"
        const val NAME = "name"
        const val NOTES = "notes"
        const val USERNAME = "username"
        const val ACCESSED = "accessed"
        const val CREATED = "created"
        const val MODIFIED = "modified"
        val FLAG_OPTIONS = listOf<OptionDescriptor>(
            OptionDescriptor(CASE, "Treat upper and lower case as distinct."),
            OptionDescriptor(FIXED, "Match fixed substrings instead of regular expressions."),
            OptionDescriptor(HELP, "Print this help message."),
            OptionDescriptor(LONG, "Long format listing.")
        )
        val ABBREV_STRING_OPTIONS = listOf<OptionDescriptor>(
            OptionDescriptor(NAME, "Match site name.")
        )
        val STRING_OPTIONS = listOf<OptionDescriptor>(
            OptionDescriptor(NOTES, "Match notes."),
            OptionDescriptor(USERNAME, "Match username.")
        )
        val TIME_OPTIONS = listOf<OptionDescriptor>(
            OptionDescriptor(ACCESSED, "Match time password last read."),
            OptionDescriptor(CREATED, "Match time created."),
            OptionDescriptor(MODIFIED, "Match time last modified.")
        )
        val REDACTED = "(redacted)".toCharArray()
    }
    private lateinit var commandLine: CommandLine
    private val matchers = mutableMapOf<String, MutableList<(Any?) -> Boolean>>()

    private data class OptionDescriptor(val name: String, val help: String)

    override fun run(args: Array<String>) {
        parseArgs(args)
        runQuery()
    }

    private fun parseArgs(args: Array<String>): Unit {
        val options = Options().apply {
            FLAG_OPTIONS.forEach {
                addOption(it.name.first().toString(), it.name, false, it.help)
            }
            ABBREV_STRING_OPTIONS.forEach() {
                addOption(Option(it.name.first().toString(), it.name, true, it.help).apply
                    { setArgs(Option.UNLIMITED_VALUES) })
            }
            (STRING_OPTIONS + TIME_OPTIONS).forEach {
                addOption(Option(null, it.name, true, it.help).apply
                    { setArgs(Option.UNLIMITED_VALUES) })
            }
        }
        try {
            commandLine = DefaultParser().parse(options, args)
        } catch (e: ParseException) {
            throw SubcommandException(message = e.message ?: "syntax error", cause = e, status = 2)
        }
        if (commandLine.hasOption(HELP)) {
            HelpFormatter().printHelp("$SHORTNAME list [options]", options)
            throw SubcommandException(status = 0)
        }
        if (commandLine.args.isNotEmpty()) {
            throw SubcommandException(message = "unexpected trailing arguments", status = 2)
        }

        (ABBREV_STRING_OPTIONS + STRING_OPTIONS).forEach {
            commandLine.getOptionValues(it.name)?.forEach { value ->
                val regexOptions = mutableSetOf<RegexOption>()
                if (commandLine.hasOption(FIXED)) {
                    regexOptions.add(RegexOption.LITERAL)
                }
                if (!commandLine.hasOption(CASE)) {
                    regexOptions.add(RegexOption.IGNORE_CASE)
                }
                try {
                    if (it.name !in matchers) {
                        matchers[it.name] = mutableListOf<(Any?) -> Boolean>()
                    }
                    matchers[it.name]!! += { x -> x is String && x.contains(Regex(value, regexOptions)) }
                } catch (e: PatternSyntaxException) {
                    throw SubcommandException(message = "${see(value)} - invalid regular expression", cause = e, status = 2)
                }
            }
        }

        TIME_OPTIONS.forEach {
            commandLine.getOptionValues(it.name)?.forEach { rawValue ->
                if (rawValue.isEmpty()) {
                    throw SubcommandException(message = "empty string is not a valid time expression")
                }
                val (op, exp) = when(rawValue.first()) {
                    '+', '>' -> Pair<Char, String>('>', rawValue.substring(1))
                    '='      -> Pair<Char, String>('=', rawValue.substring(1))
                    '-', '<' -> Pair<Char, String>('<', rawValue.substring(1))
                    else     -> Pair<Char, String>('=', rawValue)
                }
                val value = parseDateTime(exp)
                if (value == null) {
                    throw SubcommandException(message = "${see(rawValue)} - invalid time expression")
                }
                if (it.name !in matchers) {
                    matchers[it.name] = mutableListOf<(Any?) -> Boolean>()
                }
                when(op) {
                    '>' -> matchers[it.name]!! += { x -> x is java.util.Date && x.time > value }
                    '=' -> matchers[it.name]!! += { x -> x is java.util.Date && (x.time/1000L) == (value/1000L) }
                    '<' -> matchers[it.name]!! += { x -> x is java.util.Date && x.time < value }
                    else -> throw RuntimeException("should never happen")
                }
            }
        }
    }

    private fun runQuery(): Unit {
        val db = Database.default

        db.connection.prepareStatement("select $NAME, $USERNAME, $NOTES, $CREATED, $MODIFIED, $ACCESSED from passwords").use {
            val results = it.executeQuery()
            val printer = if (commandLine.hasOption(LONG)) Entry::printLong else Entry::print
            var count = 0;
            while (results.next()) {
                val entry = Entry(
                    name = results.getDecryptedString(1, db.encryption)!!,
                    username = results.getDecryptedString(2, db.encryption)!!,
                    password = REDACTED,
                    notes = results.getDecryptedString(3, db.encryption),
                    created = results.getDate(4),
                    modified = results.getDate(5),
                    accessed = results.getDate(6)
                )
                val passed = matchers
                    .map { x -> x.value.fold(true) {
                            acc, pred -> acc && pred(entry.getField(x.key)) } }
                    .reduceOrNull() { a, b -> a && b }
                    ?: true
                if (passed) {
                    if (count > 0) {
                        println()
                    }
                    printer(entry, null)
                    count++
                }
            }
            val s = if (count == 1) "" else "s"
            if (count > 0) {
                println()
            }
            println("$count record$s found")
        }
    }
}