1 | #! /usr/bin/env python |
---|
2 | # -*- coding: ISO-8859-15 -*- |
---|
3 | # |
---|
4 | # PyKota |
---|
5 | # |
---|
6 | # PyKota : Print Quotas for CUPS and LPRng |
---|
7 | # |
---|
8 | # (c) 2003, 2004, 2005, 2006, 2007 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 | |
---|
27 | import sys |
---|
28 | import os |
---|
29 | import stat |
---|
30 | import tempfile |
---|
31 | import pwd |
---|
32 | import grp |
---|
33 | |
---|
34 | nowready = """ |
---|
35 | |
---|
36 | |
---|
37 | PyKota is now ready to run ! |
---|
38 | |
---|
39 | Before printing, you still have to manually modify CUPS' printers.conf |
---|
40 | to manually prepend cupspykota:// in front of each DeviceURI. |
---|
41 | |
---|
42 | Once this is done, just restart CUPS and all should work fine. |
---|
43 | |
---|
44 | Please report any problem to : alet@librelogiciel.com |
---|
45 | |
---|
46 | Thanks in advance. |
---|
47 | """ |
---|
48 | |
---|
49 | pghbaconf = """local\tall\tpostgres\t\tident sameuser |
---|
50 | local\tall\tall\t\tident sameuser |
---|
51 | host\tall\tall\t127.0.0.1\t255.255.255.255\tident sameuser |
---|
52 | host\tall\tall\t::1\tffff:ffff:ffff:ffff:ffff:ffff:ffff:ffff\treject |
---|
53 | host\tall\tall\t::ffff:127.0.0.1/128\treject |
---|
54 | host\tall\tall\t0.0.0.0\t0.0.0.0\treject""" |
---|
55 | |
---|
56 | pykotadminconf = """[global] |
---|
57 | storageadmin: pykotaadmin |
---|
58 | storageadminpw: readwritepw""" |
---|
59 | |
---|
60 | pykotaconf = """# |
---|
61 | # This is a generated configuration file for PyKota |
---|
62 | # |
---|
63 | # IMPORTANT : many more directives can be used, and some of the directives |
---|
64 | # below accept different and/or more complex parameters. Please read |
---|
65 | # /usr/share/pykota/conf/pykota.conf.sample for more details about the |
---|
66 | # numerous possibilities allowed. |
---|
67 | # |
---|
68 | [global] |
---|
69 | |
---|
70 | # Database settings |
---|
71 | storagebackend : pgstorage |
---|
72 | storageserver : %(storageserver)s |
---|
73 | storagename : pykota |
---|
74 | storageuser : pykotauser |
---|
75 | storageuserpw : readonlypw |
---|
76 | storagecaching : No |
---|
77 | disablehistory : No |
---|
78 | |
---|
79 | # Logging method |
---|
80 | logger : system |
---|
81 | |
---|
82 | # Set debug to Yes during installation and testing |
---|
83 | debug : Yes |
---|
84 | |
---|
85 | # Who should receive automatic bug reports ? |
---|
86 | crashrecipient : pykotacrashed@librelogiciel.com |
---|
87 | |
---|
88 | # Should we keep temporary files on disk ? |
---|
89 | # Set this to yes for debugging software accounting problems |
---|
90 | keepfiles : no |
---|
91 | |
---|
92 | # Logos for banners and CGI scripts |
---|
93 | logourl : http://www.pykota.com/pykota.png |
---|
94 | logolink : http://www.pykota.com/ |
---|
95 | |
---|
96 | # SMTP |
---|
97 | smtpserver : %(smtpserver)s |
---|
98 | maildomain : %(dnsdomain)s |
---|
99 | |
---|
100 | # Print Administrator |
---|
101 | admin : %(adminname)s |
---|
102 | adminmail : %(adminemail)s |
---|
103 | |
---|
104 | # Use usernames as-is or convert them to lowercase ? |
---|
105 | utolower : No |
---|
106 | |
---|
107 | # Should we hide some fields in the history (title, filename) ? |
---|
108 | privacy : no |
---|
109 | |
---|
110 | # Should we charge end users when an error occurs ? |
---|
111 | onbackenderror : nocharge |
---|
112 | |
---|
113 | # Default accounting methods : |
---|
114 | preaccounter : software() |
---|
115 | accounter : software() |
---|
116 | onaccountererror : stop |
---|
117 | |
---|
118 | # Who will receive warning messages ? |
---|
119 | # both means admin and user. |
---|
120 | mailto : both |
---|
121 | |
---|
122 | # Grace delay for pages based quotas, works the same |
---|
123 | # as for disk quotas |
---|
124 | gracedelay : 7 |
---|
125 | |
---|
126 | # Configurable zero, to give free credits |
---|
127 | balancezero : 0.0 |
---|
128 | |
---|
129 | # Warning limit for credit based quotas |
---|
130 | poorman : 1.0 |
---|
131 | |
---|
132 | # Warning messages to use |
---|
133 | poorwarn : Your Print Quota account balance is low. |
---|
134 | Soon you'll not be allowed to print anymore. |
---|
135 | |
---|
136 | softwarn : Your Print Quota Soft Limit is reached. |
---|
137 | This means that you may still be allowed to print for some |
---|
138 | time, but you must contact your administrator to purchase |
---|
139 | more print quota. |
---|
140 | |
---|
141 | hardwarn : Your Print Quota Hard Limit is reached. |
---|
142 | This means that you are not allowed to print anymore. |
---|
143 | Please contact your administrator at root@localhost |
---|
144 | as soon as possible to solve the problem. |
---|
145 | |
---|
146 | # Number of banners allowed to be printed by users |
---|
147 | # who are over quota |
---|
148 | maxdenybanners : 0 |
---|
149 | |
---|
150 | # Should we allow users to ever be over quota on their last job ? |
---|
151 | # strict means no. |
---|
152 | enforcement : strict |
---|
153 | |
---|
154 | # Should we trust printers' internal page counter ? |
---|
155 | trustjobsize : yes |
---|
156 | |
---|
157 | # How to handle duplicate jobs |
---|
158 | denyduplicates : no |
---|
159 | duplicatesdelay : 0 |
---|
160 | |
---|
161 | # What should we do when an unknown user prints ? |
---|
162 | # The policy below will automatically create a printing account |
---|
163 | # for unknown users, allowing them to print with no limit on the |
---|
164 | # current printer. |
---|
165 | policy : external(pkusers --add --skipexisting --limitby noquota --description \"Added automatically\" \$PYKOTAUSERNAME && edpykota --add --skipexisting --printer \$PYKOTAPRINTERNAME \$PYKOTAUSERNAME) |
---|
166 | |
---|
167 | """ |
---|
168 | |
---|
169 | |
---|
170 | class PyKotaSetup : |
---|
171 | """Base class for PyKota installers.""" |
---|
172 | backendsdirectory = "/usr/lib/cups/backend" # overload it if needed |
---|
173 | pykotadirectory = "/usr/share/pykota" # overload it if needed |
---|
174 | pgrestart = "/etc/init.d/postgresql* restart" # overload it if needed |
---|
175 | cupsrestart = "/etc/init.d/cupsys restart" # overload it if needed |
---|
176 | adduser = "adduser --system --group --home /etc/pykota --gecos PyKota pykota" # overload it if needed |
---|
177 | packages = [ "wget", |
---|
178 | "bzip2", |
---|
179 | "subversion", |
---|
180 | "postgresql", |
---|
181 | "postgresql-client", |
---|
182 | "cupsys", |
---|
183 | "cupsys-client", |
---|
184 | "python-dev", |
---|
185 | "python-jaxml", |
---|
186 | "python-reportlab", |
---|
187 | "python-reportlab-accel", |
---|
188 | "python-psyco", |
---|
189 | "python-pygresql", |
---|
190 | "python-psyco", |
---|
191 | "python-osd", |
---|
192 | "python-egenix-mxdatetime", |
---|
193 | "python-imaging", |
---|
194 | "python-pysnmp4", |
---|
195 | "python-chardet", |
---|
196 | "python-pam" ] |
---|
197 | |
---|
198 | otherpackages = [ { "name" : "pkpgcounter", |
---|
199 | "version" : "2.17", |
---|
200 | "url" : "http://www.pykota.com/software/%(name)s/download/tarballs/%(name)s-%(version)s.tar.gz", |
---|
201 | "commands" : [ "tar -zxf %(name)s-%(version)s.tar.gz", |
---|
202 | "cd %(name)s-%(version)s", |
---|
203 | "python setup.py install", |
---|
204 | ], |
---|
205 | }, |
---|
206 | { "name" : "pkipplib", |
---|
207 | "version" : "0.07", |
---|
208 | "url" : "http://www.pykota.com/software/%(name)s/download/tarballs/%(name)s-%(version)s.tar.gz", |
---|
209 | "commands" : [ "tar -zxf %(name)s-%(version)s.tar.gz", |
---|
210 | "cd %(name)s-%(version)s", |
---|
211 | "python setup.py install", |
---|
212 | ], |
---|
213 | }, |
---|
214 | { "name" : "ghostpcl", |
---|
215 | "version" : "1.41p1", |
---|
216 | "url" : "ftp://mirror.cs.wisc.edu/pub/mirrors/ghost/AFPL/GhostPCL/%(name)s_%(version)s.tar.bz2", |
---|
217 | "commands" : [ "bunzip2 <%(name)s_%(version)s.tar.bz2 | tar -xf -", |
---|
218 | "cd %(name)s_%(version)s", |
---|
219 | "make fonts", |
---|
220 | "make product", |
---|
221 | "make install", |
---|
222 | ], |
---|
223 | }, |
---|
224 | ] |
---|
225 | |
---|
226 | def __init__(self) : |
---|
227 | """Initializes instance specific datas.""" |
---|
228 | self.launched = [] |
---|
229 | |
---|
230 | def yesno(self, message) : |
---|
231 | """Asks the end user some question and returns the answer.""" |
---|
232 | answer = raw_input("\n%s ? " % message).strip().upper()[0] |
---|
233 | return answer == 'Y' |
---|
234 | |
---|
235 | def confirmCommand(self, message, command, record=True) : |
---|
236 | """Asks for confirmation before a command is launched, and launches it if needed.""" |
---|
237 | if self.yesno("The following command will be launched %(message)s :\n%(command)s\nDo you agree" % locals()) : |
---|
238 | os.system(command) |
---|
239 | if record : |
---|
240 | self.launched.append(command) |
---|
241 | return True |
---|
242 | else : |
---|
243 | return False |
---|
244 | |
---|
245 | def confirmPipe(self, message, command) : |
---|
246 | """Asks for confirmation before a command is launched in a pipe, launches it if needed, and returns the result.""" |
---|
247 | if self.yesno("The following command will be launched %(message)s :\n%(command)s\nDo you agree" % locals()) : |
---|
248 | pipeprocess = os.popen(command, "r") |
---|
249 | result = pipeprocess.read() |
---|
250 | pipeprocess.close() |
---|
251 | return result |
---|
252 | else : |
---|
253 | return False |
---|
254 | |
---|
255 | def listPrinters(self) : |
---|
256 | """Returns a list of tuples (queuename, deviceuri) for all existing print queues.""" |
---|
257 | result = os.popen("lpstat -v", "r") |
---|
258 | lines = result.readlines() |
---|
259 | result.close() |
---|
260 | printers = [] |
---|
261 | for line in lines : |
---|
262 | (begin, end) = line.split(':', 1) |
---|
263 | deviceuri = end.strip() |
---|
264 | queuename = begin.split()[-1] |
---|
265 | printers.append((queuename, deviceuri)) |
---|
266 | return printers |
---|
267 | |
---|
268 | def downloadOtherPackages(self) : |
---|
269 | """Downloads and install additional packages from http://www.pykota.com or other websites""" |
---|
270 | olddirectory = os.getcwd() |
---|
271 | directory = tempfile.mkdtemp() |
---|
272 | print "\nDownloading additional software not available as packages in %(directory)s" % locals() |
---|
273 | os.chdir(directory) |
---|
274 | for package in self.otherpackages : |
---|
275 | name = package["name"] |
---|
276 | version = package["version"] |
---|
277 | url = package["url"] % locals() |
---|
278 | commands = " && ".join(package["commands"]) % locals() |
---|
279 | if url.startswith("svn://") : |
---|
280 | download = 'svn export "%(url)s" %(name)s' % locals() |
---|
281 | else : |
---|
282 | download = 'wget "%(url)s"' % locals() |
---|
283 | if self.confirmCommand("to download %(name)s" % locals(), download) : |
---|
284 | self.confirmCommand("to install %(name)s" % locals(), commands) |
---|
285 | self.confirmCommand("to remove the temporary directory %(directory)s" % locals(), |
---|
286 | "rm -fr %(directory)s" % locals(), |
---|
287 | record=False) |
---|
288 | os.chdir(olddirectory) |
---|
289 | |
---|
290 | def waitPrintersOnline(self) : |
---|
291 | """Asks the admin to switch all printers ON.""" |
---|
292 | while not self.yesno("First you MUST switch ALL your printers ON. Are ALL your printers ON") : |
---|
293 | pass |
---|
294 | |
---|
295 | def setupDatabase(self) : |
---|
296 | """Creates the database.""" |
---|
297 | pykotadirectory = self.pykotadirectory |
---|
298 | self.confirmCommand("to create PyKota's database in PostgreSQL", 'su - postgres -c "psql -f %(pykotadirectory)s/postgresql/pykota-postgresql.sql template1"' % locals()) |
---|
299 | |
---|
300 | def configurePostgreSQL(self) : |
---|
301 | """Configures PostgreSQL for PyKota to work.""" |
---|
302 | pgconffiles = self.confirmPipe("to find PostgreSQL's configuration files", "find /etc -name postgresql.conf 2>/dev/null") |
---|
303 | if pgconffiles is not False : |
---|
304 | pgconffiles = [part.strip() for part in pgconffiles.split()] |
---|
305 | pgconfdirs = [os.path.split(pgconffile)[0] for pgconffile in pgconffiles] |
---|
306 | for i in range(len(pgconfdirs)) : |
---|
307 | pgconfdir = pgconfdirs[i] |
---|
308 | pgconffile = pgconffiles[i] |
---|
309 | if (len(pgconfdirs) == 1) or self.yesno("Do PostgreSQL configuration files reside in %(pgconfdir)s" % locals()) : |
---|
310 | answer = self.confirmPipe("to see if PostgreSQL accepts TCP/IP connections", "grep ^tcpip_socket %(pgconffile)s" % locals()) |
---|
311 | conflines = pghbaconf.split("\n") |
---|
312 | if answer is not False : |
---|
313 | tcpip = answer.strip().lower().endswith("true") |
---|
314 | else : |
---|
315 | tcpip = False |
---|
316 | if tcpip : |
---|
317 | conflines.insert(2, "host\tpykota\tpykotaadmin,pykotauser\t127.0.0.1\t255.255.255.255\tmd5") |
---|
318 | else : |
---|
319 | conflines.insert(1, "local\tpykota\tpykotaadmin,pykotauser\t\tmd5") |
---|
320 | conf = "\n".join(conflines) |
---|
321 | self.confirmCommand("to configure PostgreSQL correctly for PyKota", 'echo "%(conf)s" >%(pgconfdir)s/pg_hba.conf' % locals()) |
---|
322 | self.confirmCommand("to make PostgreSQL take the changes into account", self.pgrestart) |
---|
323 | return tcpip |
---|
324 | return None |
---|
325 | |
---|
326 | def genConfig(self, adminname, adminemail, dnsdomain, smtpserver, home, tcpip) : |
---|
327 | """Generates minimal configuration files for PyKota.""" |
---|
328 | if tcpip : |
---|
329 | storageserver = "localhost" |
---|
330 | else : |
---|
331 | storageserver = "" |
---|
332 | conf = pykotaconf % locals() |
---|
333 | self.confirmCommand("to generate PyKota's main configuration file", 'echo "%(conf)s" >%(home)s/pykota.conf' % locals()) |
---|
334 | conf = pykotadminconf % locals() |
---|
335 | self.confirmCommand("to generate PyKota's administrators configuration file", 'echo "%(conf)s" >%(home)s/pykotadmin.conf' % locals()) |
---|
336 | self.confirmCommand("to change permissions on PyKota's administrators configuration file", "chmod 640 %(home)s/pykotadmin.conf" % locals()) |
---|
337 | self.confirmCommand("to change permissions on PyKota's main configuration file", "chmod 644 %(home)s/pykota.conf" % locals()) |
---|
338 | self.confirmCommand("to change ownership of PyKota's configuration files", "chown pykota.pykota %(home)s/pykota.conf %(home)s/pykotadmin.conf" % locals()) |
---|
339 | answer = self.confirmPipe("to automatically detect the best settings for your printers", "pkturnkey --doconf 2>/dev/null") |
---|
340 | if answer is not False : |
---|
341 | lines = answer.split("\n") |
---|
342 | begin = end = None |
---|
343 | for i in range(len(lines)) : |
---|
344 | line = lines[i] |
---|
345 | if line.strip().startswith("--- CUT ---") : |
---|
346 | if begin is None : |
---|
347 | begin = i |
---|
348 | else : |
---|
349 | end = i |
---|
350 | |
---|
351 | if (begin is not None) and (end is not None) : |
---|
352 | suffix = "\n".join(lines[begin+1:end]) |
---|
353 | self.confirmCommand("to improve PyKota's configuration wrt your existing printers", 'echo "%(suffix)s" >>%(home)s/pykota.conf' % locals()) |
---|
354 | |
---|
355 | def addPyKotaUser(self) : |
---|
356 | """Adds a system user named pykota, returns its home directory or None""" |
---|
357 | try : |
---|
358 | user = pwd.getpwnam("pykota") |
---|
359 | except KeyError : |
---|
360 | if self.confirmCommand("to create a system user named 'pykota'", self.adduser) : |
---|
361 | try : |
---|
362 | return pwd.getpwnam("pykota")[5] |
---|
363 | except KeyError : |
---|
364 | return None |
---|
365 | else : |
---|
366 | return None |
---|
367 | else : |
---|
368 | return user[5] |
---|
369 | |
---|
370 | def setupBackend(self) : |
---|
371 | """Installs the cupspykota backend.""" |
---|
372 | backendsdirectory = self.backendsdirectory |
---|
373 | pykotadirectory = self.pykotadirectory |
---|
374 | self.confirmCommand("to make PyKota known to CUPS", "ln -s %(pykotadirectory)s/cupspykota %(backendsdirectory)s/cupspykota" % locals()) |
---|
375 | self.confirmCommand("to restart CUPS for the changes to take effect", self.cupsrestart) |
---|
376 | |
---|
377 | def managePrinters(self, printers) : |
---|
378 | """For each printer, asks if it should be managed with PyKota or not.""" |
---|
379 | for (queuename, deviceuri) in printers : |
---|
380 | command = 'pkprinters --add --cups --description "Printer created with pksetup" "%(queuename)s"' % locals() |
---|
381 | self.confirmCommand("to import the %(queuename)s print queue into PyKota's database and reroute it through PyKota" % locals(), command) |
---|
382 | |
---|
383 | def installPyKotaFiles(self) : |
---|
384 | """Installs PyKota files through Python's Distutils mechanism.""" |
---|
385 | pksetupdir = os.path.split(os.path.abspath(sys.argv[0]))[0] |
---|
386 | pykotadir = os.path.abspath(os.path.join(pksetupdir, "..")) |
---|
387 | setuppy = os.path.join(pykotadir, "setup.py") |
---|
388 | if os.path.exists(setuppy) : |
---|
389 | self.confirmCommand("to install PyKota files on your system", "python %(setuppy)s install" % locals()) |
---|
390 | |
---|
391 | def setup(self) : |
---|
392 | """Installation procedure.""" |
---|
393 | self.installPyKotaFiles() |
---|
394 | self.waitPrintersOnline() |
---|
395 | adminname = raw_input("What is the name of the print administrator ? ").strip() |
---|
396 | adminemail = raw_input("What is the email address of the print administrator ? ").strip() |
---|
397 | dnsdomain = raw_input("What is your DNS domain name ? ").strip() |
---|
398 | smtpserver = raw_input("What is the hostname or IP address of your SMTP server ? ").strip() |
---|
399 | homedirectory = self.addPyKotaUser() |
---|
400 | if homedirectory is None : |
---|
401 | sys.stderr.write("Installation can't proceed. You MUST create a system user named 'pykota'.\n") |
---|
402 | else : |
---|
403 | self.upgradeSystem() |
---|
404 | self.setupPackages() |
---|
405 | self.downloadOtherPackages() |
---|
406 | tcpip = self.configurePostgreSQL() |
---|
407 | self.genConfig(adminname, adminemail, dnsdomain, smtpserver, homedirectory, tcpip) |
---|
408 | self.setupDatabase() |
---|
409 | self.setupBackend() |
---|
410 | self.managePrinters(self.listPrinters()) |
---|
411 | print nowready |
---|
412 | print "The script %s can be used to reinstall in unattended mode.\n" % self.genInstaller() |
---|
413 | |
---|
414 | def genInstaller(self) : |
---|
415 | """Generates an installer script.""" |
---|
416 | scriptname = "/tmp/pykota-installer.sh" |
---|
417 | commands = [ "#! /bin/sh", |
---|
418 | "#", |
---|
419 | "# PyKota installer script.", |
---|
420 | "#", |
---|
421 | "# This script was automatically generated.", |
---|
422 | "#", |
---|
423 | ] + self.launched |
---|
424 | script = open(scriptname, "w") |
---|
425 | script.write("\n".join(commands)) |
---|
426 | script.close() |
---|
427 | os.chmod(scriptname, \ |
---|
428 | stat.S_IRWXU | stat.S_IRGRP | stat.S_IXGRP | stat.S_IROTH | stat.S_IXOTH) |
---|
429 | return scriptname |
---|
430 | |
---|
431 | |
---|
432 | class Debian(PyKotaSetup) : |
---|
433 | """Class for Debian installer.""" |
---|
434 | def setupPackages(self) : |
---|
435 | """Installs missing Debian packages.""" |
---|
436 | self.confirmCommand("to install missing dependencies", "apt-get install %s" % " ".join(self.packages)) |
---|
437 | |
---|
438 | def upgradeSystem(self) : |
---|
439 | """Upgrades the Debian setup.""" |
---|
440 | if self.confirmCommand("to grab an up-to-date list of available packages", "apt-get update") : |
---|
441 | self.confirmCommand("to put your system up-to-date", "apt-get -y dist-upgrade") |
---|
442 | |
---|
443 | class Ubuntu(Debian) : |
---|
444 | """Class for Ubuntu installer.""" |
---|
445 | pass |
---|
446 | |
---|
447 | if __name__ == "__main__" : |
---|
448 | retcode = 0 |
---|
449 | if (len(sys.argv) != 2) or (sys.argv[1] == "-h") or (sys.argv[1] == "--help") : |
---|
450 | print "pksetup v0.1 (c) 2003-2007 Jerome Alet - alet@librelogiciel.com\n\nusage : pksetup distribution\n\ne.g. : pksetup debian\n\nIMPORTANT : only Debian and Ubuntu are currently supported." |
---|
451 | elif (sys.argv[1] == "-v") or (sys.argv[1] == "--version") : |
---|
452 | print "0.1" # pksetup's own version number |
---|
453 | else : |
---|
454 | classname = sys.argv[1].strip().title() |
---|
455 | try : |
---|
456 | installer = globals()[classname]() |
---|
457 | except KeyError : |
---|
458 | sys.stderr.write("There's currently no support for the %s distribution, sorry.\n" % sys.argv[1]) |
---|
459 | retcode = -1 |
---|
460 | else : |
---|
461 | try : |
---|
462 | retcode = installer.setup() |
---|
463 | except KeyboardInterrupt : |
---|
464 | sys.stderr.write("\n\n\nWARNING : Setup was aborted at user's request !\n\n") |
---|
465 | retcode = -1 |
---|
466 | sys.exit(retcode) |
---|