diff --git a/data/flows.json b/data/flows.json
new file mode 100644
index 0000000000000000000000000000000000000000..0d842c4b57b6b2dd32282d6f7fa21b94cd12ff03
--- /dev/null
+++ b/data/flows.json
@@ -0,0 +1,713 @@
+[
+    {
+        "id": "d99ca802.0da3f8",
+        "type": "tab",
+        "label": "AI",
+        "disabled": false,
+        "info": ""
+    },
+    {
+        "id": "4c731c77a159b1a4",
+        "type": "ui_base",
+        "theme": {
+            "name": "theme-light",
+            "lightTheme": {
+                "default": "#0094CE",
+                "baseColor": "#0069b4",
+                "baseFont": "-apple-system,BlinkMacSystemFont,Segoe UI,Roboto,Oxygen-Sans,Ubuntu,Cantarell,Helvetica Neue,sans-serif",
+                "edited": true,
+                "reset": false
+            },
+            "darkTheme": {
+                "default": "#097479",
+                "baseColor": "#097479",
+                "baseFont": "-apple-system,BlinkMacSystemFont,Segoe UI,Roboto,Oxygen-Sans,Ubuntu,Cantarell,Helvetica Neue,sans-serif",
+                "edited": true,
+                "reset": false
+            },
+            "customTheme": {
+                "name": "Untitled Theme 1",
+                "default": "#4B7930",
+                "baseColor": "#4B7930",
+                "baseFont": "-apple-system,BlinkMacSystemFont,Segoe UI,Roboto,Oxygen-Sans,Ubuntu,Cantarell,Helvetica Neue,sans-serif"
+            },
+            "themeState": {
+                "base-color": {
+                    "default": "#0094CE",
+                    "value": "#0069b4",
+                    "edited": true
+                },
+                "page-titlebar-backgroundColor": {
+                    "value": "#0069b4",
+                    "edited": false
+                },
+                "page-backgroundColor": {
+                    "value": "#fafafa",
+                    "edited": false
+                },
+                "page-sidebar-backgroundColor": {
+                    "value": "#333333",
+                    "edited": false
+                },
+                "group-textColor": {
+                    "value": "#0195ff",
+                    "edited": false
+                },
+                "group-borderColor": {
+                    "value": "#ffffff",
+                    "edited": false
+                },
+                "group-backgroundColor": {
+                    "value": "#ffffff",
+                    "edited": false
+                },
+                "widget-textColor": {
+                    "value": "#111111",
+                    "edited": false
+                },
+                "widget-backgroundColor": {
+                    "value": "#0069b4",
+                    "edited": false
+                },
+                "widget-borderColor": {
+                    "value": "#ffffff",
+                    "edited": false
+                },
+                "base-font": {
+                    "value": "-apple-system,BlinkMacSystemFont,Segoe UI,Roboto,Oxygen-Sans,Ubuntu,Cantarell,Helvetica Neue,sans-serif"
+                }
+            },
+            "angularTheme": {
+                "primary": "indigo",
+                "accents": "blue",
+                "warn": "red",
+                "background": "grey",
+                "palette": "light"
+            }
+        },
+        "site": {
+            "name": "Sorting Line AI",
+            "hideToolbar": "false",
+            "allowSwipe": "mouse",
+            "lockMenu": "false",
+            "allowTempTheme": "false",
+            "dateFormat": "DD.MM.YYYY",
+            "sizes": {
+                "sx": 48,
+                "sy": 48,
+                "gx": 6,
+                "gy": 6,
+                "cx": 6,
+                "cy": 6,
+                "px": 0,
+                "py": 0
+            }
+        }
+    },
+    {
+        "id": "fa600c20.8a9c9",
+        "type": "mqtt-broker",
+        "name": "",
+        "broker": "localhost",
+        "port": "2883",
+        "clientid": "Node-RED",
+        "autoConnect": true,
+        "usetls": false,
+        "protocolVersion": "4",
+        "keepalive": "60",
+        "cleansession": true,
+        "birthTopic": "",
+        "birthQos": "0",
+        "birthPayload": "",
+        "birthMsg": {},
+        "closeTopic": "",
+        "closeQos": "0",
+        "closePayload": "",
+        "closeMsg": {},
+        "willTopic": "",
+        "willQos": "0",
+        "willPayload": "",
+        "willMsg": {},
+        "sessionExpiry": ""
+    },
+    {
+        "id": "61bbcb7d.153964",
+        "type": "ui_tab",
+        "name": "AI - Sorting Line",
+        "icon": "fa-magic",
+        "order": 1,
+        "disabled": false,
+        "hidden": false
+    },
+    {
+        "id": "269d9bd9.87bff4",
+        "type": "ui_group",
+        "name": "Picture Analysis",
+        "tab": "61bbcb7d.153964",
+        "order": 2,
+        "disp": true,
+        "width": "8",
+        "collapse": false,
+        "className": ""
+    },
+    {
+        "id": "5b21df6a.1635d",
+        "type": "mqtt-broker",
+        "name": "",
+        "broker": "127.0.0.1",
+        "port": "2883",
+        "clientid": "Node-RED",
+        "usetls": false,
+        "compatmode": true,
+        "keepalive": "60",
+        "cleansession": true,
+        "birthTopic": "",
+        "birthQos": "0",
+        "birthPayload": "",
+        "closeTopic": "",
+        "closeQos": "0",
+        "closePayload": "",
+        "willTopic": "",
+        "willQos": "0",
+        "willPayload": ""
+    },
+    {
+        "id": "1a1958760184e8e0",
+        "type": "ui_group",
+        "name": "Camera",
+        "tab": "61bbcb7d.153964",
+        "order": 1,
+        "disp": true,
+        "width": 8,
+        "collapse": false,
+        "className": ""
+    },
+    {
+        "id": "1e83b6b8321ff2a0",
+        "type": "ui_group",
+        "name": "3D-Model",
+        "tab": "453c284bb88a24a8",
+        "order": 3,
+        "disp": true,
+        "width": "20",
+        "collapse": true,
+        "className": ""
+    },
+    {
+        "id": "453c284bb88a24a8",
+        "type": "ui_tab",
+        "name": "Info",
+        "icon": "dashboard",
+        "order": 2,
+        "disabled": false,
+        "hidden": false
+    },
+    {
+        "id": "ff8d5e074a8d453a",
+        "type": "ui_spacer",
+        "z": "d99ca802.0da3f8",
+        "name": "spacer",
+        "group": "1a1958760184e8e0",
+        "order": 3,
+        "width": 8,
+        "height": 1
+    },
+    {
+        "id": "fa6b2a5657d4b008",
+        "type": "ui_spacer",
+        "z": "d99ca802.0da3f8",
+        "name": "spacer",
+        "group": "269d9bd9.87bff4",
+        "order": 5,
+        "width": 8,
+        "height": 1
+    },
+    {
+        "id": "cee19750d5a4dbc1",
+        "type": "ui_group",
+        "name": "Image",
+        "tab": "453c284bb88a24a8",
+        "order": 2,
+        "disp": true,
+        "width": 1,
+        "collapse": false,
+        "className": ""
+    },
+    {
+        "id": "8dc43d562e83e63b",
+        "type": "ui_spacer",
+        "z": "d99ca802.0da3f8",
+        "name": "spacer",
+        "group": "1e83b6b8321ff2a0",
+        "order": 1,
+        "width": 13,
+        "height": 1
+    },
+    {
+        "id": "b355756.8a49388",
+        "type": "ui_template",
+        "z": "d99ca802.0da3f8",
+        "group": "1a1958760184e8e0",
+        "name": "Camera view",
+        "order": 1,
+        "width": 8,
+        "height": 7,
+        "format": "<img width=\"100%\" src=\"{{msg.payload}}\">",
+        "storeOutMessages": true,
+        "fwdInMessages": true,
+        "resendOnRefresh": true,
+        "templateScope": "local",
+        "className": "",
+        "x": 850,
+        "y": 320,
+        "wires": [
+            []
+        ]
+    },
+    {
+        "id": "fa038b47.d0ae88",
+        "type": "mqtt in",
+        "z": "d99ca802.0da3f8",
+        "name": "MQTT-Input",
+        "topic": "i/cam",
+        "qos": "0",
+        "datatype": "json",
+        "broker": "fa600c20.8a9c9",
+        "nl": false,
+        "rap": false,
+        "inputs": 0,
+        "x": 350,
+        "y": 220,
+        "wires": [
+            [
+                "ceec5f61.0f45d",
+                "2d7f9f4beb3e4506",
+                "57cc37a48caf664e",
+                "9e1b0a65d9fd2c5b",
+                "a5d28a603d92c7b1",
+                "cc14f456d4859747",
+                "d03fb41.9611448"
+            ]
+        ]
+    },
+    {
+        "id": "ceec5f61.0f45d",
+        "type": "function",
+        "z": "d99ca802.0da3f8",
+        "name": "color",
+        "func": "if(msg.payload.color == \"3\"){\n    msg.payload = \"Blue\";\n} else if(msg.payload.color == \"2\"){\n    msg.payload = \"Red\";\n}else{\n    msg.payload = \"White\";\n}\nreturn msg;",
+        "outputs": 1,
+        "noerr": 0,
+        "initialize": "",
+        "finalize": "",
+        "libs": [],
+        "x": 590,
+        "y": 200,
+        "wires": [
+            [
+                "a1b6297b.a8ecf8"
+            ]
+        ]
+    },
+    {
+        "id": "a1b6297b.a8ecf8",
+        "type": "ui_text",
+        "z": "d99ca802.0da3f8",
+        "group": "269d9bd9.87bff4",
+        "order": 1,
+        "width": 0,
+        "height": 0,
+        "name": "",
+        "label": "Color of part",
+        "format": "{{msg.payload}}",
+        "layout": "row-spread",
+        "className": "",
+        "x": 850,
+        "y": 200,
+        "wires": []
+    },
+    {
+        "id": "2d7f9f4beb3e4506",
+        "type": "function",
+        "z": "d99ca802.0da3f8",
+        "name": "description",
+        "func": "msg.payload = msg.payload.description\nreturn msg;",
+        "outputs": 1,
+        "noerr": 0,
+        "initialize": "",
+        "finalize": "",
+        "libs": [],
+        "x": 610,
+        "y": 160,
+        "wires": [
+            [
+                "7ce0f3ee9e447bed"
+            ]
+        ]
+    },
+    {
+        "id": "7ce0f3ee9e447bed",
+        "type": "ui_text",
+        "z": "d99ca802.0da3f8",
+        "group": "269d9bd9.87bff4",
+        "order": 2,
+        "width": 0,
+        "height": 0,
+        "name": "",
+        "label": "Description",
+        "format": "{{msg.payload}}",
+        "layout": "row-spread",
+        "className": "",
+        "x": 850,
+        "y": 160,
+        "wires": []
+    },
+    {
+        "id": "57cc37a48caf664e",
+        "type": "function",
+        "z": "d99ca802.0da3f8",
+        "name": "bay",
+        "func": "msg.payload = msg.payload.bay\nreturn msg;",
+        "outputs": 1,
+        "noerr": 0,
+        "initialize": "",
+        "finalize": "",
+        "libs": [],
+        "x": 590,
+        "y": 280,
+        "wires": [
+            [
+                "775b2c1ea45ef95e"
+            ]
+        ]
+    },
+    {
+        "id": "775b2c1ea45ef95e",
+        "type": "ui_text",
+        "z": "d99ca802.0da3f8",
+        "group": "269d9bd9.87bff4",
+        "order": 3,
+        "width": 0,
+        "height": 0,
+        "name": "",
+        "label": "Bay / Destination",
+        "format": "{{msg.payload}}",
+        "layout": "row-spread",
+        "className": "",
+        "x": 870,
+        "y": 280,
+        "wires": []
+    },
+    {
+        "id": "9e1b0a65d9fd2c5b",
+        "type": "function",
+        "z": "d99ca802.0da3f8",
+        "name": "timestamp",
+        "func": "var date = new Date(msg.payload.ts);\nmsg.payload = date.toLocaleString('de-DE');\nreturn msg;",
+        "outputs": 1,
+        "noerr": 0,
+        "initialize": "",
+        "finalize": "",
+        "libs": [],
+        "x": 610,
+        "y": 120,
+        "wires": [
+            [
+                "89a7e957f94b3bc8"
+            ]
+        ]
+    },
+    {
+        "id": "89a7e957f94b3bc8",
+        "type": "ui_text",
+        "z": "d99ca802.0da3f8",
+        "group": "1a1958760184e8e0",
+        "order": 4,
+        "width": 0,
+        "height": 0,
+        "name": "",
+        "label": "Timestamp Picture",
+        "format": "{{msg.payload}}",
+        "layout": "row-spread",
+        "className": "",
+        "x": 870,
+        "y": 120,
+        "wires": []
+    },
+    {
+        "id": "a5d28a603d92c7b1",
+        "type": "function",
+        "z": "d99ca802.0da3f8",
+        "name": "confidence",
+        "func": "msg.payload = (parseFloat(msg.payload.confidence)*100).toFixed(2);\nreturn msg;",
+        "outputs": 1,
+        "noerr": 0,
+        "initialize": "",
+        "finalize": "",
+        "libs": [],
+        "x": 610,
+        "y": 240,
+        "wires": [
+            [
+                "aa886023ef6da84c"
+            ]
+        ]
+    },
+    {
+        "id": "aa886023ef6da84c",
+        "type": "ui_gauge",
+        "z": "d99ca802.0da3f8",
+        "name": "",
+        "group": "269d9bd9.87bff4",
+        "order": 4,
+        "width": 0,
+        "height": 0,
+        "gtype": "gage",
+        "title": "Confidence",
+        "label": "%",
+        "format": "{{msg.payload}}",
+        "min": "50",
+        "max": "100",
+        "colors": [
+            "#d13f00",
+            "#f0d105",
+            "#0ed100"
+        ],
+        "seg1": "70",
+        "seg2": "85",
+        "className": "",
+        "x": 850,
+        "y": 240,
+        "wires": []
+    },
+    {
+        "id": "d5f01e9c9767a3f7",
+        "type": "ui_text",
+        "z": "d99ca802.0da3f8",
+        "group": "1a1958760184e8e0",
+        "order": 2,
+        "width": 8,
+        "height": 1,
+        "name": "fischertechnik Logo",
+        "label": "",
+        "format": "{{msg.payload}}",
+        "layout": "row-left",
+        "className": "width:100%;",
+        "x": 870,
+        "y": 400,
+        "wires": []
+    },
+    {
+        "id": "cc14f456d4859747",
+        "type": "function",
+        "z": "d99ca802.0da3f8",
+        "name": "data",
+        "func": "msg.payload = (((parseFloat(msg.payload.duration)).toFixed(0)) / 1000);\nreturn msg;",
+        "outputs": 1,
+        "noerr": 0,
+        "initialize": "",
+        "finalize": "",
+        "libs": [],
+        "x": 590,
+        "y": 360,
+        "wires": [
+            [
+                "05494373e012e40b"
+            ]
+        ]
+    },
+    {
+        "id": "05494373e012e40b",
+        "type": "ui_text",
+        "z": "d99ca802.0da3f8",
+        "group": "269d9bd9.87bff4",
+        "order": 6,
+        "width": 0,
+        "height": 0,
+        "name": "",
+        "label": "Processing Time",
+        "format": "{{msg.payload}}s",
+        "layout": "row-spread",
+        "className": "",
+        "x": 860,
+        "y": 360,
+        "wires": []
+    },
+    {
+        "id": "c7e341a0.381cc",
+        "type": "http in",
+        "z": "d99ca802.0da3f8",
+        "name": "",
+        "url": "/3d-model",
+        "method": "get",
+        "upload": false,
+        "swaggerDoc": "",
+        "x": 360,
+        "y": 720,
+        "wires": [
+            [
+                "2fb1c354.d04e3c"
+            ]
+        ]
+    },
+    {
+        "id": "2fb1c354.d04e3c",
+        "type": "file in",
+        "z": "d99ca802.0da3f8",
+        "name": "3d-model",
+        "filename": "/opt/ft/workspaces/webfiles/Sortierstrecke9V_KI.htm",
+        "format": "",
+        "allProps": false,
+        "x": 520,
+        "y": 720,
+        "wires": [
+            [
+                "c9e28681.361d78"
+            ]
+        ]
+    },
+    {
+        "id": "c9e28681.361d78",
+        "type": "change",
+        "z": "d99ca802.0da3f8",
+        "name": "Set Headers",
+        "rules": [
+            {
+                "t": "set",
+                "p": "headers",
+                "pt": "msg",
+                "to": "{}",
+                "tot": "json"
+            },
+            {
+                "t": "set",
+                "p": "headers.content-type",
+                "pt": "msg",
+                "to": "text/html",
+                "tot": "str"
+            }
+        ],
+        "action": "",
+        "property": "",
+        "from": "",
+        "to": "",
+        "reg": false,
+        "x": 670,
+        "y": 720,
+        "wires": [
+            [
+                "88974243.7768c"
+            ]
+        ]
+    },
+    {
+        "id": "88974243.7768c",
+        "type": "http response",
+        "z": "d99ca802.0da3f8",
+        "name": "",
+        "x": 830,
+        "y": 720,
+        "wires": []
+    },
+    {
+        "id": "c0d87ee367e93a5e",
+        "type": "ui_template",
+        "z": "d99ca802.0da3f8",
+        "group": "1e83b6b8321ff2a0",
+        "name": "3d Model",
+        "order": 3,
+        "width": 20,
+        "height": 13,
+        "format": "<iframe style=\"overflow:none;border:none;\" height=\"100%\" width=\"100%\" src=\"/3d-model\"></iframe>",
+        "storeOutMessages": true,
+        "fwdInMessages": true,
+        "resendOnRefresh": true,
+        "templateScope": "local",
+        "className": "",
+        "x": 840,
+        "y": 580,
+        "wires": [
+            []
+        ]
+    },
+    {
+        "id": "d03fb41.9611448",
+        "type": "function",
+        "z": "d99ca802.0da3f8",
+        "name": "data",
+        "func": "msg.payload = msg.payload.data;\nreturn msg;",
+        "outputs": 1,
+        "noerr": 0,
+        "initialize": "",
+        "finalize": "",
+        "libs": [],
+        "x": 590,
+        "y": 320,
+        "wires": [
+            [
+                "b355756.8a49388"
+            ]
+        ]
+    },
+    {
+        "id": "16d94aa113a07b80",
+        "type": "ui_ui_control",
+        "z": "d99ca802.0da3f8",
+        "name": "UI Change",
+        "events": "all",
+        "x": 350,
+        "y": 540,
+        "wires": [
+            [
+                "41df6c4dcad9807e"
+            ]
+        ]
+    },
+    {
+        "id": "41df6c4dcad9807e",
+        "type": "template",
+        "z": "d99ca802.0da3f8",
+        "name": "Base64 Logo",
+        "field": "payload",
+        "fieldType": "msg",
+        "format": "handlebars",
+        "syntax": "mustache",
+        "template": "<a href=\"https://www.fischertechnik.de\" target=\"_blank\"><img width=\"100%\" loading=\"lazy\" defer src=\"\"/></a>",
+        "output": "str",
+        "x": 610,
+        "y": 540,
+        "wires": [
+            [
+                "f5a051caac403ce1",
+                "d5f01e9c9767a3f7"
+            ]
+        ]
+    },
+    {
+        "id": "f5a051caac403ce1",
+        "type": "ui_text",
+        "z": "d99ca802.0da3f8",
+        "group": "1e83b6b8321ff2a0",
+        "order": 2,
+        "width": 7,
+        "height": 1,
+        "name": "fischertechnik Logo",
+        "label": "",
+        "format": "{{msg.payload}}",
+        "layout": "row-left",
+        "className": "width:100%;",
+        "x": 870,
+        "y": 540,
+        "wires": []
+    },
+    {
+        "id": "c570ec00e003b3b2",
+        "type": "comment",
+        "z": "d99ca802.0da3f8",
+        "name": "Flow Version: 2022/10/24",
+        "info": "",
+        "x": 390,
+        "y": 40,
+        "wires": []
+    }
+]
\ No newline at end of file
diff --git a/lib/display.qml b/lib/display.qml
index 0c7cb5af50c305c32bce050185a261193e50c95d..01bfff5d43b2de4f6128219e071edc1544ee0aa3 100644
--- a/lib/display.qml
+++ b/lib/display.qml
@@ -16,8 +16,8 @@ TXTWindow {
     text: ""
     font.pixelSize: 16
     elide: Text.ElideRight
-    x: 17
-    y: 19
+    x: 20
+    y: 20
     width: 200
     height: 150
   }
