diff shelltools/query-pr/query.py @ 50:bb0ece71355e

Untabify.
author David A. Holland
date Sat, 02 Apr 2022 18:10:52 -0400
parents 4b7f0ee35994
children ef6d572c4e1e
line wrap: on
line diff
--- a/shelltools/query-pr/query.py	Sat Apr 02 18:10:30 2022 -0400
+++ b/shelltools/query-pr/query.py	Sat Apr 02 18:10:52 2022 -0400
@@ -30,190 +30,190 @@
 #
 
 class QueryBuilder:
-	# these fields are in the PRs table
-	prtable_fields = [
-		"id", "synopsis", "confidential", "state", "locked",
-		"timeout_date", "timeout_state",
-		"arrival_schemaversion", "arrival_date", "modified_date",
-		"closed_date",
-		"release", "environment"
-	]
+        # these fields are in the PRs table
+        prtable_fields = [
+                "id", "synopsis", "confidential", "state", "locked",
+                "timeout_date", "timeout_state",
+                "arrival_schemaversion", "arrival_date", "modified_date",
+                "closed_date",
+                "release", "environment"
+        ]
 
-	# these fields are aliases for others
-	alias_fields = {
-		"number" : "id",
-		"date" : "arrival_date",
-	}
+        # these fields are aliases for others
+        alias_fields = {
+                "number" : "id",
+                "date" : "arrival_date",
+        }
 
-	def __init__(self):
-		self.present = {}
-		self.joined = {}
-		self.fromitems = []
-		self.whereitems = []
-		self.order = None
+        def __init__(self):
+                self.present = {}
+                self.joined = {}
+                self.fromitems = []
+                self.whereitems = []
+                self.order = None
 
-	def setorder(self, order):
-		self.order = order
+        def setorder(self, order):
+                self.order = order
 
-	# add to present{} and return the value for convenience (internal)
-	def makepresent(self, field, name):
-		self.present[field] = name
-		return name
+        # add to present{} and return the value for convenience (internal)
+        def makepresent(self, field, name):
+                self.present[field] = name
+                return name
 
-	# add a join item (once only) (internal)
-	def addjoin(self, table, as_ = None):
-		if as_ is not None:
-			key = table + "-" + as_
-			val = table + " AS " + as_
-		else:
-			key = table
-			val = table
-		if key not in self.joined:
-			self.joined[key] = True
-			self.fromitems.append(val)
+        # add a join item (once only) (internal)
+        def addjoin(self, table, as_ = None):
+                if as_ is not None:
+                        key = table + "-" + as_
+                        val = table + " AS " + as_
+                else:
+                        key = table
+                        val = table
+                if key not in self.joined:
+                        self.joined[key] = True
+                        self.fromitems.append(val)
 
-	# returns a sql expression for the field
-	def getfield(self, field):
-		# already-fetched fields
-		if field in self.present:
-			return self.present[field]
+        # returns a sql expression for the field
+        def getfield(self, field):
+                # already-fetched fields
+                if field in self.present:
+                        return self.present[field]
 
-		# aliases for other fields
-		if field in alias_fields:
-			return self.getfield(alias_fields[field])
+                # aliases for other fields
+                if field in alias_fields:
+                        return self.getfield(alias_fields[field])
 
-		# simple fields directly in the PRs table
-		if field in prtable_fields:
-			self.addjoin("PRs")
-			return self.makepresent(field, "PRs." + field)
+                # simple fields directly in the PRs table
+                if field in prtable_fields:
+                        self.addjoin("PRs")
+                        return self.makepresent(field, "PRs." + field)
 
-		# now it gets more interesting...
-		if field == "closed":
-			self.addjoin("PRs")
-			self.addjoin("states")
-			self.addwhere("PRs.state = states.name")
-			return self.makepresent(field, "states.closed")
+                # now it gets more interesting...
+                if field == "closed":
+                        self.addjoin("PRs")
+                        self.addjoin("states")
+                        self.addwhere("PRs.state = states.name")
+                        return self.makepresent(field, "states.closed")
 
-		# XXX let's pick one set of names and use them everywhere
-		# (e.g. change "posttime" in the schema to "message_date"
-		# or something)
-		if field == "comment_date" or field == "posttime":
-			self.addjoin("PRs")
-			self.addjoin("messages")
-			self.addwhere("PRs.id = messages.pr")
-			return self.makepresent(field, "messages.posttime")
+                # XXX let's pick one set of names and use them everywhere
+                # (e.g. change "posttime" in the schema to "message_date"
+                # or something)
+                if field == "comment_date" or field == "posttime":
+                        self.addjoin("PRs")
+                        self.addjoin("messages")
+                        self.addwhere("PRs.id = messages.pr")
+                        return self.makepresent(field, "messages.posttime")
 
-		if field == "comment" or field == "message" or field == "post":
-			self.addjoin("PRs")
-			self.addjoin("messages")
-			self.addwhere("PRs.id = messages.pr")
-			return self.makepresent(field, "messages.body")
+                if field == "comment" or field == "message" or field == "post":
+                        self.addjoin("PRs")
+                        self.addjoin("messages")
+                        self.addwhere("PRs.id = messages.pr")
+                        return self.makepresent(field, "messages.body")
 
-		if field == "attachment":
-			self.addjoin("PRs")
-			self.addjoin("messages")
-			self.addjoin("attachments")
-			self.addwhere("PRs.id = messages.pr")
-			self.addwhere("messages.id = attachments.msgid")
-			return self.makepresent(field, "attachments.body")
+                if field == "attachment":
+                        self.addjoin("PRs")
+                        self.addjoin("messages")
+                        self.addjoin("attachments")
+                        self.addwhere("PRs.id = messages.pr")
+                        self.addwhere("messages.id = attachments.msgid")
+                        return self.makepresent(field, "attachments.body")
 
-		if field == "patch":
-			self.addjoin("PRs")
-			self.addjoin("messages")
-			self.addjoin("attachments", "patches")
-			self.addwhere("PRs.id = messages.pr")
-			self.addwhere("messages.id = patches.msgid")
-			self.addwhere("patches.mimetype = " +
-					"'application/x-patch'")
-			return self.makepresent(field, "patches.body")
+                if field == "patch":
+                        self.addjoin("PRs")
+                        self.addjoin("messages")
+                        self.addjoin("attachments", "patches")
+                        self.addwhere("PRs.id = messages.pr")
+                        self.addwhere("messages.id = patches.msgid")
+                        self.addwhere("patches.mimetype = " +
+                                        "'application/x-patch'")
+                        return self.makepresent(field, "patches.body")
 
-		if field == "mimetype":
-			subquery = "((SELECT mtmessages1.pr as pr, " + \
-				"mtmessages1.mimetype as mimetype " + \
-				"FROM messages as mtmessages1) " + \
-				"UNION " + \
-				"(SELECT mtmessages2.pr as pr, " + \
-				"mtattach2.mimetype as mimetype " + \
-				"FROM messages as mtmessages2, " + \
-				"     attachments as mtattach2 " + \
-				"WHERE mtmessages2.id = mtattach2.msgid))"
-			self.addjoin("PRs")
-			self.addjoin(subquery, "mimetypes")
-			self.addwhere("PRs.id = mimetypes.pr")
-			return self.makepresent(field, "mimetypes.mimetype")
+                if field == "mimetype":
+                        subquery = "((SELECT mtmessages1.pr as pr, " + \
+                                "mtmessages1.mimetype as mimetype " + \
+                                "FROM messages as mtmessages1) " + \
+                                "UNION " + \
+                                "(SELECT mtmessages2.pr as pr, " + \
+                                "mtattach2.mimetype as mimetype " + \
+                                "FROM messages as mtmessages2, " + \
+                                "     attachments as mtattach2 " + \
+                                "WHERE mtmessages2.id = mtattach2.msgid))"
+                        self.addjoin("PRs")
+                        self.addjoin(subquery, "mimetypes")
+                        self.addwhere("PRs.id = mimetypes.pr")
+                        return self.makepresent(field, "mimetypes.mimetype")
 
-		# XXX: need view userstrings
-		# select (id, username as name) from users
-		# union select (id, realname as name) from users
-		# (allow searching emails? ugh)
-		if field == "originator" or field == "submitter":
-			self.addjoin("PRs")
-			self.addjoin("userstrings", "originators")
-			self.addwhere("PRs.originator = originators.id")
-			return self.makepresent(field, "originators.name")
+                # XXX: need view userstrings
+                # select (id, username as name) from users
+                # union select (id, realname as name) from users
+                # (allow searching emails? ugh)
+                if field == "originator" or field == "submitter":
+                        self.addjoin("PRs")
+                        self.addjoin("userstrings", "originators")
+                        self.addwhere("PRs.originator = originators.id")
+                        return self.makepresent(field, "originators.name")
 
