mirror of
https://github.com/Ultimaker/Cura.git
synced 2025-07-20 21:27:50 -06:00
Some scaffolding and implementation for cloud output device manager
This commit is contained in:
parent
228325eb89
commit
10576d1242
3 changed files with 41 additions and 16 deletions
|
@ -23,14 +23,13 @@ from plugins.UM3NetworkPrinting.src.UM3PrintJobOutputModel import UM3PrintJobOut
|
||||||
# Note that this device represents a single remote cluster, not a list of multiple clusters.
|
# Note that this device represents a single remote cluster, not a list of multiple clusters.
|
||||||
#
|
#
|
||||||
# TODO: figure our how the QML interface for the cluster networking should operate with this limited functionality.
|
# TODO: figure our how the QML interface for the cluster networking should operate with this limited functionality.
|
||||||
# TODO: figure out how to pair remote clusters, local networked clusters and local cura printer presets.
|
|
||||||
class CloudOutputDevice(NetworkedPrinterOutputDevice):
|
class CloudOutputDevice(NetworkedPrinterOutputDevice):
|
||||||
|
|
||||||
# The translation catalog for this device.
|
# The translation catalog for this device.
|
||||||
I18N_CATALOG = i18nCatalog("cura")
|
I18N_CATALOG = i18nCatalog("cura")
|
||||||
|
|
||||||
# The cloud URL to use for remote clusters.
|
# The cloud URL to use for this remote cluster.
|
||||||
API_ROOT_PATH_FORMAT = "https://api.ultimaker.com/connect/v1/clusters/{cluster_id}"
|
API_ROOT_PATH_FORMAT = "https://api-staging.ultimaker.com/connect/v1/clusters/{cluster_id}"
|
||||||
|
|
||||||
# Signal triggered when the printers in the remote cluster were changed.
|
# Signal triggered when the printers in the remote cluster were changed.
|
||||||
printersChanged = pyqtSignal()
|
printersChanged = pyqtSignal()
|
||||||
|
@ -79,8 +78,6 @@ class CloudOutputDevice(NetworkedPrinterOutputDevice):
|
||||||
def requestWrite(self, nodes: List[SceneNode], file_name: Optional[str] = None, limit_mime_types: bool = False,
|
def requestWrite(self, nodes: List[SceneNode], file_name: Optional[str] = None, limit_mime_types: bool = False,
|
||||||
file_handler: Optional[FileHandler] = None, **kwargs: str) -> None:
|
file_handler: Optional[FileHandler] = None, **kwargs: str) -> None:
|
||||||
self.writeStarted.emit(self)
|
self.writeStarted.emit(self)
|
||||||
|
|
||||||
# TODO: actually implement this
|
|
||||||
self._addPrintJobToQueue()
|
self._addPrintJobToQueue()
|
||||||
|
|
||||||
## Get remote printers.
|
## Get remote printers.
|
||||||
|
@ -102,6 +99,8 @@ class CloudOutputDevice(NetworkedPrinterOutputDevice):
|
||||||
super()._update()
|
super()._update()
|
||||||
self.get("/status", on_finished = self._onStatusCallFinished)
|
self.get("/status", on_finished = self._onStatusCallFinished)
|
||||||
|
|
||||||
|
## Method called when HTTP request to status endpoint is finished.
|
||||||
|
# Contains both printers and print jobs statuses in a single response.
|
||||||
def _onStatusCallFinished(self, reply: QNetworkReply) -> None:
|
def _onStatusCallFinished(self, reply: QNetworkReply) -> None:
|
||||||
status_code = reply.attribute(QNetworkRequest.HttpStatusCodeAttribute)
|
status_code = reply.attribute(QNetworkRequest.HttpStatusCodeAttribute)
|
||||||
if status_code != 200:
|
if status_code != 200:
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
# Copyright (c) 2018 Ultimaker B.V.
|
# Copyright (c) 2018 Ultimaker B.V.
|
||||||
# Cura is released under the terms of the LGPLv3 or higher.
|
# Cura is released under the terms of the LGPLv3 or higher.
|
||||||
from typing import TYPE_CHECKING
|
from typing import TYPE_CHECKING, Dict
|
||||||
|
|
||||||
from plugins.UM3NetworkPrinting.src.Cloud.CloudOutputDevice import CloudOutputDevice
|
from plugins.UM3NetworkPrinting.src.Cloud.CloudOutputDevice import CloudOutputDevice
|
||||||
|
|
||||||
|
@ -11,21 +11,47 @@ if TYPE_CHECKING:
|
||||||
|
|
||||||
## The cloud output device manager is responsible for using the Ultimaker Cloud APIs to manage remote clusters.
|
## The cloud output device manager is responsible for using the Ultimaker Cloud APIs to manage remote clusters.
|
||||||
# Keeping all cloud related logic in this class instead of the UM3OutputDevicePlugin results in more readable code.
|
# Keeping all cloud related logic in this class instead of the UM3OutputDevicePlugin results in more readable code.
|
||||||
|
#
|
||||||
|
# TODO: figure out how to pair remote clusters, local networked clusters and local cura printer presets.
|
||||||
class CloudOutputDeviceManager:
|
class CloudOutputDeviceManager:
|
||||||
|
|
||||||
|
# The cloud URL to use for remote clusters.
|
||||||
|
API_ROOT_PATH = "https://api-staging.ultimaker.com/connect/v1"
|
||||||
|
|
||||||
def __init__(self, application: "CuraApplication"):
|
def __init__(self, application: "CuraApplication"):
|
||||||
|
self._application = application
|
||||||
self._output_device_manager = application.getOutputDeviceManager()
|
self._output_device_manager = application.getOutputDeviceManager()
|
||||||
self._account = application.getCuraAPI().account
|
self._account = application.getCuraAPI().account
|
||||||
self._getRemoteClusters()
|
|
||||||
|
|
||||||
# For testing:
|
# Persistent dict containing the remote clusters for the authenticated user.
|
||||||
application.globalContainerStackChanged.connect(self._addCloudOutputDevice)
|
self._remote_clusters = {} # type: Dict[str, CloudOutputDevice]
|
||||||
|
|
||||||
|
# When switching machines we check if we have to activate a remote cluster.
|
||||||
|
self._application.globalContainerStackChanged.connect(self._activeMachineChanged)
|
||||||
|
|
||||||
|
# Fetch all remote clusters for the authenticated user.
|
||||||
|
self._getRemoteClusters()
|
||||||
|
|
||||||
def _getRemoteClusters(self):
|
def _getRemoteClusters(self):
|
||||||
# TODO: get list of remote clusters and create an output device for each.
|
# TODO: get list of remote clusters and create an output device for each.
|
||||||
pass
|
# For testing we add a dummy device:
|
||||||
|
self._addCloudOutputDevice({"cluster_id": "LJ0tciiuZZjarrXAvFLEZ6ox4Cvx8FvtXUlQv4vIhV6w"})
|
||||||
|
|
||||||
def _addCloudOutputDevice(self):
|
def _addCloudOutputDevice(self, cluster_data: Dict[str, any]):
|
||||||
device = CloudOutputDevice("xxxx-xxxx-xxxx-xxxx")
|
# TODO: use model or named tuple for cluster_data
|
||||||
|
device = CloudOutputDevice(cluster_data["cluster_id"])
|
||||||
self._output_device_manager.addOutputDevice(device)
|
self._output_device_manager.addOutputDevice(device)
|
||||||
device.connect()
|
self._remote_clusters[cluster_data["cluster_id"]] = device
|
||||||
|
|
||||||
|
def _activeMachineChanged(self):
|
||||||
|
active_machine = self._application.getGlobalContainerStack()
|
||||||
|
if not active_machine:
|
||||||
|
return
|
||||||
|
|
||||||
|
stored_cluster_id = active_machine.getMetaDataEntry("um_cloud_cluster_id")
|
||||||
|
if stored_cluster_id not in self._remote_clusters.keys():
|
||||||
|
# Currently authenticated user does not have access to stored cluster or no user is signed in.
|
||||||
|
return
|
||||||
|
|
||||||
|
# We found the active machine as remote cluster so let's connect to it.
|
||||||
|
self._remote_clusters.get(stored_cluster_id).connect()
|
||||||
|
|
|
@ -40,6 +40,9 @@ class UM3OutputDevicePlugin(OutputDevicePlugin):
|
||||||
self._zero_conf = None
|
self._zero_conf = None
|
||||||
self._zero_conf_browser = None
|
self._zero_conf_browser = None
|
||||||
|
|
||||||
|
# Create a cloud output device manager that abstract all cloud connection logic away.
|
||||||
|
self._cloud_output_device_manager = CloudOutputDeviceManager(self._application)
|
||||||
|
|
||||||
# Because the model needs to be created in the same thread as the QMLEngine, we use a signal.
|
# Because the model needs to be created in the same thread as the QMLEngine, we use a signal.
|
||||||
self.addDeviceSignal.connect(self._onAddDevice)
|
self.addDeviceSignal.connect(self._onAddDevice)
|
||||||
self.removeDeviceSignal.connect(self._onRemoveDevice)
|
self.removeDeviceSignal.connect(self._onRemoveDevice)
|
||||||
|
@ -77,9 +80,6 @@ class UM3OutputDevicePlugin(OutputDevicePlugin):
|
||||||
self._service_changed_request_thread = Thread(target=self._handleOnServiceChangedRequests, daemon=True)
|
self._service_changed_request_thread = Thread(target=self._handleOnServiceChangedRequests, daemon=True)
|
||||||
self._service_changed_request_thread.start()
|
self._service_changed_request_thread.start()
|
||||||
|
|
||||||
# Create a cloud output device manager that abstract all cloud connection logic away.
|
|
||||||
self._cloud_output_device_manager = CloudOutputDeviceManager(self._application)
|
|
||||||
|
|
||||||
def getDiscoveredDevices(self):
|
def getDiscoveredDevices(self):
|
||||||
return self._discovered_devices
|
return self._discovered_devices
|
||||||
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue