2021-01-16 20:22:10 +03:00
|
|
|
#!/usr/bin/env python3
|
2020-07-02 04:21:35 +03:00
|
|
|
# GCode metadata extraction utility
|
|
|
|
#
|
|
|
|
# Copyright (C) 2020 Eric Callahan <arksine.code@gmail.com>
|
|
|
|
#
|
|
|
|
# This file may be distributed under the terms of the GNU GPLv3 license.
|
|
|
|
|
|
|
|
import json
|
|
|
|
import argparse
|
|
|
|
import re
|
|
|
|
import os
|
|
|
|
import sys
|
2020-10-30 15:56:00 +03:00
|
|
|
import base64
|
2020-08-11 21:20:21 +03:00
|
|
|
import traceback
|
2020-10-30 15:56:00 +03:00
|
|
|
import io
|
|
|
|
from PIL import Image
|
2020-07-02 04:21:35 +03:00
|
|
|
|
2021-02-23 20:47:34 +03:00
|
|
|
def log_to_stderr(msg):
|
|
|
|
sys.stderr.write(f"{msg}\n")
|
|
|
|
sys.stderr.flush()
|
|
|
|
|
2020-07-02 04:21:35 +03:00
|
|
|
# regex helpers
|
|
|
|
def _regex_find_floats(pattern, data, strict=False):
|
|
|
|
# If strict is enabled, pattern requires a floating point
|
|
|
|
# value, otherwise it can be an integer value
|
|
|
|
fptrn = r'\d+\.\d*' if strict else r'\d+\.?\d*'
|
|
|
|
matches = re.findall(pattern, data)
|
|
|
|
if matches:
|
|
|
|
# return the maximum height value found
|
|
|
|
try:
|
|
|
|
return [float(h) for h in re.findall(
|
|
|
|
fptrn, " ".join(matches))]
|
|
|
|
except Exception:
|
|
|
|
pass
|
|
|
|
return []
|
|
|
|
|
|
|
|
def _regex_find_ints(pattern, data):
|
|
|
|
matches = re.findall(pattern, data)
|
|
|
|
if matches:
|
|
|
|
# return the maximum height value found
|
|
|
|
try:
|
|
|
|
return [int(h) for h in re.findall(
|
|
|
|
r'\d+', " ".join(matches))]
|
|
|
|
except Exception:
|
|
|
|
pass
|
|
|
|
return []
|
|
|
|
|
2020-09-08 02:37:43 +03:00
|
|
|
def _regex_find_first(pattern, data, cast=float):
|
|
|
|
match = re.search(pattern, data)
|
|
|
|
val = None
|
|
|
|
if match:
|
|
|
|
try:
|
|
|
|
val = cast(match.group(1))
|
|
|
|
except Exception:
|
|
|
|
return None
|
|
|
|
return val
|
|
|
|
|
2020-07-02 04:21:35 +03:00
|
|
|
# Slicer parsing implementations
|
|
|
|
class BaseSlicer(object):
|
2020-10-30 15:56:00 +03:00
|
|
|
def __init__(self, file_path):
|
|
|
|
self.path = file_path
|
2021-02-23 20:47:34 +03:00
|
|
|
self.header_data = self.footer_data = None
|
2021-01-16 20:38:37 +03:00
|
|
|
self.layer_height = None
|
2020-07-02 04:21:35 +03:00
|
|
|
|
2021-02-23 20:47:34 +03:00
|
|
|
def set_data(self, header_data, footer_data, fsize):
|
2020-07-02 04:21:35 +03:00
|
|
|
self.header_data = header_data
|
|
|
|
self.footer_data = footer_data
|
2020-11-12 22:37:21 +03:00
|
|
|
self.size = fsize
|
2020-07-02 04:21:35 +03:00
|
|
|
|
2020-09-08 19:25:46 +03:00
|
|
|
def _parse_min_float(self, pattern, data, strict=False):
|
|
|
|
result = _regex_find_floats(pattern, data, strict)
|
2020-07-02 04:21:35 +03:00
|
|
|
if result:
|
|
|
|
return min(result)
|
|
|
|
else:
|
|
|
|
return None
|
|
|
|
|
2020-09-08 19:25:46 +03:00
|
|
|
def _parse_max_float(self, pattern, data, strict=False):
|
|
|
|
result = _regex_find_floats(pattern, data, strict)
|
2020-07-02 04:21:35 +03:00
|
|
|
if result:
|
|
|
|
return max(result)
|
|
|
|
else:
|
|
|
|
return None
|
|
|
|
|
2020-09-09 15:27:41 +03:00
|
|
|
def check_identity(self, data):
|
|
|
|
return None
|
|
|
|
|
2020-11-12 22:37:21 +03:00
|
|
|
def parse_gcode_start_byte(self):
|
|
|
|
m = re.search(r"\n[MG]\d+\s.*\n", self.header_data)
|
|
|
|
if m is None:
|
|
|
|
return None
|
|
|
|
return m.start()
|
|
|
|
|
|
|
|
def parse_gcode_end_byte(self):
|
|
|
|
rev_data = self.footer_data[::-1]
|
|
|
|
m = re.search(r"\n.*\s\d+[MG]\n", rev_data)
|
|
|
|
if m is None:
|
|
|
|
return None
|
|
|
|
return self.size - m.start()
|
|
|
|
|
2020-09-08 00:01:01 +03:00
|
|
|
def parse_first_layer_height(self):
|
|
|
|
return None
|
|
|
|
|
|
|
|
def parse_layer_height(self):
|
|
|
|
return None
|
|
|
|
|
|
|
|
def parse_object_height(self):
|
|
|
|
return None
|
|
|
|
|
|
|
|
def parse_filament_total(self):
|
|
|
|
return None
|
|
|
|
|
2021-04-09 00:58:09 +03:00
|
|
|
def parse_filament_weight_total(self):
|
|
|
|
return None
|
|
|
|
|
2020-09-08 00:01:01 +03:00
|
|
|
def parse_estimated_time(self):
|
|
|
|
return None
|
|
|
|
|
|
|
|
def parse_first_layer_bed_temp(self):
|
|
|
|
return None
|
|
|
|
|
|
|
|
def parse_first_layer_extr_temp(self):
|
|
|
|
return None
|
|
|
|
|
|
|
|
def parse_thumbnails(self):
|
|
|
|
return None
|
|
|
|
|
|
|
|
class UnknownSlicer(BaseSlicer):
|
2020-09-09 15:27:41 +03:00
|
|
|
def check_identity(self, data):
|
|
|
|
return {'slicer': "Unknown"}
|
2020-09-08 00:01:01 +03:00
|
|
|
|
|
|
|
def parse_first_layer_height(self):
|
|
|
|
return self._parse_min_float(r"G1\sZ\d+\.\d*", self.header_data)
|
|
|
|
|
|
|
|
def parse_object_height(self):
|
|
|
|
return self._parse_max_float(r"G1\sZ\d+\.\d*", self.footer_data)
|
|
|
|
|
2020-09-08 02:37:43 +03:00
|
|
|
def parse_first_layer_extr_temp(self):
|
|
|
|
return _regex_find_first(
|
|
|
|
r"M109 S(\d+\.?\d*)", self.header_data)
|
|
|
|
|
|
|
|
def parse_first_layer_bed_temp(self):
|
|
|
|
return _regex_find_first(
|
|
|
|
r"M190 S(\d+\.?\d*)", self.header_data)
|
|
|
|
|
2020-07-02 04:21:35 +03:00
|
|
|
class PrusaSlicer(BaseSlicer):
|
2020-09-09 15:27:41 +03:00
|
|
|
def check_identity(self, data):
|
|
|
|
match = re.search(r"PrusaSlicer\s(.*)\son", data)
|
|
|
|
if match:
|
|
|
|
return {
|
|
|
|
'slicer': "PrusaSlicer",
|
|
|
|
'slicer_version': match.group(1)
|
|
|
|
}
|
|
|
|
return None
|
2020-07-02 04:21:35 +03:00
|
|
|
|
|
|
|
def parse_first_layer_height(self):
|
2021-01-16 20:38:37 +03:00
|
|
|
# Check percentage
|
|
|
|
pct = _regex_find_first(
|
|
|
|
r"; first_layer_height = (\d+)%", self.footer_data)
|
|
|
|
if pct is not None:
|
|
|
|
if self.layer_height is None:
|
|
|
|
# Failed to parse the original layer height, so it is not
|
|
|
|
# possible to calculate a percentage
|
|
|
|
return None
|
|
|
|
return round(pct / 100. * self.layer_height, 6)
|
2020-09-09 00:29:08 +03:00
|
|
|
return _regex_find_first(
|
|
|
|
r"; first_layer_height = (\d+\.?\d*)", self.footer_data)
|
2020-07-02 04:21:35 +03:00
|
|
|
|
|
|
|
def parse_layer_height(self):
|
2021-01-16 20:38:37 +03:00
|
|
|
self.layer_height = _regex_find_first(
|
2020-09-09 00:29:08 +03:00
|
|
|
r"; layer_height = (\d+\.?\d*)", self.footer_data)
|
2021-01-16 20:38:37 +03:00
|
|
|
return self.layer_height
|
2020-07-02 04:21:35 +03:00
|
|
|
|
|
|
|
def parse_object_height(self):
|
2020-09-26 18:43:16 +03:00
|
|
|
matches = re.findall(
|
|
|
|
r";BEFORE_LAYER_CHANGE\n(?:.*\n)?;(\d+\.?\d*)", self.footer_data)
|
|
|
|
if matches:
|
|
|
|
try:
|
|
|
|
matches = [float(m) for m in matches]
|
|
|
|
except Exception:
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
return max(matches)
|
2020-07-02 04:21:35 +03:00
|
|
|
return self._parse_max_float(r"G1\sZ\d+\.\d*\sF", self.footer_data)
|
|
|
|
|
|
|
|
def parse_filament_total(self):
|
2020-09-09 00:29:08 +03:00
|
|
|
return _regex_find_first(
|
|
|
|
r"filament\sused\s\[mm\]\s=\s(\d+\.\d*)", self.footer_data)
|
2020-07-02 04:21:35 +03:00
|
|
|
|
2021-04-09 00:58:09 +03:00
|
|
|
def parse_filament_weight_total(self):
|
|
|
|
return _regex_find_first(
|
|
|
|
r"total\sfilament\sused\s\[g\]\s=\s(\d+\.\d*)", self.footer_data)
|
|
|
|
|
2020-07-02 04:21:35 +03:00
|
|
|
def parse_estimated_time(self):
|
2020-09-09 00:29:08 +03:00
|
|
|
time_match = re.search(
|
2020-07-02 04:21:35 +03:00
|
|
|
r';\sestimated\sprinting\stime.*', self.footer_data)
|
2020-09-09 00:29:08 +03:00
|
|
|
if not time_match:
|
2020-07-02 04:21:35 +03:00
|
|
|
return None
|
|
|
|
total_time = 0
|
2020-09-09 00:29:08 +03:00
|
|
|
time_match = time_match.group()
|
|
|
|
time_patterns = [(r"(\d+)d", 24*60*60), (r"(\d+)h", 60*60),
|
|
|
|
(r"(\d+)m", 60), (r"(\d+)s", 1)]
|
|
|
|
try:
|
|
|
|
for pattern, multiplier in time_patterns:
|
|
|
|
t = re.search(pattern, time_match)
|
|
|
|
if t:
|
|
|
|
total_time += int(t.group(1)) * multiplier
|
|
|
|
except Exception:
|
|
|
|
return None
|
2020-07-02 04:21:35 +03:00
|
|
|
return round(total_time, 2)
|
|
|
|
|
|
|
|
def parse_thumbnails(self):
|
|
|
|
thumb_matches = re.findall(
|
|
|
|
r"; thumbnail begin[;/\+=\w\s]+?; thumbnail end", self.header_data)
|
|
|
|
if not thumb_matches:
|
|
|
|
return None
|
2021-03-14 19:48:45 +03:00
|
|
|
thumb_dir = os.path.join(os.path.dirname(self.path), "thumbs")
|
|
|
|
if not os.path.exists(thumb_dir):
|
|
|
|
try:
|
|
|
|
os.mkdir(thumb_dir)
|
|
|
|
except Exception:
|
|
|
|
log_to_stderr(f"Unable to create thumb dir: {thumb_dir}")
|
|
|
|
return
|
|
|
|
thumb_base = os.path.splitext(os.path.basename(self.path))[0]
|
2020-07-02 04:21:35 +03:00
|
|
|
parsed_matches = []
|
|
|
|
for match in thumb_matches:
|
|
|
|
lines = re.split(r"\r?\n", match.replace('; ', ''))
|
|
|
|
info = _regex_find_ints(r".*", lines[0])
|
|
|
|
data = "".join(lines[1:-1])
|
|
|
|
if len(info) != 3:
|
2021-02-23 20:47:34 +03:00
|
|
|
log_to_stderr(
|
2020-08-11 21:20:21 +03:00
|
|
|
f"MetadataError: Error parsing thumbnail"
|
|
|
|
f" header: {lines[0]}")
|
2020-07-02 04:21:35 +03:00
|
|
|
continue
|
|
|
|
if len(data) != info[2]:
|
2021-02-23 20:47:34 +03:00
|
|
|
log_to_stderr(
|
2020-08-11 21:20:21 +03:00
|
|
|
f"MetadataError: Thumbnail Size Mismatch: "
|
|
|
|
f"detected {info[2]}, actual {len(data)}")
|
2020-07-02 04:21:35 +03:00
|
|
|
continue
|
2021-03-14 19:48:45 +03:00
|
|
|
thumb_name = f"{thumb_base}-{info[0]}x{info[1]}.png"
|
|
|
|
thumb_path = os.path.join(thumb_dir, thumb_name)
|
|
|
|
rel_thumb_path = os.path.join("thumbs", thumb_name)
|
|
|
|
with open(thumb_path, "wb") as f:
|
|
|
|
f.write(base64.b64decode(data.encode()))
|
2020-07-02 04:21:35 +03:00
|
|
|
parsed_matches.append({
|
|
|
|
'width': info[0], 'height': info[1],
|
2021-03-14 19:48:45 +03:00
|
|
|
'size': info[2], 'data': data,
|
|
|
|
'relative_path': rel_thumb_path})
|
2020-07-02 04:21:35 +03:00
|
|
|
return parsed_matches
|
|
|
|
|
2020-09-08 02:37:43 +03:00
|
|
|
def parse_first_layer_extr_temp(self):
|
|
|
|
return _regex_find_first(
|
|
|
|
r"; first_layer_temperature = (\d+\.?\d*)", self.footer_data)
|
|
|
|
|
|
|
|
def parse_first_layer_bed_temp(self):
|
|
|
|
return _regex_find_first(
|
|
|
|
r"; first_layer_bed_temperature = (\d+\.?\d*)", self.footer_data)
|
|
|
|
|
2020-07-02 04:21:35 +03:00
|
|
|
class Slic3rPE(PrusaSlicer):
|
2020-09-09 15:27:41 +03:00
|
|
|
def check_identity(self, data):
|
|
|
|
match = re.search(r"Slic3r\sPrusa\sEdition\s(.*)\son", data)
|
|
|
|
if match:
|
|
|
|
return {
|
|
|
|
'slicer': "Slic3r PE",
|
|
|
|
'slicer_version': match.group(1)
|
|
|
|
}
|
|
|
|
return None
|
2020-07-02 04:21:35 +03:00
|
|
|
|
|
|
|
def parse_filament_total(self):
|
2020-09-09 00:29:08 +03:00
|
|
|
return _regex_find_first(
|
|
|
|
r"filament\sused\s=\s(\d+\.\d+)mm", self.footer_data)
|
2020-07-02 04:21:35 +03:00
|
|
|
|
|
|
|
def parse_thumbnails(self):
|
|
|
|
return None
|
|
|
|
|
|
|
|
class Slic3r(Slic3rPE):
|
2020-09-09 15:27:41 +03:00
|
|
|
def check_identity(self, data):
|
|
|
|
match = re.search(r"Slic3r\s(\d.*)\son", data)
|
|
|
|
if match:
|
|
|
|
return {
|
|
|
|
'slicer': "Slic3r",
|
|
|
|
'slicer_version': match.group(1)
|
|
|
|
}
|
|
|
|
return None
|
2020-07-02 04:21:35 +03:00
|
|
|
|
|
|
|
def parse_estimated_time(self):
|
|
|
|
return None
|
|
|
|
|
|
|
|
class SuperSlicer(PrusaSlicer):
|
2020-09-09 15:27:41 +03:00
|
|
|
def check_identity(self, data):
|
|
|
|
match = re.search(r"SuperSlicer\s(.*)\son", data)
|
|
|
|
if match:
|
|
|
|
return {
|
|
|
|
'slicer': "SuperSlicer",
|
|
|
|
'slicer_version': match.group(1)
|
|
|
|
}
|
|
|
|
return None
|
2020-07-02 04:21:35 +03:00
|
|
|
|
2021-02-06 16:11:51 +03:00
|
|
|
class Cura(PrusaSlicer):
|
2020-09-09 15:27:41 +03:00
|
|
|
def check_identity(self, data):
|
|
|
|
match = re.search(r"Cura_SteamEngine\s(.*)", data)
|
|
|
|
if match:
|
|
|
|
return {
|
|
|
|
'slicer': "Cura",
|
|
|
|
'slicer_version': match.group(1)
|
|
|
|
}
|
|
|
|
return None
|
2020-07-02 04:21:35 +03:00
|
|
|
|
|
|
|
def parse_first_layer_height(self):
|
2020-09-09 00:29:08 +03:00
|
|
|
return _regex_find_first(r";MINZ:(\d+\.?\d*)", self.header_data)
|
2020-07-02 04:21:35 +03:00
|
|
|
|
|
|
|
def parse_layer_height(self):
|
2021-01-16 20:38:37 +03:00
|
|
|
self.layer_height = _regex_find_first(
|
2020-09-09 15:27:41 +03:00
|
|
|
r";Layer\sheight:\s(\d+\.?\d*)", self.header_data)
|
2021-01-16 20:38:37 +03:00
|
|
|
return self.layer_height
|
2020-07-02 04:21:35 +03:00
|
|
|
|
|
|
|
def parse_object_height(self):
|
2020-09-09 00:29:08 +03:00
|
|
|
return _regex_find_first(r";MAXZ:(\d+\.?\d*)", self.header_data)
|
2020-07-02 04:21:35 +03:00
|
|
|
|
|
|
|
def parse_filament_total(self):
|
2020-09-09 00:29:08 +03:00
|
|
|
filament = _regex_find_first(
|
|
|
|
r";Filament\sused:\s(\d+\.?\d*)m", self.header_data)
|
2020-07-02 04:21:35 +03:00
|
|
|
if filament is not None:
|
|
|
|
filament *= 1000
|
|
|
|
return filament
|
|
|
|
|
|
|
|
def parse_estimated_time(self):
|
|
|
|
return self._parse_max_float(r";TIME:.*", self.header_data)
|
|
|
|
|
2020-09-08 02:37:43 +03:00
|
|
|
def parse_first_layer_extr_temp(self):
|
|
|
|
return _regex_find_first(
|
|
|
|
r"M109 S(\d+\.?\d*)", self.header_data)
|
|
|
|
|
|
|
|
def parse_first_layer_bed_temp(self):
|
|
|
|
return _regex_find_first(
|
|
|
|
r"M190 S(\d+\.?\d*)", self.header_data)
|
|
|
|
|
2020-10-30 15:56:00 +03:00
|
|
|
def parse_thumbnails(self):
|
2021-02-06 16:11:51 +03:00
|
|
|
# Attempt to parse thumbnails from file metadata
|
|
|
|
thumbs = super().parse_thumbnails()
|
|
|
|
if thumbs is not None:
|
|
|
|
return thumbs
|
|
|
|
# Check for thumbnails extracted from the ufp
|
2021-03-14 19:48:45 +03:00
|
|
|
thumb_dir = os.path.join(os.path.dirname(self.path), "thumbs")
|
|
|
|
thumb_base = os.path.splitext(os.path.basename(self.path))[0]
|
|
|
|
thumb_path = os.path.join(thumb_dir, f"{thumb_base}.png")
|
|
|
|
rel_path_full = os.path.join("thumbs", f"{thumb_base}.png")
|
|
|
|
rel_path_small = os.path.join("thumbs", f"{thumb_base}-32x32.png")
|
|
|
|
thumb_path_small = os.path.join(thumb_dir, f"{thumb_base}-32x32.png")
|
|
|
|
if not os.path.isfile(thumb_path):
|
2020-10-30 15:56:00 +03:00
|
|
|
return None
|
|
|
|
# read file
|
|
|
|
thumbs = []
|
|
|
|
try:
|
2021-03-14 19:48:45 +03:00
|
|
|
with open(thumb_path, 'rb') as thumb_file:
|
|
|
|
fbytes = thumb_file.read()
|
2020-10-30 15:56:00 +03:00
|
|
|
with Image.open(io.BytesIO(fbytes)) as im:
|
2021-03-14 19:48:45 +03:00
|
|
|
thumb_full_b64 = base64.b64encode(fbytes).decode()
|
2020-10-30 15:56:00 +03:00
|
|
|
thumbs.append({
|
|
|
|
'width': im.width, 'height': im.height,
|
2021-03-14 19:48:45 +03:00
|
|
|
'size': len(thumb_full_b64), 'data': thumb_full_b64,
|
|
|
|
'relative_path': rel_path_full
|
2020-10-30 15:56:00 +03:00
|
|
|
})
|
|
|
|
# Create 32x32 thumbnail
|
|
|
|
im.thumbnail((32, 32), Image.ANTIALIAS)
|
2021-03-14 19:48:45 +03:00
|
|
|
tmp_thumb = io.BytesIO()
|
|
|
|
im.save(tmp_thumb, format="PNG")
|
|
|
|
im.save(thumb_path_small, format="PNG")
|
|
|
|
thumb_small_b64 = base64.b64encode(
|
|
|
|
tmp_thumb.getbuffer()).decode()
|
|
|
|
tmp_thumb.close()
|
2020-10-30 15:56:00 +03:00
|
|
|
thumbs.insert(0, {
|
|
|
|
'width': im.width, 'height': im.height,
|
2021-03-14 19:48:45 +03:00
|
|
|
'size': len(thumb_small_b64), 'data': thumb_small_b64,
|
|
|
|
'relative_path': rel_path_small
|
2020-10-30 15:56:00 +03:00
|
|
|
})
|
|
|
|
except Exception as e:
|
2021-02-23 20:47:34 +03:00
|
|
|
log_to_stderr(str(e))
|
2020-10-30 15:56:00 +03:00
|
|
|
return None
|
|
|
|
return thumbs
|
2020-07-02 04:21:35 +03:00
|
|
|
|
|
|
|
class Simplify3D(BaseSlicer):
|
2020-09-09 15:27:41 +03:00
|
|
|
def check_identity(self, data):
|
|
|
|
match = re.search(r"Simplify3D\(R\)\sVersion\s(.*)", data)
|
|
|
|
if match:
|
|
|
|
return {
|
|
|
|
'slicer': "Simplify3D",
|
|
|
|
'slicer_version': match.group(1)
|
|
|
|
}
|
|
|
|
return None
|
2020-07-02 04:21:35 +03:00
|
|
|
|
|
|
|
def parse_first_layer_height(self):
|
|
|
|
return self._parse_min_float(r"G1\sZ\d+\.\d*", self.header_data)
|
|
|
|
|
|
|
|
def parse_layer_height(self):
|
2021-01-16 20:38:37 +03:00
|
|
|
self.layer_height = _regex_find_first(
|
2020-09-09 00:29:08 +03:00
|
|
|
r";\s+layerHeight,(\d+\.?\d*)", self.header_data)
|
2021-01-16 20:38:37 +03:00
|
|
|
return self.layer_height
|
2020-07-02 04:21:35 +03:00
|
|
|
|
|
|
|
def parse_object_height(self):
|
|
|
|
return self._parse_max_float(r"G1\sZ\d+\.\d*", self.footer_data)
|
|
|
|
|
|
|
|
def parse_filament_total(self):
|
2020-09-09 00:29:08 +03:00
|
|
|
return _regex_find_first(
|
|
|
|
r";\s+Filament\slength:\s(\d+\.?\d*)\smm", self.footer_data)
|
2020-07-02 04:21:35 +03:00
|
|
|
|
2021-04-09 00:58:09 +03:00
|
|
|
def parse_filament_weight_total(self):
|
|
|
|
return _regex_find_first(
|
|
|
|
r";\s+Plastic\sweight:\s(\d+\.?\d*)\sg", self.footer_data)
|
|
|
|
|
2020-07-02 04:21:35 +03:00
|
|
|
def parse_estimated_time(self):
|
2020-09-09 00:29:08 +03:00
|
|
|
time_match = re.search(
|
2020-07-02 04:21:35 +03:00
|
|
|
r';\s+Build time:.*', self.footer_data)
|
2020-09-09 00:29:08 +03:00
|
|
|
if not time_match:
|
2020-07-02 04:21:35 +03:00
|
|
|
return None
|
|
|
|
total_time = 0
|
2020-09-09 00:29:08 +03:00
|
|
|
time_match = time_match.group()
|
|
|
|
time_patterns = [(r"(\d+)\shours", 60*60), (r"(\d+)\smin", 60),
|
|
|
|
(r"(\d+)\ssec", 1)]
|
|
|
|
try:
|
|
|
|
for pattern, multiplier in time_patterns:
|
|
|
|
t = re.search(pattern, time_match)
|
|
|
|
if t:
|
|
|
|
total_time += int(t.group(1)) * multiplier
|
|
|
|
except Exception:
|
|
|
|
return None
|
2020-07-02 04:21:35 +03:00
|
|
|
return round(total_time, 2)
|
|
|
|
|
2020-09-08 02:37:43 +03:00
|
|
|
def _get_temp_items(self, pattern):
|
|
|
|
match = re.search(pattern, self.header_data)
|
|
|
|
if match is None:
|
|
|
|
return []
|
|
|
|
return match.group().split(",")[1:]
|
|
|
|
|
|
|
|
def _get_first_layer_temp(self, heater):
|
|
|
|
heaters = self._get_temp_items(r"temperatureName.*")
|
|
|
|
temps = self._get_temp_items(r"temperatureSetpointTemperatures.*")
|
|
|
|
for h, temp in zip(heaters, temps):
|
|
|
|
if h == heater:
|
|
|
|
try:
|
|
|
|
return float(temp)
|
|
|
|
except Exception:
|
|
|
|
return None
|
|
|
|
return None
|
|
|
|
|
|
|
|
def parse_first_layer_extr_temp(self):
|
|
|
|
return self._get_first_layer_temp("Extruder 1")
|
|
|
|
|
|
|
|
def parse_first_layer_bed_temp(self):
|
|
|
|
return self._get_first_layer_temp("Heated Bed")
|
2020-07-02 04:21:35 +03:00
|
|
|
|
|
|
|
class KISSlicer(BaseSlicer):
|
2020-09-09 15:27:41 +03:00
|
|
|
def check_identity(self, data):
|
|
|
|
match = re.search(r";\sKISSlicer", data)
|
|
|
|
if match:
|
|
|
|
ident = {'slicer': "KISSlicer"}
|
|
|
|
vmatch = re.search(r";\sversion\s(.*)", data)
|
|
|
|
if vmatch:
|
|
|
|
version = vmatch.group(1).replace(" ", "-")
|
|
|
|
ident['slicer_version'] = version
|
|
|
|
return ident
|
|
|
|
return None
|
2020-07-02 04:21:35 +03:00
|
|
|
|
|
|
|
def parse_first_layer_height(self):
|
2020-09-09 00:29:08 +03:00
|
|
|
return _regex_find_first(
|
|
|
|
r";\s+first_layer_thickness_mm\s=\s(\d+\.?\d*)", self.header_data)
|
2020-07-02 04:21:35 +03:00
|
|
|
|
|
|
|
def parse_layer_height(self):
|
2021-01-16 20:38:37 +03:00
|
|
|
self.layer_height = _regex_find_first(
|
2020-09-09 00:29:08 +03:00
|
|
|
r";\s+max_layer_thickness_mm\s=\s(\d+\.?\d*)", self.header_data)
|
2021-01-16 20:38:37 +03:00
|
|
|
return self.layer_height
|
2020-07-02 04:21:35 +03:00
|
|
|
|
|
|
|
def parse_object_height(self):
|
|
|
|
return self._parse_max_float(
|
|
|
|
r";\sEND_LAYER_OBJECT\sz.*", self.footer_data)
|
|
|
|
|
|
|
|
def parse_filament_total(self):
|
|
|
|
filament = _regex_find_floats(
|
|
|
|
r";\s+Ext\s.*mm", self.footer_data, strict=True)
|
|
|
|
if filament:
|
|
|
|
return sum(filament)
|
|
|
|
return None
|
|
|
|
|
|
|
|
def parse_estimated_time(self):
|
2020-09-09 00:29:08 +03:00
|
|
|
time = _regex_find_first(
|
|
|
|
r";\sCalculated.*Build\sTime:\s(\d+\.?\d*)\sminutes",
|
|
|
|
self.footer_data)
|
2020-07-02 04:21:35 +03:00
|
|
|
if time is not None:
|
|
|
|
time *= 60
|
2020-09-09 00:29:08 +03:00
|
|
|
return round(time, 2)
|
|
|
|
return None
|
2020-07-02 04:21:35 +03:00
|
|
|
|
2020-09-08 02:37:43 +03:00
|
|
|
def parse_first_layer_extr_temp(self):
|
|
|
|
return _regex_find_first(
|
|
|
|
r"; first_layer_C = (\d+\.?\d*)", self.header_data)
|
|
|
|
|
|
|
|
def parse_first_layer_bed_temp(self):
|
|
|
|
return _regex_find_first(
|
|
|
|
r"; bed_C = (\d+\.?\d*)", self.header_data)
|
|
|
|
|
2020-07-02 04:21:35 +03:00
|
|
|
|
|
|
|
class IdeaMaker(BaseSlicer):
|
2020-09-09 15:27:41 +03:00
|
|
|
def check_identity(self, data):
|
|
|
|
match = re.search(r"\sideaMaker\s(.*),", data)
|
|
|
|
if match:
|
|
|
|
return {
|
|
|
|
'slicer': "IdeaMaker",
|
|
|
|
'slicer_version': match.group(1)
|
|
|
|
}
|
|
|
|
return None
|
2020-07-02 04:21:35 +03:00
|
|
|
|
|
|
|
def parse_first_layer_height(self):
|
|
|
|
layer_info = _regex_find_floats(
|
|
|
|
r";LAYER:0\s*.*\s*;HEIGHT.*", self.header_data)
|
|
|
|
if len(layer_info) >= 3:
|
|
|
|
return layer_info[2]
|
|
|
|
return None
|
|
|
|
|
|
|
|
def parse_layer_height(self):
|
|
|
|
layer_info = _regex_find_floats(
|
|
|
|
r";LAYER:1\s*.*\s*;HEIGHT.*", self.header_data)
|
|
|
|
if len(layer_info) >= 3:
|
2021-01-16 20:38:37 +03:00
|
|
|
self.layer_height = layer_info[2]
|
|
|
|
return self.layer_height
|
2020-07-02 04:21:35 +03:00
|
|
|
return None
|
|
|
|
|
|
|
|
def parse_object_height(self):
|
|
|
|
bounds = _regex_find_floats(
|
2020-09-09 00:29:08 +03:00
|
|
|
r";Bounding Box:.*", self.header_data)
|
2020-07-02 04:21:35 +03:00
|
|
|
if len(bounds) >= 6:
|
|
|
|
return bounds[5]
|
|
|
|
return None
|
|
|
|
|
|
|
|
def parse_filament_total(self):
|
|
|
|
filament = _regex_find_floats(
|
2021-04-10 17:52:58 +03:00
|
|
|
r";Material.\d\sUsed:.*", self.footer_data, strict=True)
|
2020-07-02 04:21:35 +03:00
|
|
|
if filament:
|
|
|
|
return sum(filament)
|
|
|
|
return None
|
|
|
|
|
2021-04-09 00:58:09 +03:00
|
|
|
def parse_filament_weight_total(self):
|
|
|
|
pi = 3.141592653589793
|
|
|
|
length = _regex_find_floats(
|
|
|
|
r";Material.\d\sUsed:.*", self.footer_data, strict=True)
|
|
|
|
diameter = _regex_find_floats(
|
|
|
|
r";Filament\sDiameter\s.\d:.*", self.header_data, strict=True)
|
|
|
|
density = _regex_find_floats(
|
|
|
|
r";Filament\sDensity\s.\d:.*", self.header_data, strict=True)
|
|
|
|
if len(length) == len(density) == len(diameter):
|
|
|
|
# calc individual weight for each filament with m=pi/4*d²*l*rho
|
|
|
|
weights = [(pi/4 * diameter[i]**2 * length[i] * density[i]/10**6)
|
|
|
|
for i in range(len(length))]
|
|
|
|
return sum(weights)
|
|
|
|
return None
|
|
|
|
|
2020-07-02 04:21:35 +03:00
|
|
|
def parse_estimated_time(self):
|
2020-09-09 00:29:08 +03:00
|
|
|
return _regex_find_first(
|
|
|
|
r";Print\sTime:\s(\d+\.?\d*)", self.footer_data)
|
2020-07-02 04:21:35 +03:00
|
|
|
|
2020-09-08 02:37:43 +03:00
|
|
|
def parse_first_layer_extr_temp(self):
|
|
|
|
return _regex_find_first(
|
|
|
|
r"M109 T0 S(\d+\.?\d*)", self.header_data)
|
|
|
|
|
|
|
|
def parse_first_layer_bed_temp(self):
|
|
|
|
return _regex_find_first(
|
|
|
|
r"M190 S(\d+\.?\d*)", self.header_data)
|
|
|
|
|
2020-09-08 19:25:46 +03:00
|
|
|
class IceSL(BaseSlicer):
|
2020-09-09 15:27:41 +03:00
|
|
|
def check_identity(self, data):
|
|
|
|
match = re.search(r"; <IceSL.*>", data)
|
|
|
|
if match:
|
|
|
|
return {'slicer': "IceSL"}
|
|
|
|
return None
|
2020-09-08 19:25:46 +03:00
|
|
|
|
|
|
|
def parse_first_layer_height(self):
|
|
|
|
return _regex_find_first(
|
|
|
|
r"; z_layer_height_first_layer_mm :\s+(\d+\.\d+)",
|
|
|
|
self.header_data, float)
|
|
|
|
|
|
|
|
def parse_layer_height(self):
|
2021-01-16 20:38:37 +03:00
|
|
|
self.layer_height = _regex_find_first(
|
2020-09-08 19:25:46 +03:00
|
|
|
r"; z_layer_height_mm :\s+(\d+\.\d+)",
|
|
|
|
self.header_data, float)
|
2021-01-16 20:38:37 +03:00
|
|
|
return self.layer_height
|
2020-09-08 19:25:46 +03:00
|
|
|
|
|
|
|
def parse_object_height(self):
|
|
|
|
return self._parse_max_float(
|
|
|
|
r"G0 F\d+ Z\d+\.\d+", self.footer_data, strict=True)
|
|
|
|
|
|
|
|
def parse_first_layer_extr_temp(self):
|
|
|
|
return _regex_find_first(
|
|
|
|
r"; extruder_temp_degree_c_0 :\s+(\d+\.?\d*)", self.header_data)
|
|
|
|
|
|
|
|
def parse_first_layer_bed_temp(self):
|
|
|
|
return _regex_find_first(
|
|
|
|
r"; bed_temp_degree_c :\s+(\d+\.?\d*)", self.header_data)
|
|
|
|
|
2020-07-02 04:21:35 +03:00
|
|
|
|
|
|
|
READ_SIZE = 512 * 1024
|
|
|
|
SUPPORTED_SLICERS = [
|
|
|
|
PrusaSlicer, Slic3rPE, Slic3r, SuperSlicer,
|
2020-09-08 19:25:46 +03:00
|
|
|
Cura, Simplify3D, KISSlicer, IdeaMaker, IceSL]
|
2020-07-02 04:21:35 +03:00
|
|
|
SUPPORTED_DATA = [
|
2021-01-16 20:38:37 +03:00
|
|
|
'layer_height', 'first_layer_height', 'object_height',
|
2021-04-09 00:58:09 +03:00
|
|
|
'filament_total', 'filament_weight_total', 'estimated_time',
|
|
|
|
'thumbnails', 'first_layer_bed_temp', 'first_layer_extr_temp',
|
2020-11-12 22:37:21 +03:00
|
|
|
'gcode_start_byte', 'gcode_end_byte']
|
2020-07-02 04:21:35 +03:00
|
|
|
|
2021-02-23 20:47:34 +03:00
|
|
|
def extract_metadata(file_path):
|
2020-08-11 21:20:21 +03:00
|
|
|
metadata = {}
|
2020-10-30 15:56:00 +03:00
|
|
|
slicers = [s(file_path) for s in SUPPORTED_SLICERS]
|
2020-08-11 21:20:21 +03:00
|
|
|
header_data = footer_data = slicer = None
|
|
|
|
size = os.path.getsize(file_path)
|
|
|
|
metadata['size'] = size
|
2020-10-22 00:53:45 +03:00
|
|
|
metadata['modified'] = os.path.getmtime(file_path)
|
2020-08-11 21:20:21 +03:00
|
|
|
with open(file_path, 'r') as f:
|
|
|
|
# read the default size, which should be enough to
|
|
|
|
# identify the slicer
|
|
|
|
header_data = f.read(READ_SIZE)
|
|
|
|
for s in slicers:
|
2020-09-09 15:27:41 +03:00
|
|
|
ident = s.check_identity(header_data)
|
|
|
|
if ident is not None:
|
2020-08-11 21:20:21 +03:00
|
|
|
slicer = s
|
2020-09-09 15:27:41 +03:00
|
|
|
metadata.update(ident)
|
2020-08-11 21:20:21 +03:00
|
|
|
break
|
2020-09-08 00:01:01 +03:00
|
|
|
if slicer is None:
|
2020-10-30 15:56:00 +03:00
|
|
|
slicer = UnknownSlicer(file_path)
|
2020-09-09 15:27:41 +03:00
|
|
|
metadata['slicer'] = "Unknown"
|
2020-09-08 00:01:01 +03:00
|
|
|
if size > READ_SIZE * 2:
|
|
|
|
f.seek(size - READ_SIZE)
|
|
|
|
footer_data = f.read()
|
|
|
|
elif size > READ_SIZE:
|
|
|
|
remaining = size - READ_SIZE
|
|
|
|
footer_data = header_data[remaining - READ_SIZE:] + f.read()
|
2020-08-30 22:27:46 +03:00
|
|
|
else:
|
2020-09-08 00:01:01 +03:00
|
|
|
footer_data = header_data
|
2021-02-23 20:47:34 +03:00
|
|
|
slicer.set_data(header_data, footer_data, size)
|
2020-09-08 00:01:01 +03:00
|
|
|
for key in SUPPORTED_DATA:
|
|
|
|
func = getattr(slicer, "parse_" + key)
|
|
|
|
result = func()
|
|
|
|
if result is not None:
|
|
|
|
metadata[key] = result
|
2020-08-11 21:20:21 +03:00
|
|
|
return metadata
|
2020-07-02 04:21:35 +03:00
|
|
|
|
|
|
|
def main(path, filename):
|
|
|
|
file_path = os.path.join(path, filename)
|
|
|
|
metadata = {}
|
|
|
|
if not os.path.isfile(file_path):
|
2021-02-23 20:47:34 +03:00
|
|
|
log_to_stderr(f"File Not Found: {file_path}")
|
|
|
|
sys.exit(-1)
|
|
|
|
try:
|
|
|
|
metadata = extract_metadata(file_path)
|
|
|
|
except Exception:
|
|
|
|
log_to_stderr(traceback.format_exc())
|
|
|
|
sys.exit(-1)
|
2020-07-02 04:21:35 +03:00
|
|
|
fd = sys.stdout.fileno()
|
|
|
|
data = json.dumps(
|
2021-02-23 20:47:34 +03:00
|
|
|
{'file': filename, 'metadata': metadata}).encode()
|
2020-07-02 04:21:35 +03:00
|
|
|
while data:
|
|
|
|
try:
|
|
|
|
ret = os.write(fd, data)
|
|
|
|
except OSError:
|
|
|
|
continue
|
|
|
|
data = data[ret:]
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
# Parse start arguments
|
|
|
|
parser = argparse.ArgumentParser(
|
|
|
|
description="GCode Metadata Extraction Utility")
|
|
|
|
parser.add_argument(
|
|
|
|
"-f", "--filename", metavar='<filename>',
|
|
|
|
help="name gcode file to parse")
|
|
|
|
parser.add_argument(
|
|
|
|
"-p", "--path", default=os.path.abspath(os.path.dirname(__file__)),
|
|
|
|
metavar='<path>',
|
|
|
|
help="optional absolute path for file"
|
|
|
|
)
|
|
|
|
args = parser.parse_args()
|
|
|
|
main(args.path, args.filename)
|