-		if field == "reporter" or field == "respondent":
-			self.addjoin("PRs")
-			self.addjoin("subscriptions")
-			self.addjoin("userstrings", "reporters")
-			self.addwhere("subscriptions.userid = reporters.id")
-			self.addwhere("subscriptions.reporter")
-			return self.makepresent(field, "reporters.name")
+                if field == "reporter" or field == "respondent":
+                        self.addjoin("PRs")
+                        self.addjoin("subscriptions")
+                        self.addjoin("userstrings", "reporters")
+                        self.addwhere("subscriptions.userid = reporters.id")
+                        self.addwhere("subscriptions.reporter")
+                        return self.makepresent(field, "reporters.name")
 
-		if field == "responsible":
-			self.addjoin("PRs")
-			self.addjoin("subscriptions")
-			self.addjoin("userstrings", "responsibles")
-			self.addwhere("subscriptions.userid = responsibles.id")
-			self.addwhere("subscriptions.responsible")
-			return self.makepresent(field, "responsibles.name")
+                if field == "responsible":
+                        self.addjoin("PRs")
+                        self.addjoin("subscriptions")
+                        self.addjoin("userstrings", "responsibles")
+                        self.addwhere("subscriptions.userid = responsibles.id")
+                        self.addwhere("subscriptions.responsible")
+                        return self.makepresent(field, "responsibles.name")
 
-		if field in hierclasses:
-			col = field + "_data"
-			self.addjoin("PRs")
-			self.addjoin("hierclass_data", col)
-			self.addwhere("PRs.id = %s.pr" % col)
-			self.addwhere("%s.scheme = '%s'" % (col, field))
-			return self.makepresent(field, "%s.value" % col)
+                if field in hierclasses:
+                        col = field + "_data"
+                        self.addjoin("PRs")
+                        self.addjoin("hierclass_data", col)
+                        self.addwhere("PRs.id = %s.pr" % col)
+                        self.addwhere("%s.scheme = '%s'" % (col, field))
+                        return self.makepresent(field, "%s.value" % col)
 
-		if field in flatclasses:
-			col = field + "_data"
-			self.addjoin("PRs")
-			self.addjoin("flatclass_data", col)
-			self.addwhere("PRs.id = %s.pr" % col)
-			self.addwhere("%s.scheme = '%s'" % (col, field))
-			return self.makepresent(field, "%s.value" % col)
+                if field in flatclasses:
+                        col = field + "_data"
+                        self.addjoin("PRs")
+                        self.addjoin("flatclass_data", col)
+                        self.addwhere("PRs.id = %s.pr" % col)
+                        self.addwhere("%s.scheme = '%s'" % (col, field))
+                        return self.makepresent(field, "%s.value" % col)
 
-		if field in textclasses:
-			col = field + "_data"
-			self.addjoin("PRs")
-			self.addjoin("textclass_data", col)
-			self.addwhere("PRs.id = %s.pr" % col)
-			self.addwhere("%s.scheme = '%s'" % (col, field))
-			return self.makepresent(field, "%s.value" % col)
+                if field in textclasses:
+                        col = field + "_data"
+                        self.addjoin("PRs")
+                        self.addjoin("textclass_data", col)
+                        self.addwhere("PRs.id = %s.pr" % col)
+                        self.addwhere("%s.scheme = '%s'" % (col, field))
+                        return self.makepresent(field, "%s.value" % col)
 
-		if field in tagclasses:
-			col = field + "_data"
-			self.addjoin("PRs")
-			self.addjoin("tagclass_data", col)
-			self.addwhere("PRs.id = %s.pr" % col)
-			self.addwhere("%s.scheme = '%s'" % (col, field))
-			return self.makepresent(field, "%s.value" % col)
+                if field in tagclasses:
+                        col = field + "_data"
+                        self.addjoin("PRs")
+                        self.addjoin("tagclass_data", col)
+                        self.addwhere("PRs.id = %s.pr" % col)
+                        self.addwhere("%s.scheme = '%s'" % (col, field))
+                        return self.makepresent(field, "%s.value" % col)
 
-		sys.stderr.write("Unknown field %s" % field)
-		exit(1)
-	# end getfield
+                sys.stderr.write("Unknown field %s" % field)
+                exit(1)
+        # end getfield
 
-	# emit sql
-	def build(self, sels):
-		s = ", ".join(sels)
-		f = ", ".join(self.fromitems)
-		w = " and ".join(self.whereitems)
-		q = "SELECT %s\nFROM %s\nWHERE %s\n" % (s, f, w)
-		if self.order is not None:
-			q = q + "ORDER BY " + self.order + "\n"
-		return q
-	# endif
+        # emit sql
+        def build(self, sels):
+                s = ", ".join(sels)
+                f = ", ".join(self.fromitems)
+                w = " and ".join(self.whereitems)
+                q = "SELECT %s\nFROM %s\nWHERE %s\n" % (s, f, w)
+                if self.order is not None:
+                        q = q + "ORDER BY " + self.order + "\n"
+                return q
+        # endif
 
 # end class QueryBuilder
 
@@ -229,33 +229,33 @@
 dblink = None
 
 def opendb():
-	global dblink
+        global dblink
 
-	host = "localhost"
-	user = "swallowtail"
-	database = "swallowtail"
-	dblink = psycopg2.connect("host=%s user=%s dbname=%s" %
-					(host, user, database))
+        host = "localhost"
+        user = "swallowtail"
+        database = "swallowtail"
+        dblink = psycopg2.connect("host=%s user=%s dbname=%s" %
+                                        (host, user, database))
 # end opendb
 
 def closedb():
-	global dblink
+        global dblink
 
-	dblink.close()
-	dblink = None
+        dblink.close()
+        dblink = None
 # end closedb
 
 def querydb(qtext, args):
-	print "Executing this query:"
-	print qtext
-	print "Args are:"
-	print args
+        print "Executing this query:"
+        print qtext
+        print "Args are:"
+        print args
 
-	cursor = dblink.cursor()
-	cursor.execute(qtext, args)
-	result = cursor.fetchall()
-	cursor.close()
-	return result
+        cursor = dblink.cursor()
+        cursor.execute(qtext, args)
+        result = cursor.fetchall()
+        cursor.close()
+        return result
 # end querydb
 
 ############################################################
@@ -263,58 +263,58 @@
 # XXX: obsolete, remove
 
 class Query:
-	def __init__(self):
-		self.selections = []
-		self.tables = []
-		self.constraints = []
-		self.args = []
-	prtables = ["PRs"]
-	prconstraints = []
+        def __init__(self):
+                self.selections = []
+                self.tables = []
+                self.constraints = []
+                self.args = []
+        prtables = ["PRs"]
+        prconstraints = []
 
-	def select(self, s):
-		self.selections.append(s)
+        def select(self, s):
+                self.selections.append(s)
 
-	def addtable(self, t):
-		assert(t not in self.tables)
-		self.tables.append(t)
+        def addtable(self, t):
+                assert(t not in self.tables)
+                self.tables.append(t)
 
-	def constrain(self, expr):
-		self.constraints.append(t)
+        def constrain(self, expr):
+                self.constraints.append(t)
 
-	def internval(self, val):
-		num = len(self.args)
-		self.args[num] = val
-		return "$%d" % num
+        def internval(self, val):
+                num = len(self.args)
+                self.args[num] = val
+                return "$%d" % num
 
-	def textify(self):
-		s = "SELECT %s\n" % ",".join(self.selections)
-		f = "FROM %s\n" % ",".join(self.tables)
-		w = "WHERE %s\n" % " AND ".join(self.constraints)
-		return s + f + w
+        def textify(self):
+                s = "SELECT %s\n" % ",".join(self.selections)
+                f = "FROM %s\n" % ",".join(self.tables)
+                w = "WHERE %s\n" % " AND ".join(self.constraints)
+                return s + f + w
 # end class Query
 
 def regexp_constraint(q, field, value):
-	cleanval = q.internval(value)
-	if not isregexp(value):
-		return "%s = %s" % (field, cleanval)
-	else:
-		# XXX what's the right operator again?
-		return "%s ~= %s" % (field, cleanval)
+        cleanval = q.internval(value)
+        if not isregexp(value):
+                return "%s = %s" % (field, cleanval)
+        else:
+                # XXX what's the right operator again?
+                return "%s ~= %s" % (field, cleanval)
 # end regexp_constraint
 
 def intrange_constraint(q, field, value):
