123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325 |
- # Copyright (c) 2018 Ultimaker B.V.
- # Cura is released under the terms of the LGPLv3 or higher.
- from UM.Application import Application
- from UM.Logger import Logger
- from cura.PrinterOutputDevice import PrinterOutputDevice, ConnectionState
- from PyQt5.QtNetwork import QHttpMultiPart, QHttpPart, QNetworkRequest, QNetworkAccessManager, QNetworkReply
- from PyQt5.QtCore import pyqtProperty, pyqtSignal, pyqtSlot, pyqtSignal, QUrl, QCoreApplication
- from time import time
- from typing import Callable, Any, Optional, Dict, Tuple
- from enum import IntEnum
- from typing import List
- import os # To get the username
- import gzip
- class AuthState(IntEnum):
- NotAuthenticated = 1
- AuthenticationRequested = 2
- Authenticated = 3
- AuthenticationDenied = 4
- AuthenticationReceived = 5
- class NetworkedPrinterOutputDevice(PrinterOutputDevice):
- authenticationStateChanged = pyqtSignal()
- def __init__(self, device_id, address: str, properties, parent = None) -> None:
- super().__init__(device_id = device_id, parent = parent)
- self._manager = None # type: QNetworkAccessManager
- self._last_manager_create_time = None # type: float
- self._recreate_network_manager_time = 30
- self._timeout_time = 10 # After how many seconds of no response should a timeout occur?
- self._last_response_time = None # type: float
- self._last_request_time = None # type: float
- self._api_prefix = ""
- self._address = address
- self._properties = properties
- self._user_agent = "%s/%s " % (Application.getInstance().getApplicationName(), Application.getInstance().getVersion())
- self._onFinishedCallbacks = {} # type: Dict[str, Callable[[QNetworkReply], None]]
- self._authentication_state = AuthState.NotAuthenticated
- # QHttpMultiPart objects need to be kept alive and not garbage collected during the
- # HTTP which uses them. We hold references to these QHttpMultiPart objects here.
- self._kept_alive_multiparts = {} # type: Dict[QNetworkReply, QHttpMultiPart]
- self._sending_gcode = False
- self._compressing_gcode = False
- self._gcode = [] # type: List[str]
- self._connection_state_before_timeout = None # type: Optional[ConnectionState]
- printer_type = self._properties.get(b"machine", b"").decode("utf-8")
- printer_type_identifiers = {
- "9066": "ultimaker3",
- "9511": "ultimaker3_extended",
- "9051": "ultimaker_s5"
- }
- self._printer_type = "Unknown"
- for key, value in printer_type_identifiers.items():
- if printer_type.startswith(key):
- self._printer_type = value
- break
- def requestWrite(self, nodes, file_name=None, filter_by_machine=False, file_handler=None, **kwargs) -> None:
- raise NotImplementedError("requestWrite needs to be implemented")
- def setAuthenticationState(self, authentication_state) -> None:
- if self._authentication_state != authentication_state:
- self._authentication_state = authentication_state
- self.authenticationStateChanged.emit()
- @pyqtProperty(int, notify=authenticationStateChanged)
- def authenticationState(self) -> int:
- return self._authentication_state
- def _compressDataAndNotifyQt(self, data_to_append: str) -> bytes:
- compressed_data = gzip.compress(data_to_append.encode("utf-8"))
- self._progress_message.setProgress(-1) # Tickle the message so that it's clear that it's still being used.
- QCoreApplication.processEvents() # Ensure that the GUI does not freeze.
- # Pretend that this is a response, as zipping might take a bit of time.
- # If we don't do this, the device might trigger a timeout.
- self._last_response_time = time()
- return compressed_data
- def _compressGCode(self) -> Optional[bytes]:
- self._compressing_gcode = True
- ## Mash the data into single string
- max_chars_per_line = int(1024 * 1024 / 4) # 1/4 MB per line.
- file_data_bytes_list = []
- batched_lines = []
- batched_lines_count = 0
- for line in self._gcode:
- if not self._compressing_gcode:
- self._progress_message.hide()
- # Stop trying to zip / send as abort was called.
- return None
- # if the gcode was read from a gcode file, self._gcode will be a list of all lines in that file.
- # Compressing line by line in this case is extremely slow, so we need to batch them.
- batched_lines.append(line)
- batched_lines_count += len(line)
- if batched_lines_count >= max_chars_per_line:
- file_data_bytes_list.append(self._compressDataAndNotifyQt("".join(batched_lines)))
- batched_lines = []
- batched_lines_count = 0
- # Don't miss the last batch (If any)
- if len(batched_lines) != 0:
- file_data_bytes_list.append(self._compressDataAndNotifyQt("".join(batched_lines)))
- self._compressing_gcode = False
- return b"".join(file_data_bytes_list)
- def _update(self) -> bool:
- if self._last_response_time:
- time_since_last_response = time() - self._last_response_time
- else:
- time_since_last_response = 0
- if self._last_request_time:
- time_since_last_request = time() - self._last_request_time
- else:
- time_since_last_request = float("inf") # An irrelevantly large number of seconds
- if time_since_last_response > self._timeout_time >= time_since_last_request:
- # Go (or stay) into timeout.
- if self._connection_state_before_timeout is None:
- self._connection_state_before_timeout = self._connection_state
- self.setConnectionState(ConnectionState.closed)
- # We need to check if the manager needs to be re-created. If we don't, we get some issues when OSX goes to
- # sleep.
- if time_since_last_response > self._recreate_network_manager_time:
- if self._last_manager_create_time is None:
- self._createNetworkManager()
- if time() - self._last_manager_create_time > self._recreate_network_manager_time:
- self._createNetworkManager()
- elif self._connection_state == ConnectionState.closed:
- # Go out of timeout.
- self.setConnectionState(self._connection_state_before_timeout)
- self._connection_state_before_timeout = None
- return True
- def _createEmptyRequest(self, target, content_type: Optional[str] = "application/json") -> QNetworkRequest:
- url = QUrl("http://" + self._address + self._api_prefix + target)
- request = QNetworkRequest(url)
- if content_type is not None:
- request.setHeader(QNetworkRequest.ContentTypeHeader, "application/json")
- request.setHeader(QNetworkRequest.UserAgentHeader, self._user_agent)
- return request
- def _createFormPart(self, content_header, data, content_type = None) -> QHttpPart:
- part = QHttpPart()
- if not content_header.startswith("form-data;"):
- content_header = "form_data; " + content_header
- part.setHeader(QNetworkRequest.ContentDispositionHeader, content_header)
- if content_type is not None:
- part.setHeader(QNetworkRequest.ContentTypeHeader, content_type)
- part.setBody(data)
- return part
- ## Convenience function to get the username from the OS.
- # The code was copied from the getpass module, as we try to use as little dependencies as possible.
- def _getUserName(self) -> str:
- for name in ("LOGNAME", "USER", "LNAME", "USERNAME"):
- user = os.environ.get(name)
- if user:
- return user
- return "Unknown User" # Couldn't find out username.
- def _clearCachedMultiPart(self, reply: QNetworkReply) -> None:
- if reply in self._kept_alive_multiparts:
- del self._kept_alive_multiparts[reply]
- def put(self, target: str, data: str, onFinished: Optional[Callable[[Any, QNetworkReply], None]]) -> None:
- if self._manager is None:
- self._createNetworkManager()
- request = self._createEmptyRequest(target)
- self._last_request_time = time()
- reply = self._manager.put(request, data.encode())
- self._registerOnFinishedCallback(reply, onFinished)
- def get(self, target: str, onFinished: Optional[Callable[[Any, QNetworkReply], None]]) -> None:
- if self._manager is None:
- self._createNetworkManager()
- request = self._createEmptyRequest(target)
- self._last_request_time = time()
- reply = self._manager.get(request)
- self._registerOnFinishedCallback(reply, onFinished)
- def post(self, target: str, data: str, onFinished: Optional[Callable[[Any, QNetworkReply], None]], onProgress: Callable = None) -> None:
- if self._manager is None:
- self._createNetworkManager()
- request = self._createEmptyRequest(target)
- self._last_request_time = time()
- reply = self._manager.post(request, data)
- if onProgress is not None:
- reply.uploadProgress.connect(onProgress)
- self._registerOnFinishedCallback(reply, onFinished)
- def postFormWithParts(self, target:str, parts: List[QHttpPart], onFinished: Optional[Callable[[Any, QNetworkReply], None]], onProgress: Callable = None) -> None:
- if self._manager is None:
- self._createNetworkManager()
- request = self._createEmptyRequest(target, content_type=None)
- multi_post_part = QHttpMultiPart(QHttpMultiPart.FormDataType)
- for part in parts:
- multi_post_part.append(part)
- self._last_request_time = time()
- reply = self._manager.post(request, multi_post_part)
- self._kept_alive_multiparts[reply] = multi_post_part
- if onProgress is not None:
- reply.uploadProgress.connect(onProgress)
- self._registerOnFinishedCallback(reply, onFinished)
- return reply
- def postForm(self, target: str, header_data: str, body_data: bytes, onFinished: Optional[Callable[[Any, QNetworkReply], None]], onProgress: Callable = None) -> None:
- post_part = QHttpPart()
- post_part.setHeader(QNetworkRequest.ContentDispositionHeader, header_data)
- post_part.setBody(body_data)
- self.postFormWithParts(target, [post_part], onFinished, onProgress)
- def _onAuthenticationRequired(self, reply, authenticator) -> None:
- Logger.log("w", "Request to {url} required authentication, which was not implemented".format(url = reply.url().toString()))
- def _createNetworkManager(self) -> None:
- Logger.log("d", "Creating network manager")
- if self._manager:
- self._manager.finished.disconnect(self.__handleOnFinished)
- self._manager.authenticationRequired.disconnect(self._onAuthenticationRequired)
- self._manager = QNetworkAccessManager()
- self._manager.finished.connect(self.__handleOnFinished)
- self._last_manager_create_time = time()
- self._manager.authenticationRequired.connect(self._onAuthenticationRequired)
- if self._properties.get(b"temporary", b"false") != b"true":
- Application.getInstance().getMachineManager().checkCorrectGroupName(self.getId(), self.name)
- def _registerOnFinishedCallback(self, reply: QNetworkReply, onFinished: Optional[Callable[[Any, QNetworkReply], None]]) -> None:
- if onFinished is not None:
- self._onFinishedCallbacks[reply.url().toString() + str(reply.operation())] = onFinished
- def __handleOnFinished(self, reply: QNetworkReply) -> None:
- # Due to garbage collection, we need to cache certain bits of post operations.
- # As we don't want to keep them around forever, delete them if we get a reply.
- if reply.operation() == QNetworkAccessManager.PostOperation:
- self._clearCachedMultiPart(reply)
- if reply.attribute(QNetworkRequest.HttpStatusCodeAttribute) is None:
- # No status code means it never even reached remote.
- return
- self._last_response_time = time()
- if self._connection_state == ConnectionState.connecting:
- self.setConnectionState(ConnectionState.connected)
- callback_key = reply.url().toString() + str(reply.operation())
- try:
- if callback_key in self._onFinishedCallbacks:
- self._onFinishedCallbacks[callback_key](reply)
- except Exception:
- Logger.logException("w", "something went wrong with callback")
- @pyqtSlot(str, result=str)
- def getProperty(self, key: str) -> str:
- bytes_key = key.encode("utf-8")
- if bytes_key in self._properties:
- return self._properties.get(bytes_key, b"").decode("utf-8")
- else:
- return ""
- def getProperties(self):
- return self._properties
- ## Get the unique key of this machine
- # \return key String containing the key of the machine.
- @pyqtProperty(str, constant=True)
- def key(self) -> str:
- return self._id
- ## The IP address of the printer.
- @pyqtProperty(str, constant=True)
- def address(self) -> str:
- return self._properties.get(b"address", b"").decode("utf-8")
- ## Name of the printer (as returned from the ZeroConf properties)
- @pyqtProperty(str, constant=True)
- def name(self) -> str:
- return self._properties.get(b"name", b"").decode("utf-8")
- ## Firmware version (as returned from the ZeroConf properties)
- @pyqtProperty(str, constant=True)
- def firmwareVersion(self) -> str:
- return self._properties.get(b"firmware_version", b"").decode("utf-8")
- @pyqtProperty(str, constant=True)
- def printerType(self) -> str:
- return self._printer_type
- ## IPadress of this printer
- @pyqtProperty(str, constant=True)
- def ipAddress(self) -> str:
- return self._address
|