123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142 |
- # Copyright (c) 2015 Ultimaker B.V.
- # Cura is released under the terms of the AGPLv3 or higher.
- from cura.CuraApplication import CuraApplication
- from UM.Extension import Extension
- from UM.Application import Application
- from UM.Preferences import Preferences
- from UM.Scene.Iterator.DepthFirstIterator import DepthFirstIterator
- from UM.Scene.SceneNode import SceneNode
- from UM.Message import Message
- from UM.i18n import i18nCatalog
- from UM.Logger import Logger
- from UM.Platform import Platform
- from UM.Qt.Duration import DurationFormat
- from UM.Job import Job
- import platform
- import math
- import urllib.request
- import urllib.parse
- import ssl
- import hashlib
- catalog = i18nCatalog("cura")
- class SliceInfoJob(Job):
- data = None
- url = None
- def __init__(self, url, data):
- super().__init__()
- self.url = url
- self.data = data
- def run(self):
- if not self.url or not self.data:
- Logger.log("e", "URL or DATA for sending slice info was not set!")
- return
- # Submit data
- kwoptions = {"data" : self.data,
- "timeout" : 5
- }
- if Platform.isOSX():
- kwoptions["context"] = ssl._create_unverified_context()
- Logger.log("d", "Sending anonymous slice info to [%s]...", self.url)
- try:
- f = urllib.request.urlopen(self.url, **kwoptions)
- Logger.log("i", "Sent anonymous slice info.")
- f.close()
- except urllib.error.HTTPError as http_exception:
- Logger.log("e", "An HTTP error occurred while trying to send slice information: %s" % http_exception)
- except Exception as e: # We don't want any exception to cause problems
- Logger.log("e", "An exception occurred while trying to send slice information: %s" % e)
- ## This Extension runs in the background and sends several bits of information to the Ultimaker servers.
- # The data is only sent when the user in question gave permission to do so. All data is anonymous and
- # no model files are being sent (Just a SHA256 hash of the model).
- class SliceInfo(Extension):
- info_url = "http://stats.youmagine.com/curastats/slice"
- def __init__(self):
- super().__init__()
- Application.getInstance().getOutputDeviceManager().writeStarted.connect(self._onWriteStarted)
- Preferences.getInstance().addPreference("info/send_slice_info", True)
- Preferences.getInstance().addPreference("info/asked_send_slice_info", False)
- if not Preferences.getInstance().getValue("info/asked_send_slice_info"):
- self.send_slice_info_message = Message(catalog.i18nc("@info", "Cura collects anonymised slicing statistics. You can disable this in preferences"), lifetime = 0, dismissable = False)
- self.send_slice_info_message.addAction("Dismiss", catalog.i18nc("@action:button", "Dismiss"), None, "")
- self.send_slice_info_message.actionTriggered.connect(self.messageActionTriggered)
- self.send_slice_info_message.show()
- def messageActionTriggered(self, message_id, action_id):
- self.send_slice_info_message.hide()
- Preferences.getInstance().setValue("info/asked_send_slice_info", True)
- def _onWriteStarted(self, output_device):
- try:
- if not Preferences.getInstance().getValue("info/send_slice_info"):
- Logger.log("d", "'info/send_slice_info' is turned off.")
- return # Do nothing, user does not want to send data
- # Listing all files placed on the buildplate
- modelhashes = []
- for node in DepthFirstIterator(CuraApplication.getInstance().getController().getScene().getRoot()):
- if type(node) is not SceneNode or not node.getMeshData():
- continue
- modelhashes.append(node.getMeshData().getHash())
- # Creating md5sums and formatting them as discussed on JIRA
- modelhash_formatted = ",".join(modelhashes)
- global_container_stack = Application.getInstance().getGlobalContainerStack()
- # Get total material used (in mm^3)
- print_information = Application.getInstance().getPrintInformation()
- material_radius = 0.5 * global_container_stack.getProperty("material_diameter", "value")
- # TODO: Send material per extruder instead of mashing it on a pile
- material_used = math.pi * material_radius * material_radius * sum(print_information.materialLengths) #Volume of all materials used
- # Bundle the collected data
- submitted_data = {
- "processor": platform.processor(),
- "machine": platform.machine(),
- "platform": platform.platform(),
- "settings": global_container_stack.serialize(), # global_container with references on used containers
- "version": Application.getInstance().getVersion(),
- "modelhash": modelhash_formatted,
- "printtime": print_information.currentPrintTime.getDisplayString(DurationFormat.Format.ISO8601),
- "filament": material_used,
- "language": Preferences.getInstance().getValue("general/language"),
- }
- for container in global_container_stack.getContainers():
- container_id = container.getId()
- try:
- container_serialized = container.serialize()
- except NotImplementedError:
- Logger.log("w", "Container %s could not be serialized!", container_id)
- continue
- if container_serialized:
- submitted_data["settings_%s" %(container_id)] = container_serialized # This can be anything, eg. INI, JSON, etc.
- else:
- Logger.log("i", "No data found in %s to be serialized!", container_id)
- # Convert data to bytes
- submitted_data = urllib.parse.urlencode(submitted_data)
- binary_data = submitted_data.encode("utf-8")
- # Sending slice info non-blocking
- reportJob = SliceInfoJob(self.info_url, binary_data)
- reportJob.start()
- except Exception as e:
- # We really can't afford to have a mistake here, as this would break the sending of g-code to a device
- # (Either saving or directly to a printer). The functionality of the slice data is not *that* important.
- Logger.log("e", "Exception raised while sending slice info: %s" %(repr(e))) # But we should be notified about these problems of course.
|