Navigation

    Logo
    • Register
    • Login
    • Search
    • Recent
    • Tags
    • Unread
    • Categories
    • Unreplied
    • Popular
    • GitHub
    • Docu
    • Hilfe
    1. Home
    2. Deutsch
    3. Off Topic
    4. Meshtastic

    NEWS

    • ioBroker@Smart Living Forum Solingen, 14.06. - Agenda added

    • ioBroker goes Matter ... Matter Adapter in Stable

    • Monatsrückblick - April 2025

    Meshtastic

    This topic has been deleted. Only users with topic management privileges can see it.
    • M
      mark77 last edited by Homoran

      Moin,

      seit ein paar Tagen experimentiere ich mit meshtastic (meshtastic.org).

      Im Forum wollte ich mal schauen ob es gleichgesinnte gibt, die das System erfolgreich in ioBroker eingebunden haben und wenn ja, wie?

      Gruß,
      Mark

      T J 2 Replies Last reply Reply Quote 0
      • T
        ToGe88 Developer @mark77 last edited by

        @mark77 Moin, zwar schon ein bisschen älter der Beitrag aber ich antworte mal. Ich teste gerade auch ein wenig mit Meshtastic herum und habe mir ein Script erstellt welches über die Python Meshtastic CLI die Daten in den ioBroker holt. (Könnte man ggfs. auch zu einem Adapter umbauen zu einem späteren Zeitpunkt).

        Das Script erzeugt Datenpunkte um die Telemetrie Daten der Nodes zu überwachen, dafür werden diese alle 30 Sekunden vom Modul ausgelesen. Zudem ist es möglich Nachrichten an Chats, Direktnachrichten an Nodes, Traceroutes und Pings zu versenden. Der Empfang von Nachrichten ist aktuell nicht möglich. Ist alles noch mehr eine Bastellösung und im Testbetrieb. Die Integration ist aber schon sehr hilfreich wenn man zum Reichweitentest aus der Ferne z.B. Nachrichten von der "Basis" auslösen will oder prüfen möchte ob der mobile Node empfangen wird obwohl man dort kein ACK erhält. Möglich wäre auch eine Begrüßung von neuen Nodes auf dem Default Channel wenn sie in Reichweite der Basis mit ioBroker kommen 😉

        Wichtig ist das die Meshtastic CLI (https://meshtastic.org/docs/software/python/cli/installation/) für den User installiert wird unter welchem auch der ioBroker läuft. Ich nutze aktuell die Verbindung per WLAN, wenn es über die USB Schnittstelle laufen soll müsste man an den entsprechenden stellen die Befehle für exec anpassen.

        var deviceIp = '192.168.1.117';
        
        var chats = [
            { name: 'Default', id: 0 },
            { name: 'Benutzerkanal 1', id: 1 },
            { name: 'Benutzerkanal 2', id: 2 }
        ];
        
        
        function updateNodes() {
            exec('/home/iobroker/.local/bin/meshtastic --host '+deviceIp+' --nodes', async function (error, result, stderr) {
                if (result.includes('Connected to radio')) {
                    var nodes = parseData(result);
                    handleNodes(nodes);
                }
            });
        }
        
        function handleNodes(nodes) {
            nodes.forEach(node => {
                node.ID = node.ID.replace("!", "");
                if (nodeIsKnown(node.ID)) {
                    updateNode(node);
                } else {
                    createNode(node);
                    setTimeout(function() {
                        updateNode(node);
                    },4000);
                }
            });
        }
        
        function updateNode(data) {
        var nodeInfoStates = [
                {id: 'id', name: 'NodeID', type: 'string', unit: null, log: false},
                {id: 'user', name: 'User', type: 'string', unit: null, log: false},
                {id: 'alias', name: 'Alias', type: 'string', unit: null, log: false},
                {id: 'location', name: 'Location', type: 'string', unit: null, log: true},
                {id: 'altitude', name: 'Altitude', type: 'number', unit: 'm', log: true},
                {id: 'chanUtil', name: 'Channel util.', type: 'number', unit: '%', log: true},
                {id: 'txAir', name: 'Tx air util.', type: 'number', unit: '%', log: true},
                {id: 'snr', name: 'SNR', type: 'number', unit: 'db', log: true},
                {id: 'channel', name: 'Channel', type: 'string', unit: null, log: false},
                {id: 'lastHeard', name: 'Last heard', type: 'string', unit: null, log: true},
                {id: 'battery', name: 'Battery', type: 'number', unit: '%', log: true}
            ];
            setState('0_userdata.0.Meshtastic.Nodes.' + data.ID + '.info.id', data.N, true);
            setState('0_userdata.0.Meshtastic.Nodes.' + data.ID + '.info.user', data.User, true);
            setState('0_userdata.0.Meshtastic.Nodes.' + data.ID + '.info.alias', data.AKA, true);
            setState('0_userdata.0.Meshtastic.Nodes.' + data.ID + '.info.location', data.Latitude.replace('°', '') + ', ' + data.Longitude.replace('°', ''), true);
            setState('0_userdata.0.Meshtastic.Nodes.' + data.ID + '.info.altitude', parseFloat(data.Altitude.replace(' m', '')), true);
            setState('0_userdata.0.Meshtastic.Nodes.' + data.ID + '.info.chanUtil', parseFloat(data['Channel util.'].replace('%', '')), true);
            setState('0_userdata.0.Meshtastic.Nodes.' + data.ID + '.info.txAir', parseFloat(data['Tx air util.'].replace('%', '')), true);
            setState('0_userdata.0.Meshtastic.Nodes.' + data.ID + '.info.snr', parseFloat(data.SNR.replace(' dB', '')), true);
            setState('0_userdata.0.Meshtastic.Nodes.' + data.ID + '.info.channel', data.Channel, true);
            setState('0_userdata.0.Meshtastic.Nodes.' + data.ID + '.info.lastHeard', data.LastHeard, true);
            setState('0_userdata.0.Meshtastic.Nodes.' + data.ID + '.info.battery', parseFloat(data.Battery.replace('%', '')), true);
        }
        
        function createNode(data) {
            log('creating new node', 'info');
            var newNode = {
                "type": 'channel',
                "common": {
                    "name": data.User,
                }
            };
            setObject('0_userdata.0.Meshtastic.Nodes.' + data.ID, newNode);
            createNodeStates(data.ID);
        }
        
        function createNodeStates(id) {
        
            var infoChannel = {
                "type": 'channel',
                "common": {
                    "name": 'Info',
                }
            };
            setObject('0_userdata.0.Meshtastic.Nodes.' + id + '.info', infoChannel);
        
            var nodeInfoStates = [
                {id: 'id', name: 'NodeID', type: 'string', unit: null, log: false},
                {id: 'user', name: 'User', type: 'string', unit: null, log: false},
                {id: 'alias', name: 'Alias', type: 'string', unit: null, log: false},
                {id: 'location', name: 'Location', type: 'string', unit: null, log: true},
                {id: 'altitude', name: 'Altitude', type: 'number', unit: 'm', log: true},
                {id: 'chanUtil', name: 'Channel util.', type: 'number', unit: '%', log: true},
                {id: 'txAir', name: 'Tx air util.', type: 'number', unit: '%', log: true},
                {id: 'snr', name: 'SNR', type: 'number', unit: 'db', log: true},
                {id: 'channel', name: 'Channel', type: 'string', unit: null, log: false},
                {id: 'lastHeard', name: 'Last heard', type: 'string', unit: null, log: true},
                {id: 'battery', name: 'Battery', type: 'number', unit: '%', log: true}
            ];
        
            nodeInfoStates.forEach(state => {
                var currentObject = {
                    "type": 'state',
                    "common": {
                        "name": state.name,
                        "type": state.type,
                        "unit": state.unit,
                        "read": true,
                        "write": false,
                    }
                };
                setObject('0_userdata.0.Meshtastic.Nodes.' + id + '.info.' + state.id, currentObject);
            });
        
        
            var commandChannel = {
                "type": 'channel',
                "common": {
                    "name": 'Command',
                }
            };
            setObject('0_userdata.0.Meshtastic.Nodes.' + id + '.command', commandChannel);
        
            var nodeCommandStates = [
                {id: 'sendMessage', name: 'Direktnachricht an Node senden', type: 'string', role: 'value'},
                {id: 'sendPing', name: 'Ping an Node senden', type: 'boolean', role: 'button'},
                {id: 'sendTraceRoute', name: 'Traceroute auf Node starten', type: 'boolean', role: 'button'},
                {id: 'getLocation', name: 'Standort anfordern', type: 'boolean', role: 'button'},
                {id: 'getTelemetry', name: 'Telemetrie anfordern', type: 'boolean', role: 'button'},
            ];
        
            nodeCommandStates.forEach(state => {
                var currentObject = {
                    "type": 'state',
                    "common": {
                        "name": state.name,
                        "type": state.type,
                        "read": true,
                        "write": true,
                        "role": state.role,
                    }
                };
                setObject('0_userdata.0.Meshtastic.Nodes.' + id + '.command.' + state.id, currentObject);
            });
        } 
        
        function nodeIsKnown(id) {
            var nodeObject = getObject('0_userdata.0.Meshtastic.Nodes.' + id);
            if (nodeObject) { return true };
            return false;
        }
        
        function parseData(data) {
            // Split the data into lines
            const lines = data.trim().split('\n');
        
            // Extract the header line and data lines
            const headerLine = lines[2]; // Header line is the second line
            const dataLines = lines.slice(4, -1); // Skip the header, separator, and footer lines
        
            // Split the header line into keys
            const keys = headerLine.split('│').map(key => key.trim()).filter(key => key.length > 0);
        
            // Parse each data line into an object
            const objects = dataLines.map(line => {
                const values = line.split('│').map(value => value.trim()).filter(value => value.length > 0);
        
                // Skip lines that do not have the correct number of values
                if (values.length !== keys.length) {
                    return null;
                }
        
                let obj = {};
                keys.forEach((key, index) => {
                    obj[key] = values[index];
                });
                return obj;
            }).filter(obj => obj !== null); // Filter out null objects
        
            return objects;
        }
        
        function createChannels() {
            var baseChannel = {
                "type": 'channel',
                "common": {
                    "name": 'Meshtastic Server',
                }
            };
            setObject('0_userdata.0.Meshtastic', baseChannel);
        
            var baseChannel = {
                "type": 'channel',
                "common": {
                    "name": 'Nodes',
                }
            };
            setObject('0_userdata.0.Meshtastic.Nodes', baseChannel);
        
            var baseChannel = {
                "type": 'channel',
                "common": {
                    "name": 'Chats',
                }
            };
            setObject('0_userdata.0.Meshtastic.Chats', baseChannel);
        }
        
        function createChats() {
            chats.forEach(chatObj => {
                var chat = {
                    "type": 'channel',
                    "common": {
                        "name": chatObj.name,
                    }
                };
                setObject('0_userdata.0.Meshtastic.Chats.' + chatObj.id, chat);
                var chatCommandStates = [
                    {id: 'sendMessage', name: 'Nachricht an Chat senden', type: 'string', role: 'value'},
                ];
                chatCommandStates.forEach(state => {
                    var currentObject = {
                        "type": 'state',
                        "common": {
                            "name": state.name,
                            "type": state.type,
                            "read": true,
                            "write": true,
                            "role": state.role,
                        }
                    };
                    setObject('0_userdata.0.Meshtastic.Chats.' + chatObj.id + '.sendMessage', currentObject);
                });
            });
        }
        
        function registerEndpointListeners() {
            $('state[id=0_userdata.0.Meshtastic.Nodes.*.command.*]').each(function (id, i) {
                on({id: id, change: "any"}, function (obj) {
                    var id = obj.id;
                    var idParts = id.split('.');
                    var nodeId = idParts[4];
                    log(id, 'debug');
        
                    if (idParts[6] == 'getTelemetry' || idParts[6] == 'getLocation') {
                        
                        requestTelemetry(nodeId);
                    }
        
                    if (idParts[6] == 'sendPing') {
                        
                        sendPing(nodeId);
                    }
        
                    if (idParts[6] == 'sendTraceRoute') {
                        
                        startTraceroute(nodeId);
                    }
        
                    if (idParts[6] == 'sendMessage') {
                        var message = getState(id).val;
                        
                        sendDirectMessage(nodeId, message);
                    }
                });
            });
            $('state[id=0_userdata.0.Meshtastic.Chats.*.sendMessage]').each(function (id, i) {
                on({id: id, change: "any"}, function (obj) {
                    var id = obj.id;
                    var idParts = id.split('.');
                    var chatId = idParts[4];
                    var message = getState(id).val;
                    sendChatMessage(chatId, message);
                });
            });
        }
        
        function requestTelemetry(target, callback = null, counter = 0) {
            log('Requested Telemetry for node ' + target, 'info');
            exec('/home/iobroker/.local/bin/meshtastic --host '+deviceIp+' --request-telemetry --dest \'!'+target+'\'', async function (error, result, stderr) {
                if (result.includes('Connected to radio')) {
                    log('Telemetry success', 'info');
                    if (callback) {
                        callback();
                    }
                } else {
                    // Erneut versuchen
                    log(result, 'error');
                    counter = counter + 1;
                    if (counter <= 5) {
                        requestTelemetry(target, callback, counter);
                    }
                }
            });
        }
        
        function startTraceroute(target, callback = null, counter = 0) {
            log('Start traceroute for node ' + target, 'info');
            exec('/home/iobroker/.local/bin/meshtastic --host '+deviceIp+' --traceroute --dest \'!'+target+'\'', async function (error, result, stderr) {
                if (result.includes('Connected to radio')) {
                    log('Traceroute success', 'info');
                    if (callback) {
                        callback();
                    }
                } else {
                    // Erneut versuchen
                    log(result, 'error');
                    counter = counter + 1;
                    if (counter <= 5) {
                        startTraceroute(target, callback, counter);
                    }
                }
            });
        }
        
        function sendPing(target, callback = null, counter = 0) {
            log('Send ping for node ' + target, 'info');
            exec('/home/iobroker/.local/bin/meshtastic --host '+deviceIp+' --sendping --dest \'!'+target+'\'', async function (error, result, stderr) {
                if (result.includes('Connected to radio')) {
                    log('Ping success', 'info');
                    if (callback) {
                        callback();
                    }
                } else {
                    // Erneut versuchen
                    log(result, 'error');
                    counter = counter + 1;
                    if (counter <= 5) {
                        sendPing(target, callback, counter++);
                    }
                }
            });
        }
        
        function sendDirectMessage(target, message, callback = null, counter = 0) {
            log('Send message for node ' + target + ' message: ' + message, 'info');
            exec('/home/iobroker/.local/bin/meshtastic --host '+deviceIp+' --dest \'!'+target+'\' --sendtext "'+message+'"', async function (error, result, stderr) {
                if (result.includes('Connected to radio')) {
                    log('Message success', 'info');
                    if (callback) {
                        callback();
                    }
                } else {
                    // Erneut versuchen
                    log(result, 'error');
                    counter = counter + 1;
                    if (counter <= 5) {
                        sendDirectMessage(target, message, callback, counter);
                    }
                }
            });
        }
        
        function sendChatMessage(chatId, message, callback = null, counter = 0) {
            log('Send message for chat ' + chatId + ' message: ' + message, 'info');
            exec('/home/iobroker/.local/bin/meshtastic --host '+deviceIp+' --ch-index '+chatId+' --sendtext "'+message+'"', async function (error, result, stderr) {
                if (result.includes('Connected to radio')) {
                    log('Message success', 'info');
                    if (callback) {
                        callback();
                    }
                } else {
                    // Erneut versuchen
                    log(result, 'error');
                    counter = counter + 1;
                    if (counter <= 5) {
                        sendChatMessage(chatId, message, callback, counter);
                    }
                }
            });
        }
        
        registerEndpointListeners();
        createChats();
        createChannels();
        updateNodes();
        
        setInterval(function() {
            updateNodes();
        },30000);
        
        
        
        M 1 Reply Last reply Reply Quote 0
        • M
          mark77 @ToGe88 last edited by

          @toge88 sehr schön!

          Ein Adapter wäre natürlich traumhaft.

          Bei meinem Modul (diymore ESP32 LoRa [Amazon]) ärgert mich im Moment, dass wenn er die WLAN Verbindung verloren hat, neu gestartet werden muss.
          Wie ist das bei dir?

          Gruß,
          Mark

          1 Reply Last reply Reply Quote 0
          • J
            JimmyBondi @mark77 last edited by

            Ich reihe mich mal hier ein, auch wenn der Thread schon über 1 Jahr alt ist.

            Spiele ebenfalls aktuell ein wenig mit Mashtastic herum.
            Ein Adapter wäre schon prima.

            @ToGe88
            Danke für das Script !
            Habe alles erfolgreich installieren können und das Script läuft so weit.
            Allerdings werden mir nur Nodes angelegt, die in der App als "unknown" deklariert sind.
            Wenn ich mir die Node Liste in der Bash (via CLI) anzeigen lassen, dann haben diese den Channel 0 und bei allen anderen ist der Channel leer.
            Hast Du da ggf eine aktuellere Variante ?

            1 Reply Last reply Reply Quote 0
            • First post
              Last post

            Support us

            ioBroker
            Community Adapters
            Donate

            950
            Online

            31.7k
            Users

            79.7k
            Topics

            1.3m
            Posts

            3
            4
            592
            Loading More Posts
            • Oldest to Newest
            • Newest to Oldest
            • Most Votes
            Reply
            • Reply as topic
            Log in to reply
            Community
            Impressum | Datenschutz-Bestimmungen | Nutzungsbedingungen
            The ioBroker Community 2014-2023
            logo