mirror of
https://github.com/Ultimaker/Cura.git
synced 2025-07-07 06:57:28 -06:00

add it to the list of unique configurations. Remove some connections to signals and reuse already defined listeners.
463 lines
No EOL
19 KiB
Python
463 lines
No EOL
19 KiB
Python
# Copyright (c) 2017 Ultimaker B.V.
|
|
# Cura is released under the terms of the LGPLv3 or higher.
|
|
|
|
from UM.Logger import Logger
|
|
from UM.Application import Application
|
|
from UM.Settings.ContainerRegistry import ContainerRegistry
|
|
from UM.i18n import i18nCatalog
|
|
from UM.Message import Message
|
|
from UM.Qt.Duration import Duration, DurationFormat
|
|
|
|
from cura.PrinterOutput.NetworkedPrinterOutputDevice import NetworkedPrinterOutputDevice, AuthState
|
|
from cura.PrinterOutput.PrinterOutputModel import PrinterOutputModel
|
|
from cura.PrinterOutput.PrintJobOutputModel import PrintJobOutputModel
|
|
from cura.PrinterOutput.MaterialOutputModel import MaterialOutputModel
|
|
from cura.PrinterOutput.NetworkCamera import NetworkCamera
|
|
|
|
from .ClusterUM3PrinterOutputController import ClusterUM3PrinterOutputController
|
|
|
|
from PyQt5.QtNetwork import QNetworkRequest, QNetworkReply
|
|
from PyQt5.QtGui import QDesktopServices
|
|
from PyQt5.QtCore import pyqtSlot, QUrl, pyqtSignal, pyqtProperty, QObject
|
|
|
|
from time import time
|
|
from datetime import datetime
|
|
from typing import Optional
|
|
|
|
import json
|
|
import os
|
|
|
|
i18n_catalog = i18nCatalog("cura")
|
|
|
|
|
|
class ClusterUM3OutputDevice(NetworkedPrinterOutputDevice):
|
|
printJobsChanged = pyqtSignal()
|
|
activePrinterChanged = pyqtSignal()
|
|
|
|
# This is a bit of a hack, as the notify can only use signals that are defined by the class that they are in.
|
|
# Inheritance doesn't seem to work. Tying them together does work, but i'm open for better suggestions.
|
|
clusterPrintersChanged = pyqtSignal()
|
|
|
|
def __init__(self, device_id, address, properties, parent = None):
|
|
super().__init__(device_id = device_id, address = address, properties=properties, parent = parent)
|
|
self._api_prefix = "/cluster-api/v1/"
|
|
|
|
self._number_of_extruders = 2
|
|
|
|
self._print_jobs = []
|
|
|
|
self._monitor_view_qml_path = os.path.join(os.path.dirname(os.path.abspath(__file__)), "ClusterMonitorItem.qml")
|
|
self._control_view_qml_path = os.path.join(os.path.dirname(os.path.abspath(__file__)), "ClusterControlItem.qml")
|
|
|
|
# See comments about this hack with the clusterPrintersChanged signal
|
|
self.printersChanged.connect(self.clusterPrintersChanged)
|
|
|
|
self._accepts_commands = True
|
|
|
|
# Cluster does not have authentication, so default to authenticated
|
|
self._authentication_state = AuthState.Authenticated
|
|
|
|
self._error_message = None
|
|
self._progress_message = None
|
|
|
|
self._active_printer = None # type: Optional[PrinterOutputModel]
|
|
|
|
self._printer_selection_dialog = None
|
|
|
|
self.setPriority(3) # Make sure the output device gets selected above local file output
|
|
self.setName(self._id)
|
|
self.setShortDescription(i18n_catalog.i18nc("@action:button Preceded by 'Ready to'.", "Print over network"))
|
|
self.setDescription(i18n_catalog.i18nc("@properties:tooltip", "Print over network"))
|
|
|
|
self.setConnectionText(i18n_catalog.i18nc("@info:status", "Connected over the network"))
|
|
|
|
self._printer_uuid_to_unique_name_mapping = {}
|
|
|
|
self._finished_jobs = []
|
|
|
|
self._cluster_size = int(properties.get(b"cluster_size", 0))
|
|
|
|
self._latest_reply_handler = None
|
|
|
|
|
|
def requestWrite(self, nodes, file_name=None, filter_by_machine=False, file_handler=None, **kwargs):
|
|
self.writeStarted.emit(self)
|
|
|
|
gcode_dict = getattr(Application.getInstance().getController().getScene(), "gcode_dict", [])
|
|
active_build_plate_id = Application.getInstance().getMultiBuildPlateModel().activeBuildPlate
|
|
gcode_list = gcode_dict[active_build_plate_id]
|
|
|
|
if not gcode_list:
|
|
# Unable to find g-code. Nothing to send
|
|
return
|
|
|
|
self._gcode = gcode_list
|
|
|
|
is_job_sent = True
|
|
if len(self._printers) > 1:
|
|
self._spawnPrinterSelectionDialog()
|
|
else:
|
|
is_job_sent = self.sendPrintJob()
|
|
|
|
# Notify the UI that a switch to the print monitor should happen
|
|
if is_job_sent:
|
|
Application.getInstance().getController().setActiveStage("MonitorStage")
|
|
|
|
def _spawnPrinterSelectionDialog(self):
|
|
if self._printer_selection_dialog is None:
|
|
path = os.path.join(os.path.dirname(os.path.abspath(__file__)), "PrintWindow.qml")
|
|
self._printer_selection_dialog = Application.getInstance().createQmlComponent(path, {"OutputDevice": self})
|
|
if self._printer_selection_dialog is not None:
|
|
self._printer_selection_dialog.show()
|
|
|
|
@pyqtProperty(int, constant=True)
|
|
def clusterSize(self):
|
|
return self._cluster_size
|
|
|
|
@pyqtSlot()
|
|
@pyqtSlot(str)
|
|
def sendPrintJob(self, target_printer = ""):
|
|
Logger.log("i", "Sending print job to printer.")
|
|
if self._sending_gcode:
|
|
self._error_message = Message(
|
|
i18n_catalog.i18nc("@info:status",
|
|
"Sending new jobs (temporarily) blocked, still sending the previous print job."))
|
|
self._error_message.show()
|
|
return False
|
|
|
|
self._sending_gcode = True
|
|
|
|
self._progress_message = Message(i18n_catalog.i18nc("@info:status", "Sending data to printer"), 0, False, -1,
|
|
i18n_catalog.i18nc("@info:title", "Sending Data"))
|
|
self._progress_message.addAction("Abort", i18n_catalog.i18nc("@action:button", "Cancel"), None, "")
|
|
self._progress_message.actionTriggered.connect(self._progressMessageActionTriggered)
|
|
self._progress_message.show()
|
|
|
|
compressed_gcode = self._compressGCode()
|
|
if compressed_gcode is None:
|
|
# Abort was called.
|
|
return False
|
|
|
|
parts = []
|
|
|
|
# If a specific printer was selected, it should be printed with that machine.
|
|
if target_printer:
|
|
target_printer = self._printer_uuid_to_unique_name_mapping[target_printer]
|
|
parts.append(self._createFormPart("name=require_printer_name", bytes(target_printer, "utf-8"), "text/plain"))
|
|
|
|
# Add user name to the print_job
|
|
parts.append(self._createFormPart("name=owner", bytes(self._getUserName(), "utf-8"), "text/plain"))
|
|
|
|
file_name = "%s.gcode.gz" % Application.getInstance().getPrintInformation().jobName
|
|
|
|
parts.append(self._createFormPart("name=\"file\"; filename=\"%s\"" % file_name, compressed_gcode))
|
|
|
|
self._latest_reply_handler = self.postFormWithParts("print_jobs/", parts, onFinished=self._onPostPrintJobFinished, onProgress=self._onUploadPrintJobProgress)
|
|
|
|
return True
|
|
|
|
@pyqtProperty(QObject, notify=activePrinterChanged)
|
|
def activePrinter(self) -> Optional["PrinterOutputModel"]:
|
|
return self._active_printer
|
|
|
|
@pyqtSlot(QObject)
|
|
def setActivePrinter(self, printer):
|
|
if self._active_printer != printer:
|
|
if self._active_printer and self._active_printer.camera:
|
|
self._active_printer.camera.stop()
|
|
self._active_printer = printer
|
|
self.activePrinterChanged.emit()
|
|
|
|
def _onPostPrintJobFinished(self, reply):
|
|
self._progress_message.hide()
|
|
self._compressing_gcode = False
|
|
self._sending_gcode = False
|
|
|
|
def _onUploadPrintJobProgress(self, bytes_sent, bytes_total):
|
|
if bytes_total > 0:
|
|
new_progress = bytes_sent / bytes_total * 100
|
|
# Treat upload progress as response. Uploading can take more than 10 seconds, so if we don't, we can get
|
|
# timeout responses if this happens.
|
|
self._last_response_time = time()
|
|
if new_progress > self._progress_message.getProgress():
|
|
self._progress_message.show() # Ensure that the message is visible.
|
|
self._progress_message.setProgress(bytes_sent / bytes_total * 100)
|
|
else:
|
|
self._progress_message.setProgress(0)
|
|
self._progress_message.hide()
|
|
|
|
def _progressMessageActionTriggered(self, message_id=None, action_id=None):
|
|
if action_id == "Abort":
|
|
Logger.log("d", "User aborted sending print to remote.")
|
|
self._progress_message.hide()
|
|
self._compressing_gcode = False
|
|
self._sending_gcode = False
|
|
Application.getInstance().getController().setActiveStage("PrepareStage")
|
|
|
|
# After compressing the sliced model Cura sends data to printer, to stop receiving updates from the request
|
|
# the "reply" should be disconnected
|
|
if self._latest_reply_handler:
|
|
self._latest_reply_handler.disconnect()
|
|
self._latest_reply_handler = None
|
|
|
|
|
|
@pyqtSlot()
|
|
def openPrintJobControlPanel(self):
|
|
Logger.log("d", "Opening print job control panel...")
|
|
QDesktopServices.openUrl(QUrl("http://" + self._address + "/print_jobs"))
|
|
|
|
@pyqtSlot()
|
|
def openPrinterControlPanel(self):
|
|
Logger.log("d", "Opening printer control panel...")
|
|
QDesktopServices.openUrl(QUrl("http://" + self._address + "/printers"))
|
|
|
|
@pyqtProperty("QVariantList", notify=printJobsChanged)
|
|
def printJobs(self):
|
|
return self._print_jobs
|
|
|
|
@pyqtProperty("QVariantList", notify=printJobsChanged)
|
|
def queuedPrintJobs(self):
|
|
return [print_job for print_job in self._print_jobs if print_job.assignedPrinter is None or print_job.state == "queued"]
|
|
|
|
@pyqtProperty("QVariantList", notify=printJobsChanged)
|
|
def activePrintJobs(self):
|
|
return [print_job for print_job in self._print_jobs if print_job.assignedPrinter is not None and print_job.state != "queued"]
|
|
|
|
@pyqtProperty("QVariantList", notify=clusterPrintersChanged)
|
|
def connectedPrintersTypeCount(self):
|
|
printer_count = {}
|
|
for printer in self._printers:
|
|
if printer.type in printer_count:
|
|
printer_count[printer.type] += 1
|
|
else:
|
|
printer_count[printer.type] = 1
|
|
result = []
|
|
for machine_type in printer_count:
|
|
result.append({"machine_type": machine_type, "count": printer_count[machine_type]})
|
|
return result
|
|
|
|
@pyqtSlot(int, result=str)
|
|
def formatDuration(self, seconds):
|
|
return Duration(seconds).getDisplayString(DurationFormat.Format.Short)
|
|
|
|
@pyqtSlot(int, result=str)
|
|
def getTimeCompleted(self, time_remaining):
|
|
current_time = time()
|
|
datetime_completed = datetime.fromtimestamp(current_time + time_remaining)
|
|
return "{hour:02d}:{minute:02d}".format(hour=datetime_completed.hour, minute=datetime_completed.minute)
|
|
|
|
@pyqtSlot(int, result=str)
|
|
def getDateCompleted(self, time_remaining):
|
|
current_time = time()
|
|
datetime_completed = datetime.fromtimestamp(current_time + time_remaining)
|
|
return (datetime_completed.strftime("%a %b ") + "{day}".format(day=datetime_completed.day)).upper()
|
|
|
|
def _printJobStateChanged(self):
|
|
username = self._getUserName()
|
|
|
|
if username is None:
|
|
return # We only want to show notifications if username is set.
|
|
|
|
finished_jobs = [job for job in self._print_jobs if job.state == "wait_cleanup"]
|
|
|
|
newly_finished_jobs = [job for job in finished_jobs if job not in self._finished_jobs and job.owner == username]
|
|
for job in newly_finished_jobs:
|
|
if job.assignedPrinter:
|
|
job_completed_text = i18n_catalog.i18nc("@info:status", "Printer '{printer_name}' has finished printing '{job_name}'.".format(printer_name=job.assignedPrinter.name, job_name = job.name))
|
|
else:
|
|
job_completed_text = i18n_catalog.i18nc("@info:status", "The print job '{job_name}' was finished.".format(job_name = job.name))
|
|
job_completed_message = Message(text=job_completed_text, title = i18n_catalog.i18nc("@info:status", "Print finished"))
|
|
job_completed_message.show()
|
|
|
|
# Ensure UI gets updated
|
|
self.printJobsChanged.emit()
|
|
|
|
# Keep a list of all completed jobs so we know if something changed next time.
|
|
self._finished_jobs = finished_jobs
|
|
|
|
def _update(self):
|
|
if not super()._update():
|
|
return
|
|
self.get("printers/", onFinished=self._onGetPrintersDataFinished)
|
|
self.get("print_jobs/", onFinished=self._onGetPrintJobsFinished)
|
|
|
|
def _onGetPrintJobsFinished(self, reply: QNetworkReply):
|
|
if not checkValidGetReply(reply):
|
|
return
|
|
|
|
result = loadJsonFromReply(reply)
|
|
if result is None:
|
|
return
|
|
|
|
print_jobs_seen = []
|
|
job_list_changed = False
|
|
for print_job_data in result:
|
|
print_job = findByKey(self._print_jobs, print_job_data["uuid"])
|
|
|
|
if print_job is None:
|
|
print_job = self._createPrintJobModel(print_job_data)
|
|
job_list_changed = True
|
|
|
|
self._updatePrintJob(print_job, print_job_data)
|
|
|
|
if print_job.state != "queued": # Print job should be assigned to a printer.
|
|
if print_job.state == "failed":
|
|
# Print job was failed, so don't attach it to a printer.
|
|
printer = None
|
|
else:
|
|
printer = self._getPrinterByKey(print_job_data["printer_uuid"])
|
|
else: # The job can "reserve" a printer if some changes are required.
|
|
printer = self._getPrinterByKey(print_job_data["assigned_to"])
|
|
|
|
if printer:
|
|
printer.updateActivePrintJob(print_job)
|
|
|
|
print_jobs_seen.append(print_job)
|
|
|
|
# Check what jobs need to be removed.
|
|
removed_jobs = [print_job for print_job in self._print_jobs if print_job not in print_jobs_seen]
|
|
|
|
for removed_job in removed_jobs:
|
|
job_list_changed |= self._removeJob(removed_job)
|
|
|
|
if job_list_changed:
|
|
self.printJobsChanged.emit() # Do a single emit for all print job changes.
|
|
|
|
def _onGetPrintersDataFinished(self, reply: QNetworkReply):
|
|
if not checkValidGetReply(reply):
|
|
return
|
|
|
|
result = loadJsonFromReply(reply)
|
|
if result is None:
|
|
return
|
|
|
|
printer_list_changed = False
|
|
printers_seen = []
|
|
|
|
for printer_data in result:
|
|
printer = findByKey(self._printers, printer_data["uuid"])
|
|
|
|
if printer is None:
|
|
printer = self._createPrinterModel(printer_data)
|
|
printer_list_changed = True
|
|
|
|
printers_seen.append(printer)
|
|
|
|
self._updatePrinter(printer, printer_data)
|
|
|
|
removed_printers = [printer for printer in self._printers if printer not in printers_seen]
|
|
for printer in removed_printers:
|
|
self._removePrinter(printer)
|
|
|
|
if removed_printers or printer_list_changed:
|
|
self.printersChanged.emit()
|
|
|
|
def _createPrinterModel(self, data):
|
|
printer = PrinterOutputModel(output_controller=ClusterUM3PrinterOutputController(self),
|
|
number_of_extruders=self._number_of_extruders)
|
|
printer.setCamera(NetworkCamera("http://" + data["ip_address"] + ":8080/?action=stream"))
|
|
self._printers.append(printer)
|
|
return printer
|
|
|
|
def _createPrintJobModel(self, data):
|
|
print_job = PrintJobOutputModel(output_controller=ClusterUM3PrinterOutputController(self),
|
|
key=data["uuid"], name= data["name"])
|
|
print_job.stateChanged.connect(self._printJobStateChanged)
|
|
self._print_jobs.append(print_job)
|
|
return print_job
|
|
|
|
def _updatePrintJob(self, print_job, data):
|
|
print_job.updateTimeTotal(data["time_total"])
|
|
print_job.updateTimeElapsed(data["time_elapsed"])
|
|
print_job.updateState(data["status"])
|
|
print_job.updateOwner(data["owner"])
|
|
|
|
def _updatePrinter(self, printer, data):
|
|
# For some unknown reason the cluster wants UUID for everything, except for sending a job directly to a printer.
|
|
# Then we suddenly need the unique name. So in order to not have to mess up all the other code, we save a mapping.
|
|
self._printer_uuid_to_unique_name_mapping[data["uuid"]] = data["unique_name"]
|
|
machine_definition = ContainerRegistry.getInstance().findDefinitionContainers(name = data["machine_variant"])[0]
|
|
|
|
printer.updateName(data["friendly_name"])
|
|
printer.updateKey(data["uuid"])
|
|
printer.updateType(data["machine_variant"])
|
|
|
|
# Do not store the buildplate information that comes from connect if the current printer has not buildplate information
|
|
if "build_plate" in data and machine_definition.getMetaDataEntry("has_variant_buildplates", False):
|
|
printer.updateBuildplateName(data["build_plate"]["type"])
|
|
if not data["enabled"]:
|
|
printer.updateState("disabled")
|
|
else:
|
|
printer.updateState(data["status"])
|
|
|
|
for index in range(0, self._number_of_extruders):
|
|
extruder = printer.extruders[index]
|
|
try:
|
|
extruder_data = data["configuration"][index]
|
|
except IndexError:
|
|
break
|
|
|
|
extruder.updateHotendID(extruder_data.get("print_core_id", ""))
|
|
|
|
material_data = extruder_data["material"]
|
|
if extruder.activeMaterial is None or extruder.activeMaterial.guid != material_data["guid"]:
|
|
containers = ContainerRegistry.getInstance().findInstanceContainers(type="material",
|
|
GUID=material_data["guid"])
|
|
if containers:
|
|
color = containers[0].getMetaDataEntry("color_code")
|
|
brand = containers[0].getMetaDataEntry("brand")
|
|
material_type = containers[0].getMetaDataEntry("material")
|
|
name = containers[0].getName()
|
|
else:
|
|
Logger.log("w",
|
|
"Unable to find material with guid {guid}. Using data as provided by cluster".format(
|
|
guid=material_data["guid"]))
|
|
color = material_data["color"]
|
|
brand = material_data["brand"]
|
|
material_type = material_data["material"]
|
|
name = "Empty" if material_data["material"] == "empty" else "Unknown"
|
|
|
|
material = MaterialOutputModel(guid=material_data["guid"], type=material_type,
|
|
brand=brand, color=color, name=name)
|
|
extruder.updateActiveMaterial(material)
|
|
|
|
def _removeJob(self, job):
|
|
if job not in self._print_jobs:
|
|
return False
|
|
|
|
if job.assignedPrinter:
|
|
job.assignedPrinter.updateActivePrintJob(None)
|
|
job.stateChanged.disconnect(self._printJobStateChanged)
|
|
self._print_jobs.remove(job)
|
|
|
|
return True
|
|
|
|
def _removePrinter(self, printer):
|
|
self._printers.remove(printer)
|
|
if self._active_printer == printer:
|
|
self._active_printer = None
|
|
self.activePrinterChanged.emit()
|
|
|
|
|
|
def loadJsonFromReply(reply):
|
|
try:
|
|
result = json.loads(bytes(reply.readAll()).decode("utf-8"))
|
|
except json.decoder.JSONDecodeError:
|
|
Logger.logException("w", "Unable to decode JSON from reply.")
|
|
return
|
|
return result
|
|
|
|
|
|
def checkValidGetReply(reply):
|
|
status_code = reply.attribute(QNetworkRequest.HttpStatusCodeAttribute)
|
|
|
|
if status_code != 200:
|
|
Logger.log("w", "Got status code {status_code} while trying to get data".format(status_code=status_code))
|
|
return False
|
|
return True
|
|
|
|
|
|
def findByKey(list, key):
|
|
for item in list:
|
|
if item.key == key:
|
|
return item |