Mercurial > cgi-bin > hgweb.cgi > PassMan
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") } } }