@@ -25,7 +25,7 @@ TXTWindow {
     id: red
     color: "#EC1600"
     active: false
-    x: 71
+    x: 70
     y: 195
     width: 35
     height: 35
@@ -34,8 +34,8 @@ TXTWindow {
     id: white
     color: "#FFFFFF"
     active: false
-    x: 1
-    y: 194
+    x: 0
+    y: 195
     width: 35
     height: 35
   }
@@ -43,8 +43,8 @@ TXTWindow {
     id: blue
     color: "#005693"
     active: false
-    x: 137
-    y: 194
+    x: 135
+    y: 195
     width: 35
     height: 35
   }
@@ -52,19 +52,19 @@ TXTWindow {
     id: fail
     color: "#E7AE13"
     active: false
-    x: 204
+    x: 205
     y: 195
     width: 35
     height: 35
   }
   TXTLabel {
     id: version_label
-    text: "<small>Sorting Line AI: Version 2022/11/24</small>"
+    text: "Sorting Line AI: Version 2023/01/17"
     font.pixelSize: 16
     elide: Text.ElideRight
-    x: 19
+    x: 20
     y: 0
-    width: 201
+    width: 200
     height: 20
   }
   TXTLabel {
@@ -75,6 +75,6 @@ TXTWindow {
     x: 20
     y: 170
     width: 200
-    height: 22
+    height: 20
   }
 }
diff --git a/lib/display.xml b/lib/display.xml
index da339d6323f36988e334b4f3c642a053c3a69441..00c1619be5820997cdb23c8c8632e7df4e839b11 100644
--- a/lib/display.xml
+++ b/lib/display.xml
@@ -1 +1 @@
-<xml type="display" version="2"><item id="33" class="TXTLabel"><name>img_label</name><text/><geometry><x>17</x><y>19</y><width>200</width><height>150</height></geometry></item><item id="20" class="StatusIndicator"><name>red</name><color>#EC1600</color><active>false</active><geometry><x>71</x><y>195</y><width>35</width><height>35</height></geometry></item><item id="21" class="StatusIndicator"><name>white</name><color>#FFFFFF</color><active>false</active><geometry><x>1</x><y>194</y><width>35</width><height>35</height></geometry></item><item id="22" class="StatusIndicator"><name>blue</name><color>#005693</color><active>false</active><geometry><x>137</x><y>194</y><width>35</width><height>35</height></geometry></item><item id="23" class="StatusIndicator"><name>fail</name><color>#E7AE13</color><active>false</active><geometry><x>204</x><y>195</y><width>35</width><height>35</height></geometry></item><item id="24" class="TXTLabel"><name>version_label</name><text>&lt;small&gt;Sorting Line AI: Version 2022/11/24&lt;/small&gt;</text><geometry><x>19</x><y>0</y><width>201</width><height>20</height></geometry></item><item id="28" class="TXTLabel"><name>part_pass_fail</name><text>&lt;font&gt;Webadress loading...&lt;/font&gt;</text><geometry><x>20</x><y>170</y><width>200</width><height>22</height></geometry></item></xml>
\ No newline at end of file
+<xml type="display" version="2"><item id="33" class="TXTLabel"><name>img_label</name><text/><geometry><x>20</x><y>20</y><width>200</width><height>150</height></geometry></item><item id="20" class="StatusIndicator"><name>red</name><color>#EC1600</color><active>false</active><geometry><x>70</x><y>195</y><width>35</width><height>35</height></geometry></item><item id="21" class="StatusIndicator"><name>white</name><color>#FFFFFF</color><active>false</active><geometry><x>0</x><y>195</y><width>35</width><height>35</height></geometry></item><item id="22" class="StatusIndicator"><name>blue</name><color>#005693</color><active>false</active><geometry><x>135</x><y>195</y><width>35</width><height>35</height></geometry></item><item id="23" class="StatusIndicator"><name>fail</name><color>#E7AE13</color><active>false</active><geometry><x>205</x><y>195</y><width>35</width><height>35</height></geometry></item><item id="24" class="TXTLabel"><name>version_label</name><text>Sorting Line AI: Version 2023/01/17</text><geometry><x>20</x><y>0</y><width>200</width><height>20</height></geometry></item><item id="28" class="TXTLabel"><name>part_pass_fail</name><text>&lt;font&gt;Webadress loading...&lt;/font&gt;</text><geometry><x>20</x><y>170</y><width>200</width><height>20</height></geometry></item></xml>
\ No newline at end of file
diff --git a/lib/machine_learning.py b/lib/machine_learning.py
index 9671f7e84f9a6a76b6201dddc7bd4095f8c32ff2..6552c7fd2c290b57b3115babb1a2b48ce4002f42 100644
--- a/lib/machine_learning.py
+++ b/lib/machine_learning.py
@@ -14,6 +14,7 @@ from lib.camera import *
 from lib.controller import *
 from lib.display import *
 from lib.node_red import *
+from lib.node_red import *
 
 
 tag = None
@@ -29,27 +30,16 @@ prob = None
 keytext = None
 pos = None
 frame = None
+ts_process0 = None
 detector = None
 result = None
+ts_proces = None
 key = None
-
-
-def reset_inteface():
-    global tag, value, num, color, ts, filename, sat, hue, duration, prob, keytext, pos, frame, detector, result, key
-    display.set_attr("part_pass_fail.text", str(containInHTML('i', 'Not analysed yet')))
-    display.set_attr("red.active", str(False).lower())
-    display.set_attr("white.active", str(False).lower())
-    display.set_attr("blue.active", str(False).lower())
-    display.set_attr("fail.active", str(False).lower())
-
-
-def containInHTML(tag, value):
-    global num, color, ts, filename, sat, hue, duration, prob, keytext, pos, frame, detector, result, key
-    return ''.join([str(x) for x in ['<', tag, '>', value, '</', tag, '>']])
+log = None
 
 
 def MakePictureRunKiReturnFoundPart():
-    global tag, value, num, color, ts, filename, sat, hue, duration, prob, keytext, pos, frame, detector, result, key
+    global tag, value, num, color, ts, filename, sat, hue, duration, prob, keytext, pos, frame, ts_process0, detector, result, ts_proces, key, log
     reset_inteface()
     TXT_SLD_M_O4_led.set_brightness(int(512))
     time.sleep(0.2)
@@ -69,9 +59,12 @@ def MakePictureRunKiReturnFoundPart():
     sat = color[2] # range 0-255
     TXT_SLD_M_C1_motor_step_counter.reset()
     TXT_SLD_M_M1_encodermotor.set_speed(int(160), Motor.CCW)
-    TXT_SLD_M_M1_encodermotor.set_distance(int(240))
+    TXT_SLD_M_M1_encodermotor.set_distance(int(200))
+    ts_process0 = (time.time() * 1000)
     detector = ObjectDetector('/opt/ft/workspaces/machine-learning/object-detection/sorting_line/model.tflite', '/opt/ft/workspaces/machine-learning/object-detection/sorting_line/labels.txt')
     result = detector.process_image(frame)
+    ts_proces = (time.time() * 1000)
+    print('processing time: {:.0f} ms'.format(ts_proces - ts_process0))
     color = get_color()
     TXT_SLD_M_O4_led.set_brightness(int(0))
     print(result)
@@ -79,6 +72,10 @@ def MakePictureRunKiReturnFoundPart():
         prob = result[0]['probability']
         pos = result[0]['position']
         key = result[0]['label']
+        if True:
+            log = open('/opt/ft/workspaces/log.txt', 'a', encoding='utf8')
+            log.write('{} ms: {:.0f} key: {} prob: {:.2f} color: {}'.format(timestamp(), ts_proces - ts_process0, key, prob, color) + '\n')
+            log.close()
         keytext = ''
         if key == 'CRACK':
             keytext = 'Cracks in Workpiece'
@@ -122,8 +119,22 @@ def MakePictureRunKiReturnFoundPart():
     return num
 
 
+def reset_inteface():
+    global tag, value, num, color, ts, filename, sat, hue, duration, prob, keytext, pos, frame, ts_process0, detector, result, ts_proces, key, log
+    display.set_attr("part_pass_fail.text", str(containInHTML('i', 'Not analysed yet')))
+    display.set_attr("red.active", str(False).lower())
+    display.set_attr("white.active", str(False).lower())
+    display.set_attr("blue.active", str(False).lower())
+    display.set_attr("fail.active", str(False).lower())
+
+
+def containInHTML(tag, value):
+    global num, color, ts, filename, sat, hue, duration, prob, keytext, pos, frame, ts_process0, detector, result, ts_proces, key, log
+    return ''.join([str(x) for x in ['<', tag, '>', value, '</', tag, '>']])
+
+
 def get_color():
-    global tag, value, num, color, ts, filename, sat, hue, duration, prob, keytext, pos, frame, detector, result, key
+    global tag, value, num, color, ts, filename, sat, hue, duration, prob, keytext, pos, frame, ts_process0, detector, result, ts_proces, key, log
     if hue >= 85 and hue < 130 and sat >= 40:
         color = 3
     elif (hue >= 130 and hue <= 180 or hue >= 0 and hue < 15) and sat >= 40:
@@ -134,13 +145,13 @@ def get_color():
 
 
 def timestamp():
-    global tag, value, num, color, ts, filename, sat, hue, duration, prob, keytext, pos, frame, detector, result, key
+    global tag, value, num, color, ts, filename, sat, hue, duration, prob, keytext, pos, frame, ts_process0, detector, result, ts_proces, key, log
     ts = datetime.now().strftime("%Y-%m-%dT%H:%M:%S.%f")[:-3] + "Z"
     return ts
 
 
 def saveFileandPublish():
-    global tag, value, num, color, ts, filename, sat, hue, duration, prob, keytext, pos, frame, detector, result, key
+    global tag, value, num, color, ts, filename, sat, hue, duration, prob, keytext, pos, frame, ts_process0, detector, result, ts_proces, key, log
     filename = '/opt/ft/workspaces/last-image.png'
     if(pos != ""):
         image = cv2.rectangle(frame, (pos[0], pos[1]), (pos[2], pos[3]), (180,105,0), 2)