Weiter zum Inhalt

Skripten / Logik

16.7k Themen 215.1k 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
    madingM
    Hi zusammen, ich möchte mir im Laufe des Jahres ein Batteriespeicher zulegen. Ich habe bereits eine (amortisierte) PV mit einem Fronius Wechselrichter und auch eine openWB. Vor längerem bin ich auf E3DC aufmerksam geworden und habe ein HK One angeboten bekommen. Aus Amortisationssicht ist ein Stromspeicher zu einem gewissen Teil Liebhaberei, da ich relativ wenig fahre, die openWB habe, die PV only laden kann und auch eine Kopplung zwischen PV und Sole WP habe (bei Einspeisung wird das Brauch/ Heizwasser überhöht). Mir wurde ein HK One mit 15 kwh angeboten, bei den Daten spuckt die KI eher was Richtung 8kwh aus, wenn man etwas mehr auf die Amortisation schaut. Bis jetzt habe ich auch einen grossen Bogen um Tibber, Awatar und Konsorten gemacht, da es sich für mich nicht lohnt. Im Sommer habe ich komplett ca. 3kwh Bezug für alles (Heizung, Auto, Haushaltsstrom). Im Winter sieht das natürlich anders aus. Ein sehr interessanter Punkt für mich wäre zb mit einem dynamischen Stromtarif z.B. per Blockly Skript im Winter den Speicher vom Netz laden zu lassen, wenn der Strompreis unter x cent/ kWh liegt. Es gibt ja das Sktipt bzw. den Adapter. Ist das Szenario damit möglich (dem Akku den Befehl zu geben, vom Netz zu laden)? Wie funktioniert das Zusammenspiel mit der openWB (es gibt ja die Möglichkeit die Batterie vor der Wallbox zu füllen) etc. Wie zufrieden seid ihr mit E3DC aus der Heimautomatisierungs-Sicht?
  • Hilfe für Skripterstellung mit Blockly

    7k 80k
    7k Themen
    80k Beiträge
    T
    @Gonzokalle sagte: Nun hatte ich doch eine Idee. So scheint es zu funktionieren. Kann man das Script auch nehmen, wenn man zwei Wechselrichter hat?
  • Hilfe für Skripterstellung mit Node-RED

    960 13k
    960 Themen
    13k Beiträge
    M
    Hallo, nachdem ich es mit Eurer Hilfe geschafft habe mein Zigbee-Netzwerk mit dem SLZB-MR3U zu erstellen und der Adapter fleißig arbeitet, konnte ich das Sonoff SWV-ZFE Ventil dem Zigbee-Netzwerk hinzufügen. Allerdings kam folgenden Fehlermeldung im Log New device joined '0xa4c138145f7cffff' model SWV-ZFE New device: '0xa4c138145f7cffff' does not have a known model. please provide an external converter for 'SWV-ZFE'. Ich konnte mit der Node "IoBroker in" auch schon Werte von dem Device lesen. Wie kann ich jetzt das Ventil dazu bewegen zu öffnen und zu schließen? "ioBroker out" höchstwahrscheinlich, aber wie muss ich das konfigurieren? [image: 1780853385321-71758951-c14a-444b-a2bc-d1d7e5ad3ba8-image.jpeg] Ich habe schon mit einer vorangestellten function node per msg.payload = { "state": "ON", }; return msg; bzw. "off" versucht, aber es tut sich einfach nichts.....
  • 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
    14k Aufrufe
    NegaleinN
    @Ro75 sagte: Vielen Dank. Ro75. hab es soeben in die Sammlung aufgenommen :)
  • Heizstab per %-Werte flexibel schalten je nach Einspeisung

    10
    0 Stimmen
    10 Beiträge
    123 Aufrufe
    paul53P
    @rtwl [sagte]: bis 60° geheizt werden. anschließend soll wieder nichts passieren bis wieder <45° im Puffer sind. Das sollte so funktionieren: [image: 1780866023500-blockly_temp.jpg]
  • mein Kampf mit true und false

    14
    0 Stimmen
    14 Beiträge
    141 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
  • Alexa Shopping List mit Bring synchronisieren

    183
    0 Stimmen
    183 Beiträge
    43k Aufrufe
    S
    Hallo zusammen, nach dem ich hier viel über IO Broker in Verbindung mit Bring und der Alexa Shopping Liste gelernt habe und immer mal wieder "tweaks" an meinem abgewandelten Script vorgenommen habe und es nun super geschmeidigt läuft, möchte ich gerne das aktuelle ALEXA BRING SCRIPT mit euch teilen und der hervorragenden Community auf diesem Weg etwas zurück geben und einfach nur DANKE sagen. Zum Script: Bei der brindBaseID müsst ihr XXX wie gewohnt durch eure Bring Shopping List ID ersetzen, damit die richtige Liste befüllt wird. Ansonste alles wie immer: Alexa2.0 Adapter und bring.0 Adapter sind Voraussetzung, dass es funktioniert. Steht aber auch auskommentiert im Script. // ===================================================================== // Alexa Shopping List <-> Bring! Zwei-Wege-Sync // Engine: 3-Wege-Merge gegen persistierten Schatten + Zeitstempel-Konfliktlösung // Behandelt korrekt: Hinzufügen, Abhaken, Reaktivieren, Löschen (beidseitig) // ===================================================================== // --- KONFIGURATION: an die eigene Installation anpassen --- // Bring-Listen-ID findest du in den Objekten unter "bring.0" -> Channel deiner Liste (UUID): const bringBaseId = 'bring.0.xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx'; // Alexa-Einkaufsliste, bei den meisten Setups: const alexa2BaseId = 'alexa2.0.Lists.SHOPPING_LIST'; // Voraussetzungen: Adapter "alexa2" + "bring" eingerichtet, 0_userdata.0 vorhanden. // Persistenter Speicher für den Schatten (0_userdata.0 existiert in modernen ioBroker-Installs) const SHADOW_STATE = '0_userdata.0.alexaBringSync.shadow'; // Timing const DEBOUNCE_MS = 2000; // warte nach der letzten Änderung, bevor synchronisiert wird const LOCK_MS = 5000; // Beruhigungsfenster, bis async Schreibvorgänge der Adapter durch sind // Bei einem echten Konflikt (beide Seiten gleichzeitig geändert, Zeitstempel uneindeutig): // true = im Zweifel "active" gewinnen lassen (nichts verlieren, was man evtl. noch braucht) const CONFLICT_PREFER_ACTIVE = true; // Optionaler Watchdog: warnt im Log, wenn die alexa2-Cloud-Verbindung wegbricht const WATCH_ALEXA_CONNECTION = true; // --- ABGELEITETE IDs --- const bringListId = bringBaseId + '.content'; const bringListCompletedId = bringBaseId + '.recentContent'; const bringAddToList = bringBaseId + '.saveItem'; // hinzufügen / aus recent reaktivieren const bringCompleteItem = bringBaseId + '.moveToRecentContent'; // abhaken (-> recent) const bringRemoveItem = bringBaseId + '.removeItem'; // komplett entfernen (content + recent) const alexaAddToList = alexa2BaseId + '.#New'; const alexaListId = alexa2BaseId + '.json'; // --- LAUFZEIT-ZUSTAND --- let isSyncing = false; let pendingSync = false; let syncTimeout = null; let debounceTimer = null; // --- HILFSFUNKTIONEN --- function cleanName(str) { return str ? String(str).trim().toLowerCase() : ''; } function formatName(str) { if (!str) return ''; str = String(str).trim(); return str.charAt(0).toUpperCase() + str.slice(1); } function loadShadow() { try { const s = getState(SHADOW_STATE); return (s && s.val) ? JSON.parse(s.val) : {}; } catch (e) { log('Schatten nicht lesbar, starte leer: ' + e, 'warn'); return {}; } } function saveShadow(sh) { setState(SHADOW_STATE, JSON.stringify(sh), true); } // Konfliktlösung: jüngerer Zeitstempel gewinnt, sonst konfigurierte Vorzugsregel function resolveConflict(aStatus, bStatus, aTs, bTs, displayName) { let winner; if (aTs > bTs) winner = aStatus; else if (bTs > aTs) winner = bStatus; else winner = CONFLICT_PREFER_ACTIVE ? ((aStatus === 'active' || bStatus === 'active') ? 'active' : 'done') : ((aStatus === 'done' || bStatus === 'done') ? 'done' : 'active'); log(`[KONFLIKT] "${displayName}": Alexa=${aStatus}(${aTs}) vs Bring=${bStatus}(${bTs}) -> ${winner}`, 'warn'); return winner; } function finishLock() { if (syncTimeout) clearTimeout(syncTimeout); syncTimeout = setTimeout(() => { isSyncing = false; if (pendingSync) { pendingSync = false; triggerSync(); } // Nachlauf für während der Sperre Eingetroffenes }, LOCK_MS); } // --- HAUPTLOGIK --- function doSync() { if (isSyncing) { pendingSync = true; return; } isSyncing = true; try { const aState = getState(alexaListId); const bState = getState(bringListId); const rState = getState(bringListCompletedId); if (!aState || aState.val == null || !bState || bState.val == null) { finishLock(); return; } const alexaList = JSON.parse(aState.val); const bringActive = JSON.parse(bState.val); const bringRecent = (rState && rState.val) ? JSON.parse(rState.val) : []; // grobe Bring-Zeitstempel (Listenebene) nur für die Konfliktlösung const bringContentLc = bState.lc || bState.ts || 0; const bringRecentLc = (rState && (rState.lc || rState.ts)) || 0; // --- Maps der aktuellen Zustände --- const alexaByName = {}; // active gewinnt bei Namens-Duplikaten alexaList.forEach(it => { const k = cleanName(it.value); if (!k) return; const st = it.completed ? 'done' : 'active'; if (!alexaByName[k] || st === 'active') alexaByName[k] = it; }); const bringActiveMap = {}; bringActive.forEach(it => { const k = cleanName(it.name); if (k) bringActiveMap[k] = it; }); const bringRecentMap = {}; bringRecent.forEach(it => { const k = cleanName(it.name); if (k) bringRecentMap[k] = it; }); const shadow = loadShadow(); const newShadow = {}; const names = new Set([ ...Object.keys(alexaByName), ...Object.keys(bringActiveMap), ...Object.keys(bringRecentMap), ...Object.keys(shadow) ]); names.forEach(name => { const aItem = alexaByName[name]; const aStatus = aItem ? (aItem.completed ? 'done' : 'active') : 'absent'; const bStatus = bringActiveMap[name] ? 'active' : (bringRecentMap[name] ? 'done' : 'absent'); const prev = shadow[name] || { a: 'absent', b: 'absent', c: 'absent' }; const aChanged = aStatus !== prev.a; const bChanged = bStatus !== prev.b; const displayName = (bringActiveMap[name] || bringRecentMap[name] || {}).name || (aItem ? aItem.value : null) || prev.name || formatName(name); const aTs = aItem ? (aItem.updatedDateTime || 0) : 0; const bTs = (bStatus === 'active') ? bringContentLc : (bStatus === 'done' ? bringRecentLc : 0); // --- Zielzustand T bestimmen --- let T; if (aChanged && !bChanged) T = aStatus; else if (bChanged && !aChanged) T = bStatus; else if (aChanged && bChanged) T = (aStatus === bStatus) ? aStatus : resolveConflict(aStatus, bStatus, aTs, bTs, displayName); else T = prev.c; // nichts geändert -> Kanon halten if (!T) T = (aStatus !== 'absent') ? aStatus : bStatus; // --- Alexa angleichen --- let aResult = aStatus; if (T === 'active') { if (aStatus === 'absent') { log(`[ALEXA +] ${displayName}`); setState(alexaAddToList, displayName); aResult = 'active'; } else if (aStatus === 'done') { log(`[ALEXA reaktiv.] ${displayName}`); setState(`${alexa2BaseId}.items.${aItem.id}.completed`, false); aResult = 'active'; } } else if (T === 'done') { if (aStatus === 'active') { log(`[ALEXA abgehakt] ${displayName}`); setState(`${alexa2BaseId}.items.${aItem.id}.completed`, true); aResult = 'done'; } // aStatus 'absent' -> erledigtes Item lässt sich in Alexa nicht erzeugen, bleibt absent } else if (T === 'absent') { if (aStatus !== 'absent') { log(`[ALEXA gelöscht] ${displayName}`); setState(`${alexa2BaseId}.items.${aItem.id}.#delete`, true); aResult = 'absent'; } } // --- Bring angleichen --- let bResult = bStatus; if (T === 'active') { if (bStatus !== 'active') { log(`[BRING +] ${displayName}`); setState(bringAddToList, displayName); bResult = 'active'; } } else if (T === 'done') { if (bStatus === 'active') { log(`[BRING abgehakt] ${displayName}`); setState(bringCompleteItem, displayName); bResult = 'done'; } // bStatus 'absent' -> kann nicht als erledigt erzeugt werden, bleibt absent } else if (T === 'absent') { if (bStatus !== 'absent') { log(`[BRING gelöscht] ${displayName}`); setState(bringRemoveItem, displayName); bResult = 'absent'; } } // --- neuen Schatten schreiben (komplett verschwundene Items fallen raus) --- if (!(aResult === 'absent' && bResult === 'absent')) { newShadow[name] = { a: aResult, b: bResult, c: T, name: displayName, ts: Date.now() }; } }); saveShadow(newShadow); } catch (e) { log('Fehler im Sync-Skript: ' + e, 'error'); } finishLock(); } // --- TRIGGER mit Debounce --- function triggerSync() { if (debounceTimer) clearTimeout(debounceTimer); debounceTimer = setTimeout(doSync, DEBOUNCE_MS); } on({ id: bringListId, change: 'ne' }, triggerSync); on({ id: bringListCompletedId, change: 'ne' }, triggerSync); on({ id: alexaListId, change: 'ne' }, triggerSync); if (WATCH_ALEXA_CONNECTION) { on({ id: 'alexa2.0.info.connection', change: 'ne' }, obj => { if (obj.state.val === false) log('alexa2-Cloud-Verbindung verloren – Sync evtl. inaktiv bis zum Reconnect.', 'warn'); }); } // --- START: Schatten-State sicherstellen, dann initial abgleichen --- createState(SHADOW_STATE, '{}', false, { name: 'Alexa-Bring Sync Shadow', type: 'string', read: true, write: true }, () => { log('Alexa <-> Bring Sync (Shadow-Merge) gestartet. Initialer Abgleich …'); triggerSync(); }); Das Script müsste ihr einfach als JS Script in IO Broker kopieren und starten. Der Rest funktioniert dann automatisch. WICHTIG NOCHMAL: Die Bring Base ID bzw. der Platzhalter XXX mit eurer echten Listen ID ersetzen. DANKE nochmal an die gesamte Community, mit deren Hilfe dieses angepasste und für meine Zwecke hervorragend funktionierende Script entstanden ist. KUDOS an alle die ihre Scripts geteilt haben.
  • VCF Datei auslesen

    2
    0 Stimmen
    2 Beiträge
    78 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
    156 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
    320 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
    295 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
    537 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
    11k 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
    8k 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
    44 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
    691 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, '\\='); }
  • [Vorlage] Anwesenheitssimulation mit dauerhaftem Lernen

    javascript template security
    21
    1 Stimmen
    21 Beiträge
    1k Aufrufe
    NashraN
    @mrMuppet Danke für die gute Erklärung
  • verschiedene Skripte

    javascript
    1
    1
    0 Stimmen
    1 Beiträge
    121 Aufrufe
    Niemand hat geantwortet
  • [Vorlage] Luftqualitätswerte abrufen

    55
    2
    4 Stimmen
    55 Beiträge
    9k Aufrufe
    Siggi0904S
    @Boronsbruder sagte: Wenn ich mal Zeit habe, schau ich mir APIv4 an ;) Wenn man in die News zur V4 schaut, sind "nur" Statistiken dazu gekommen. Ich habe bei mir auf die V4 mit den aktuellen Einstellungen umgestellt und es funktioniert bisher. Vielleicht kann man das Script noch ausbauen und weitere Funktionen der API implementieren. Danke und frohe Ostern.
  • Bambu Lab A1 - Status "fertig"

    6
    0 Stimmen
    6 Beiträge
    249 Aufrufe
    skvarelS
    Es gibt einen Datenpunkt im Bambu Adapter auf den ich reagiere ... zumindest für den P1S Ich prüfe alle 15 Minuten ob der Druck fertig ist und ob die Temperaturen niedrig genug sind. In der VIS habe ich einen Switch um das automatische Abschalten zu steuern. Man will ja nicht nach jedem Druck abschalten ;) Hier mal mein Script: [image: 1775111135695-0cb2eba1-8d7c-46d7-b489-669dc835de7e-image.jpeg] Die View dazu: [image: 1775111001000-fb8b3d6e-705a-4553-8ab5-8660c2ab8d28-image.jpeg]
  • Blockly: Astro-Block - (Zeit)-Versatz wird nicht ausgeführt

    4
    0 Stimmen
    4 Beiträge
    202 Aufrufe
    HomoranH
    @w00dy sagte: demnach passt es. Was immer du da vorhast, -45 Minuten ist vor Sonnenuntergang!

310

Online

32.9k

Benutzer

83.1k

Themen

1.3m

Beiträge