Restructure the code and implement a printing queue #29
54
src/main.py
54
src/main.py
@@ -23,7 +23,6 @@ import sys
|
|||||||
import os # For VARS from the shell.
|
import os # For VARS from the shell.
|
||||||
import pprint # To pretty print JSON
|
import pprint # To pretty print JSON
|
||||||
import toml # Used for the config file parsing
|
import toml # Used for the config file parsing
|
||||||
import threading
|
|
||||||
from flask import (
|
from flask import (
|
||||||
Flask,
|
Flask,
|
||||||
request,
|
request,
|
||||||
@@ -43,7 +42,6 @@ from web import Web # Wrapper for the web routes and API
|
|||||||
from print_queue import PrintQueue
|
from print_queue import PrintQueue
|
||||||
from worker import PrintWorker
|
from worker import PrintWorker
|
||||||
|
|
||||||
|
|
||||||
# We create the main Flask object
|
# We create the main Flask object
|
||||||
app = Flask(__name__)
|
app = Flask(__name__)
|
||||||
socketio = SocketIO(app, cors_allowed_origins="*")
|
socketio = SocketIO(app, cors_allowed_origins="*")
|
||||||
@@ -87,7 +85,7 @@ except FileExistsError:
|
|||||||
app.logger.debug("Directory %s already exists.", UPLOAD_FOLDER)
|
app.logger.debug("Directory %s already exists.", UPLOAD_FOLDER)
|
||||||
except PermissionError:
|
except PermissionError:
|
||||||
app.logger.error("Permission denied: Unable to create %s", UPLOAD_FOLDER)
|
app.logger.error("Permission denied: Unable to create %s", UPLOAD_FOLDER)
|
||||||
exit(77)
|
sys.exit(77)
|
||||||
|
|
||||||
# Output the config file
|
# Output the config file
|
||||||
if os.getenv("FLASK_DEBUG"):
|
if os.getenv("FLASK_DEBUG"):
|
||||||
@@ -137,6 +135,7 @@ limiter = Limiter(
|
|||||||
|
|
||||||
# General routes
|
# General routes
|
||||||
|
|
||||||
|
|
||||||
@app.route("/")
|
@app.route("/")
|
||||||
@limiter.limit("1/second", override_defaults=False)
|
@limiter.limit("1/second", override_defaults=False)
|
||||||
def index():
|
def index():
|
||||||
@@ -152,8 +151,10 @@ def webcam():
|
|||||||
app.logger.debug("Loading webcam interface")
|
app.logger.debug("Loading webcam interface")
|
||||||
return render_template("webcam.html")
|
return render_template("webcam.html")
|
||||||
|
|
||||||
|
|
||||||
# Form treatement
|
# Form treatement
|
||||||
|
|
||||||
|
|
||||||
@app.route("/web/print/sms", methods=["POST"])
|
@app.route("/web/print/sms", methods=["POST"])
|
||||||
@limiter.limit("6/minute", override_defaults=False)
|
@limiter.limit("6/minute", override_defaults=False)
|
||||||
def web_print_sms():
|
def web_print_sms():
|
||||||
@@ -164,7 +165,7 @@ def web_print_sms():
|
|||||||
txt = request.form["txt"]
|
txt = request.form["txt"]
|
||||||
except werkzeug.exceptions.BadRequestKeyError as e:
|
except werkzeug.exceptions.BadRequestKeyError as e:
|
||||||
app.logger.error("Whoops, we are missing the txt input field. : %s ", str(e))
|
app.logger.error("Whoops, we are missing the txt input field. : %s ", str(e))
|
||||||
flash("Whoops, no forms submitted or missing signature : " + str(e), 'error')
|
flash("Whoops, no forms submitted or missing signature : " + str(e), "error")
|
||||||
return redirect(url_for("index"))
|
return redirect(url_for("index"))
|
||||||
|
|
||||||
try:
|
try:
|
||||||
@@ -172,23 +173,23 @@ def web_print_sms():
|
|||||||
sign = request.form["signature"]
|
sign = request.form["signature"]
|
||||||
except werkzeug.exceptions.BadRequestKeyError as e:
|
except werkzeug.exceptions.BadRequestKeyError as e:
|
||||||
app.logger.warning(
|
app.logger.warning(
|
||||||
"No signature found for this print, using default signature.", str(e)
|
"No signature found for this print, using default signature : %s ", str(e)
|
||||||
)
|
)
|
||||||
sign = configuration_file["defaults"]["signature"]
|
sign = configuration_file["defaults"]["signature"]
|
||||||
|
|
||||||
try:
|
try:
|
||||||
web.print_sms(txt, sign)
|
web.print_sms(txt, sign)
|
||||||
except Exception as e:
|
except RuntimeError as e:
|
||||||
app.logger.error("Whoops, we could not print an SMS because : %s ", str(e))
|
app.logger.error("Whoops, we could not print an SMS because : %s ", str(e))
|
||||||
flash("Whoops, we could not print an SMS because :" + str(e), 'error')
|
flash("Whoops, we could not print an SMS because :" + str(e), "error")
|
||||||
return redirect(url_for("index"))
|
return redirect(url_for("index"))
|
||||||
|
|
||||||
# end try
|
# end try
|
||||||
flash("The SMS has been printed !", 'info')
|
flash("The SMS has been printed !", "info")
|
||||||
return redirect(url_for("index"))
|
return redirect(url_for("index"))
|
||||||
|
|
||||||
|
|
||||||
@app.route("/web/print/img", methods=["POST"])
|
@app.route("/web/print/img", methods=["POST"])
|
||||||
@limiter.limit("1/second", override_defaults=False)
|
@limiter.limit("1/second", override_defaults=False)
|
||||||
def web_print_img():
|
def web_print_img():
|
||||||
"""Prints an image on a printer"""
|
"""Prints an image on a printer"""
|
||||||
@@ -199,7 +200,7 @@ def web_print_img():
|
|||||||
sign = request.form["signature"]
|
sign = request.form["signature"]
|
||||||
except werkzeug.exceptions.BadRequestKeyError as e:
|
except werkzeug.exceptions.BadRequestKeyError as e:
|
||||||
app.logger.warning(
|
app.logger.warning(
|
||||||
"No signature found for this print, using default signature.", str(e)
|
"No signature found for this print, using default signature : %s", str(e)
|
||||||
)
|
)
|
||||||
sign = configuration_file["defaults"]["signature"]
|
sign = configuration_file["defaults"]["signature"]
|
||||||
|
|
||||||
@@ -207,7 +208,7 @@ def web_print_img():
|
|||||||
if "img" not in request.files:
|
if "img" not in request.files:
|
||||||
app.logger.error("Whoops, no images submitted : %s ", str(e))
|
app.logger.error("Whoops, no images submitted : %s ", str(e))
|
||||||
app.logger.error("Error getting the files : %s", str(e))
|
app.logger.error("Error getting the files : %s", str(e))
|
||||||
flash("Whoops, no images submitted : " + str(e), 'error')
|
flash("Whoops, no images submitted : " + str(e), "error")
|
||||||
return redirect(url_for("index"))
|
return redirect(url_for("index"))
|
||||||
|
|
||||||
file = request.files["img"]
|
file = request.files["img"]
|
||||||
@@ -215,20 +216,21 @@ def web_print_img():
|
|||||||
# empty file without a filename.
|
# empty file without a filename.
|
||||||
if file.filename == "":
|
if file.filename == "":
|
||||||
app.logger.error("Submitted file has no filename !")
|
app.logger.error("Submitted file has no filename !")
|
||||||
flash("Submitted file has no filename !", 'error')
|
flash("Submitted file has no filename !", "error")
|
||||||
return redirect(url_for("index"))
|
return redirect(url_for("index"))
|
||||||
|
|
||||||
try:
|
try:
|
||||||
app.logger.debug("Sending the image to the printer.")
|
app.logger.debug("Sending the image to the printer.")
|
||||||
web.print_image(file, sign)
|
web.print_image(file, sign)
|
||||||
except Exception as e:
|
except RuntimeError as e:
|
||||||
app.logger.error("The image could not be printed because : %s ", str(e))
|
app.logger.error("The image could not be printed because : %s ", str(e))
|
||||||
flash("The image could not be printed because : " + str(e), 'error')
|
flash("The image could not be printed because : " + str(e), "error")
|
||||||
return redirect(url_for("index"))
|
return redirect(url_for("index"))
|
||||||
|
|
||||||
flash("Picture printed !", 'info')
|
flash("Picture printed !", "info")
|
||||||
return redirect(url_for("index"))
|
return redirect(url_for("index"))
|
||||||
|
|
||||||
|
|
||||||
# API routes
|
# API routes
|
||||||
# The api has the following methods
|
# The api has the following methods
|
||||||
# api/print/{sms,img,letter,qr,barcode}
|
# api/print/{sms,img,letter,qr,barcode}
|
||||||
@@ -267,11 +269,12 @@ def api_print_sms():
|
|||||||
try:
|
try:
|
||||||
# comment: We try to print the SMS
|
# comment: We try to print the SMS
|
||||||
web.print_sms(txt, sign)
|
web.print_sms(txt, sign)
|
||||||
except Exception as e:
|
except RuntimeError as e:
|
||||||
return str(e), 500
|
return str(e), 500
|
||||||
# end try
|
# end try
|
||||||
return "OK", 200
|
return "OK", 200
|
||||||
|
|
||||||
|
|
||||||
@app.route("/api/print/img", methods=["POST"])
|
@app.route("/api/print/img", methods=["POST"])
|
||||||
@limiter.limit("6/minute", override_defaults=False)
|
@limiter.limit("6/minute", override_defaults=False)
|
||||||
def api_print_image():
|
def api_print_image():
|
||||||
@@ -302,38 +305,43 @@ def api_print_image():
|
|||||||
try:
|
try:
|
||||||
app.logger.debug("Sending the image to the printer.")
|
app.logger.debug("Sending the image to the printer.")
|
||||||
web.print_image(file, sign)
|
web.print_image(file, sign)
|
||||||
except Exception as e:
|
except RuntimeError as e:
|
||||||
return str(e), 500
|
return str(e), 500
|
||||||
|
|
||||||
return "OK", 200
|
return "OK", 200
|
||||||
|
|
||||||
|
|
||||||
@app.route("/api/camera/picture", methods=["GET"])
|
@app.route("/api/camera/picture", methods=["GET"])
|
||||||
def camera_picture():
|
def camera_picture():
|
||||||
"""Returns a picture taken by the camera on a raspberry pi"""
|
"""Returns a picture taken by the camera on a raspberry pi"""
|
||||||
if RASPBERRY_PI_CONNECTED:
|
if RASPBERRY_PI_CONNECTED:
|
||||||
try:
|
try:
|
||||||
return rpi.camera_picture()
|
return rpi.camera_picture()
|
||||||
except Exception as e:
|
except RuntimeError as e:
|
||||||
return jsonify({"message": "Error getting the stream : " + e}), 500
|
return jsonify({"message": "Error getting the stream : " + e}), 500
|
||||||
else:
|
else:
|
||||||
return jsonify({"message": "No camera present"}), 500
|
return jsonify({"message": "No camera present"}), 500
|
||||||
|
|
||||||
@app.route('/api/queue', methods=["GET"])
|
|
||||||
|
@app.route("/api/queue", methods=["GET"])
|
||||||
def api_queue_status():
|
def api_queue_status():
|
||||||
"""API endpoint for entire queue"""
|
"""API endpoint for entire queue"""
|
||||||
return jsonify(web.get_queue_state())
|
return jsonify(web.get_queue_state())
|
||||||
|
|
||||||
@app.route('/api/worker', methods=["GET"])
|
|
||||||
|
@app.route("/api/worker", methods=["GET"])
|
||||||
def api_worker_state():
|
def api_worker_state():
|
||||||
"""API endpoint to get the worker state"""
|
"""API endpoint to get the worker state"""
|
||||||
return jsonify(worker.current_state())
|
return jsonify(worker.current_state())
|
||||||
|
|
||||||
@app.route('/api/worker/start')
|
|
||||||
|
@app.route("/api/worker/start")
|
||||||
def api_worker_start():
|
def api_worker_start():
|
||||||
worker.start_worker()
|
worker.start_worker()
|
||||||
return jsonify(worker.current_state())
|
return jsonify(worker.current_state())
|
||||||
|
|
||||||
@app.route('/api/worker/stop')
|
|
||||||
|
@app.route("/api/worker/stop")
|
||||||
def api_worker_stop():
|
def api_worker_stop():
|
||||||
worker.stop_worker()
|
worker.stop_worker()
|
||||||
return jsonify(worker.current_state())
|
return jsonify(worker.current_state())
|
||||||
@@ -341,6 +349,7 @@ def api_worker_stop():
|
|||||||
|
|
||||||
## Authentification
|
## Authentification
|
||||||
|
|
||||||
|
|
||||||
@app.route("/login")
|
@app.route("/login")
|
||||||
@limiter.limit("1/second", override_defaults=False)
|
@limiter.limit("1/second", override_defaults=False)
|
||||||
def login_page():
|
def login_page():
|
||||||
@@ -393,5 +402,6 @@ def camera_status():
|
|||||||
else:
|
else:
|
||||||
socketio.emit("camera_status", False)
|
socketio.emit("camera_status", False)
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
app.run(debug=True, use_reloader=False, host="0.0.0.0", ssl_context="adhoc")
|
app.run(debug=True, use_reloader=False, host="0.0.0.0", ssl_context="adhoc")
|
||||||
|
|||||||
@@ -1,6 +1,8 @@
|
|||||||
# This class has the method by which we manage the Tasks
|
"""
|
||||||
# It's a printing queue, so we need to add, remove and get information on where
|
This class has the method by which we manage the Tasks
|
||||||
# the queue is
|
It's a printing queue, so we need to add, remove and get information on where
|
||||||
|
the queue is
|
||||||
|
"""
|
||||||
|
|
||||||
from collections import deque
|
from collections import deque
|
||||||
|
|
||||||
@@ -9,12 +11,14 @@ from collections import deque
|
|||||||
import threading
|
import threading
|
||||||
|
|
||||||
from datetime import datetime
|
from datetime import datetime
|
||||||
from task import TaskType, CutTask
|
from task import TaskType
|
||||||
|
|
||||||
|
|
||||||
class PrintQueue:
|
class PrintQueue:
|
||||||
"""
|
"""
|
||||||
A Double-ended Queue to manage the printing Tasks
|
A Double-ended Queue to manage the printing Tasks
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(self, app):
|
def __init__(self, app):
|
||||||
self.app = app
|
self.app = app
|
||||||
self._queue = deque()
|
self._queue = deque()
|
||||||
@@ -34,7 +38,11 @@ class PrintQueue:
|
|||||||
self._queue.append(task)
|
self._queue.append(task)
|
||||||
position = self._queue.index(task)
|
position = self._queue.index(task)
|
||||||
# We return the current position of the task if it was added
|
# We return the current position of the task if it was added
|
||||||
self.app.logger.debug("Added a new task %s to the queue at position %s", task.task_id, position)
|
self.app.logger.debug(
|
||||||
|
"Added a new task %s to the queue at position %s",
|
||||||
|
task.task_id,
|
||||||
|
position,
|
||||||
|
)
|
||||||
return position
|
return position
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
self.app.logger.error("Could not add a task to the queue : %s ", e)
|
self.app.logger.error("Could not add a task to the queue : %s ", e)
|
||||||
@@ -85,32 +93,32 @@ class PrintQueue:
|
|||||||
return {
|
return {
|
||||||
"task_id": task_id,
|
"task_id": task_id,
|
||||||
"status": task.status,
|
"status": task.status,
|
||||||
"type" : task.task_type,
|
"type": task.task_type,
|
||||||
"position": index,
|
"position": index,
|
||||||
"in_queue": True,
|
"in_queue": True,
|
||||||
"content" : task.content,
|
"content": task.content,
|
||||||
"signature": task.signature
|
"signature": task.signature,
|
||||||
}
|
}
|
||||||
|
|
||||||
if task.task_type == TaskType.TEXT:
|
if task.task_type == TaskType.TEXT:
|
||||||
return {
|
return {
|
||||||
"task_id": task_id,
|
"task_id": task_id,
|
||||||
"status": task.status,
|
"status": task.status,
|
||||||
"type" : task.task_type,
|
"type": task.task_type,
|
||||||
"position": index,
|
"position": index,
|
||||||
"in_queue": True,
|
"in_queue": True,
|
||||||
"image_path" : str(task.image_path),
|
"image_path": str(task.image_path),
|
||||||
"signature" : task.signature,
|
"signature": task.signature,
|
||||||
"process" : str(task.process)
|
"process": str(task.process),
|
||||||
}
|
}
|
||||||
|
|
||||||
if task.task_type == TaskType.CUT:
|
if task.task_type == TaskType.CUT:
|
||||||
return {
|
return {
|
||||||
"task_id": task_id,
|
"task_id": task_id,
|
||||||
"status": task.status,
|
"status": task.status,
|
||||||
"type" : task.task_type,
|
"type": task.task_type,
|
||||||
"position": index,
|
"position": index,
|
||||||
"in_queue": True
|
"in_queue": True,
|
||||||
}
|
}
|
||||||
|
|
||||||
return None
|
return None
|
||||||
@@ -123,5 +131,5 @@ class PrintQueue:
|
|||||||
"status": task_status,
|
"status": task_status,
|
||||||
"position": None,
|
"position": None,
|
||||||
"in_queue": False,
|
"in_queue": False,
|
||||||
"completed_at": datetime.now().isoformat()
|
"completed_at": datetime.now().isoformat(),
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,10 +1,14 @@
|
|||||||
|
|
||||||
|
# import brother_ql
|
||||||
|
from time import sleep
|
||||||
|
import os.path
|
||||||
|
|
||||||
|
from PIL import Image, ImageEnhance
|
||||||
|
import numpy as np
|
||||||
|
|
||||||
# Importing the module to manage the connection to the printer.
|
# Importing the module to manage the connection to the printer.
|
||||||
import escpos.printer
|
import escpos.printer
|
||||||
# import brother_ql
|
|
||||||
from time import sleep, gmtime, strftime
|
|
||||||
import os.path
|
|
||||||
from PIL import Image, ImageEnhance, ImageOps
|
|
||||||
import numpy as np
|
|
||||||
|
|
||||||
class Printer(object):
|
class Printer(object):
|
||||||
"""
|
"""
|
||||||
@@ -75,10 +79,14 @@ class Printer(object):
|
|||||||
try:
|
try:
|
||||||
# This also calls open(), which we need to close()
|
# This also calls open(), which we need to close()
|
||||||
# or else the device will appear as busy.
|
# or else the device will appear as busy.
|
||||||
p = escpos.printer.Usb(self.device_id, self.vendor_id, 0, profile="TM-P80")
|
p = escpos.printer.Usb(
|
||||||
|
self.device_id, self.vendor_id, 0, profile="TM-P80"
|
||||||
|
)
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
self.app.logger.error(
|
self.app.logger.error(
|
||||||
"The USB device is not plugged in, trying again %s : %s",waiting_elapsed, str(e)
|
"The USB device is not plugged in, trying again %s : %s",
|
||||||
|
waiting_elapsed,
|
||||||
|
str(e),
|
||||||
)
|
)
|
||||||
pass
|
pass
|
||||||
|
|
||||||
@@ -87,7 +95,10 @@ class Printer(object):
|
|||||||
self.ready = True
|
self.ready = True
|
||||||
self.app.logger.debug("Printer online !")
|
self.app.logger.debug("Printer online !")
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
self.app.logger.error("Error while getting the printer online %s : %s",waiting_elapsed, str(e)
|
self.app.logger.error(
|
||||||
|
"Error while getting the printer online %s : %s",
|
||||||
|
waiting_elapsed,
|
||||||
|
str(e),
|
||||||
)
|
)
|
||||||
pass
|
pass
|
||||||
|
|
||||||
@@ -128,8 +139,10 @@ class Printer(object):
|
|||||||
|
|
||||||
def _print_sms(self, msg, signature="", bold=False):
|
def _print_sms(self, msg, signature="", bold=False):
|
||||||
|
|
||||||
if not isinstance(msg,str):
|
if not isinstance(msg, str):
|
||||||
self.app.logger.error("It is not possible to print a " + str(type(msg)) + ", only strings.")
|
self.app.logger.error(
|
||||||
|
"It is not possible to print a " + str(type(msg)) + ", only strings."
|
||||||
|
)
|
||||||
raise ValueError
|
raise ValueError
|
||||||
|
|
||||||
# We make sure that the signature is not something too goofy
|
# We make sure that the signature is not something too goofy
|
||||||
@@ -168,7 +181,9 @@ class Printer(object):
|
|||||||
self.printer.close()
|
self.printer.close()
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
self.app.logger.error("Unable to print because : " + str(e))
|
self.app.logger.error("Unable to print because : " + str(e))
|
||||||
raise RuntimeError("Unable to print a SMS, the printer couldn't do it.") from e
|
raise RuntimeError(
|
||||||
|
"Unable to print a SMS, the printer couldn't do it."
|
||||||
|
) from e
|
||||||
|
|
||||||
self.app.logger.info("Printed text")
|
self.app.logger.info("Printed text")
|
||||||
return True
|
return True
|
||||||
@@ -201,7 +216,9 @@ class Printer(object):
|
|||||||
self.app.logger.debug("Proccessing the image")
|
self.app.logger.debug("Proccessing the image")
|
||||||
path = _process_image(self, path)
|
path = _process_image(self, path)
|
||||||
except RuntimeError as e:
|
except RuntimeError as e:
|
||||||
self.app.logger.error("Error while processing the image, aborting print : %s",str(e))
|
self.app.logger.error(
|
||||||
|
"Error while processing the image, aborting print : %s", str(e)
|
||||||
|
)
|
||||||
raise e
|
raise e
|
||||||
else:
|
else:
|
||||||
self.app.logger.warning("Not proccessing the image")
|
self.app.logger.warning("Not proccessing the image")
|
||||||
@@ -226,7 +243,9 @@ class Printer(object):
|
|||||||
try:
|
try:
|
||||||
self.printer.close()
|
self.printer.close()
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
self.app.logger.error("Could not close the printer connexion %s", str(e))
|
self.app.logger.error(
|
||||||
|
"Could not close the printer connexion %s", str(e)
|
||||||
|
)
|
||||||
raise RuntimeError("Could not close the printer connexion. ") from e
|
raise RuntimeError("Could not close the printer connexion. ") from e
|
||||||
|
|
||||||
self.app.logger.info("Printed a picture")
|
self.app.logger.info("Printed a picture")
|
||||||
@@ -261,11 +280,13 @@ class Printer(object):
|
|||||||
def print_task(self, task_type, data):
|
def print_task(self, task_type, data):
|
||||||
"""Execute actual print based on task type"""
|
"""Execute actual print based on task type"""
|
||||||
match (task_type.value):
|
match (task_type.value):
|
||||||
case ("text"):
|
case "text":
|
||||||
self._print_sms(data["txt"],signature=data["sign"])
|
self._print_sms(data["txt"], signature=data["sign"])
|
||||||
case ("image"):
|
case "image":
|
||||||
self._print_img(data["img"], signature=data["sign"],process=data["process"])
|
self._print_img(
|
||||||
case ("cut"):
|
data["img"], signature=data["sign"], process=data["process"]
|
||||||
|
)
|
||||||
|
case "cut":
|
||||||
self._cut()
|
self._cut()
|
||||||
case _:
|
case _:
|
||||||
raise RuntimeError("This task type is not supported")
|
raise RuntimeError("This task type is not supported")
|
||||||
@@ -324,6 +345,7 @@ def _process_image(self, path):
|
|||||||
|
|
||||||
return jpeg_path
|
return jpeg_path
|
||||||
|
|
||||||
|
|
||||||
def discover_printers():
|
def discover_printers():
|
||||||
"""
|
"""
|
||||||
We try to find all the connected printers ( 0 or n ) to this system.
|
We try to find all the connected printers ( 0 or n ) to this system.
|
||||||
@@ -336,8 +358,8 @@ def discover_printers():
|
|||||||
04f9 Brother Industries, Ltd
|
04f9 Brother Industries, Ltd
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def find_and_parse_borther_ql_printer():
|
|
||||||
|
|
||||||
|
def find_and_parse_borther_ql_printer():
|
||||||
|
|
||||||
## We might be able to no use this because there is a `discover` command in https://github.com/pklaus/brother_ql#usage
|
## We might be able to no use this because there is a `discover` command in https://github.com/pklaus/brother_ql#usage
|
||||||
|
|
||||||
@@ -352,18 +374,18 @@ def find_and_parse_borther_ql_printer():
|
|||||||
|
|
||||||
for backend_name in ["pyusb", "linux_kernel"]:
|
for backend_name in ["pyusb", "linux_kernel"]:
|
||||||
try:
|
try:
|
||||||
#print(f"Trying backend: {backend_name}")
|
# print(f"Trying backend: {backend_name}")
|
||||||
backend = backend_factory(backend_name)
|
backend = backend_factory(backend_name)
|
||||||
available_devices = backend["list_available_devices"]()
|
available_devices = backend["list_available_devices"]()
|
||||||
#print(f"Found {len(available_devices)} devices with {backend_name} backend")
|
# print(f"Found {len(available_devices)} devices with {backend_name} backend")
|
||||||
|
|
||||||
for printer in available_devices:
|
for printer in available_devices:
|
||||||
#print(f"Found device: {printer}")
|
# print(f"Found device: {printer}")
|
||||||
identifier = printer["identifier"]
|
identifier = printer["identifier"]
|
||||||
parts = identifier.split("/")
|
parts = identifier.split("/")
|
||||||
|
|
||||||
if len(parts) < 4:
|
if len(parts) < 4:
|
||||||
#print(f"Skipping device with invalid identifier format: {identifier}")
|
# print(f"Skipping device with invalid identifier format: {identifier}")
|
||||||
continue
|
continue
|
||||||
|
|
||||||
protocol = parts[0]
|
protocol = parts[0]
|
||||||
@@ -373,7 +395,7 @@ def find_and_parse_borther_ql_printer():
|
|||||||
try:
|
try:
|
||||||
vendor_id, product_id = device_info.split(":")
|
vendor_id, product_id = device_info.split(":")
|
||||||
except ValueError:
|
except ValueError:
|
||||||
#print(f"Invalid device info format: {device_info}")
|
# print(f"Invalid device info format: {device_info}")
|
||||||
continue
|
continue
|
||||||
|
|
||||||
# Default model
|
# Default model
|
||||||
@@ -386,9 +408,9 @@ def find_and_parse_borther_ql_printer():
|
|||||||
if m.product_id == product_id_int:
|
if m.product_id == product_id_int:
|
||||||
model = m.identifier
|
model = m.identifier
|
||||||
break
|
break
|
||||||
#print(f"Matched printer model: {model}")
|
# print(f"Matched printer model: {model}")
|
||||||
except ValueError:
|
except ValueError:
|
||||||
#print(f"Invalid product ID format: {product_id}")
|
# print(f"Invalid product ID format: {product_id}")
|
||||||
continue
|
continue
|
||||||
|
|
||||||
printer_info = {
|
printer_info = {
|
||||||
@@ -400,15 +422,16 @@ def find_and_parse_borther_ql_printer():
|
|||||||
"product_id": product_id,
|
"product_id": product_id,
|
||||||
"serial_number": serial_number,
|
"serial_number": serial_number,
|
||||||
}
|
}
|
||||||
#print(f"Found printer: {printer_info}")
|
# print(f"Found printer: {printer_info}")
|
||||||
return printer_info
|
return printer_info
|
||||||
|
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
#print(f"Error with backend {backend_name}: {str(e)}")
|
# print(f"Error with backend {backend_name}: {str(e)}")
|
||||||
continue
|
continue
|
||||||
|
|
||||||
print("No Brother QL printer found")
|
print("No Brother QL printer found")
|
||||||
return None
|
return None
|
||||||
|
|
||||||
|
|
||||||
def fint_and_parse_epson_printer():
|
def fint_and_parse_epson_printer():
|
||||||
pass
|
pass
|
||||||
|
|||||||
@@ -1,13 +1,12 @@
|
|||||||
from flask_socketio import SocketIO
|
|
||||||
from gpiozero import Button, LED, DigitalOutputDevice
|
|
||||||
from time import sleep, gmtime, strftime
|
|
||||||
from PIL import Image
|
|
||||||
import io # To check if we are on a Raspberry Pi
|
import io # To check if we are on a Raspberry Pi
|
||||||
import subprocess
|
import subprocess
|
||||||
import os
|
import os
|
||||||
|
from time import sleep, gmtime, strftime
|
||||||
|
from flask_socketio import SocketIO
|
||||||
|
from gpiozero import Button, LED, DigitalOutputDevice
|
||||||
|
from PIL import Image
|
||||||
|
|
||||||
|
class Raspberry():
|
||||||
class Raspberry(object):
|
|
||||||
"""
|
"""
|
||||||
This class will manage three things :
|
This class will manage three things :
|
||||||
- Connecting to a USB webcam
|
- Connecting to a USB webcam
|
||||||
|
|||||||
47
src/task.py
47
src/task.py
@@ -1,27 +1,33 @@
|
|||||||
# Here we define the types of tasks
|
"""
|
||||||
# We are using Abstract Base Classes,
|
Here we define the types of tasks
|
||||||
# like this we can define types of tasks ( text, images, ... )
|
We are using Abstract Base Classes,
|
||||||
# that all work with the same basic options
|
like this we can define types of tasks ( text, images, ... )
|
||||||
|
that all work with the same basic options
|
||||||
|
|
||||||
# The tasks are going to be injected into a Queue.
|
The tasks are going to be injected into a Queue.
|
||||||
# It's a usefull way of storing information in our
|
It's a usefull way of storing information in our
|
||||||
# program, while making sure that things are indeed printed.
|
program, while making sure that things are indeed printed.
|
||||||
# It's also a way to prevent two concurrent connexions creating
|
It's also a way to prevent two concurrent connexions creating
|
||||||
# a access conflict on a single printer, like two people wanting
|
a access conflict on a single printer, like two people wanting
|
||||||
# to print at the same time.
|
to print at the same time.
|
||||||
|
|
||||||
# We can also delay and store printing tasks until a printer becomes
|
|
||||||
# available if none is online.
|
|
||||||
|
|
||||||
|
We can also delay and store printing tasks until a printer becomes
|
||||||
|
available if none is online.
|
||||||
|
"""
|
||||||
from abc import ABC, abstractmethod
|
from abc import ABC, abstractmethod
|
||||||
|
|
||||||
## See https://docs.python.org/3/library/abc.html to learn more about this
|
## See https://docs.python.org/3/library/abc.html to learn more about this
|
||||||
|
|
||||||
# from dataclasses import dataclass
|
# from dataclasses import dataclass
|
||||||
from enum import Enum
|
from enum import Enum
|
||||||
import uuid
|
import uuid
|
||||||
|
|
||||||
|
|
||||||
## You can expand this if you want to take other types of tasks into account
|
## You can expand this if you want to take other types of tasks into account
|
||||||
class TaskType(Enum):
|
class TaskType(Enum):
|
||||||
|
"""
|
||||||
|
The different tasks supported by the printers
|
||||||
|
"""
|
||||||
TEXT = "text"
|
TEXT = "text"
|
||||||
IMAGE = "image"
|
IMAGE = "image"
|
||||||
CUT = "cut"
|
CUT = "cut"
|
||||||
@@ -31,6 +37,7 @@ class PrintTask(ABC):
|
|||||||
"""
|
"""
|
||||||
A print task holds information about what we are looking to print.
|
A print task holds information about what we are looking to print.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(self, task_type):
|
def __init__(self, task_type):
|
||||||
self.task_id = self._generate_id()
|
self.task_id = self._generate_id()
|
||||||
self.task_type = task_type
|
self.task_type = task_type
|
||||||
@@ -41,10 +48,10 @@ class PrintTask(ABC):
|
|||||||
@abstractmethod
|
@abstractmethod
|
||||||
def get_print_data(self):
|
def get_print_data(self):
|
||||||
"""Return data formatted for printer"""
|
"""Return data formatted for printer"""
|
||||||
pass
|
|
||||||
|
|
||||||
def _generate_id(self):
|
def _generate_id(self):
|
||||||
# Generate unique task ID
|
# Generate unique task ID
|
||||||
return str(uuid.uuid4())
|
return str(uuid.uuid4())
|
||||||
|
|
||||||
|
|
||||||
@@ -52,18 +59,21 @@ class TextTask(PrintTask):
|
|||||||
"""
|
"""
|
||||||
This tasks represents a texte content, and it's signature.
|
This tasks represents a texte content, and it's signature.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(self, content, signature):
|
def __init__(self, content, signature):
|
||||||
super().__init__(TaskType.TEXT)
|
super().__init__(TaskType.TEXT)
|
||||||
self.content = content
|
self.content = content
|
||||||
self.signature = signature
|
self.signature = signature
|
||||||
|
|
||||||
def get_print_data(self):
|
def get_print_data(self):
|
||||||
return { "txt": self.content, "sign": self.signature }
|
return {"txt": self.content, "sign": self.signature}
|
||||||
|
|
||||||
|
|
||||||
class ImageTask(PrintTask):
|
class ImageTask(PrintTask):
|
||||||
"""
|
"""
|
||||||
This tasks represents a image content ( in the form of it's path ), and it's signature.
|
This tasks represents a image content ( in the form of it's path ), and it's signature.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(self, image_path, signature, process):
|
def __init__(self, image_path, signature, process):
|
||||||
super().__init__(TaskType.IMAGE)
|
super().__init__(TaskType.IMAGE)
|
||||||
self.image_path = image_path
|
self.image_path = image_path
|
||||||
@@ -72,7 +82,8 @@ class ImageTask(PrintTask):
|
|||||||
|
|
||||||
def get_print_data(self):
|
def get_print_data(self):
|
||||||
# Return image data in printer-compatible format
|
# Return image data in printer-compatible format
|
||||||
return { "img": self.image_path, "sign": self.signature, "process" : self.process }
|
return {"img": self.image_path, "sign": self.signature, "process": self.process}
|
||||||
|
|
||||||
|
|
||||||
class CutTask(PrintTask):
|
class CutTask(PrintTask):
|
||||||
"""
|
"""
|
||||||
@@ -85,4 +96,4 @@ class CutTask(PrintTask):
|
|||||||
# There is no print data,
|
# There is no print data,
|
||||||
# the task existence in itself is indication of what to do
|
# the task existence in itself is indication of what to do
|
||||||
def get_print_data(self):
|
def get_print_data(self):
|
||||||
return None
|
return None
|
||||||
|
|||||||
70
src/user.py
70
src/user.py
@@ -1,41 +1,41 @@
|
|||||||
class User(object):
|
# class User(object):
|
||||||
"""docstring for User."""
|
# """docstring for User."""
|
||||||
|
|
||||||
def __init__(self, arg):
|
# def __init__(self, arg):
|
||||||
super(User, self).__init__()
|
# super(User, self).__init__()
|
||||||
self.arg = arg
|
# self.arg = arg
|
||||||
|
|
||||||
|
|
||||||
# @app.route('/login', methods=['POST','GET'])
|
# # @app.route('/login', methods=['POST','GET'])
|
||||||
# @limiter.limit("100 per minute", error_message=error_handler_limiter)
|
# # @limiter.limit("100 per minute", error_message=error_handler_limiter)
|
||||||
def login():
|
# def login():
|
||||||
if request.method == "POST":
|
# if request.method == "POST":
|
||||||
if not session.get("logged_in"):
|
# if not session.get("logged_in"):
|
||||||
if request.form["username"] and request.form["password"]:
|
# if request.form["username"] and request.form["password"]:
|
||||||
# Get the json
|
# # Get the json
|
||||||
with open("users.json") as f:
|
# with open("users.json") as f:
|
||||||
users_file = json.load(f)
|
# users_file = json.load(f)
|
||||||
for user in users_file["users"]:
|
# for user in users_file["users"]:
|
||||||
if users_file["users"][user] == request.form["password"]:
|
# if users_file["users"][user] == request.form["password"]:
|
||||||
session["logged_in"] = True
|
# session["logged_in"] = True
|
||||||
session["user"] = request.form["username"]
|
# session["user"] = request.form["username"]
|
||||||
|
|
||||||
if not session.get("logged_in"):
|
# if not session.get("logged_in"):
|
||||||
flash("Mot de passe ou pseudo invalide.", "danger")
|
# flash("Mot de passe ou pseudo invalide.", "danger")
|
||||||
return redirect(url_for("login"))
|
# return redirect(url_for("login"))
|
||||||
else:
|
# else:
|
||||||
return redirect(url_for("display_index_page"))
|
# return redirect(url_for("display_index_page"))
|
||||||
else:
|
# else:
|
||||||
flash("Incorrect logins")
|
# flash("Incorrect logins")
|
||||||
return render_template("password.html")
|
# return render_template("password.html")
|
||||||
else:
|
# else:
|
||||||
return render_template("password.html")
|
# return render_template("password.html")
|
||||||
else:
|
# else:
|
||||||
return render_template("password.html")
|
# return render_template("password.html")
|
||||||
|
|
||||||
|
|
||||||
@app.route("/logout")
|
# @app.route("/logout")
|
||||||
def logout():
|
# def logout():
|
||||||
session["logged_in"] = False
|
# session["logged_in"] = False
|
||||||
flash("Tu est déconnecté", "info")
|
# flash("Tu est déconnecté", "info")
|
||||||
return redirect(url_for("login"))
|
# return redirect(url_for("login"))
|
||||||
|
|||||||
29
src/web.py
29
src/web.py
@@ -30,7 +30,7 @@ class Web(object):
|
|||||||
except Exception as e:
|
except Exception as e:
|
||||||
self.app.logger.error(e)
|
self.app.logger.error(e)
|
||||||
raise RuntimeError("Could not add SMS to queue, " + str(e)) from e
|
raise RuntimeError("Could not add SMS to queue, " + str(e)) from e
|
||||||
self.app.logger.info("Added two new tasks at position %s and %s", sms, cut )
|
self.app.logger.info("Added two new tasks at position %s and %s", sms, cut)
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def print_image(self, image, sign: str) -> bool:
|
def print_image(self, image, sign: str) -> bool:
|
||||||
@@ -46,19 +46,22 @@ class Web(object):
|
|||||||
if file_uploaded:
|
if file_uploaded:
|
||||||
self.app.logger.debug("File has been uploaded, printing...")
|
self.app.logger.debug("File has been uploaded, printing...")
|
||||||
try:
|
try:
|
||||||
img = self.print_queue.enqueue(ImageTask(os.path.join(
|
img = self.print_queue.enqueue(
|
||||||
self.app.config["UPLOAD_FOLDER"],
|
ImageTask(
|
||||||
secure_filename(image.filename),
|
os.path.join(
|
||||||
),
|
self.app.config["UPLOAD_FOLDER"],
|
||||||
signature=sign,
|
secure_filename(image.filename),
|
||||||
process=True))
|
),
|
||||||
|
signature=sign,
|
||||||
|
process=True,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
cut = self.print_queue.enqueue(CutTask())
|
cut = self.print_queue.enqueue(CutTask())
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
raise RuntimeError("Could not add IMG to queue" + str(e)) from e
|
raise RuntimeError("Could not add IMG to queue" + str(e)) from e
|
||||||
|
|
||||||
self.app.logger.info("Added two new tasks at position %s and %s", img, cut )
|
self.app.logger.info("Added two new tasks at position %s and %s", img, cut)
|
||||||
|
|
||||||
|
|
||||||
return True
|
return True
|
||||||
|
|
||||||
@@ -96,10 +99,14 @@ class Web(object):
|
|||||||
)
|
)
|
||||||
return True
|
return True
|
||||||
else:
|
else:
|
||||||
self.app.logger.error("Could not save file because the filename is forbidden")
|
self.app.logger.error(
|
||||||
|
"Could not save file because the filename is forbidden"
|
||||||
|
)
|
||||||
return False
|
return False
|
||||||
else:
|
else:
|
||||||
self.app.logger.error("Could not save file, it seems to be null ? : " + str(filename))
|
self.app.logger.error(
|
||||||
|
"Could not save file, it seems to be null ? : " + str(filename)
|
||||||
|
)
|
||||||
return False
|
return False
|
||||||
|
|
||||||
def get_queue_state(self):
|
def get_queue_state(self):
|
||||||
|
|||||||
@@ -4,7 +4,7 @@
|
|||||||
|
|
||||||
import threading
|
import threading
|
||||||
import time
|
import time
|
||||||
from task import TaskType
|
|
||||||
|
|
||||||
class PrintWorker(threading.Thread):
|
class PrintWorker(threading.Thread):
|
||||||
def __init__(self, app, print_queue, printer, socketio=None):
|
def __init__(self, app, print_queue, printer, socketio=None):
|
||||||
@@ -14,7 +14,7 @@ class PrintWorker(threading.Thread):
|
|||||||
self.printer = printer
|
self.printer = printer
|
||||||
self.socketio = socketio # Optional
|
self.socketio = socketio # Optional
|
||||||
self.running = True
|
self.running = True
|
||||||
self.state = "idle" # idle, printing, dead, drinking-a-beer
|
self.state = "idle" # idle, printing, dead, drinking-a-beer
|
||||||
|
|
||||||
self.app.logger.debug("Ho great, I'm alive... I'm ready to work another day...")
|
self.app.logger.debug("Ho great, I'm alive... I'm ready to work another day...")
|
||||||
|
|
||||||
@@ -22,23 +22,17 @@ class PrintWorker(threading.Thread):
|
|||||||
"""Background thread that processes queue items"""
|
"""Background thread that processes queue items"""
|
||||||
self.app.logger.info("Worker started working.")
|
self.app.logger.info("Worker started working.")
|
||||||
while True:
|
while True:
|
||||||
if not self.running:
|
if not self.running or not self.printer.ready:
|
||||||
time.sleep(0.2)
|
time.sleep(0.2)
|
||||||
continue
|
continue
|
||||||
|
|
||||||
# TODO: This could be improved to simply no start
|
|
||||||
# the while loop as long as the printer is not ready.
|
|
||||||
# and maybe get out of it when the printer is not ready anymore ?
|
|
||||||
if not self.printer.ready:
|
|
||||||
self.app.logger.debug("Waiting for the printer to be ready...")
|
|
||||||
time.sleep(1)
|
|
||||||
continue
|
|
||||||
|
|
||||||
try:
|
try:
|
||||||
task = self.print_queue.dequeue()
|
task = self.print_queue.dequeue()
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
self.app.logger.error("Could not get a new task ! %s ", str(e))
|
self.app.logger.error("Could not get a new task ! %s ", str(e))
|
||||||
raise RuntimeError("We could not get a new task because " + str(e)) from e
|
raise RuntimeError(
|
||||||
|
"We could not get a new task because " + str(e)
|
||||||
|
) from e
|
||||||
|
|
||||||
if task:
|
if task:
|
||||||
try:
|
try:
|
||||||
@@ -51,7 +45,7 @@ class PrintWorker(threading.Thread):
|
|||||||
print_data = task.get_print_data()
|
print_data = task.get_print_data()
|
||||||
try:
|
try:
|
||||||
self.printer.print_task(task.task_type, print_data)
|
self.printer.print_task(task.task_type, print_data)
|
||||||
except Exception as e:
|
except RuntimeError as e:
|
||||||
self.app.logger.error("Could not print : %s", str(e))
|
self.app.logger.error("Could not print : %s", str(e))
|
||||||
raise e
|
raise e
|
||||||
|
|
||||||
@@ -59,7 +53,7 @@ class PrintWorker(threading.Thread):
|
|||||||
self.print_queue.mark_completed(task.task_id, "completed")
|
self.print_queue.mark_completed(task.task_id, "completed")
|
||||||
self._emit_status(task.task_id, "completed")
|
self._emit_status(task.task_id, "completed")
|
||||||
|
|
||||||
except Exception as e:
|
except RuntimeError as e:
|
||||||
task.status = "failed"
|
task.status = "failed"
|
||||||
self.print_queue.mark_completed(task.task_id, "failed")
|
self.print_queue.mark_completed(task.task_id, "failed")
|
||||||
self._emit_status(task.task_id, "failed", error=str(e))
|
self._emit_status(task.task_id, "failed", error=str(e))
|
||||||
@@ -78,18 +72,19 @@ class PrintWorker(threading.Thread):
|
|||||||
data = {
|
data = {
|
||||||
"task_id": task_id,
|
"task_id": task_id,
|
||||||
"status": status,
|
"status": status,
|
||||||
"position": None # Task no longer in queue
|
"position": None, # Task no longer in queue
|
||||||
}
|
}
|
||||||
|
|
||||||
if error:
|
if error:
|
||||||
data["error"] = error
|
data["error"] = error
|
||||||
|
|
||||||
self.socketio.emit('task_status', data, room=room)
|
self.socketio.emit("task_status", data, room=room)
|
||||||
|
|
||||||
def stop_worker(self):
|
def stop_worker(self):
|
||||||
"""
|
"""
|
||||||
Give the worker a break
|
Give the worker a break
|
||||||
"""
|
"""
|
||||||
|
self.app.logger.debug("Giving the worker a break")
|
||||||
self.state = "drinking-a-beer"
|
self.state = "drinking-a-beer"
|
||||||
self.running = False
|
self.running = False
|
||||||
|
|
||||||
@@ -97,6 +92,7 @@ class PrintWorker(threading.Thread):
|
|||||||
"""
|
"""
|
||||||
Get the worker back to it
|
Get the worker back to it
|
||||||
"""
|
"""
|
||||||
|
self.app.logger.debug("Time to work !")
|
||||||
self.state = "idle"
|
self.state = "idle"
|
||||||
self.running = True
|
self.running = True
|
||||||
|
|
||||||
@@ -107,5 +103,5 @@ class PrintWorker(threading.Thread):
|
|||||||
return {
|
return {
|
||||||
"is_running": self.running,
|
"is_running": self.running,
|
||||||
"queue_size": len(self.print_queue),
|
"queue_size": len(self.print_queue),
|
||||||
"state" : self.state
|
"state": self.state,
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user