-	(lower, upper) = args.number
-	if lower is not None:
-		assert(typeof(lower) == int)
-		prq.constrain("%s >= %d" % (field, lower))
-	if upper is not None:
-		assert(typeof(upper) == int)
-		prq.constrain("%s <= %d" % (field, upper))
+        (lower, upper) = args.number
+        if lower is not None:
+                assert(typeof(lower) == int)
+                prq.constrain("%s >= %d" % (field, lower))
+        if upper is not None:
+                assert(typeof(upper) == int)
+                prq.constrain("%s <= %d" % (field, upper))
 # end intrange_constraint
 
 def daterange_constraint(q, field, value):
-	# XXX
-	assert(0)
+        # XXX
+        assert(0)
 # end daterange_constraint
 
 ############################################################
@@ -322,513 +322,513 @@
 # this is old code that needs to be merged or deleted into the new stuff
 def oldstuff():
 
-	# If we're doing something other than a search, do it now
-	if args.attach is not None:
-		get_attachment(args.attach)
-		exit(0)
-	if args.message is not None:
-		get_message(args.message)
-		exit(0)
+        # If we're doing something other than a search, do it now
+        if args.attach is not None:
+                get_attachment(args.attach)
+                exit(0)
+        if args.message is not None:
+                get_message(args.message)
+                exit(0)
 
-	if args.prs is not None and len(args.prs) > 0:
-		show_prs(args.prs)
-		exit(0)
+        if args.prs is not None and len(args.prs) > 0:
+                show_prs(args.prs)
+                exit(0)
 
-	#
-	# Collect up the search constraints
-	#
-	
-	# 1. Constraints on the PRs table
-	checkprtable = False
-	prq = Query()
-	prq.select("PRs.id as id")
-	prq.addtable("PRs")
-	if not args.closed:
-		checkprtable = True
-		prq.addtable("states")
-		prq.constrain("PRs.state = states.name")
-		prq.constrain("states.closed = FALSE")
-	if args.public:
-		checkprtable = True
-		prq.constrain("NOT PRs.confidential")
-	if args.number is not None:
-		checkprtable = True
-		intrange_constraint(prq, "PRs.id", args.number)
-	if args.synopsis is not None:
-		checkprtable = True
-		regexp_constraint(prq, "PRs.synopsis", args.synopsis)
-	if args.confidential is not None:
-		checkprtable = True
-		assert(typeof(args.confidential) == bool)
-		if args.confidential:
-			prq.constrain("PRs.confidential")
-		else:
-			prq.constrain("not PRs.confidential")
-	if args.state is not None:
-		checkprtable = True
-		regexp_constraint(prq, "PRs.state", args.state)
-	if args.locked is not None:
-		checkprtable = True
-		assert(typeof(args.locked) == bool)
-		if args.locked:
-			prq.constrain("PRs.locked")
-		else:
-			prq.constrain("not PRs.locked")
-	if args.arrival_schemaversion is not None:
-		checkprtable = True
-		intrange_constraint(prq, "PRs.arrival_schemaversion",
-					args.arrival_schemaversion)
-	if args.arrival_date is not None:
-		checkprtable = True
-		daterange_constraint(prq, "PRs.arrival_date",
-					args.arrival_date)
-	if args.closed_date is not None:
-		checkprtable = True
-		daterange_constraint(prq, "PRs.closed_date",
-					args.closed_date)
-	if args.last_modified is not None:
-		checkprtable = True
-		daterange_constraint(prq, "PRs.last_modified",
-					args.last_modified)
-	if args.release is not None:
-		checkprtable = True
-		regexp_constraint(prq, "PRs.release", args.release)
-	if args.environment is not None:
-		checkprtable = True
-		regexp_constraint(prq, "PRs.environment", args.environment)
+        #
+        # Collect up the search constraints
+        #
+        
+        # 1. Constraints on the PRs table
+        checkprtable = False
+        prq = Query()
+        prq.select("PRs.id as id")
+        prq.addtable("PRs")
+        if not args.closed:
+                checkprtable = True
+                prq.addtable("states")
+                prq.constrain("PRs.state = states.name")
+                prq.constrain("states.closed = FALSE")
+        if args.public:
+                checkprtable = True
+                prq.constrain("NOT PRs.confidential")
+        if args.number is not None:
+                checkprtable = True
+                intrange_constraint(prq, "PRs.id", args.number)
+        if args.synopsis is not None:
+                checkprtable = True
+                regexp_constraint(prq, "PRs.synopsis", args.synopsis)
+        if args.confidential is not None:
+                checkprtable = True
+                assert(typeof(args.confidential) == bool)
+                if args.confidential:
+                        prq.constrain("PRs.confidential")
+                else:
+                        prq.constrain("not PRs.confidential")
+        if args.state is not None:
+                checkprtable = True
+                regexp_constraint(prq, "PRs.state", args.state)
+        if args.locked is not None:
+                checkprtable = True
+                assert(typeof(args.locked) == bool)
+                if args.locked:
+                        prq.constrain("PRs.locked")
+                else:
+                        prq.constrain("not PRs.locked")
+        if args.arrival_schemaversion is not None:
+                checkprtable = True
+                intrange_constraint(prq, "PRs.arrival_schemaversion",
+                                        args.arrival_schemaversion)
+        if args.arrival_date is not None:
+                checkprtable = True
+                daterange_constraint(prq, "PRs.arrival_date",
+                                        args.arrival_date)
+        if args.closed_date is not None:
+                checkprtable = True
+                daterange_constraint(prq, "PRs.closed_date",
+                                        args.closed_date)
+        if args.last_modified is not None:
+                checkprtable = True
+                daterange_constraint(prq, "PRs.last_modified",
+                                        args.last_modified)
+        if args.release is not None:
+                checkprtable = True
+                regexp_constraint(prq, "PRs.release", args.release)
+        if args.environment is not None:
+                checkprtable = True
+                regexp_constraint(prq, "PRs.environment", args.environment)
 
-	if args.originator_name is not None or \
-			args.originator_email is not None:
-		prq.addtable("usermail as originator")
-		prq.constrain("PRs.originator = originator.id")
-	if args.originator_name is not None:
-		checkprtable = True
-		regexp_constraint(prq, "originator.realname",
-					args.originator_name)
-	if args.originator_email is not None:
-		checkprtable = True
-		regexp_constraint(prq, "originator.email",
-					args.originator_name)
-	if args.originator_id is not None:
-		checkprtable = True
-		intrange_constraint(prq, "PRs.originator", args.originator_id)
+        if args.originator_name is not None or \
+                        args.originator_email is not None:
+                prq.addtable("usermail as originator")
+                prq.constrain("PRs.originator = originator.id")
+        if args.originator_name is not None:
+                checkprtable = True
+                regexp_constraint(prq, "originator.realname",
+                                        args.originator_name)
+        if args.originator_email is not None:
+                checkprtable = True
+                regexp_constraint(prq, "originator.email",
+                                        args.originator_name)
+        if args.originator_id is not None:
+                checkprtable = True
+                intrange_constraint(prq, "PRs.originator", args.originator_id)
 
-	queries = []
-	if checkprtable:
-		queries.append(prq)
+        queries = []
+        if checkprtable:
+                queries.append(prq)
 
