1
+ import datetime
1
2
import logging
2
3
import math
3
4
import os
4
5
import re
5
6
import subprocess
6
7
import tempfile
7
8
from io import BytesIO
8
- from typing import Dict , Optional
9
+ from typing import Dict , Tuple , Optional
9
10
10
11
import magic
11
12
from django .conf import settings
@@ -62,6 +63,29 @@ def set_user_ratelimit_status(username: str) -> None:
62
63
cache .incr (cache_key )
63
64
64
65
66
+ def parse_alerts (alerts : str ) -> Tuple [str , bool ]:
67
+ known_alerts = {
68
+ "paused" : "unavailable" ,
69
+ "media-empty-error" : "out of paper" ,
70
+ "media-empty-warning" : "out of paper" ,
71
+ "media-jam-error" : "jammed" ,
72
+ "media-jam-warning" : "jammed" ,
73
+ "none" : "working" ,
74
+ }
75
+ alerts = alerts .split ()
76
+ alerts_text = ", " .join (known_alerts .get (alert , "error" ) for alert in alerts )
77
+ error_alerts = ["paused" ]
78
+ broken_alerts = ["media-empty-error" , "media-empty-warning" , "media-jam-error" , "media-jam-warning" ]
79
+ printer_class = "working"
80
+ for alert in alerts :
81
+ if alert in error_alerts or alert not in known_alerts :
82
+ printer_class = "error"
83
+ break
84
+ if alert in broken_alerts :
85
+ printer_class = "broken"
86
+ return alerts_text , printer_class
87
+
88
+
65
89
def get_printers () -> Dict [str , str ]:
66
90
"""Returns a dictionary mapping name:description for available printers.
67
91
@@ -86,8 +110,9 @@ def get_printers() -> Dict[str, str]:
86
110
except (subprocess .CalledProcessError , subprocess .TimeoutExpired ):
87
111
return []
88
112
89
- PRINTER_LINE_RE = re .compile (r"^printer\s+(\w+)\s+(?!disabled) " , re .ASCII )
113
+ PRINTER_LINE_RE = re .compile (r"^printer\s+(\w+)" , re .ASCII )
90
114
DESCRIPTION_LINE_RE = re .compile (r"^\s+Description:\s+(.*)\s*$" , re .ASCII )
115
+ ALERTS_LINE_RE = re .compile (r"^\s+Alerts:\s+(.*)\s*$" , re .ASCII )
91
116
92
117
printers = {}
93
118
last_name = None
@@ -98,19 +123,23 @@ def get_printers() -> Dict[str, str]:
98
123
name = match .group (1 )
99
124
if name != "Please_Select_a_Printer" :
100
125
# By default, use the name of the printer instead of the description
101
- printers [name ] = name
126
+ printers [name ] = [ name ]
102
127
# Record the name of the printer so when we parse the rest of the
103
128
# extended description we know which printer it's referring to.
104
129
last_name = name
105
- elif last_name is not None :
106
- match = DESCRIPTION_LINE_RE .match (line )
107
- if match is not None :
108
- # Pull out the description
109
- description = match .group (1 )
110
- # And make sure we don't set an empty description
111
- if description :
112
- printers [last_name ] = description
113
- last_name = None
130
+ elif last_name is not None :
131
+ description_match = DESCRIPTION_LINE_RE .match (line )
132
+ if description_match is not None :
133
+ # Pull out the description
134
+ description = description_match .group (1 )
135
+ # And make sure we don't set an empty description
136
+ if description :
137
+ printers [last_name ] = [description ]
138
+ alerts_match = ALERTS_LINE_RE .match (line )
139
+ if alerts_match is not None :
140
+ alerts = alerts_match .group (1 )
141
+ printers [last_name ].append (alerts )
142
+ last_name = None
114
143
115
144
cache .set (key , printers , timeout = settings .CACHE_AGE ["printers_list" ])
116
145
return printers
@@ -300,8 +329,11 @@ def html_to_pdf(template_src, filename, context=None):
300
329
301
330
def print_job (obj : PrintJob , do_print : bool = True ):
302
331
printer = obj .printer
303
- if printer not in get_printers ().keys ():
332
+ all_printers = get_printers ()
333
+ if printer not in all_printers :
304
334
raise Exception ("Printer not authorized." )
335
+ if parse_alerts (all_printers [printer ][1 ])[1 ] == "error" :
336
+ raise Exception ("Printer unavailable." )
305
337
306
338
if not obj .file :
307
339
raise InvalidInputPrintingError ("No file given to print." )
@@ -467,6 +499,9 @@ def print_view(request):
467
499
messages .error (request , "You don't have printer access outside of the TJ network." )
468
500
return redirect ("index" )
469
501
502
+ if request .method == "GET" and "refresh" in request .GET and request .user .is_printing_admin :
503
+ cache .delete ("printing:printers" )
504
+
470
505
printers = get_printers ()
471
506
if request .method == "POST" :
472
507
form = PrintJobForm (request .POST , request .FILES , printers = printers )
@@ -494,5 +529,10 @@ def print_view(request):
494
529
)
495
530
else :
496
531
form = PrintJobForm (printers = printers )
497
- context = {"form" : form }
532
+ alerts = {}
533
+ for printer in printers :
534
+ alerts [printer ] = parse_alerts (printers [printer ][1 ])
535
+ elapsed_seconds = settings .CACHE_AGE ["printers_list" ] - cache .ttl ("printing:printers" )
536
+ start_time = datetime .datetime .now () - datetime .timedelta (seconds = elapsed_seconds )
537
+ context = {"form" : form , "alerts" : alerts , "updated_time" : start_time .strftime ("%-I:%M:%S %p" )}
498
538
return render (request , "printing/print.html" , context )
0 commit comments