#!/usr/bin/env python # Microscopic modification and query tool for overrides in projectb # Copyright (C) 2004 Daniel Silverstone # $Id: alicia,v 1.6 2004-11-27 17:58:13 troup Exp $ # This program is free software; you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation; either version 2 of the License, or # (at your option) any later version. # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # You should have received a copy of the GNU General Public License # along with this program; if not, write to the Free Software # Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA ################################################################################ ## So line up your soldiers and she'll shoot them all down ## Coz Alisha Rules The World ## You think you found a dream, then it shatters and it seems, ## That Alisha Rules The World ################################################################################ import pg, sys; import utils, db_access; import apt_pkg, logging; ################################################################################ Cnf = None; projectB = None; ################################################################################ # Shamelessly stolen from melanie. Should probably end up in utils.py def game_over(): answer = utils.our_raw_input("Continue (y/N)? ").lower(); if answer != "y": print "Aborted." sys.exit(1); def usage (exit_code=0): print """Usage: alicia [OPTIONS] package [section] [priority] Make microchanges or microqueries of the overrides -h, --help show this help and exit -d, --done=BUG# send priority/section change as closure to bug# -n, --no-action don't do anything -s, --suite specify the suite to use """ sys.exit(exit_code) def main (): global Cnf, projectB; Cnf = utils.get_conf() Arguments = [('h',"help","Alicia::Options::Help"), ('d',"done","Alicia::Options::Done", "HasArg"), ('n',"no-action","Alicia::Options::No-Action"), ('s',"suite","Alicia::Options::Suite", "HasArg"), ]; for i in ["help", "no-action"]: if not Cnf.has_key("Alicia::Options::%s" % (i)): Cnf["Alicia::Options::%s" % (i)] = ""; if not Cnf.has_key("Alicia::Options::Suite"): Cnf["Alicia::Options::Suite"] = "unstable"; arguments = apt_pkg.ParseCommandLine(Cnf,Arguments,sys.argv); Options = Cnf.SubTree("Alicia::Options") if Options["Help"]: usage(); projectB = pg.connect(Cnf["DB::Name"], Cnf["DB::Host"], int(Cnf["DB::Port"])); db_access.init(Cnf, projectB); if not arguments: utils.fubar("package name is a required argument."); package = arguments.pop(0); suite = Options["Suite"] if arguments and len(arguments) > 2: utils.fubar("Too many arguments"); if arguments and len(arguments) == 1: # Determine if the argument is a priority or a section... arg = arguments.pop(); q = projectB.query(""" SELECT ( SELECT COUNT(*) FROM section WHERE section=%s ) AS secs, ( SELECT COUNT(*) FROM priority WHERE priority=%s ) AS prios """ % ( pg._quote(arg,"str"), pg._quote(arg,"str"))); r = q.getresult(); if r[0][0] == 1: arguments = (arg,"."); elif r[0][1] == 1: arguments = (".",arg); else: utils.fubar("%s is not a valid section or priority" % (arg)); # Retrieve current section/priority... q = projectB.query(""" SELECT priority.priority AS prio, section.section AS sect FROM override, priority, section, suite WHERE override.priority = priority.id AND override.section = section.id AND override.package = %s AND override.suite = suite.id AND suite.suite_name = %s """ % (pg._quote(package,"str"), pg._quote(suite,"str"))); if q.ntuples() == 0: utils.fubar("Unable to find package %s" % (package)); if q.ntuples() > 1: utils.fubar("%s is ambiguous. Matches %d packages" % (package,q.ntuples())); r = q.getresult(); oldsection = r[0][1]; oldpriority = r[0][0]; if not arguments: print "%s is in section '%s' at priority '%s'" % ( package,oldsection,oldpriority); sys.exit(0); # At this point, we have a new section and priority... check they're valid... newsection, newpriority = arguments; if newsection == ".": newsection = oldsection; if newpriority == ".": newpriority = oldpriority; q = projectB.query("SELECT id FROM section WHERE section=%s" % ( pg._quote(newsection,"str"))); if q.ntuples() == 0: utils.fubar("Supplied section %s is invalid" % (newsection)); newsecid = q.getresult()[0][0]; q = projectB.query("SELECT id FROM priority WHERE priority=%s" % ( pg._quote(newpriority,"str"))); if q.ntuples() == 0: utils.fubar("Supplied priority %s is invalid" % (newpriority)); newprioid = q.getresult()[0][0]; if newpriority == oldpriority and newsection == oldsection: print "I: Doing nothing" sys.exit(0); # If we're in no-action mode if Options["No-Action"]: if newpriority != oldpriority: print "I: Would change priority from %s to %s" % (oldpriority,newpriority); if newsection != oldsection: print "I: Would change section from %s to %s" % (oldsection,newsection); if Options.has_key("Done"): print "I: Would also close bug(s): %s" % (Options["Done"]); sys.exit(0); if newpriority != oldpriority: print "I: Will change priority from %s to %s" % (oldpriority,newpriority); if newsection != oldsection: print "I: Will change section from %s to %s" % (oldsection,newsection); if not Options.has_key("Done"): pass; #utils.warn("No bugs to close have been specified. Noone will know you have done this."); else: print "I: Will close bug(s): %s" % (Options["Done"]); game_over(); Logger = logging.Logger(Cnf, "alicia"); projectB.query("BEGIN WORK"); # We're in "do it" mode, we have something to do... do it if newpriority != oldpriority: q = projectB.query(""" UPDATE override SET priority=%d WHERE package=%s AND suite = (SELECT id FROM suite WHERE suite_name=%s)""" % ( newprioid, pg._quote(package,"str"), pg._quote(suite,"str") )); Logger.log(["changed priority",package,oldpriority,newpriority]); if newsection != oldsection: q = projectB.query(""" UPDATE override SET section=%d WHERE package=%s AND suite = (SELECT id FROM suite WHERE suite_name=%s)""" % ( newsecid, pg._quote(package,"str"), pg._quote(suite,"str") )); Logger.log(["changed priority",package,oldsection,newsection]); projectB.query("COMMIT WORK"); if Options.has_key("Done"): Subst = {}; Subst["__ALICIA_ADDRESS__"] = Cnf["Alicia::MyEmailAddress"]; Subst["__BUG_SERVER__"] = Cnf["Dinstall::BugServer"]; bcc = []; if Cnf.Find("Dinstall::Bcc") != "": bcc.append(Cnf["Dinstall::Bcc"]); if Cnf.Find("Alicia::Bcc") != "": bcc.append(Cnf["Alicia::Bcc"]); if bcc: Subst["__BCC__"] = "Bcc: " + ", ".join(bcc); else: Subst["__BCC__"] = "X-Filler: 42"; Subst["__CC__"] = "X-Katie: alicia $Revision: 1.6 $"; Subst["__ADMIN_ADDRESS__"] = Cnf["Dinstall::MyAdminAddress"]; Subst["__DISTRO__"] = Cnf["Dinstall::MyDistribution"]; Subst["__WHOAMI__"] = utils.whoami(); summary = "Concerning package %s...\n" % (package); summary += "Operating on the %s suite\n" % (suite); if newpriority != oldpriority: summary += "Changed priority from %s to %s\n" % (oldpriority,newpriority); if newsection != oldsection: summary += "Changed section from %s to %s\n" % (oldsection,newsection); Subst["__SUMMARY__"] = summary; for bug in utils.split_args(Options["Done"]): Subst["__BUG_NUMBER__"] = bug; mail_message = utils.TemplateSubst( Subst,Cnf["Dir::Templates"]+"/alicia.bug-close"); utils.send_mail(mail_message); Logger.log(["closed bug",bug]); Logger.close(); print "Done"; ################################################################################# if __name__ == '__main__': main()