-	if args.responsible is not None:
-		sq = Query()
-		sq.select("subscriptions.pr as id")
-		sq.addtable("subscriptions")
-		sq.addtable("users")
-		sq.constrain("subscriptions.userid = users.id")
-		regexp_constraint(sq, "users.realname", args.responsible)
-		sq.constrain("subscriptions.responsible")
-		queries.append(sq)
-	if args.respondent is not None:
-		sq = Query()
-		sq.select("subscriptions.pr as id")
-		sq.addtable("subscriptions")
-		sq.addtable("users as subscribed")
-		sq.constrain("subscriptions.userid = users.id")
-		regexp_constraint(sq, "users.realname", args.respondent)
-		sq.constrain("subscriptions.reporter")
-		queries.append(sq)
-	if args.subscribed is not None:
-		sq = Query()
-		sq.select("subscriptions.pr as id")
-		sq.addtable("subscriptions")
-		sq.addtable("users as subscribed")
-		sq.constrain("subscriptions.userid = users.id")
-		regexp_constraint(sq, "users.realname", args.subscribed)
-		queries.append(sq)
+        if args.responsible is not None:
+                sq = Query()
+                sq.select("subscriptions.pr as id")
+                sq.addtable("subscriptions")
+                sq.addtable("users")
+                sq.constrain("subscriptions.userid = users.id")
+                regexp_constraint(sq, "users.realname", args.responsible)
+                sq.constrain("subscriptions.responsible")
+                queries.append(sq)
+        if args.respondent is not None:
+                sq = Query()
+                sq.select("subscriptions.pr as id")
+                sq.addtable("subscriptions")
+                sq.addtable("users as subscribed")
+                sq.constrain("subscriptions.userid = users.id")
+                regexp_constraint(sq, "users.realname", args.respondent)
+                sq.constrain("subscriptions.reporter")
+                queries.append(sq)
+        if args.subscribed is not None:
+                sq = Query()
+                sq.select("subscriptions.pr as id")
+                sq.addtable("subscriptions")
+                sq.addtable("users as subscribed")
+                sq.constrain("subscriptions.userid = users.id")
+                regexp_constraint(sq, "users.realname", args.subscribed)
+                queries.append(sq)
 
-	if args.messages is not None:
-		mq = Query()
-		mq.select("messages.pr as id")
-		mq.addtable("messages")
-		regexp_constraint(sq, "messages.text", args.messages)
-		queries.append(mq)
+        if args.messages is not None:
+                mq = Query()
+                mq.select("messages.pr as id")
+                mq.addtable("messages")
+                regexp_constraint(sq, "messages.text", args.messages)
+                queries.append(mq)
 
-	if args.adminlog is not None:
-		aq = Query()
-		aq.select("adminlog.pr as id")
-		aq.addtable("adminlog")
-		regexp_constraint(sq, "adminlog.change", args.adminlog)
-		regexp_constraint(sq, "adminlog.comment", args.adminlog)
-		assert(len(aq.constraints) == 2)
-		x = "%s OR %s" % (aq.constraints[0], aq.constraints[1])
-		aq.constraints = [x]
-		queries.append(aq)
+        if args.adminlog is not None:
+                aq = Query()
+                aq.select("adminlog.pr as id")
+                aq.addtable("adminlog")
+                regexp_constraint(sq, "adminlog.change", args.adminlog)
+                regexp_constraint(sq, "adminlog.comment", args.adminlog)
+                assert(len(aq.constraints) == 2)
+                x = "%s OR %s" % (aq.constraints[0], aq.constraints[1])
+                aq.constraints = [x]
+                queries.append(aq)
 
-	if args.anytext is not None:
-		choke("--anytext isn't supported yet")
+        if args.anytext is not None:
+                choke("--anytext isn't supported yet")
 
-	for scheme in classification_schemes:	
-		if args[scheme] is not None:
-			schemetype = classification_schemetypes[scheme]
-			tbl = "%sclass_data" % schemetype
-			cq = Query()
-			cq.select("scheme.pr as id")
-			cq.addtable("%s as scheme" % schemetype)
-			cq.constrain("scheme.scheme = '%s'" % scheme)
-			regexp_constraint(cq, "scheme.value", args[scheme])
-			queries.append(cq)
-	# end loop
+        for scheme in classification_schemes:   
+                if args[scheme] is not None:
+                        schemetype = classification_schemetypes[scheme]
+                        tbl = "%sclass_data" % schemetype
+                        cq = Query()
+                        cq.select("scheme.pr as id")
+                        cq.addtable("%s as scheme" % schemetype)
+                        cq.constrain("scheme.scheme = '%s'" % scheme)
+                        regexp_constraint(cq, "scheme.value", args[scheme])
+                        queries.append(cq)
+        # end loop
 
-	querytexts = [q.textify() for q in queries]
-	return "INTERSECT\n".join(querytexts)
+        querytexts = [q.textify() for q in queries]
+        return "INTERSECT\n".join(querytexts)
 
 ############################################################
 # printing
 
 class PrintText:
-	def __init__(self, output):
-		self.lines = (output == "RAW" or output == "LIST")
-	def printheader(self, row):
-		# nothing
+        def __init__(self, output):
+                self.lines = (output == "RAW" or output == "LIST")
+        def printheader(self, row):
+                # nothing
                 pass
-	def printrow(self, row):
-		# XXX
-		print row
-	def printfooter(self, row):
-		# nothing
+        def printrow(self, row):
+                # XXX
+                print row
+        def printfooter(self, row):
+                # nothing
                 pass
 # end class PrintText
 
 class PrintCsv:
-	def __init__(self, output):
-		# nothing
+        def __init__(self, output):
+                # nothing
                 pass
-	def printheader(self, row):
-		# XXX
+        def printheader(self, row):
+                # XXX
                 pass
-	def printrow(self, row):
-		# XXX
+        def printrow(self, row):
+                # XXX
                 pass
-	def printfooter(self, row):
-		# nothing
+        def printfooter(self, row):
+                # nothing
                 pass
 # end class PrintCsv
 
 class PrintXml:
-	def __init__(self, output):
-		# nothing
+        def __init__(self, output):
+                # nothing
                 pass
-	def printheader(self, row):
-		# XXX
+        def printheader(self, row):
+                # XXX
                 pass
-	def printrow(self, row):
-		# XXX
+        def printrow(self, row):
+                # XXX
                 pass
-	def printfooter(self, row):
-		# XXX
+        def printfooter(self, row):
+                # XXX
                 pass
 # end class PrintXml
 
 class PrintJson:
-	def __init__(self, output):
-		# nothing
+        def __init__(self, output):
+                # nothing
                 pass
-	def printheader(self, row):
-		# XXX
+        def printheader(self, row):
+                # XXX
                 pass
-	def printrow(self, row):
-		# XXX
+        def printrow(self, row):
+                # XXX
                 pass
-	def printfooter(self, row):
-		# XXX
+        def printfooter(self, row):
+                # XXX
                 pass
 # end class PrintJson
 
 class PrintRdf:
-	def __init__(self, output):
-		# nothing
+        def __init__(self, output):
+                # nothing
                 pass
-	def printheader(self, row):
-		# XXX
+        def printheader(self, row):
+                # XXX
                 pass
-	def printrow(self, row):
-		# XXX
+        def printrow(self, row):
+                # XXX
                 pass
-	def printfooter(self, row):
-		# XXX
+        def printfooter(self, row):
+                # XXX
                 pass
 # end class PrintRdf
 
 class PrintRdflike:
-	def __init__(self, output):
-		# nothing
+        def __init__(self, output):
+                # nothing
                 pass
-	def printheader(self, row):
-		# XXX
+        def printheader(self, row):
+                # XXX
                 pass
-	def printrow(self, row):
-		# XXX
+        def printrow(self, row):
+                # XXX
                 pass
-	def printfooter(self, row):
-		# XXX
+        def printfooter(self, row):
+                # XXX
                 pass
 # end class PrintRdflike
 
 def print_prs(ids):
-	if sel.outformat == "TEXT":
-		mkprinter = PrintText
-	elif sel.outformat == "CSV":
-		mkprinter = PrintCsv
-	elif sel.outformat == "XML":
-		mkprinter = PrintXml
-	elif sel.outformat == "JSON":
-		mkprinter = PrintJson
-	elif sel.outformat == "RDF":
-		mkprinter = PrintRdf
-	elif sel.outformat == "RDFLIKE":
-		mkprinter = PrintRdflike
-	else:
-		assert(False)
+        if sel.outformat == "TEXT":
+                mkprinter = PrintText
+        elif sel.outformat == "CSV":
+                mkprinter = PrintCsv
+        elif sel.outformat == "XML":
+                mkprinter = PrintXml
+        elif sel.outformat == "JSON":
+                mkprinter = PrintJson
+        elif sel.outformat == "RDF":
+                mkprinter = PrintRdf
+        elif sel.outformat == "RDFLIKE":
+                mkprinter = PrintRdflike
+        else:
+                assert(False)
 
-	# reset the printer
-	printer = mkprinter(sel.output)
+        # reset the printer
+        printer = mkprinter(sel.output)
 
