root / pykota / trunk / bin / pkprinters @ 2724

Revision 2724, 14.6 kB (checked in by jerome, 18 years ago)

Skip lines between entries while listing them.

  • Property svn:eol-style set to native
  • Property svn:executable set to *
  • Property svn:keywords set to Author Date Id Revision
Line 
1#! /usr/bin/env python
2# -*- coding: ISO-8859-15 -*-
3
4# PyKota Printers Manager
5#
6# PyKota - Print Quotas for CUPS and LPRng
7#
8# (c) 2003, 2004, 2005, 2006 Jerome Alet <alet@librelogiciel.com>
9# This program is free software; you can redistribute it and/or modify
10# it under the terms of the GNU General Public License as published by
11# the Free Software Foundation; either version 2 of the License, or
12# (at your option) any later version.
13#
14# This program is distributed in the hope that it will be useful,
15# but WITHOUT ANY WARRANTY; without even the implied warranty of
16# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
17# GNU General Public License for more details.
18#
19# You should have received a copy of the GNU General Public License
20# along with this program; if not, write to the Free Software
21# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
22#
23# $Id$
24#
25#
26
27import os
28import sys
29import pwd
30
31from pykota.tool import PyKotaTool, PyKotaToolError, PyKotaCommandLineError, crashed, N_
32
33__doc__ = N_("""pkprinters v%(__version__)s (c) %(__years__)s %(__author__)s
34
35A Printers Manager for PyKota.
36
37command line usage :
38
39  pkprinters [options] printer1 printer2 printer3 ... printerN
40
41options :
42
43  -v | --version       Prints pkprinters's version number then exits.
44  -h | --help          Prints this message then exits.
45 
46  -a | --add           Adds printers if they don't exist on the Quota
47                       Storage Server. If they exist, they are modified
48                       unless -s|--skipexisting is also used.
49                       
50  -d | --delete        Deletes printers from the quota storage.
51 
52  -D | --description d Adds a textual description to printers.
53                       
54  -c | --charge p[,j]  Sets the price per page and per job to charge.
55                       Job price is optional.
56                       If both are to be set, separate them with a comma.
57                       Floating point and negative values are allowed.
58 
59  -g | --groups pg1[,pg2...] Adds or Remove the printer(s) to the printer
60                       groups pg1, pg2, etc... which must already exist.
61                       A printer group is just like a normal printer,
62                       only that it is usually unknown from the printing
63                       system. Create printer groups exactly the same
64                       way that you create printers, then add other
65                       printers to them with this option.
66                       Accounting is done on a printer and on all
67                       the printer groups it belongs to, quota checking
68                       is done on a printer and on all the printer groups
69                       it belongs to.
70                       If the --remove option below is not used, the
71                       default action is to add printers to the specified
72                       printer groups.
73                       
74  -l | --list          List informations about the printer(s) and the
75                       printers groups it is a member of.
76                       
77  -r | --remove        In combination with the --groups option above,                       
78                       remove printers from the specified printers groups.
79                       
80  -s | --skipexisting  In combination with the --add option above, tells
81                       pkprinters to not modify existing printers.
82                       
83  -m | --maxjobsize s  Sets the maximum job size allowed on the printer
84                       to s pages.
85                       
86  -p | --passthrough   Activate passthrough mode for the printer. In this
87                       mode, users are allowed to print without any impact
88                       on their quota or account balance.
89                       
90  -n | --nopassthrough Deactivate passthrough mode for the printer.
91                       Without -p or -n, printers are created in
92                       normal mode, i.e. no passthrough.
93 
94  printer1 through printerN can contain wildcards if the --add option
95  is not set.
96 
97examples :                             
98
99  $ pkprinters --add -D "HP Printer" --charge 0.05,0.1 hp2100 hp2200 hp8000
100 
101  Will create three printers named hp2100, hp2200 and hp8000.
102  Their price per page will be set at 0.05 unit, and their price
103  per job will be set at 0.1 unit. Units are in your own currency,
104  or whatever you want them to mean.
105  All of their descriptions will be set to the string "HP Printer".
106  If any of these printers already exists, it will also be modified
107  unless the -s|--skipexisting command line option is also used.
108           
109  $ pkprinters --delete "*"
110 
111  This will completely delete all printers and associated quota information,
112  as well as their job history. USE WITH CARE !
113 
114  $ pkprinters --groups Laser,HP "hp*"
115 
116  This will put all printers which name matches "hp*" into printers groups
117  Laser and HP, which MUST already exist.
118 
119  $ pkprinters --groups LexMark --remove hp2200
120 
121  This will remove the hp2200 printer from the LexMark printer group.
122""")
123       
124class PKPrinters(PyKotaTool) :       
125    """A class for a printers manager."""
126    def main(self, names, options) :
127        """Manage printers."""
128        if (not self.config.isAdmin) and (not options["list"]) :
129            raise PyKotaCommandLineError, "%s : %s" % (pwd.getpwuid(os.geteuid())[0], _("You're not allowed to use this command."))
130           
131        if options["delete"] :   
132            self.display("%s...\n" % _("Deletion"))
133            todelete = self.storage.getMatchingPrinters(",".join(names))
134            nbtotal = len(todelete)
135            for i in range(nbtotal) :
136                entry = todelete[i]
137                if entry.Exists :
138                    entry.delete()
139                percent = 100.0 * float(i) / float(nbtotal)
140                self.display("\r%.02f%%" % percent)
141        else :
142            if options["add"] :   
143                self.display("%s...\n" % _("Creation"))
144                printers = []
145                nbtotal = len(names)
146                for i in range(nbtotal) :
147                    pname = names[i]
148                    printer = self.storage.getPrinter(pname)
149                    if printer.Exists :
150                        if options["skipexisting"] :
151                            self.printInfo(_("Printer %s already exists, skipping.") % printer.Name)
152                        else :   
153                            self.printInfo(_("Printer %s already exists, will be modified.") % printer.Name)
154                            printers.append(printer)
155                    else :
156                        if self.isValidName(pname) :
157                            printer = self.storage.addPrinter(pname)
158                            if not printer.Exists :
159                                raise PyKotaToolError, _("Impossible to add printer %s") % pname
160                            else :   
161                                printers.append(printer)
162                        else :   
163                            raise PyKotaCommandLineError, _("Invalid printer name %s") % pname
164                    percent = 100.0 * float(i) / float(nbtotal)
165                    self.display("\r%.02f%%" % percent)
166                self.display("\r100.00%%\r        \r%s\n" % _("Done."))
167            else :       
168                if not names :
169                    names = ["*"]
170                printers = self.storage.getMatchingPrinters(",".join(names))
171                if not printers :
172                    raise PyKotaCommandLineError, _("There's no printer matching %s") % " ".join(names)
173                       
174            if options["list"] :
175                for printer in printers :
176                    parents = ", ".join([p.Name for p in self.storage.getParentPrinters(printer)])
177                    if parents : 
178                        parents = "%s %s" % (_("in"), parents)
179                    print "%s [%s] (%s + #*%s)" % \
180                          (printer.Name, printer.Description, printer.PricePerJob, \
181                           printer.PricePerPage)
182                    print "    %s" % (_("Passthrough mode : %s") % ((printer.PassThrough and _("ON")) or _("OFF")))
183                    print "    %s" % (_("Maximum job size : %s") % ((printer.MaxJobSize and (_("%s pages") % printer.MaxJobSize)) or _("Unlimited")))
184                    if parents :       
185                        print "    %s" % parents
186                    print   
187            else :
188                self.display("%s...\n" % _("Modification"))
189               
190                if options["groups"] :       
191                    printersgroups = self.storage.getMatchingPrinters(options["groups"])
192                    if not printersgroups :
193                        raise PyKotaCommandLineError, _("There's no printer matching %s") % " ".join(options["groups"].split(','))
194                else :         
195                    printersgroups = []
196                       
197                if options["charge"] :
198                    try :
199                        charges = [float(part) for part in options["charge"].split(',', 1)]
200                    except ValueError :   
201                        raise PyKotaCommandLineError, _("Invalid charge amount value %s") % options["charge"]
202                    else :   
203                        if len(charges) > 2 :
204                            charges = charges[:2]
205                        if len(charges) != 2 :
206                            charges = [charges[0], None]
207                        (perpage, perjob) = charges
208                       
209                if options["maxjobsize"] :       
210                    try :
211                        maxjobsize = int(options["maxjobsize"])
212                        if maxjobsize < 0 :
213                            raise ValueError
214                    except ValueError :   
215                        raise PyKotaCommandLineError, _("Invalid maximum job size value %s") % options["maxjobsize"]
216                else :       
217                    maxjobsize = None
218                       
219                description = options["description"]
220                if description :
221                    description = description.strip()
222                   
223                nopassthrough = options["nopassthrough"]   
224                passthrough = options["passthrough"]
225                nbtotal = len(printers)
226                for i in range(nbtotal) :       
227                    printer = printers[i]
228                    if options["charge"] :
229                        printer.setPrices(perpage, perjob)   
230                    if description is not None :        # NB : "" is allowed !
231                        printer.setDescription(description)
232                    if nopassthrough and printer.PassThrough :   
233                        printer.setPassThrough(False)
234                    if passthrough and not printer.PassThrough :   
235                        printer.setPassThrough(True)
236                    if (maxjobsize is not None) and (printer.MaxJobSize != maxjobsize) :   
237                        printer.setMaxJobSize(maxjobsize)
238                    printer.save()   
239                    for pgroup in printersgroups :
240                        if options["remove"] :
241                            pgroup.delPrinterFromGroup(printer)
242                        else :
243                            pgroup.addPrinterToGroup(printer)   
244                    percent = 100.0 * float(i) / float(nbtotal)
245                    self.display("\r%.02f%%" % percent)
246                               
247        if not options["list"] :               
248            self.display("\r100.00%%\r        \r%s\n" % _("Done."))
249                     
250if __name__ == "__main__" : 
251    retcode = 0
252    try :
253        short_options = "hvac:D:dg:lrsnpm:"
254        long_options = ["help", "version", "add", "charge=", "description=", \
255                        "delete", "groups=", "list", "remove", \
256                        "skipexisting", "passthrough", "nopassthrough", \
257                        "maxjobsize="]
258       
259        # Initializes the command line tool
260        manager = PKPrinters(doc=__doc__)
261        manager.deferredInit()
262       
263        # parse and checks the command line
264        (options, args) = manager.parseCommandline(sys.argv[1:], short_options, long_options)
265       
266        # sets long options
267        options["help"] = options["h"] or options["help"]
268        options["version"] = options["v"] or options["version"]
269        options["add"] = options["a"] or options["add"]
270        options["charge"] = options["c"] or options["charge"]
271        options["description"] = options["D"] or options["description"]
272        options["delete"] = options["d"] or options["delete"] 
273        options["groups"] = options["g"] or options["groups"]
274        options["list"] = options["l"] or options["list"]
275        options["remove"] = options["r"] or options["remove"]
276        options["skipexisting"] = options["s"] or options["skipexisting"]
277        options["maxjobsize"] = options["m"] or options["maxjobsize"]
278        options["passthrough"] = options["p"] or options["passthrough"]
279        options["nopassthrough"] = options["n"] or options["nopassthrough"]
280       
281        if options["help"] :
282            manager.display_usage_and_quit()
283        elif options["version"] :
284            manager.display_version_and_quit()
285        elif (options["delete"] and (options["add"] or options["groups"] or options["charge"] or options["remove"] or options["description"])) \
286           or (options["skipexisting"] and not options["add"]) \
287           or (options["list"] and (options["add"] or options["delete"] or options["groups"] or options["charge"] or options["remove"] or options["description"])) \
288           or (options["passthrough"] and options["nopassthrough"]) :
289            raise PyKotaCommandLineError, _("incompatible options, see help.")
290        elif options["remove"] and not options["groups"] :   
291            raise PyKotaCommandLineError, _("You have to pass printer groups names on the command line")
292        elif (not args) and (not options["list"]) :   
293            raise PyKotaCommandLineError, _("You have to pass printer names on the command line")
294        else :
295            retcode = manager.main(args, options)
296    except KeyboardInterrupt :       
297        sys.stderr.write("\nInterrupted with Ctrl+C !\n")
298        retcode = -3
299    except PyKotaCommandLineError, msg :   
300        sys.stderr.write("%s : %s\n" % (sys.argv[0], msg))
301        retcode = -2
302    except SystemExit :       
303        pass
304    except :
305        try :
306            manager.crashed("pkprinters failed")
307        except :   
308            crashed("pkprinters failed")
309        retcode = -1
310
311    try :
312        manager.storage.close()
313    except (TypeError, NameError, AttributeError) :   
314        pass
315       
316    sys.exit(retcode)   
Note: See TracBrowser for help on using the browser.