-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathPrintWeightLabels.py
executable file
·154 lines (115 loc) · 4.48 KB
/
PrintWeightLabels.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
143
144
145
146
147
148
149
150
151
152
153
154
#!/usr/bin/python3
"""
author : berteh, for co-labor
licence : CC-BY-SA
homepage, contributions : https://github.com/berteh/print-weight-barcodes
"""
import serial, re, random, cups, time, yaml, os
from string import Template
from tempfile import NamedTemporaryFile
config = yaml.safe_load(open(os.path.realpath(os.path.dirname(__file__))+"/config.yaml"))
DEFAULT_STATION = config['default CLI station']
DEBUG = config['debug']
def get_config_GUI_details() :
global config
return config['gui stations']
class PrintWeightLabels:
"""
constructor overloading based on number of arguments: either
PrintWeightLabels(scale, printer, label_template_name, clean_print_queue_on_run) or
PrintWeightLabels(station name)
"""
def __init__(self, *args):
global DEBUG, DEFAULT_STATION, config
DEBUG = DEBUG | config['debug']
if len(args) > 1 :
if DEBUG : print(f"configuring new PrintWeightLabel station : {args[0]} > {args[1]}")
self.scale = args[0]
self.printer = args[1]
self.zpl_template = config['templates'][args[2]]['zpl']
self.cleanQueue = args[3]
elif len(args) == 1 :
if DEBUG : print(f"configuring new PrintWeightLabel {args[0]}")
self.printer = config[args[0]]['printer name']
self.scale = config[args[0]]['scale port']
self.zpl_template = config['templates'][config[args[0]]['label template']]['zpl']
self.cleanQueue = config[args[0]]['empty print queue on launch']
else : raise RuntimeError(f"Number of arguments must be 1 (station) or 3 (printer, scale, label), it is {len(args)}")
if self.cleanQueue :
self.purge_printer_queue()
def get_weight_from_scale(self) :
""" Raises Error if cannot read from scale
"""
global DEBUG, config
if DEBUG : print("getting weight from scale")
try :
s = serial.Serial(port=self.scale, baudrate=9600, bytesize=serial.EIGHTBITS, timeout=2, stopbits=serial.STOPBITS_ONE, parity=serial.PARITY_NONE)
b = s.read(config['scale bytes'])
except serial.SerialException as ex :
raise RuntimeError(f"scale configuration/connection problem : could not read weight from {self.scale}")
data = b.decode('ascii', 'backslashreplace') # or use b.decode('cp437') to avoir overhead of replacing unknown characters
if DEBUG : print(f"data: {data}")
if (data == "") : raise RuntimeError(f"Timeout getting weight from scale {self.scale}")
weight = re.search(f"\s{{{config['scale empties']}}}(\S+)\skg\s", data)
return(weight.group(1))
def get_zpl(self, kgs = "0,251") :
global DEBUG, config
if DEBUG : print("preparing label content")
grams = kgs.replace(",","").zfill(5);
t = Template(self.zpl_template)
lzpl = t.safe_substitute(kgs=kgs, grams=grams, cheers=random.choice(config['cheers']))
return(lzpl)
def send_to_printer(self, lzpl) :
""" Raises cups.IPPError if printer not found.
Pools the provided ZPL string if printer queue exists, even if printer is not connected
Returns print job ID
"""
global DEBUG
if DEBUG : print("printing")
f = NamedTemporaryFile(mode='w+b')
try:
conn = cups.Connection()
b = lzpl.encode(encoding = 'utf-16')
f.write(b)
f.seek(0) # rewind needed to allow cups to read file
jobID = conn.printFile(self.printer, f.name, 'Weigh label', {} )
finally :
f.close()
conn = None
return jobID
def purge_printer_queue(self) :
global DEBUG
if DEBUG : print("cleaning printing jobs in queue")
try:
conn = cups.Connection()
conn.cancelAllJobs(self.printer, my_jobs=False, purge_jobs=True)
except cups.IPPError as err:
print(f"Cannot connect with the printer {self.printer}: {err}")
finally :
conn = None
def printer_pending_jobs(self) : # TODO does not check printer in particular as of now, only CUPS connection > useless as-is.
try:
conn = cups.Connection()
open_jobs = conn.getJobs(which_jobs='not-completed', my_jobs=False) #TODO filter only self.printer
except cups.IPPError as err:
print(f"Cannot connect with the printer: {err}")
finally :
conn = None
return(len(open_jobs))
def weight_and_print(self) :
try :
kgs = self.get_weight_from_scale()
except Exception as err:
raise RuntimeError('Scale connection problem') from err
zpl = self.get_zpl(kgs)
try :
self.send_to_printer(zpl)
except Exception as err:
raise RuntimeError('Printer connection problem') from err
if __name__ == "__main__":
if DEBUG : print(f"configuring {DEFAULT_STATION}")
station = PrintWeightLabels(DEFAULT_STATION)
if TEST :
station.test()
else :
station.weight_and_print()