-	if sel.output == "RAW":
-		printer.printheader(ids[0])
-		for id in ids:
-			printer(id)
-		printer.printfooter(ids[0])
-		return
-	elif sel.output == "LIST":
-		# XXX is there a clean way to do this passing the
-		# whole list of ids at once?
-		query = "SELECT id, synopsis\n" + \
-			"FROM PRs\n" + \
-			"WHERE id = $1"
-	elif sel.output == "HEADERS":
-		query = None # XXX
-	elif sel.output == "META":
-		query = None # XXX
-	elif sel.output == "FULL":
-		query = None # XXX
-	else:
-		assert(False)
+        if sel.output == "RAW":
+                printer.printheader(ids[0])
+                for id in ids:
+                        printer(id)
+                printer.printfooter(ids[0])
+                return
+        elif sel.output == "LIST":
+                # XXX is there a clean way to do this passing the
+                # whole list of ids at once?
+                query = "SELECT id, synopsis\n" + \
+                        "FROM PRs\n" + \
+                        "WHERE id = $1"
+        elif sel.output == "HEADERS":
+                query = None # XXX
+        elif sel.output == "META":
+                query = None # XXX
+        elif sel.output == "FULL":
+                query = None # XXX
+        else:
+                assert(False)
 
-	first = True
-	for id in ids:
-		results = querydb(query, [id])
-		if first:
-			printer.printheader(results[0])
-			first = False
-		for r in results:
-			printer.printrow(r)
-	printer.printfooter(results[0])
+        first = True
+        for id in ids:
+                results = querydb(query, [id])
+                if first:
+                        printer.printheader(results[0])
+                        first = False
+                for r in results:
+                        printer.printrow(r)
+        printer.printfooter(results[0])
 # end print_prs
 
 # XXX if in public mode we need to check if the PR is public
 def print_message(pr, msgnum):
-	query = "SELECT users.username AS username,\n" + \
-		"       users.realname AS realname,\n" + \
-		"       messages.id AS id, parent_id,\n" + \
-		"       posttime, mimetype, body\n" + \
-		"FROM messages, users\n" + \
-		"WHERE messages.who = users.id\n" + \
-		"  AND messages.pr = $1\n" + \
-		"  AND messages.number_in_pr = $2\n"
-	# Note that while pr is safe, msgnum came from the commandline
-	# and may not be.
-	results = querydb(query, [pr, msgnum])
-	[result] = results
-	(username, realname, id, parent_id, posttime, mimetype, body) = result
-	# XXX honor mimetype
-	# XXX honor output format (e.g. html)
-	sys.stdout.write("From swallowtail@%s  %s\n" % (organization,posttime))
-	sys.stdout.write("From: %s (%s)\n" % (username, realname))
-	sys.stdout.write("References: %s\n" % parent_id)
-	sys.stdout.write("Date: %s\n" % posttime)
-	sys.stdout.write("Content-Type: %s\n" % mimetype)
-	sys.stdout.write("\n")
-	sys.stdout.write(body)
+        query = "SELECT users.username AS username,\n" + \
+                "       users.realname AS realname,\n" + \
+                "       messages.id AS id, parent_id,\n" + \
+                "       posttime, mimetype, body\n" + \
+                "FROM messages, users\n" + \
+                "WHERE messages.who = users.id\n" + \
+                "  AND messages.pr = $1\n" + \
+                "  AND messages.number_in_pr = $2\n"
+        # Note that while pr is safe, msgnum came from the commandline
+        # and may not be.
+        results = querydb(query, [pr, msgnum])
+        [result] = results
+        (username, realname, id, parent_id, posttime, mimetype, body) = result
+        # XXX honor mimetype
+        # XXX honor output format (e.g. html)
+        sys.stdout.write("From swallowtail@%s  %s\n" % (organization,posttime))
+        sys.stdout.write("From: %s (%s)\n" % (username, realname))
+        sys.stdout.write("References: %s\n" % parent_id)
+        sys.stdout.write("Date: %s\n" % posttime)
+        sys.stdout.write("Content-Type: %s\n" % mimetype)
+        sys.stdout.write("\n")
+        sys.stdout.write(body)
 # end print_message
 
 # XXX if in public mode we need to check if the PR is public
 def print_attachment(pr, attachnum):
-	query = "SELECT a.mimetype as mimetype, a.body as body\n" + \
-		"FROM messages, attachments as a\n" + \
-		"WHERE messages.pr = $1\n" + \
-		"  AND messages.id = a.msgid\n" + \
-		"  AND a.number_in_pr = $2\n"
-	# Note that while pr is safe, attachnum came from the
-	# commandline and may not be.
-	results = querydb(query, [pr, msgnum])
-	[result] = results
-	(mimetype, body) = result
-	# XXX honor mimetype
-	# XXX need an http output mode so we can send the mimetype!
-	sys.stdout.write(body)
+        query = "SELECT a.mimetype as mimetype, a.body as body\n" + \
+                "FROM messages, attachments as a\n" + \
+                "WHERE messages.pr = $1\n" + \
+                "  AND messages.id = a.msgid\n" + \
+                "  AND a.number_in_pr = $2\n"
+        # Note that while pr is safe, attachnum came from the
+        # commandline and may not be.
+        results = querydb(query, [pr, msgnum])
+        [result] = results
+        (mimetype, body) = result
+        # XXX honor mimetype
+        # XXX need an http output mode so we can send the mimetype!
+        sys.stdout.write(body)
 # end print_attachment
 
 ############################################################
 # AST for query
 
 class Invocation:
-	class Query:
-		Q_TERM = 1  # XXX unused so far
-		Q_SQL = 2
-		Q_AND = 3
-		Q_OR = 4
-		Q_TSTRING = 5
-		Q_QSTRING = 6
+        class Query:
+                Q_TERM = 1  # XXX unused so far
+                Q_SQL = 2
+                Q_AND = 3
+                Q_OR = 4
+                Q_TSTRING = 5
+                Q_QSTRING = 6
 
-		def __init__(self, type):
-			self.type = type
-		def doterm(term):
-			self = Query(Q_TERM)
-			self.term = term
-			return self
-		def dosql(s):
-			self = Query(Q_SQL)
-			self.sql = s
-			return self
-		def doand(qs):
-			self = Query(Q_AND)
-			self.args = qs
-			return self
-		def door(qs):
-			self = Query(Q_OR)
-			self.args = qs
-			return self
-		# query term string
-		def dotstring(q):
-			self = Query(Q_TSTRING)
-			self.string = q
-			return self
-		# whole query string
-		def doqstring(q):
-			self = Query(Q_QSTRING)
-			self.string = q
-			return self
-	# end class Query
+                def __init__(self, type):
+                        self.type = type
+                def doterm(term):
+                        self = Query(Q_TERM)
+                        self.term = term
+                        return self
+                def dosql(s):
+                        self = Query(Q_SQL)
+                        self.sql = s
+                        return self
+                def doand(qs):
+                        self = Query(Q_AND)
+                        self.args = qs
+                        return self
+                def door(qs):
+                        self = Query(Q_OR)
+                        self.args = qs
+                        return self
+                # query term string
+                def dotstring(q):
+                        self = Query(Q_TSTRING)
+                        self.string = q
+                        return self
+                # whole query string
+                def doqstring(q):
+                        self = Query(Q_QSTRING)
+                        self.string = q
+                        return self
+        # end class Query
 
-	class Order:
-		def __init__(self, field, rev = False):
-			self.field = field
-			self.rev = rev
-		def dooldest(ign):
-			return Order("number")
-		def donewest(ign):
-			return Order("number", True)
-		def dostaleness(ign):
-			return Order("modified_date", True)
-		def dofield(field):
-			return Order(field)
-		def dorevfield(field):
-			return Order(field, True)
-	# end class Order
+        class Order:
+                def __init__(self, field, rev = False):
+                        self.field = field
+                        self.rev = rev
+                def dooldest(ign):
+                        return Order("number")
+                def donewest(ign):
+                        return Order("number", True)
+                def dostaleness(ign):
+                        return Order("modified_date", True)
+                def dofield(field):
+                        return Order(field)
+                def dorevfield(field):
+                        return Order(field, True)
+        # end class Order
 
-	class Search:
-		def __init__(self, qs, openonly, publiconly, os):
-			self.queries = qs
-			self.openonly = openonly
-			self.publiconly = publiconly
-			self.orders = os
-	# end class Search
+        class Search:
+                def __init__(self, qs, openonly, publiconly, os):
+                        self.queries = qs
+                        self.openonly = openonly
+                        self.publiconly = publiconly
+                        self.orders = os
+        # end class Search
 
-	class Selection:
-		S_PR = 1
-		S_MESSAGE = 2
-		S_ATTACHMENT = 3
+        class Selection:
+                S_PR = 1
+                S_MESSAGE = 2
+                S_ATTACHMENT = 3
 
