Weiter zum Inhalt

Skripten / Logik

16.7k Themen 215.4k Beiträge

Hilfe zu JavaScript, Blockly, TypeScript, Node-RED, Scenes und text2command

NEWS

Unterkategorien


  • Hilfe für Skripterstellung mit JavaScript

    3k 50k
    3k Themen
    50k Beiträge
    maxclaudiM
    @Murphy-0 sagte: Dein Blockly ist so schön strukturiert. Chapeau ! Meins sieht mehr nach Spaghetti aus Dankeschön, jetzt habe ich Hunger :-)
  • Hilfe für Skripterstellung mit Blockly

    7k 80k
    7k Themen
    80k Beiträge
    G
    Das werde ich mir mal anschauen. Danke.
  • Hilfe für Skripterstellung mit Node-RED

    961 13k
    961 Themen
    13k Beiträge
    P
    Der "normale" TCP Request kann auch Rohdaten. Zumindest habe ich darüber damals mit einem Pioneer AVR gesprochen. Den Flow habe ich nicht mehr. Zum Testen / Probieren ein Grundgerüst: Spoiler [ { "id": "telnet_flow_group", "type": "group", "style": { "label": true }, "nodes": [ "inject_trigger", "change_add_newline", "tcp_telnet_request", "debug_response" ], "x": 34, "y": 79, "w": 712, "h": 122 }, { "id": "inject_trigger", "type": "inject", "z": "b59f3d99b19bc9e5", "g": "telnet_flow_group", "name": "Befehl senden", "props": [ { "p": "payload", "v": "help", "vt": "str" } ], "repeat": "", "crontab": "", "once": false, "onceDelay": 0.1, "topic": "", "x": 150, "y": 140, "wires": [ [ "change_add_newline" ] ] }, { "id": "change_add_newline", "type": "change", "z": "b59f3d99b19bc9e5", "g": "telnet_flow_group", "name": "Zeilenumbruch (\\r\\n)", "rules": [ { "t": "set", "p": "payload", "pt": "msg", "to": "$msg.payload & \"\\r\\n\"", "tot": "jsonata" } ], "action": "", "property": "", "from": "", "to": "", "reg": false, "x": 360, "y": 140, "wires": [ [ "tcp_telnet_request" ] ] }, { "id": "tcp_telnet_request", "type": "tcp request", "z": "b59f3d99b19bc9e5", "g": "telnet_flow_group", "server": "localhost", "port": "23", "out": "time", "splitc": "0", "name": "Telnet Server", "ret": "string", "newline": "", "tls": "", "x": 570, "y": 140, "wires": [ [ "debug_response" ] ] }, { "id": "debug_response", "type": "debug", "z": "b59f3d99b19bc9e5", "g": "telnet_flow_group", "name": "Antwort", "active": true, "tosidebar": true, "console": false, "tostatus": false, "complete": "payload", "targetType": "msg", "statusVal": "", "statusType": "auto", "x": 720, "y": 140, "wires": [] } ]
  • Scriptsammlung Vol. 2

    Angeheftet
    3
    3 Stimmen
    3 Beiträge
    5k Aufrufe
    NegaleinN
    Achtung: Diese Scripts sind teils auch ungetestet bzw. nur vom Ersteller getestet worden. Blockly diverse Scripte Schimpfwortgenerator (BananaJoe, Nikolai Radke) Ein Schimpfwortgenerator ioBroker-Forum-Thread: Schimpfwortgenerator Witze aus API (mading) Ein Witzegenerator ioBroker-Forum-Thread: Witzegenerator Bilder mittels LLM ChatGPT Vision ananalysieren (David G.) Bilder mit ChatGPT ananalysieren ioBroker-Forum-Thread: Bilder mittels LLM ChatGPT Vision ananalysieren Visualisierung Agentdvr-Aufnahmen in der Visualisierung darstellen (David G.) Agentdvr-Aufnahmen anzeigen ioBroker-Forum-Thread: Agentdvr-Aufnahmen in der Visualisierung darstellen Trash HTML Widget VIS2 (skvarel) Trash HTML Widget VIS2 ioBroker-Forum-Thread: Trash HTML Widget VIS2 GitHub GitHub
  • Scriptsammlung Vol. 2 -- Diskussion

    Angeheftet
    69
    1 Stimmen
    69 Beiträge
    15k Aufrufe
    NegaleinN
    @Ro75 sagte: Vielen Dank. Ro75. hab es soeben in die Sammlung aufgenommen :)
  • Abwesenheitszeiten für Behörden nachweisen

    6
    0 Stimmen
    6 Beiträge
    240 Aufrufe
    Dr. BakteriusD
    @horst-böttcher zwei Anregungen: Traccar - zeichnet alle Touren mit dem Auto (oder auch Handy) auf. Kann lokal gehostet werden. NFC und Handy-App. Beim Verlassen oder Betreten der Wohnung Handy kurz zu einem Button halten. Oder mit etwas mehr Aufwand einen Chip-Leser und entsprechende Schlüsselanhänger der per MQTT den Code des Anhängers an ioBroker sendet. Beim Verlassen und Betreten Schlüssel einfach kurz hinhalten. (Ich mache das so um die Alarmanlage zu steuern und die Türe zuzusperren - funktioniert seit Jahren, nach ein wenig Bastelei, sehr zuverlässig.)
  • Shuttercontrol Sonnenschutz funktioniert nicht (Homematic)

    6
    6
    0 Stimmen
    6 Beiträge
    572 Aufrufe
    M
    Ich habe leider auch das Problem, dass der Sonnenschutz nicht funktioniert. Zeitgesteuert öffnen und schließen sich die Rollladen. Nur der Sonnenschutz will einfach nicht funktionieren...ich hoffe ihr könnt mir helfen 😊 Ich verwende Homematic HM-LC-Bl1-FM Funk-Rollladenaktoren und eine CCU2. Die Steuerung über ioBroker funktioniert manuell einwandfrei: Klicke ich bei "Steuerung und Überwachung von Shuttercontrol" auf "Sonnenschutz aller Rollläden aktivieren" fahren alle Rollladen in die für den Sonnenschutz voreingestellte Position. Auch die Zeitsteuerung mit Shuttercontrol funktioniert. Ich möchte die Rollladen gerne nach Außentemperatur, Lichtsensor & Himmelsrichtung steuern, sodass nur sonnenbeschienene Fenster beschattet werden. (Außentemperatur und Helligkeitswert für den Lichtsensor liefert eine Homematic Wetterstation) Dazu habe ich folgende Einstellungen vorgenommen: [image: 1782372103978-shuttercontrol_sonnenschutzeinstellungen-resized.png] [image: 1782372115546-shuttercontrol_haupteinstellungen-resized.png] [image: 1782372029313-shuttercontrol_extraeinstellungen_sonnenschutzeinstellungen-resized.png] Bei einem Azimut von derzeit 94°, einer Außentemperatur von 30 °C und einem Helligkeitswert von 192 bleibt der Rollladen geöffnet. Könnt ihr mir sagen, was ich falsch mache?
  • Blockly funktioniert plötzlich nicht mehr

    3
    1
    0 Stimmen
    3 Beiträge
    129 Aufrufe
    H
    @mickemup Habe das jetzt mal so gemacht und einen Test gestartet. Hat funktioniert...mal schauen ob es dann morgen früh um 5 Uhr auch so ist. DP wurde nicht geändert. [image: 1782364412368-6ce793c6-45e0-44fa-abbb-fbbeb670f147-image.jpeg]
  • Keylose API für offene Stadtdaten (DWD, Pegel, Strom)

    javascript
    1
    2 Stimmen
    1 Beiträge
    79 Aufrufe
    Niemand hat geantwortet
  • Script NP-Regelung HMS-800-2t (ODTU) + HM 1950AC (KI)

    1
    0 Stimmen
    1 Beiträge
    37 Aufrufe
    Niemand hat geantwortet
  • Alexa Shopping List mit Bring synchronisieren

    184
    0 Stimmen
    184 Beiträge
    45k Aufrufe
    I
    @schiefvancleef danke, eine Umsetzung auf Groß/Klein wäre noch super
  • Heizstab per %-Werte flexibel schalten je nach Einspeisung

    13
    0 Stimmen
    13 Beiträge
    325 Aufrufe
    paul53P
    @rtwl [sagte]: eventuell sind 55° auch schon ausreichend. Davon bin ich überzeugt. Ich habe es im letzten Vorschlag angepasst.
  • mein Kampf mit true und false

    14
    0 Stimmen
    14 Beiträge
    327 Aufrufe
    HomoranH
    @OliverIO sagte: leider ist es bei vis1 so das die meisten datenpunkte immer als string ankommen Ääähm, nein! @OliverIO sagte: egal welcher typ man am datenpunkt konfiguriert hat. Das, ja! Der Typ des DP hat keinen Einfluss auf das Format des Widgets
  • VCF Datei auslesen

    2
    0 Stimmen
    2 Beiträge
    104 Aufrufe
    M
    Thema hat sich erledigt. War am Ende total easy. http://"user":"passwort"@xxx.xxx.xxx.xxx:5000/carddav/"user"/"adressbuch-id"/"kontakt-id".vcf war die Lösung. Jetzt sind auch die Anrufbilder verfügbar.
  • Habe ein Problem in Typescript ....Fehler im Script

    6
    0 Stimmen
    6 Beiträge
    210 Aufrufe
    T
    @OliverIO sagte: @ticaki Allerdings ist es meiner Meinung nach aktuell immer noch so ist, das man Grundlagen der Programmierung kennen muss. Wenn du absolut blank bist, verstehst du nichtmal die Ausgabe der KI und gibst beim 3 Versuch auf. Die Eisntiegshürden sind zwar wirklich gering geworden, aber man muss das auch lernen wollen und auch die prompts richtig formulieren. sonst kommt nur schrott raus. Das stimmt, war mir zuviel arbeit den Text rüber zu kopieren und passend zu formatieren. Umso mehr ahnung man hat umso schneller kommt man als Ziel außer man macht sowas: ❯ lass den quark und gehe es richtig an - setzte agenten darauf an im internet zu suchen wie man das macht und obs überhaupt geht - prüfe die infos und komme dann mit neuen vorschlägen :D
  • Einschaltverzögerung mit schwankenten Werten

    17
    0 Stimmen
    17 Beiträge
    473 Aufrufe
    J
    ich wollte hier keine Verwirrung auslösen, ich bin froh das hier Leute wie ihr bereit sind zu helfen und dann werden die Themen besprochen...Dank geht an Euch!
  • [gelöst] Wie Timer finden?

    10
    0 Stimmen
    10 Beiträge
    385 Aufrufe
    I
    DANKE! Ja, der Bekannte hat das Script (tatsächlich waren es noch 2 weitere verschollene) im Objektbaum javascript gesehen und dort zunächst disabled und gelöscht, und der Timer wird nicht mehr angezeigt. Jetzt warten wir noch 12:12 Uhr ab ;-) (Update: hat geklappt) Ich denke, das war's! Vielen Dank an alle für die freundliche Hilfe, eine kleine Spende für's Projekt geht gleich raus.
  • Mr Pure Salzelektrolyse

    6
    0 Stimmen
    6 Beiträge
    607 Aufrufe
    H
    also, ich habs also geschafft. in die Tuya App hinzugefügt und dann mittels Adapter. Ist allerdings ein bissl eine Spielerei mit dem Developer Account und dem verlinken...aber ich hab alle Daten DANKE
  • Proxmox-Updater (Host/LXC/VM) auch ioBroker,piHole,etc

    58
    3 Stimmen
    58 Beiträge
    12k Aufrufe
    da_WoodyD
    @Bass-T GRANULATION! hab ich schon lange geinstet, echt goil!
  • Daten Seriell von Paradigma Solaranlage lesen

    javascript communication heating
    55
    0 Stimmen
    55 Beiträge
    9k Aufrufe
    Samson71S
    @MatthiasROW Unabhängig davon, dass dieser Beitrag mit über 5 Jahren mehr als steinalt ist........ Schonmal auf der genannten Webseite selber nachgesehen? Nur weil ein (uralter) Link nicht mehr passt, existiert die Webseite und auch deren Inhalte grundsätzlich noch. Ich finde da jedenfalls ne Menge Infos zum Systa Aqua II - Logger incl. der Schaltungsbeschreibung.
  • Ecconreset bei mqtt Teilnehmer

    1
    2
    0 Stimmen
    1 Beiträge
    54 Aufrufe
    Niemand hat geantwortet
  • mqtt-Abruf WiCAN-OBD-Dongle mit mqtt-Adapter und Blockly

    12
    1
    0 Stimmen
    12 Beiträge
    2k Aufrufe
    K
    Hi, Läuft es bei dir noch? Ich bin am überlegen mir auch ein WiCAN-OBD für unseren eUP zu holen. Habe davon aber ehrlich gesagt keine Ahnung. Könntest du mich dann unterstützen? Das wichtigste für mich ist, den aktuellen Akkustand in iobroker zu bekommen. Vielej Dank vorab! :)
  • Ostrom Api auslesen

    4
    0 Stimmen
    4 Beiträge
    738 Aufrufe
    NicolomaN
    ich habe weiter ein anderes: const axios = require('axios'); // ======= OSTROM PROD ======= const CLIENT_ID = 'DEINE_ID'; const CLIENT_SECRET = 'DEIN_SECRET'; const ZIP = '59759'; // ======= INFLUXDB 2.x ======= const INFLUX_URL = 'http://192.168.178.103:8086'; const INFLUX_ORG = 'my_org'; const INFLUX_BUCKET = 'iobroker'; const INFLUX_TOKEN = 'DEIN_TOKEN'; // ======= IO BROKER ======= const BASE_DP = '0_userdata.0.ostrom'; const AUTH_URL = 'https://auth.production.ostrom-api.io/oauth2/token'; const API_URL = 'https://production.ostrom-api.io'; let isRunning = false; createStates(); // API nur selten abrufen schedule('30 14 * * *', fetchPrices); // Status stündlich aktualisieren schedule('0 * * * *', updateCurrentFromCache); // beim Start nur Status aus Cache aktualisieren setTimeout(updateCurrentFromCache, 10000); // manueller API-Abruf on({ id: `${BASE_DP}.control.runNow`, val: true }, async () => { await setStateAsync(`${BASE_DP}.control.runNow`, false, true); await fetchPrices(); }); async function fetchPrices() { if (isRunning) return; const lastFetch = getState(`${BASE_DP}.meta.lastFetch`)?.val || 0; const now = Date.now(); if (now - lastFetch < 120000) { log('Ostrom: Letzte API-Abfrage < 2 Minuten – überspringe', 'warn'); return; } isRunning = true; try { const token = await getAccessToken(); const prices = await getPrices(token); const sorted = prices .slice() .sort((a, b) => new Date(a.date).getTime() - new Date(b.date).getTime()); await setStateAsync(`${BASE_DP}.todayTomorrow.json`, JSON.stringify(sorted), true); await updateCurrentFromPrices(sorted); await updateForecast(sorted); await writeToInflux(sorted); await setStateAsync(`${BASE_DP}.meta.lastFetch`, now, true); await setStateAsync(`${BASE_DP}.meta.lastFetchReadable`, new Date(now).toLocaleString('de-DE'), true); log(`Ostrom OK: ${sorted.length} Preise`, 'info'); } catch (err) { if (err.response?.status === 429) { log('Ostrom 429: Rate Limit – später erneut versuchen', 'warn'); } else { log(`Ostrom Fehler: ${err.message}`, 'error'); } if (err.response) { log(`Status: ${err.response.status}`, 'error'); log(JSON.stringify(err.response.data), 'error'); } } finally { isRunning = false; } } async function updateCurrentFromCache() { const json = getState(`${BASE_DP}.todayTomorrow.json`)?.val; if (!json) { log('Ostrom: Kein Preis-Cache vorhanden', 'warn'); return; } try { const prices = JSON.parse(json); await updateCurrentFromPrices(prices); log('Ostrom current.isFree stündlich aktualisiert', 'info'); } catch (err) { log(`Ostrom Cache Fehler: ${err.message}`, 'error'); } } async function updateCurrentFromPrices(prices) { const current = findCurrentPrice(prices); if (!current) return; const grossKwhPrice = Number(current.grossKwhPrice ?? 0); const grossKwhTaxAndLevies = Number(current.grossKwhTaxAndLevies ?? 0); // echter variabler Bruttopreis ohne monatliche Kosten const effectiveGrossKwhPrice = grossKwhPrice + grossKwhTaxAndLevies; await setStateAsync(`${BASE_DP}.current.grossKwhPrice`, round(grossKwhPrice), true); await setStateAsync(`${BASE_DP}.current.grossKwhTaxAndLevies`, round(grossKwhTaxAndLevies), true); await setStateAsync(`${BASE_DP}.current.effectiveGrossKwhPrice`, round(effectiveGrossKwhPrice), true); await setStateAsync(`${BASE_DP}.current.date`, current.date, true); await setStateAsync(`${BASE_DP}.current.dateReadable`, new Date(current.date).toLocaleString('de-DE'), true); await setStateAsync(`${BASE_DP}.current.hour`, new Date(current.date).getHours(), true); await setStateAsync(`${BASE_DP}.current.isFree`, effectiveGrossKwhPrice <= 0, true); } async function updateForecast(prices) { const effectivePrices = prices.map(p => { const gross = Number(p.grossKwhPrice ?? 0); const tax = Number(p.grossKwhTaxAndLevies ?? 0); return { date: p.date, value: gross + tax }; }); const values = effectivePrices.map(p => p.value); const min = Math.min(...values); const max = Math.max(...values); const avg = values.reduce((a, b) => a + b, 0) / values.length; const cheapest = effectivePrices.find(p => p.value === min); const expensive = effectivePrices.find(p => p.value === max); const nextFree = effectivePrices.find(p => new Date(p.date).getTime() >= Date.now() && p.value <= 0); await setStateAsync(`${BASE_DP}.forecast.minEffectiveGrossKwhPrice`, round(min), true); await setStateAsync(`${BASE_DP}.forecast.maxEffectiveGrossKwhPrice`, round(max), true); await setStateAsync(`${BASE_DP}.forecast.avgEffectiveGrossKwhPrice`, round(avg), true); await setStateAsync(`${BASE_DP}.forecast.cheapestDate`, cheapest?.date || '', true); await setStateAsync(`${BASE_DP}.forecast.cheapestDateReadable`, cheapest ? new Date(cheapest.date).toLocaleString('de-DE') : '', true); await setStateAsync(`${BASE_DP}.forecast.mostExpensiveDate`, expensive?.date || '', true); await setStateAsync(`${BASE_DP}.forecast.mostExpensiveDateReadable`, expensive ? new Date(expensive.date).toLocaleString('de-DE') : '', true); await setStateAsync(`${BASE_DP}.forecast.nextFreeDate`, nextFree?.date || '', true); await setStateAsync(`${BASE_DP}.forecast.nextFreeDateReadable`, nextFree ? new Date(nextFree.date).toLocaleString('de-DE') : '', true); await setStateAsync(`${BASE_DP}.forecast.hasFreeHour`, !!nextFree, true); } async function getAccessToken() { const auth = Buffer.from(`${CLIENT_ID}:${CLIENT_SECRET}`).toString('base64'); const res = await axios.post( AUTH_URL, 'grant_type=client_credentials', { headers: { Authorization: `Basic ${auth}`, 'Content-Type': 'application/x-www-form-urlencoded' }, timeout: 15000 } ); return res.data.access_token; } async function getPrices(token) { const now = new Date(); const start = new Date(now.getFullYear(), now.getMonth(), now.getDate(), 0, 0, 0); const end = new Date(now.getFullYear(), now.getMonth(), now.getDate() + 2, 0, 0, 0); const res = await axios.get(`${API_URL}/spot-prices`, { params: { startDate: start.toISOString(), endDate: end.toISOString(), resolution: 'HOUR', zip: ZIP }, headers: { Authorization: `Bearer ${token}` }, timeout: 15000 }); return res.data.data || []; } async function writeToInflux(prices) { if (!prices.length) return; const lines = []; for (const p of prices) { const ts = BigInt(new Date(p.date).getTime()) * 1000000n; const grossKwhPrice = Number(p.grossKwhPrice ?? 0); const grossKwhTaxAndLevies = Number(p.grossKwhTaxAndLevies ?? 0); const effectiveGrossKwhPrice = grossKwhPrice + grossKwhTaxAndLevies; lines.push( `ostrom_prices,source=production,zip=${escapeTag(ZIP)} ` + `grossKwhPrice=${grossKwhPrice},` + `grossKwhTaxAndLevies=${grossKwhTaxAndLevies},` + `effectiveGrossKwhPrice=${effectiveGrossKwhPrice} ` + `${ts.toString()}` ); } const url = `${INFLUX_URL}/api/v2/write` + `?org=${encodeURIComponent(INFLUX_ORG)}` + `&bucket=${encodeURIComponent(INFLUX_BUCKET)}` + `&precision=ns`; await axios.post(url, lines.join('\n'), { headers: { Authorization: `Token ${INFLUX_TOKEN}`, 'Content-Type': 'text/plain' }, timeout: 15000 }); } function findCurrentPrice(prices) { const now = new Date(); return prices.find(p => { const start = new Date(p.date); const end = new Date(start.getTime() + 3600000); return now >= start && now < end; }); } async function createStates() { await ensure(`${BASE_DP}.control.runNow`, 'boolean', true, true, 'button'); await ensure(`${BASE_DP}.meta.lastFetch`, 'number', 0, true, 'value'); await ensure(`${BASE_DP}.meta.lastFetchReadable`, 'string', '', true, 'text'); await ensure(`${BASE_DP}.todayTomorrow.json`, 'string', '', true, 'json'); await ensure(`${BASE_DP}.current.grossKwhPrice`, 'number', 0, true, 'value', 'ct/kWh'); await ensure(`${BASE_DP}.current.grossKwhTaxAndLevies`, 'number', 0, true, 'value', 'ct/kWh'); await ensure(`${BASE_DP}.current.effectiveGrossKwhPrice`, 'number', 0, true, 'value', 'ct/kWh'); await ensure(`${BASE_DP}.current.date`, 'string', '', true, 'text'); await ensure(`${BASE_DP}.current.dateReadable`, 'string', '', true, 'text'); await ensure(`${BASE_DP}.current.hour`, 'number', 0, true, 'value', 'h'); await ensure(`${BASE_DP}.current.isFree`, 'boolean', false, true, 'indicator'); await ensure(`${BASE_DP}.forecast.minEffectiveGrossKwhPrice`, 'number', 0, true, 'value', 'ct/kWh'); await ensure(`${BASE_DP}.forecast.maxEffectiveGrossKwhPrice`, 'number', 0, true, 'value', 'ct/kWh'); await ensure(`${BASE_DP}.forecast.avgEffectiveGrossKwhPrice`, 'number', 0, true, 'value', 'ct/kWh'); await ensure(`${BASE_DP}.forecast.cheapestDate`, 'string', '', true, 'text'); await ensure(`${BASE_DP}.forecast.cheapestDateReadable`, 'string', '', true, 'text'); await ensure(`${BASE_DP}.forecast.mostExpensiveDate`, 'string', '', true, 'text'); await ensure(`${BASE_DP}.forecast.mostExpensiveDateReadable`, 'string', '', true, 'text'); await ensure(`${BASE_DP}.forecast.nextFreeDate`, 'string', '', true, 'text'); await ensure(`${BASE_DP}.forecast.nextFreeDateReadable`, 'string', '', true, 'text'); await ensure(`${BASE_DP}.forecast.hasFreeHour`, 'boolean', false, true, 'indicator'); } async function ensure(id, type, def, write, role, unit = '') { const exists = await existsStateAsync(id); if (!exists) { await createStateAsync(id, def, { type, role, read: true, write, unit }); } } function round(value) { return Math.round(value * 1000) / 1000; } function escapeTag(value) { return String(value) .replace(/ /g, '\\ ') .replace(/,/g, '\\,') .replace(/=/g, '\\='); }

524

Online

33.0k

Benutzer

83.3k

Themen

1.3m

Beiträge