-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathinstall_printers.py
executable file
·142 lines (125 loc) · 6.28 KB
/
install_printers.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
#!/usr/bin/python
import subprocess
import sys
import os
import re
from install_printers_config import printers
predicate_installer = os.path.join(os.path.dirname(os.path.abspath(__file__)), 'predicate_installer.py') # in this directory
if not os.path.exists(predicate_installer):
predicate_installer = os.path.join(os.path.dirname(os.path.dirname(os.path.abspath(__file__))), 'predicate_installer.py') # in parent directory
# restart CUPS and make sure it's running
subprocess.call(['/bin/launchctl', 'unload', '/System/Library/LaunchDaemons/org.cups.cupsd.plist'])
subprocess.call(['/bin/launchctl', 'load', '/System/Library/LaunchDaemons/org.cups.cupsd.plist'])
subprocess.check_call(['/bin/launchctl', 'start', 'org.cups.cupsd'])
errors = 0
for printer in printers:
lpadminflag = '-m'
try:
if 'ppd' in printer and not os.path.exists(printer['ppd']) and 'predicate' in printer:
# mark printer as connected
subprocess.check_call(['/usr/libexec/PlistBuddy', '-c', "Add InstalledPrinters: string " + printer['model'], '/Library/Printers/InstalledPrinters.plist'])
# install printer driver
subprocess.call(['/usr/bin/python', predicate_installer, printer['predicate']])
elif 'ppd' in printer and 'download' in printer:
from distutils.version import StrictVersion
try:
if not os.path.exists(printer['ppd']):
raise Exception()
installed_version = subprocess.check_output(['/usr/bin/zgrep', '^*FileVersion:', printer['ppd']]).split(':')[1].strip().replace('"','').replace("'",'')
except:
installed_version = '0.0'
if not 'version' in printer:
printer['version'] = '0.0'
if not os.path.exists(printer['ppd']) or StrictVersion(printer['version']) > StrictVersion(installed_version):
dmg = os.path.join(os.path.dirname(os.path.abspath(__file__)), os.path.basename(printer['download']))
subprocess.check_call(['/usr/bin/curl', '-L', '-o', dmg, printer['download']])
mount = subprocess.check_output(['/usr/bin/hdiutil', 'attach', '-plist', dmg]).split('\n')
mountpoint = None
for i in range(len(mount)):
if 'mount-point' in mount[i]:
mountpoint = mount[i+1].strip().replace('<string>','').replace('</string>','')
break
if mountpoint is None:
raise Exception('Could not mount DMG')
pkgpath = os.path.join(mountpoint, printer['pkgpath'])
print "Installing %s" % pkgpath
subprocess.check_call(['/usr/sbin/installer', '-package', pkgpath, '-target', '/'])
subprocess.check_call(['/usr/bin/hdiutil', 'detach', mountpoint])
elif 'ppd' not in printer and 'download' in printer and printer['download'].endswith('.ppd'):
lpadminflag = '-P'
printer['ppd'] = os.path.join('/Library/Printers/PPDs', os.path.basename(printer['download']))
print "Downloading %s" % os.path.basename(printer['ppd'])
subprocess.check_call(['/usr/bin/curl', '-L', '-o', printer['ppd'], printer['download']])
if 'download_icon' in printer:
# extract the icon path from the PPD
icon_path = subprocess.check_output(['/usr/bin/grep', '^*APPrinterIconPath: ', printer['ppd']])
icon_path = re.sub('\*APPrinterIconPath: "(.*?)"\n', r'\1', icon_path)
icon_dir = os.path.dirname(icon_path)
if not os.path.exists(icon_dir):
os.makedirs(icon_dir)
print "Downloading %s to %s" % (os.path.basename(printer['download_icon']), icon_path)
subprocess.check_call(['/usr/bin/curl', '-L', '-o', icon_path, printer['download_icon']])
print "Converting to ICNS"
subprocess.check_call(['/usr/bin/sips', '-s', 'format', 'icns', icon_path, '--out', icon_path])
if not os.path.exists(printer['ppd']):
raise Exception('PPD not found')
if printer['ppd'].startswith('/usr/share/cups/model'):
printer['ppd'] = os.path.basename(printer['ppd']) # relative path required
# delete print queue if it already exists
DEVNULL = open(os.devnull, 'w')
if subprocess.call(['/usr/bin/lpstat', '-p', printer['name']], stdout=DEVNULL, stderr=DEVNULL) == 0: # printer exists
subprocess.call(['/usr/sbin/lpadmin', '-x', printer['name']])
DEVNULL.close()
# add print queue
subprocess.check_call(['/usr/sbin/lpadmin', '-p', printer['name'], '-v', printer['uri'], lpadminflag, printer['ppd'], '-L', printer['location'], '-E', '-o', 'printer-is-shared=false'])
# set printer options
if 'options' in printer and len(printer['options']) > 0:
optionstrings = []
for k,v in printer['options'].iteritems():
optionstrings.append('-o')
optionstrings.append('%s=%s' % (k,v))
subprocess.check_call(['/usr/sbin/lpadmin', '-p', printer['name']] + optionstrings)
except Exception as e:
print "Could not add printer %s: %s" % (printer['name'], e)
errors += 1
subprocess.check_call(['/bin/launchctl', 'unload', '/System/Library/LaunchDaemons/org.cups.cupsd.plist'])
with open('/etc/cups/printers.conf') as f:
pr = f.readlines()
for printer in printers:
cache_dirty = False
ppd_path = os.path.join('/etc/cups/ppd', printer['name'] + '.ppd')
if 'remote' in printer and printer['remote'] == True:
section = False
for i,line in enumerate(pr):
if line.startswith('<Printer ' + printer['name'] + '>'):
section = True
elif line.startswith('</Printer>'):
section = False
# set CUPS_PRINTER_REMOTE in Type
if section and line.startswith('Type '):
pr[i] = 'Type %d\n' % (int(line[5:-1]) | 2)
# process the job on the server instead of on the client
f_ppd = open(ppd_path, 'a')
f_ppd.write('\n\n')
for mimetype in ["application/pdf", "image/*", "application/postscript", "application/vnd.cups-postscript", "application/vnd.cups-command"]:
f_ppd.write('*cupsFilter: "%s 0 -"\n' % mimetype)
f_ppd.close()
cache_dirty = True
if 'editppd' in printer:
import re
f_ppd = open(ppd_path, 'r+')
ppd = f_ppd.read()
for o,n in printer['editppd'].iteritems():
ppd = re.sub(o,n, ppd)
f_ppd.seek(0)
f_ppd.write(ppd)
f_ppd.close()
cache_dirty = True
ppdcache = '/var/spool/cups/cache/' + printer['name'] + '.data'
if cache_dirty and os.path.exists(ppdcache): # need to delete the PPD cache. OS X 10.8 appears to have done that automatically, but 10.9 keeps using the old cupsFilter list if we don't do this
os.unlink(ppdcache)
with open('/etc/cups/printers.conf', 'w') as f:
f.writelines(pr)
subprocess.check_call(['/bin/launchctl', 'load', '/System/Library/LaunchDaemons/org.cups.cupsd.plist'])
if errors != 0:
sys.exit(1)