-		def __init__(self, type):
-			self.type = type
-		def dopr(output, outformat):
-			self = Selection(S_PR)
-			self.output = output
-			self.outformat = outformat
-			return self
-		def domessage(arg):
-			self = Selection(S_MESSAGE)
-			self.message = arg
-			return self
-		def doattachment(arg):
-			self = Selection(S_ATTACHMENT)
-			self.attachment = arg
-			return self
-	# end class Selection
+                def __init__(self, type):
+                        self.type = type
+                def dopr(output, outformat):
+                        self = Selection(S_PR)
+                        self.output = output
+                        self.outformat = outformat
+                        return self
+                def domessage(arg):
+                        self = Selection(S_MESSAGE)
+                        self.message = arg
+                        return self
+                def doattachment(arg):
+                        self = Selection(S_ATTACHMENT)
+                        self.attachment = arg
+                        return self
+        # end class Selection
 
-	class Op:
-		# operation codes
-		OP_FIELDS = 1
-		OP_SHOW = 2
-		OP_RANGE = 3
-		OP_SEARCH = 4
+        class Op:
+                # operation codes
+                OP_FIELDS = 1
+                OP_SHOW = 2
+                OP_RANGE = 3
+                OP_SEARCH = 4
 
-		def __init__(self, type):
-			self.type = type
+                def __init__(self, type):
+                        self.type = type
 
-		def dofields():
-			return Op(OP_FIELDS)
-		def doshow(field):
-			self = Op(OP_SHOW)
-			self.field = field
-			return self
-		def dorange(field):
-			self = Op(OP_RANGE)
-			self.field = field
-			return self
-		def dosearch(s, sels):
-			self = Op(OP_SEARCH)
-			self.search = s
-			self.sels = sels
-			return self
-	# end class Op
+                def dofields():
+                        return Op(OP_FIELDS)
+                def doshow(field):
+                        self = Op(OP_SHOW)
+                        self.field = field
+                        return self
+                def dorange(field):
+                        self = Op(OP_RANGE)
+                        self.field = field
+                        return self
+                def dosearch(s, sels):
+                        self = Op(OP_SEARCH)
+                        self.search = s
+                        self.sels = sels
+                        return self
+        # end class Op
 
-	def __init__(self, ops):
-		self.ops = ops
+        def __init__(self, ops):
+                self.ops = ops
 # end class Invocation
 
 ############################################################
 # run (eval the SQL and print the results)
 
 def run_sel(sel, ids):
-	if sel.type == S_PR:
-		if ids == []:
-			sys.stderr.write("No PRs matched.\n")
-			exit(1)
+        if sel.type == S_PR:
+                if ids == []:
+                        sys.stderr.write("No PRs matched.\n")
+                        exit(1)
 
-		print_prs(ids)
-	elif sel.type == S_MESSAGE:
-		if len(ids) != 1:
-			sys.stderr.write("Cannot retrieve messages " +
-				"from multiple PRs.")
-			exit(1)
-		print_message(ids[0], sel.message)
-	elif sel.type == S_ATTACHMENT:
-		if len(ids) != 1:
-			sys.stderr.write("Cannot retrieve attachments " +
-				"from multiple PRs.")
-			exit(1)
-		print_message(ids[0], sel.attachment)
-	else:
-		assert(False)
+                print_prs(ids)
+        elif sel.type == S_MESSAGE:
+                if len(ids) != 1:
+                        sys.stderr.write("Cannot retrieve messages " +
+                                "from multiple PRs.")
+                        exit(1)
+                print_message(ids[0], sel.message)
+        elif sel.type == S_ATTACHMENT:
+                if len(ids) != 1:
+                        sys.stderr.write("Cannot retrieve attachments " +
+                                "from multiple PRs.")
+                        exit(1)
+                print_message(ids[0], sel.attachment)
+        else:
+                assert(False)
 
 def run_op(op):
-	if op.type == OP_FIELDS:
-		list_fields()
-	elif op.type == OP_SHOW:
-		describe_field(op.field)
-	elif op.type == OP_RANGE:
-		print_field_range(op.field)
-	elif op.type == OP_SEARCH:
-		sql = op.search
-		args = op.args # XXX not there!
-		ids = querydb(op.search, args)
-		for s in op.sels:
-			run_sel(s, ids)
-	else:
-		assert(False)
+        if op.type == OP_FIELDS:
+                list_fields()
+        elif op.type == OP_SHOW:
+                describe_field(op.field)
+        elif op.type == OP_RANGE:
+                print_field_range(op.field)
+        elif op.type == OP_SEARCH:
+                sql = op.search
+                args = op.args # XXX not there!
+                ids = querydb(op.search, args)
+                for s in op.sels:
+                        run_sel(s, ids)
+        else:
+                assert(False)
 
 def run(ast):
-	for op in ast.ops:
-		run_op(op)
+        for op in ast.ops:
+                run_op(op)
 
 ############################################################
 # compile (convert the AST so the searches are pure SQL)
@@ -843,127 +843,127 @@
         return True
 
 def compile_query(q):
-	if q.type == Q_QSTRING:
-		# XXX should use a split that honors quotes
-		terms = q.string.split()
-		terms = [dotstring(t) for t in terms]
-		return compile_query(doand(terms))
-	if q.type == Q_TSTRING:
-		qb = QueryBuilder()
-		s = q.string
-		if matches(s, "^[0-9]+$"):
-			f = qb.getfield("number")
-			# Note: s is user-supplied but clean to insert directly
-			qb.addwhere("%s = %s" % (f, s))
-		elif matches(s, "^[0-9]+-[0-9]+$"):
-			f = qb.getfield("number")
-			ss = s.split("-")
-			# Note: ss[] is user-supplied but clean
-			qb.addwhere("%s >= %s" % (f, ss[0]))
-			qb.addwhere("%s <= %s" % (f, ss[1]))
-		elif matches(s, "^[0-9]+-$"):
-			f = qb.getfield("number")
-			ss = s.split("-")
-			# Note: ss[] is user-supplied but clean
-			qb.addwhere("%s >= %s" % (f, ss[0]))
-		elif matches(s, "^-[0-9]+$"):
-			f = qb.getfield("number")
-			ss = s.split("-")
-			# Note: ss[] is user-supplied but clean
-			qb.addwhere("%s <= %s" % (f, ss[1]))
-		elif matches(s, "^[^:]+:[^:]+$"):
-			# XXX honor quoted terms
-			# XXX = or LIKE?
-			ss = s.split(":")
-			# ss[0] is not clean but if it's crap it won't match
-			f = qb.getfield(ss[0])
-			# ss[1] is not clean, so intern it for safety
-			s = qb.intern(ss[1])
-			qb.addwhere("%s = %s" % (f, s))
-		elif matches(s, "^-[^:]+:[^:]+$"):
-			# XXX honor quoted terms
-			# XXX <> or NOT LIKE?
-			ss = s.split(":")
-			# ss[0] is not clean but if it's crap it won't match
-			f = qb.getfield(ss[0])
-			# ss[1] is not clean, so intern it for safety
-			s = qb.intern(ss[1])
-			qb.addwhere("%s <> %s" % (f, s))
-		elif matches(s, "^-"):
-			# XXX <> or NOT LIKE?
-			f = qb.getfield("alltext")
-			# s is not clean, so intern it for safety
-			s = qb.intern(s)
-			qb.addwhere("%s <> %s" % (f, s))
-		else:
-			# XXX = or LIKE?
-			f = qb.getfield("alltext")
-			# s is not clean, so intern it for safety
-			s = qb.intern(s)
-			qb.addwhere("%s = %s" % (f, s))
+        if q.type == Q_QSTRING:
+                # XXX should use a split that honors quotes
+                terms = q.string.split()
+                terms = [dotstring(t) for t in terms]
+                return compile_query(doand(terms))
+        if q.type == Q_TSTRING:
+                qb = QueryBuilder()
+                s = q.string
+                if matches(s, "^[0-9]+$"):
+                        f = qb.getfield("number")
+                        # Note: s is user-supplied but clean to insert directly
+                        qb.addwhere("%s = %s" % (f, s))
+                elif matches(s, "^[0-9]+-[0-9]+$"):
+                        f = qb.getfield("number")
+                        ss = s.split("-")
+                        # Note: ss[] is user-supplied but clean
+                        qb.addwhere("%s >= %s" % (f, ss[0]))
+                        qb.addwhere("%s <= %s" % (f, ss[1]))
+                elif matches(s, "^[0-9]+-$"):
+                        f = qb.getfield("number")
+                        ss = s.split("-")
+                        # Note: ss[] is user-supplied but clean
+                        qb.addwhere("%s >= %s" % (f, ss[0]))
+                elif matches(s, "^-[0-9]+$"):
+                        f = qb.getfield("number")
+                        ss = s.split("-")
+                        # Note: ss[] is user-supplied but clean
+                        qb.addwhere("%s <= %s" % (f, ss[1]))
+                elif matches(s, "^[^:]+:[^:]+$"):
+                        # XXX honor quoted terms
+                        # XXX = or LIKE?
+                        ss = s.split(":")
+                        # ss[0] is not clean but if it's crap it won't match
+                        f = qb.getfield(ss[0])
+                        # ss[1] is not clean, so intern it for safety
+                        s = qb.intern(ss[1])
+                        qb.addwhere("%s = %s" % (f, s))
+                elif matches(s, "^-[^:]+:[^:]+$"):
+                        # XXX honor quoted terms
+                        # XXX <> or NOT LIKE?
+                        ss = s.split(":")
+                        # ss[0] is not clean but if it's crap it won't match
+                        f = qb.getfield(ss[0])
+                        # ss[1] is not clean, so intern it for safety
+                        s = qb.intern(ss[1])
+                        qb.addwhere("%s <> %s" % (f, s))
+                elif matches(s, "^-"):
+                        # XXX <> or NOT LIKE?
+                        f = qb.getfield("alltext")
+                        # s is not clean, so intern it for safety
+                        s = qb.intern(s)
+                        qb.addwhere("%s <> %s" % (f, s))
+                else:
+                        # XXX = or LIKE?
+                        f = qb.getfield("alltext")
+                        # s is not clean, so intern it for safety
+                        s = qb.intern(s)
+                        qb.addwhere("%s = %s" % (f, s))
 
