NEWS
Skript aufteilen möglich?
-
Hallo,
gibt es heute in 2026 eine Möglichkeit, ein größeres Skript in mehrere Module aufzuteilen?Ich habe nur einen 8 Jahre alten Thread gefunden dazu, damals ging es noch nicht.
Am besten zeigst du mal dein Script, dann kann man bessere Empfehlungen abgeben.
Bitte auch mal schreiben warum du aufteilen möchtest obwohl es so funktioniert. -
Es gibt im Forum mehrere Topics die sich damit befassen und es gelöst haben. Ich weiß aber nicht mehr wie und ich finde sie auch nicht mit der suche.
-
@bertderkleine
Meinst du in der Richtung https://forum.iobroker.net/post/1316079 -
Hallo,
gibt es heute in 2026 eine Möglichkeit, ein größeres Skript in mehrere Module aufzuteilen?Ich habe nur einen 8 Jahre alten Thread gefunden dazu, damals ging es noch nicht.
@BertDerKleine sagte in Skript aufteilen möglich?:
gibt es heute in 2026 eine Möglichkeit, ein größeres Skript in mehrere Module aufzuteilen?
Ich habe nur einen 8 Jahre alten Thread gefunden dazu, damals ging es noch nicht.
was soll das heissen.. ein script kannst du immer aufteilen, warum auch nicht
muss man nur wissen nur wie
-
@BertDerKleine sagte in Skript aufteilen möglich?:
gibt es heute in 2026 eine Möglichkeit, ein größeres Skript in mehrere Module aufzuteilen?
Ich habe nur einen 8 Jahre alten Thread gefunden dazu, damals ging es noch nicht.
was soll das heissen.. ein script kannst du immer aufteilen, warum auch nicht
muss man nur wissen nur wie
@arteck sagte in Skript aufteilen möglich?:
was soll das heissen.. ein script kannst du immer aufteilen, warum auch nicht
muss man nur wissen nur wie
Genau dem "wie" hat er doch gefragt :-) Wenn du es weist ... nur zu
-
Ich mach das z.bsp. bei meinem Unifi-API Script, da schreibe ich mir Werte in verschiedene Index.json , die ich dann mit anderen scripts auswerte. So spare ich mir States und hab das etwas getrennt, was für die Selektivität besser ist.
Andererseits ist die Struktur und Kommentarfunktion im Script wichtig, damit man weiß, was man da verbrochen hat.. solange die Funktionen nicht selbsterklärend sind… -
Alle Möglichkeiten aufzuzählen macht das ganze imho nur unübersichtlich - der TE sollte sich dazu mal melden (der javascript-adapter hat auch ein messagesystem :) )
-
@arteck sagte in Skript aufteilen möglich?:
was soll das heissen.. ein script kannst du immer aufteilen, warum auch nicht
muss man nur wissen nur wie
Genau dem "wie" hat er doch gefragt :-) Wenn du es weist ... nur zu
@BananaJoe sagte in Skript aufteilen möglich?:
Genau dem "wie" hat er doch gefragt :-) Wenn du es weist ... nur zu
Richtig!
Ich wiederhole nochmal was die Intention ist:
Langes Javaskript in zwei kleinere Skripte aufteilen. Soll trotzdem noch weiter funktionieren. zB Hauptfunktionen in A, Nebenfunktionen in B oder was auch immer als Splitkriterium.Soweit ich das in dem Uraltthread gelesen habe, werden alle globalen Skripte immer überall mitgeladen. Das gefällt mir nicht, daher sind simple globale Skripte keine Antwort auf meine Frage.
Nachdem hier bisher keine besonders hilfreichen Tips kamen, habe ich die Maschinen gefragt und die sind der Meinung, man könnte das mit "onMessage / messageToAsync." lösen.
Beispiel wäre so ein Szenario, wo ich eine Hilfsfunktion eben in ein zweites lokales Skript auslagere:
// === Skript A: History Helper === // Liefert Durchschnitt + Median eines Datenpunkts für ein Zeitfenster onMessage('getHistoryStats', async (data, callback) => { const { stateId, hours } = data; // z.B. { stateId: '0_userdata.0.temperatur', hours: 24 } if (!stateId || !hours) { return callback({ error: 'stateId und hours fehlen' }); } const end = Date.now(); const start = end - hours * 60 * 60 * 1000; getHistory(stateId, { start: start, end: end, aggregate: 'none', // alle Rohwerte holen (für Median nötig) count: 0 // kein Limit }, (err, result) => { if (err) { log(`[HistoryHelper] Fehler bei ${stateId}: ${err}`, 'error'); return callback({ error: err.toString() }); } // Nur gültige Werte const values = result .map(r => r.val) .filter(v => v !== null && v !== undefined && !isNaN(v)); if (values.length === 0) { return callback({ average: null, median: null, count: 0 }); } // Durchschnitt const average = values.reduce((a, b) => a + b, 0) / values.length; // Median const sorted = [...values].sort((a, b) => a - b); const mid = Math.floor(sorted.length / 2); const median = sorted.length % 2 === 0 ? (sorted[mid - 1] + sorted[mid]) / 2 : sorted[mid]; log(`[HistoryHelper] ${stateId} | ${values.length} Werte | Ø ${average.toFixed(2)} | Median ${median.toFixed(2)}`); callback({ average: Number(average.toFixed(2)), median: Number(median.toFixed(2)), count: values.length }); }); }); log('✅ History-Helper gestartet – wartet auf "getHistoryStats" Nachrichten');und das Hauptskript:
// === Skript B: Daily Report === // Wird jeden Tag um 23:59 ausgelöst, holt für zwei Datenpunkte die Stats // und verschickt eine E-Mail const dp1 = '0_userdata.0.temperatur_wohnzimmer'; // ← hier deine ersten DP-ID const dp2 = '0_userdata.0.feuchtigkeit_bad'; // ← hier deine zweite DP-ID // Trigger: jeden Tag um 23:59 (Tagesende) schedule('59 23 * * *', async () => { log('🕙 Tagesende – starte History-Statistik...'); const hours = 24; // letze 24 Stunden try { // Parallele Abfragen (schnell) const [res1, res2] = await Promise.all([ messageToAsync( { instance: 0, script: 'script.js.history_helper', message: 'getHistoryStats' }, { stateId: dp1, hours: hours } ), messageToAsync( { instance: 0, script: 'script.js.history_helper', message: 'getHistoryStats' }, { stateId: dp2, hours: hours } ) ]); // E-Mail zusammenbauen const text = ` Tagesbericht – letzte 24 Stunden (${new Date().toLocaleDateString('de-DE')}) ${dp1}: • Durchschnitt: ${res1.average ?? '—'} • Median: ${res1.median ?? '—'} • Werte: ${res1.count} ${dp2}: • Durchschnitt: ${res2.average ?? '—'} • Median: ${res2.median ?? '—'} • Werte: ${res2.count} `.trim(); // E-Mail versenden (E-Mail-Adapter muss konfiguriert sein!) sendTo('email.0', 'send', { to: 'deine@email.de', // ← deine Empfängeradresse subject: 'ioBroker Tagesbericht ' + new Date().toLocaleDateString('de-DE'), text: text }); log('✅ E-Mail mit History-Statistik versendet'); } catch (e) { log(`❌ Fehler im Daily-Report: ${e}`, 'error'); } }); log('✅ Daily-Report-Skript geladen – Cron-Job um 23:59 aktiv');Ist das also der beste Weg, um die Aufgabe aus Post #3 und in diesem Beispiel zu lösen, wenn ich ein Skript aufteilen will und die Funktionen ohne "globale Skripte" anderswo wiederverwenden will?
So könnte ich den Durchschnittsberechner auslagern.
-
Am besten zeigst du mal dein Script, dann kann man bessere Empfehlungen abgeben.
Bitte auch mal schreiben warum du aufteilen möchtest obwohl es so funktioniert.@BertDerKleine sagte in Skript aufteilen möglich?:
Nachdem hier bisher keine besonders hilfreichen Tips kamen
viel mitwirkung war von deiner seite allerdings auch nicht da
@oliverio sagte in Skript aufteilen möglich?:
Am besten zeigst du mal dein Script, dann kann man bessere Empfehlungen abgeben.
Bitte auch mal schreiben warum du aufteilen möchtest obwohl es so funktioniert. -
Nächstes Mal beschreibe etwas mehr was du vorhast - das Messagesystem hab ich in meinem letzten Beitrag erwähnt. Wieso soll ich aber da viel Text zu schreiben wenn ich nicht weiß ob es passen könnte.
Wäre mir aber zu komplex - würde es wie ilovegym mit states und json machen. (vielleicht macht er es auch mit fs - das wäre mir dann zu umständlich)
-
@BananaJoe sagte in Skript aufteilen möglich?:
Genau dem "wie" hat er doch gefragt :-) Wenn du es weist ... nur zu
Richtig!
Ich wiederhole nochmal was die Intention ist:
Langes Javaskript in zwei kleinere Skripte aufteilen. Soll trotzdem noch weiter funktionieren. zB Hauptfunktionen in A, Nebenfunktionen in B oder was auch immer als Splitkriterium.Soweit ich das in dem Uraltthread gelesen habe, werden alle globalen Skripte immer überall mitgeladen. Das gefällt mir nicht, daher sind simple globale Skripte keine Antwort auf meine Frage.
Nachdem hier bisher keine besonders hilfreichen Tips kamen, habe ich die Maschinen gefragt und die sind der Meinung, man könnte das mit "onMessage / messageToAsync." lösen.
Beispiel wäre so ein Szenario, wo ich eine Hilfsfunktion eben in ein zweites lokales Skript auslagere:
// === Skript A: History Helper === // Liefert Durchschnitt + Median eines Datenpunkts für ein Zeitfenster onMessage('getHistoryStats', async (data, callback) => { const { stateId, hours } = data; // z.B. { stateId: '0_userdata.0.temperatur', hours: 24 } if (!stateId || !hours) { return callback({ error: 'stateId und hours fehlen' }); } const end = Date.now(); const start = end - hours * 60 * 60 * 1000; getHistory(stateId, { start: start, end: end, aggregate: 'none', // alle Rohwerte holen (für Median nötig) count: 0 // kein Limit }, (err, result) => { if (err) { log(`[HistoryHelper] Fehler bei ${stateId}: ${err}`, 'error'); return callback({ error: err.toString() }); } // Nur gültige Werte const values = result .map(r => r.val) .filter(v => v !== null && v !== undefined && !isNaN(v)); if (values.length === 0) { return callback({ average: null, median: null, count: 0 }); } // Durchschnitt const average = values.reduce((a, b) => a + b, 0) / values.length; // Median const sorted = [...values].sort((a, b) => a - b); const mid = Math.floor(sorted.length / 2); const median = sorted.length % 2 === 0 ? (sorted[mid - 1] + sorted[mid]) / 2 : sorted[mid]; log(`[HistoryHelper] ${stateId} | ${values.length} Werte | Ø ${average.toFixed(2)} | Median ${median.toFixed(2)}`); callback({ average: Number(average.toFixed(2)), median: Number(median.toFixed(2)), count: values.length }); }); }); log('✅ History-Helper gestartet – wartet auf "getHistoryStats" Nachrichten');und das Hauptskript:
// === Skript B: Daily Report === // Wird jeden Tag um 23:59 ausgelöst, holt für zwei Datenpunkte die Stats // und verschickt eine E-Mail const dp1 = '0_userdata.0.temperatur_wohnzimmer'; // ← hier deine ersten DP-ID const dp2 = '0_userdata.0.feuchtigkeit_bad'; // ← hier deine zweite DP-ID // Trigger: jeden Tag um 23:59 (Tagesende) schedule('59 23 * * *', async () => { log('🕙 Tagesende – starte History-Statistik...'); const hours = 24; // letze 24 Stunden try { // Parallele Abfragen (schnell) const [res1, res2] = await Promise.all([ messageToAsync( { instance: 0, script: 'script.js.history_helper', message: 'getHistoryStats' }, { stateId: dp1, hours: hours } ), messageToAsync( { instance: 0, script: 'script.js.history_helper', message: 'getHistoryStats' }, { stateId: dp2, hours: hours } ) ]); // E-Mail zusammenbauen const text = ` Tagesbericht – letzte 24 Stunden (${new Date().toLocaleDateString('de-DE')}) ${dp1}: • Durchschnitt: ${res1.average ?? '—'} • Median: ${res1.median ?? '—'} • Werte: ${res1.count} ${dp2}: • Durchschnitt: ${res2.average ?? '—'} • Median: ${res2.median ?? '—'} • Werte: ${res2.count} `.trim(); // E-Mail versenden (E-Mail-Adapter muss konfiguriert sein!) sendTo('email.0', 'send', { to: 'deine@email.de', // ← deine Empfängeradresse subject: 'ioBroker Tagesbericht ' + new Date().toLocaleDateString('de-DE'), text: text }); log('✅ E-Mail mit History-Statistik versendet'); } catch (e) { log(`❌ Fehler im Daily-Report: ${e}`, 'error'); } }); log('✅ Daily-Report-Skript geladen – Cron-Job um 23:59 aktiv');Ist das also der beste Weg, um die Aufgabe aus Post #3 und in diesem Beispiel zu lösen, wenn ich ein Skript aufteilen will und die Funktionen ohne "globale Skripte" anderswo wiederverwenden will?
So könnte ich den Durchschnittsberechner auslagern.
@BertDerKleine sagte in Skript aufteilen möglich?:
Ich wiederhole nochmal was die Intention ist:
Langes Javaskript in zwei kleinere Skripte aufteilen. Soll trotzdem noch weiter funktionieren. zB Hauptfunktionen in A, Nebenfunktionen in B oder was auch immer als Splitkriterium.
Leider hat diese "Beschreibung" keinerlei Aussagekraft, was du mit "lang" und "weiter funktionieren" meinen könntest.
-
@arteck sagte in Skript aufteilen möglich?:
was soll das heissen.. ein script kannst du immer aufteilen, warum auch nicht
muss man nur wissen nur wie
Genau dem "wie" hat er doch gefragt :-) Wenn du es weist ... nur zu
@BananaJoe nenen da steht
damals ging es noch nicht.
das rauf bezog sich meine Antwort ausserdem.. wiso ging es nach seiner Aussage nicht ? das will ich wissen
wo kein Script da keine ..na du weisst schon
-
Ich muss den Thread mal kapern :-)
Ich habe da so eine ähnliche Anforderung oder besser gesagt würde mich das eh mal interessieren, ob sowas irgendwie möglich ist oder was der best pratice wäre.
Ich habe z.B. ein Login für supabase um testweise mal Daten in die Postgres DB in der cloud zu schreiben. ein weit entfernter Gedanke von mir ist, alles States oder zumindest die notwendigen in Supabase zu haben und mal schauen was man dann so machen kann.
Naja, jedenfalls habe ich ein Login um mir einen Token zu holen.// ============================================= // Login & Init // ============================================= function initSupabase() { supabase = createClient(supabaseUrl, supabaseKey, { auth: { autoRefreshToken: true, persistSession: false } }); console.log('Supabase Client initialisiert'); } async function login() { try { console.log(`Login für ${Email}...`); const { data, error } = await supabase.auth.signInWithPassword({ email: Email, password: Password }); if (error) throw error; const token = data.session?.access_token; if (!token) throw new Error('Kein Token erhalten'); const payload = JSON.parse(Buffer.from(token.split('.')[1], 'base64').toString()); currentKkunde = payload.kKunde || 0; console.log(` Login OK | kKunde = ${currentKkunde}`); authenticatedClient = createClient(supabaseUrl, supabaseKey, { global: { headers: { Authorization: `Bearer ${token}` } } }); return token; } catch (err) { console.error('Login fehlgeschlagen', err); return null; } }das wollte ich gerne in einem Script auslagern, was ich dann aufrufen kann. Klar, ist nicht viel Code, aber interessiert mich halt
Zukünftig könnte dann eine Main Function die Daten alle holen, updaten , insert, löschen, Durchschnitt und noch ne Menge mehr machen.
Wenn ich mir den SQL Adapter Main ansehe mit 3500 Zeilen, da frage ich mich, mit welchen Editor arbeitet man da? Man scrollt ja hin und her.
https://github.com/ioBroker/ioBroker.sql/blob/master/main.js -
Ich muss den Thread mal kapern :-)
Ich habe da so eine ähnliche Anforderung oder besser gesagt würde mich das eh mal interessieren, ob sowas irgendwie möglich ist oder was der best pratice wäre.
Ich habe z.B. ein Login für supabase um testweise mal Daten in die Postgres DB in der cloud zu schreiben. ein weit entfernter Gedanke von mir ist, alles States oder zumindest die notwendigen in Supabase zu haben und mal schauen was man dann so machen kann.
Naja, jedenfalls habe ich ein Login um mir einen Token zu holen.// ============================================= // Login & Init // ============================================= function initSupabase() { supabase = createClient(supabaseUrl, supabaseKey, { auth: { autoRefreshToken: true, persistSession: false } }); console.log('Supabase Client initialisiert'); } async function login() { try { console.log(`Login für ${Email}...`); const { data, error } = await supabase.auth.signInWithPassword({ email: Email, password: Password }); if (error) throw error; const token = data.session?.access_token; if (!token) throw new Error('Kein Token erhalten'); const payload = JSON.parse(Buffer.from(token.split('.')[1], 'base64').toString()); currentKkunde = payload.kKunde || 0; console.log(` Login OK | kKunde = ${currentKkunde}`); authenticatedClient = createClient(supabaseUrl, supabaseKey, { global: { headers: { Authorization: `Bearer ${token}` } } }); return token; } catch (err) { console.error('Login fehlgeschlagen', err); return null; } }das wollte ich gerne in einem Script auslagern, was ich dann aufrufen kann. Klar, ist nicht viel Code, aber interessiert mich halt
Zukünftig könnte dann eine Main Function die Daten alle holen, updaten , insert, löschen, Durchschnitt und noch ne Menge mehr machen.
Wenn ich mir den SQL Adapter Main ansehe mit 3500 Zeilen, da frage ich mich, mit welchen Editor arbeitet man da? Man scrollt ja hin und her.
https://github.com/ioBroker/ioBroker.sql/blob/master/main.js@ple sagte in Skript aufteilen möglich?:
das wollte ich gerne in einem Script auslagern, was ich dann aufrufen kann. Klar, ist nicht viel Code, aber interessiert mich halt
Zukünftig könnte dann eine Main Function die Daten alle holen, updaten , insert, löschen, Durchschnitt und noch ne Menge mehr machen.
Wenn ich mir den SQL Adapter Main ansehe mit 3500 Zeilen, da frage ich mich, mit welchen Editor arbeitet man da? Man scrollt ja hin und her.
https://github.com/ioBroker/ioBroker.sql/blob/master/main.jsAls Editor arbeiten viele mit vscode - gibt dafür auch ein Plugin um iobroker scripte zu bearbeiten.
Ausgelagerte Funktionen gehen nur über global oder ne lib
global: ist ein globales script das in jedes andere Skript kopiert wird
lib: ist ne externe js datei - hab aber vergessen wie das geht. -
Ich muss den Thread mal kapern :-)
Ich habe da so eine ähnliche Anforderung oder besser gesagt würde mich das eh mal interessieren, ob sowas irgendwie möglich ist oder was der best pratice wäre.
Ich habe z.B. ein Login für supabase um testweise mal Daten in die Postgres DB in der cloud zu schreiben. ein weit entfernter Gedanke von mir ist, alles States oder zumindest die notwendigen in Supabase zu haben und mal schauen was man dann so machen kann.
Naja, jedenfalls habe ich ein Login um mir einen Token zu holen.// ============================================= // Login & Init // ============================================= function initSupabase() { supabase = createClient(supabaseUrl, supabaseKey, { auth: { autoRefreshToken: true, persistSession: false } }); console.log('Supabase Client initialisiert'); } async function login() { try { console.log(`Login für ${Email}...`); const { data, error } = await supabase.auth.signInWithPassword({ email: Email, password: Password }); if (error) throw error; const token = data.session?.access_token; if (!token) throw new Error('Kein Token erhalten'); const payload = JSON.parse(Buffer.from(token.split('.')[1], 'base64').toString()); currentKkunde = payload.kKunde || 0; console.log(` Login OK | kKunde = ${currentKkunde}`); authenticatedClient = createClient(supabaseUrl, supabaseKey, { global: { headers: { Authorization: `Bearer ${token}` } } }); return token; } catch (err) { console.error('Login fehlgeschlagen', err); return null; } }das wollte ich gerne in einem Script auslagern, was ich dann aufrufen kann. Klar, ist nicht viel Code, aber interessiert mich halt
Zukünftig könnte dann eine Main Function die Daten alle holen, updaten , insert, löschen, Durchschnitt und noch ne Menge mehr machen.
Wenn ich mir den SQL Adapter Main ansehe mit 3500 Zeilen, da frage ich mich, mit welchen Editor arbeitet man da? Man scrollt ja hin und her.
https://github.com/ioBroker/ioBroker.sql/blob/master/main.jsDu könntest ein Service Script laufen lassen, das alles mit supabase macht, also anmelden, abrufen, schreiben etc.
So wie @ticaki oben vorgeschlagen hat horcht dieses Skript dann auf bestimmte befehlsworte, führt die jeweilige Aktion aus und gibt die Daten zurück, ohne das sichnjedes einzelne Skripte um die supabase Details kümmern muss.
Klar kann man auch einzelne Programmatiken lokal ablegen und auch einbinden. Da gibt es hier im Forum Beispiele dazu. Müsste ich suchen wenn das das Ziel wäre.
Dann könnte man noch ein öffentliches npm Paket machen, das finde ich nicht so optimal, theoretisch auch ein privates Paket, das habe ich noch nie ausprobiert.
Wenn ich etwas umfangreicheres benötige, mit mehreren Dateien, dann würde ich lieber einen eigenen Adapter machen, da hat man dann alle Möglichkeiten.