NEWS
(Gelöst) Telefonbuch ohne Fritzbox
-
Hallo zusammen,
ich würde gerne Alexa bei eingehenden Anrufen bei bestimmten Nummern anstatt der Rufnummer direkt den Namen vorlesen lassen. Ich weiss, es gibt den iobroker.fritzbox und den iobroker.tr-064-Adapter, die erfordern aber beide eine Fritzbox, die ich nicht habe und auch nicht anschaffen werde. Ich habe ein Grandstream GXP1625, welches mir bei Anrufen via Action URL direkt die anrufende Nummer via Rest-Api in einen Datenpunkt schreibt, welche Alexa dann vorliest. Das funktioniert perfekt.
Wie kann ich aber nun in iobroker für beispielsweise 10 Nummern den passenden Namen hinterlegen? Klar, ich könnte das in einem Script mit 10 Falls-Abfragen manuell machen, aber das muss doch auch irgendwie eleganter gehen, so in Richtung SVERWEIS() bei Excel? Dass ich irgendwo eine Tabelle mit Rufnummer und Name anlege und dann bei einem Match der Name anstatt der Nummer gezogen wird?!
Bevorzugt in Blockly, Javascript könnte ich zumindest für meine Zwecke anpassen, denke ich, wenn jemand was Passendes hat.Für Ideen bin ich dankbar.
Gruss, Jürgen
-
@wildbill nicht dass ich das könnte
aber würde das nicht mit einer Liste oder JSON gehen, die man mit der angezeigten Nummer durchsucht?
ATttribut nummer wäre dann der Name
{"0123-45678" : "Müller"; "0234-98765" : "Meier"}
nur mal so ins unreine -
@homoran Ja, irgendwie so stelle ich mir das auch vor. Aber, wie Du schreibst:
"Nicht, dass ich das könnte".
Mir fehlt da grad irgendwie der Ansatz, wo ich das hinterlege und wie ich das auslese. Als Liste in Blockly irgendwie vielleicht, aber da bin ich grad auch etwas überfordert, wie ich das sauber mache.Gruss, Jürgen
-
@wildbill Ich denke, ich würde es sowohl mit zwei Listen hinbekommen, als auch indem ich ein JSON erstelle. Bei folgenden zwei Seiten habe ich mich mal orientiert:
Liste und JSON.
Hinbekommen werde ich wohl irgendwie beides. Liste eben zwei Listen einmal mit den Nummern und einmal mit gleicher Reihenfolge die passenden Namen. Und dann eben die Liste durchsuchen und wenn die Nummer matcht, aus der zweiten Liste an der gleichen Stelle den Namen. JSON wäre halt alles in einem, dafür habe ich mit JSON noch nie intensiver gearbeitet.Gibt es Erfahrungsberichte oder Bestpractice, was in dem Fall einfacher/besser wäre?
Gruss, Jürgen
-
@wildbill sagte in Telefonbuch ohne Fritzbox:
Gibt es Erfahrungsberichte oder Bestpractice, was in dem Fall einfacher/besser wäre?
so leid es mir für ihn tut, aber ich fürchte da brauchen wir die Hilfe von @paul53, der hat das mit den zwei Listen schon mal irgendwo gepostet
-
@wildbill sagte: die anrufende Nummer via Rest-Api in einen Datenpunkt schreibt
Ansatz mit zwei korrespondierenden Listen:
const idNummer = '...'; const idName = '...'; const nummern = [ '1234567', '8934568' ]; const namen = [ 'Müller', 'Heinz' ]; on({id: idNummer}, function(dp) { let name = dp.state.val; // unbekannter Name = Nummer for(let i = 0; i < nummern.length; i++) { if(nummern[i] == dp.state.val) { name = namen[i]; break; } } setState(idName, name); });
-
@paul53 Kommt als Number ohne führende Nullen. Also beispielsweise 49891234567 für 0049 (Deutschland) 89 (München) 1234567 (Rufnummer). Oder 43 xxxxxxx wenn es aus Österreich kommt.
Das mit den fehlenden Nullen und abschneiden bzw, ersetzen der 49 mit 0 habe ich bereits. Ebenso das Hinzufügen von zwei 00, wenn es aus dem Ausland kommt.
Mir fehlt nur der Ansatz, um bei einigen Nummern eben stattdessen den Namen aus einer Liste oder einem JSON zu suchen bzw. wie ich das sinnvollerweise aufbaue.Gruss, Jürgen
-
@wildbill sagte: Mir fehlt nur der Ansatz
Siehe oben: Hat sich mit deiner Antwort überschnitten.
-
@paul53 ist das nicht ein Anwendungsfall für eine Map?
https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map
Key->Telefonnummer, Value->Anzeigename?
const myTelBook = new Map(); myTelBook.set( 110, 'Polizei'); myTelBook.set(112, 'Feuerwehr'); .... my Anrufername=myTelBook.get(110); // returns 'Polizei'
-
@martinp sagte: ist das nicht ein Anwendungsfall für eine Map?
Damit habe ich mich noch nicht beschäftigt.
-
@paul53 sagte: Ansatz mit zwei korrespondierenden Listen:
Habe oben korrigiert (break).
-
@paul53 Super, das probiere ich mal so umzusetzen.
@MartinP Das scheint mir dann eher in die Richtung zu gehen, in die auch JSON läuft. Muss ich mir auch nochmal anschauen. Aber ich glaube, das Script von @paul53 macht genau das, was ich will, ohne großen Aufwand mit Pflege zu betreiben. Zwei Listen erstellen, bei Bedarf anpassen und gut.Komme nur heute nicht mehr dazu, Frau hat Geburtstag, gibt sonst Mecker.
Aber ich gebe Bescheid, wie es letztendlich geklappt hat.
Danke Euch Allen. @Homoran Ich setze mal auf gelöst, sollte ja nur noch eine Sache der Umsetzung sein.
Gruss, Jürgen
-
-
@wildbill Der Ansatz mit den zwei Listen macht mir Bauchschmerzen... Das ist bei einer Handvoll Namens - Telefonnummern Paaren noch handhabbar, aber wenn es mehr werden, wird das schon eine Pein, darauf zu achten, dass es da keinen Fehler gibt ...
Wenn Großoma Erna gestorben ist, und man nur ihren Namen löscht, aber nicht ihre Telefonnummer, hat man schon einen Versatz...
Erfahrungsgemäß fängt man dann irgendwann mit dem Finger an, auf dem Bildschirm Zeilen zu zählen, kopiert die beiden Tabellen über die Zwischenablage in einen "richtigen" Editor, um sie nebeneinander anzuschauen usw ...
Wichtig ist bei der Map, dass die einem das Durchsuchen abnimmt. Man wirft eine Telefonnummer ein, und kriegt den Namen zurück.... ob überhaupt ein passendes Key/Value Paar existiert, kann man vorher mit exist(key) feststellen ...
Keine handgestrickte Suchschleife o. Ä.
Eine ganz professionelle Lösung ist das aber auch noch nicht - da würde man mit einer Datenbank arbeiten
-
@martinp Ja, das hat mir bei der Lösung mit MAP oder JSON besser gefallen, dass es jeweils ein festes Wertepaar gibt. Da es aber wohl wirklich nur auf zwei Handvoll Nummern rauslaufen wird, ist es mit einer Liste, denke ich, noch überschaubar. Ich werde vermutlich damit anfangen, aber in einer ruhigen Minute mal mit Wertepaaren probieren, nur, um vielleicht mal wieder was zu lernen.
Danke.Gruss, Jürgen
-
@wildbill sagte: JSON besser gefallen, dass es jeweils ein festes Wertepaar gibt.
Version mit Objekt (nicht JSON):
const telBuch = { '1234567': 'Müller', '8934568': 'Heinz' }; const idNummer = '...'; const idName = '...'; on({id: idNummer}, function(dp) { let name = telBuch[dp.state.val]; if(!name) name = dp.state.val // unbekannter Name = Nummer setState(idName, name); });
-
@paul53 So, abschließend noch kurz die Vollzugsmeldung. Alles funktioniert wie gehabt. Um nicht zwei Skripte (mein Blockly und das neue) laufen haben zu müssen, habe ich einfach mein bisheriges Blockly als Javascript ausgeben lassen und einfach quick&dirty vor das neue Script kopiert. Datenpunkte nochmal angepasst und läuft. Ich hatte gestern Nacht, als meine Frau im Bett war, noch versucht, das irgendwie eleganter zu machen, insbesondere nur mit einem Trigger und sauber mit idNummer und idName durchgehend, anstatt wieder auf die Datenpunkte zuzugreifen, aber da bin ich in reinem Javascript irgendwie nicht fit genug. Spätestens beim Versuch, davon dann erst aber der vierten Stelle auszulesen habe ich mich dann irgendwo in den Klammern verzettelt...
Unten das Gesamt-Script, wie es läuft, falls jemand mal sowas in der Art braucht. Und nochmal Danke an Alle. Ohne Eure Hilfe wäre das Ganze sicher noch weitaus uneleganter geworden. So gefällt es mir richtig gut und ich mach mich nun an das Eintragen von meinen Nummern und Namen.
Gruss, Jürgen
function subsequenceFromStartLast(sequence, at1) { var start = at1; var end = sequence.length - 1 + 1; return sequence.slice(start, end); } on({ id: [].concat(['0_userdata.0.Zustand.Anrufer-Name']), change: 'any' }, async (obj) => { let value = obj.state.val; let oldValue = obj.oldState.val; if (compareTime('08:00', '22:00', 'between')) { if (('' + getState('0_userdata.0.Zustand.Anrufer-Name').val).indexOf('00') + 1 == 1) { if (('' + getState('0_userdata.0.Zustand.Anrufer-Name').val).indexOf('0049') + 1 == 1) { setState('javascript.0.Alexa-Text' /* Alexa-Text */, ('Anruf von 0' + String(subsequenceFromStartLast(('' + getState('0_userdata.0.Zustand.Anrufer-Name').val), 4))), true); } else { setState('javascript.0.Alexa-Text' /* Alexa-Text */, ('Anruf von 00' + String(subsequenceFromStartLast(('' + getState('0_userdata.0.Zustand.Anrufer-Id').val), 0))), true); } } else { setState('javascript.0.Alexa-Text' /* Alexa-Text */, ('Anruf von ' + String(getState('0_userdata.0.Zustand.Anrufer-Name').val)), true); } } }); const telBuch = { '1234567': 'Müller', '498934568': 'Heinz' }; const idNummer = '0_userdata.0.Zustand.Anrufer-Id'/*Anrufer-Id*/; const idName = '0_userdata.0.Zustand.Anrufer-Name'/*Anrufer-Name*/; on({ id: idNummer, change: 'any' }, function(dp) { let name = telBuch[dp.state.val]; if(!name) name = '00' + dp.state.val // unbekannter Name = Nummer setState(idName, name); });
EDIT: Script angepasst. Es muss bei den Telefonnummern 49 davor stehen.