-		# XXX also does not handle:
-		#
-		# field: with no string (supposed to use a default
-		# search string)
-		#
-		# generated search fields that parse dates:
-		# {arrived,closed,modified,etc.}-{before,after}:date
-		#
-		# stale:time
+                # XXX also does not handle:
+                #
+                # field: with no string (supposed to use a default
+                # search string)
+                #
+                # generated search fields that parse dates:
+                # {arrived,closed,modified,etc.}-{before,after}:date
+                #
+                # stale:time
 
-		return qb.build("PRs.id")
-	# end Q_TSTRING case
-	if q.type == Q_OR:
-		subqueries = ["(" + compile_query(sq) + ")" for sq in q.args]
-		return " UNION ".join(subqueries)
-	if q.type == Q_AND:
-		subqueries = ["(" + compile_query(sq) + ")" for sq in q.args]
-		return " INTERSECT ".join(subqueries)
-	if q.type == Q_SQL:
-		return q.sql
-	assert(False)
+                return qb.build("PRs.id")
+        # end Q_TSTRING case
+        if q.type == Q_OR:
+                subqueries = ["(" + compile_query(sq) + ")" for sq in q.args]
+                return " UNION ".join(subqueries)
+        if q.type == Q_AND:
+                subqueries = ["(" + compile_query(sq) + ")" for sq in q.args]
+                return " INTERSECT ".join(subqueries)
+        if q.type == Q_SQL:
+                return q.sql
+        assert(False)
 # end compile_query
 
 def compile_order(qb, o):
-	str = qb.getfield(o.field)
-	if o.rev:
-		str = str + " DESCENDING"
-	return str
+        str = qb.getfield(o.field)
+        if o.rev:
+                str = str + " DESCENDING"
+        return str
 
 def compile_search(s):
-	qb2 = QueryBuilder()
+        qb2 = QueryBuilder()
 
-	# multiple query strings are treated as OR
-	query = door(s.queries)
-	query = compile_query(q)
+        # multiple query strings are treated as OR
+        query = door(s.queries)
+        query = compile_query(q)
 
-	if s.openonly:
-		qb2.addwhere("not %s" % qb.getfield("closed"))
-	if s.publiconly:
-		qb2.addwhere("not %s" % qb.getfield("confidential"))
+        if s.openonly:
+                qb2.addwhere("not %s" % qb.getfield("closed"))
+        if s.publiconly:
+                qb2.addwhere("not %s" % qb.getfield("confidential"))
 
-	orders = [compile_order(qb2, o) for o in s.orders]
-	order = ", ".join(orders)
-	if order != "":
-		qb2.setorder(order)
+        orders = [compile_order(qb2, o) for o in s.orders]
+        order = ", ".join(orders)
+        if order != "":
+                qb2.setorder(order)
 
-	if qb2.nonempty():
-		qb2.addjoin(query, "search")
-		qb2.addjoin("PRs")
-		qb2.addwhere("search = PRs.id")
-		query = qb2.build(["search"])
+        if qb2.nonempty():
+                qb2.addjoin(query, "search")
+                qb2.addjoin("PRs")
+                qb2.addwhere("search = PRs.id")
+                query = qb2.build(["search"])
 
-	return query
+        return query
 # end compile_search
 
 def compile_op(op):
-	if op.type == OP_SEARCH:
-		op.search = compile_search(op.search)
-	return op
+        if op.type == OP_SEARCH:
+                op.search = compile_search(op.search)
+        return op
 
 def compile(ast):
-	ast.ops = [compile_op(op) for op in ast.ops]
+        ast.ops = [compile_op(op) for op in ast.ops]
 
 ############################################################
 # arg handling
@@ -987,167 +987,167 @@
 # ctor you want to use, which seems completely insane.
 #
 class CtorAppend(argparse.Action):
-	def __call__(self, parser, namespace, values, option_string=None):
-		items = getattr(namespace, self.dest)
-		item = self.const(values)
-		items.append(item)
-		setattr(namespace, self.dest, items)
+        def __call__(self, parser, namespace, values, option_string=None):
+                items = getattr(namespace, self.dest)
+                item = self.const(values)
+                items.append(item)
+                setattr(namespace, self.dest, items)
 
 def getargs():
-	p = argparse.ArgumentParser(program_description)
+        p = argparse.ArgumentParser(program_description)
 
-	# note: -h/--help is built in by default
-	p.add_argument("-v", "--version",
-			action='version', version=program_version,
-			help="Print program version and exit")
+        # note: -h/--help is built in by default
+        p.add_argument("-v", "--version",
+                        action='version', version=program_version,
+                        help="Print program version and exit")
 
-	p.add_argument("--show", nargs=1,
-			action=CtorAppend, dest='ops',
-			const=Invocation.Op.doshow,
-			help="Show description of field")
-	p.add_argument("--range", nargs=1,
-			action=CtorAppend, dest='ops',
-			const=Invocation.Op.dorange,
-			help="Show range of extant values for field")
+        p.add_argument("--show", nargs=1,
+                        action=CtorAppend, dest='ops',
+                        const=Invocation.Op.doshow,
+                        help="Show description of field")
+        p.add_argument("--range", nargs=1,
+                        action=CtorAppend, dest='ops',
+                        const=Invocation.Op.dorange,
+                        help="Show range of extant values for field")
 
-	p.add_argument("--search", nargs=1,
-			action=CtorAppend, dest='queries',
-			const=Invocation.Query.doqstring,
-			help="Force string to be read as a search string")
-	p.add_argument("-s", "--sql", nargs=1,
-			action=CtorAppend, dest='queries',
-			const=Invocation.Query.dosql,
-			help="Supply explicit sql query as search")
+        p.add_argument("--search", nargs=1,
+                        action=CtorAppend, dest='queries',
+                        const=Invocation.Query.doqstring,
+                        help="Force string to be read as a search string")
+        p.add_argument("-s", "--sql", nargs=1,
+                        action=CtorAppend, dest='queries',
+                        const=Invocation.Query.dosql,
+                        help="Supply explicit sql query as search")
 
-	p.add_argument("--open",
-			action='store_const', dest='openonly', const="True",
-			help="Exclude closed PRs (default)")
-	p.add_argument("--closed",
-			action='store_const', dest='openonly', const="False",
-			help="Include closed PRs in search")
-	p.add_argument("--public",
-			action='store_const', dest='publiconly', const="True",
-			help="Exclude confidential PRs")
-	p.add_argument("--privileged",
-			action='store_const', dest='publiconly', const="False",
-			help="Allow confidential PRs (default)")
+        p.add_argument("--open",
+                        action='store_const', dest='openonly', const="True",
+                        help="Exclude closed PRs (default)")
+        p.add_argument("--closed",
+                        action='store_const', dest='openonly', const="False",
+                        help="Include closed PRs in search")
+        p.add_argument("--public",
+                        action='store_const', dest='publiconly', const="True",
+                        help="Exclude confidential PRs")
+        p.add_argument("--privileged",
+                        action='store_const', dest='publiconly', const="False",
+                        help="Allow confidential PRs (default)")
 
-	p.add_argument("--oldest",
-			action=CtorAppend, dest='orders',
-			const=Invocation.Order.dooldest,
-			help="Sort output with oldest PRs first")
-	p.add_argument("--newest",
-			action=CtorAppend, dest='orders',
-			const=Invocation.Order.donewest,
-			help="Sort output with newest PRs first")
-	p.add_argument("--staleness",
-			action=CtorAppend, dest='orders',
-			const=Invocation.Order.dostaleness,
-			help="Sort output by time since last modification")
-	p.add_argument("--orderby", nargs=1,
-			action=CtorAppend, dest='orders',
-			const=Invocation.Order.dofield,
-			help="Sort output by specific field")
-	p.add_argument("--revorderby", nargs=1,
-			action=CtorAppend, dest='orders',
-			const=Invocation.Order.dorevfield,
-			help="Sort output by specific field, reversed")
+        p.add_argument("--oldest",
+                        action=CtorAppend, dest='orders',
+                        const=Invocation.Order.dooldest,
+                        help="Sort output with oldest PRs first")
+        p.add_argument("--newest",
+                        action=CtorAppend, dest='orders',
+                        const=Invocation.Order.donewest,
+                        help="Sort output with newest PRs first")
+        p.add_argument("--staleness",
+                        action=CtorAppend, dest='orders',
+                        const=Invocation.Order.dostaleness,
+                        help="Sort output by time since last modification")
+        p.add_argument("--orderby", nargs=1,
+                        action=CtorAppend, dest='orders',
+                        const=Invocation.Order.dofield,
+                        help="Sort output by specific field")
+        p.add_argument("--revorderby", nargs=1,
+                        action=CtorAppend, dest='orders',
+                        const=Invocation.Order.dorevfield,
+                        help="Sort output by specific field, reversed")
 
-	p.add_argument("-m", "--message", nargs=1,
-			action=CtorAppend, dest='selections',
-			const=Invocation.Selection.domessage,
-			help="Print selected message (single PR only)")
-	p.add_argument("-a", "--attachment", nargs=1,
-			action=CtorAppend, dest='selections',
-			const=Invocation.Selection.doattachment,
-			help="Print selected attachment (single PR only)")
+        p.add_argument("-m", "--message", nargs=1,
+                        action=CtorAppend, dest='selections',
+                        const=Invocation.Selection.domessage,
+                        help="Print selected message (single PR only)")
+        p.add_argument("-a", "--attachment", nargs=1,
+                        action=CtorAppend, dest='selections',
+                        const=Invocation.Selection.doattachment,
+                        help="Print selected attachment (single PR only)")
 
-	p.add_argument("-r", "--raw",
-			action = 'store_const', const="RAW",
-			dest = 'output',
-			help="Print exactly what the database returns")
-	p.add_argument("-l", "--list",
-			action = 'store_const', const="LIST",
-			dest = 'output',
-			help="Print in list form (default)")
-	p.add_argument("--headers",
-			action = 'store_const', const="HEADERS",
-			dest = 'output',
-			help="Print header information only")
-	p.add_argument("--meta",
-			action = 'store_const', const="META",
-			dest = 'output',
-			help="Print all metadata")
-	p.add_argument("--metadata",
-			action = 'store_const', const="META",
-			dest = 'output')
-	p.add_argument("-f", "--full",
-			action = 'store_const', const="FULL",
-			dest = 'output',
-			help="Print everything")
+        p.add_argument("-r", "--raw",
+                        action = 'store_const', const="RAW",
+                        dest = 'output',
+                        help="Print exactly what the database returns")
+        p.add_argument("-l", "--list",
+                        action = 'store_const', const="LIST",
+                        dest = 'output',
+                        help="Print in list form (default)")
+        p.add_argument("--headers",
+                        action = 'store_const', const="HEADERS",
+                        dest = 'output',
+                        help="Print header information only")
+        p.add_argument("--meta",
+                        action = 'store_const', const="META",
+                        dest = 'output',
+                        help="Print all metadata")
+        p.add_argument("--metadata",
+                        action = 'store_const', const="META",
+                        dest = 'output')
+        p.add_argument("-f", "--full",
+                        action = 'store_const', const="FULL",
+                        dest = 'output',
+                        help="Print everything")
 
-	p.add_argument("--text",
-			action = 'store_const', const="TEXT",
-			dest = 'outformat',
-			help="Print in text format (default)")
-	p.add_argument("--csv",
-			action = 'store_const', const="CSV",
-			dest = 'outformat',
-			help="Print a CSV file")
-	p.add_argument("--xml",
-			action = 'store_const', const="XML",
-			dest = 'outformat',
-			help="Print in XML")
-	p.add_argument("--json",
-			action = 'store_const', const="JSON",
-			dest = 'outformat',
-			help="Print in JSON")
-	p.add_argument("--rdf",
-			action = 'store_const', const="RDF",
-			dest = 'outbformat',
-			help="Print in RDF")
-	p.add_argument("--rdflike",
-			action = 'store_const', const="RDFLIKE",
-			dest = 'outformat',
-			help="Print RDF-like text")
+        p.add_argument("--text",
+                        action = 'store_const', const="TEXT",
+                        dest = 'outformat',
+                        help="Print in text format (default)")
+        p.add_argument("--csv",
+                        action = 'store_const', const="CSV",
+                        dest = 'outformat',
+                        help="Print a CSV file")
+        p.add_argument("--xml",
+                        action = 'store_const', const="XML",
+                        dest = 'outformat',
+                        help="Print in XML")
+        p.add_argument("--json",
+                        action = 'store_const', const="JSON",
+                        dest = 'outformat',
+                        help="Print in JSON")
+        p.add_argument("--rdf",
+                        action = 'store_const', const="RDF",
+                        dest = 'outbformat',
+                        help="Print in RDF")
+        p.add_argument("--rdflike",
+                        action = 'store_const', const="RDFLIKE",
+                        dest = 'outformat',
+                        help="Print RDF-like text")
 
-	p.add_argument("TERM", nargs='*',
-			action=CtorAppend, dest='queries',
-			const=Invocation.Query.doqstring,
-			help="Search term")
+        p.add_argument("TERM", nargs='*',
+                        action=CtorAppend, dest='queries',
+                        const=Invocation.Query.doqstring,
+                        help="Search term")
 
-	args = p.parse_args()
+        args = p.parse_args()
 
-	ops = args.ops
-	if ops is None:
-		ops = []
-	queries = args.queries
-	if queries is not None:
-		openonly = args.openonly
-		if openonly is None:
-			openonly = True
-		publiconly = args.publiconly
-		if publiconly is None:
-			publiconly = False
-		orders = args.orders
-		if orders is None:
-			orders = [Invocation.Order.dooldest(None)]
-		output = args.output
-		if output is None:
-			output = "LIST"
-		outformat = args.outformat
-		if outformat is None:
-			outformat = "TEXT"
-		selections = args.selections
-		if selections is None:
-			sel = Invocation.Selection.dopr(output, outformat)
-			selections = [sel]
-		search = Search(queries, openonly, publiconly, orders)
-		op = dosearch(search, selections)
-		ops.append(op)
-	# endif
+        ops = args.ops
+        if ops is None:
+                ops = []
+        queries = args.queries
+        if queries is not None:
+                openonly = args.openonly
+                if openonly is None:
+                        openonly = True
+                publiconly = args.publiconly
+                if publiconly is None:
+                        publiconly = False
+                orders = args.orders
+                if orders is None:
+                        orders = [Invocation.Order.dooldest(None)]
+                output = args.output
+                if output is None:
+                        output = "LIST"
+                outformat = args.outformat
+                if outformat is None:
+                        outformat = "TEXT"
+                selections = args.selections
+                if selections is None:
+                        sel = Invocation.Selection.dopr(output, outformat)
+                        selections = [sel]
+                search = Search(queries, openonly, publiconly, orders)
+                op = dosearch(search, selections)
+                ops.append(op)
+        # endif
 
-	return Invocation(ops)
+        return Invocation(ops)
 # end getargs
 
 ############################################################