NEWS
TR-064 Fritzbox Anrufbeantworter
-
Ich hab jetzt etwas weiter probiert...
Abhören geht ja per Anruf daheim und Pin.
Ein Problem ist jedoch zu unterscheiden ob jemand auf den Anrufbeantworter gesprochen hat oder aufgelegt bzw. auch den Anruf auszusortieren den ich tätige um den Anrufbeantworter abzuhören.
Jemand Ideen dazu?
-
@Pihero Vielleicht über Anrufdauer und Anrufernummer?
-
@Pihero Hier gibt es ein Skript https://forum.iobroker.net/topic/15533/tr-64-erkennen-ob-ein-aufruf-auf-ab-gesprochen-wurde/5 welches ermittelt ob eine neue Nachricht auf dem AB vorhanden ist.
Bin gerade mit dem Skript am experimentieren aber auf dem ersten Blick scheint es zu funktionieren.Man muss den Zähler noch manuell zurücksetzen, da man bisher nicht aus der Fritzbox die Information herausbekommt ob es noch neue Nachrichten auf dem AB gibt.
-
Ich bin einen kleinen Schritt weiter gekommen um an Daten zum Anrufbeantworter aus der Fritzbox zu kommen.
Über den Datenpunkt
tr-064.0.states.command
kann man Befehle an die Fritzbox schicken. Das Format ist hier erklärt https://developer.aliyun.com/mirror/npm/package/iobroker.tr-064 im Kapitel "command & commandResult state"Eine Übersicht der ganzen Befehle für die verschiedenen Dienste gibt es hier
https://avm.de/service/schnittstellen/?spm=a2c6h.14275010.0.0.202628cfwq7844 für den Anrufbeantworter ist sind dann die genauen Befehle hier definiert:
https://avm.de/fileadmin/user_upload/Global/Service/Schnittstellen/x_tam.pdfWenn ich z.B. in den Datenpunkt
tr-064.0.states.command
folgendes schreibe:{"service": "urn:dslforum-org:service:X_AVM-DE_TAM:1","action": "GetList","params": {}}
aufrufe kommt in Datenpunkt
tr-064.0.states.commandResult
folgende Rückmeldung:{"NewTAMList":"<List><TAMRunning>1</TAMRunning><Stick>0</Stick><Status>0</Status><Capacity>79</Capacity><Item><Index>0</Index><Display>1</Display><Enable>1</Enable><Name>Anrufbeantworter</Name></Item><Item><Index>1</Index><Display>0</Display><Enable>0</Enable><Name></Name></Item><Item><Index>2</Index><Display>0</Display><Enable>0</Enable><Name></Name></Item><Item><Index>3</Index><Display>0</Display><Enable>0</Enable><Name></Name></Item><Item><Index>4</Index><Display>0</Display><Enable>0</Enable><Name></Name></Item></List>\n"}
Wenn ich
{"service": "urn:dslforum-org:service:X_AVM-DE_TAM:1","action": "GetMessageList","params": {"NewIndex ": "0"}}
ausführe erhalte ich Result
{"NewURL":"http://192.168.178.1:49000/tamcalllist.lua?sid=1234567890123456&tamindex=0"}
(sid hier durch einen beliebigen Wert ersetzt). Wenn ich den Link aufrufe kommt leider nur folgende Struktur mit einem Fehler zurück:
Ich vermute, dass die Authentifizierung, welche über den tr-064 Adapter gemacht wird nicht hier gültig ist und ich somit die Fehlermeldung bekomme.
Gibt es einen einfach Weg dies über ein Skript sauber zu lösen oder wird das nur funktionieren, wenn man den Adapter entsprechend erweitert? -
Ich bin einen großen Schritt weiter gekommen.
Nachdem ich den Link aus dem commandResult nicht direkt im Browser öffnen konnte kam ich nach einigen Stunden auf die Idee die Session-ID welche man im Link sieht, durch eine gültige Session-ID zu erstezen, welche man in den Links der Menüs sieht, wenn man in der Fritzbox GUI unterwegs ist. Nach diesem Schritt kam dann auf einmal ein XML zum Vorschein, welches die Daten des Anrufbeantworters enthielt.Daraufhin hab ich dann versucht über ein Skript den Link aus dem commandResult zu extrahieren und das dahinterliegende XML File zu parsen.
Mit dem folgenden Skript fülle ich jetzt den command Datenpunt, extrahiere den Link aus commandResult und parse das XML File in eine JSON Struktur die dann in einem neuen Datenpunk abgelegt wird.
Jetzt muss man die enthaltenen Informationen nur noch extrahieren, um sie z.B. in VIS weiter verarbeiten / darstellen zu können.Den Code zum parsen des XML hab ich aus diesem Beitrag von @Bluefox übernommen
Skript:
So sieht die XML Sturktur aus, welche aus der Fritzbox kommt:
Body: <?xml version="1.0" encoding="UTF-8"?> <Root> <!-- index:0 --> <!-- max:999 --> <!-- tam calls:2 --> <Message> <Index>1</Index> <Tam>0</Tam> <Called>01234567890</Called> <Date>01.09.20 11:56</Date> <Duration>0:01</Duration> <Inbook>1</Inbook> <Name>Feuersturm</Name> <New>1</New> <Number>0987654321</Number> <Path>/download.lua?path=/data/tam/rec/rec.0.001</Path> </Message> <Message> <Index>0</Index> <Tam>0</Tam> <Called>01234567890</Called> <Date>01.09.20 11:51</Date> <Duration>0:01</Duration> <Inbook>1</Inbook> <Name>Feuersturm</Name> <New>0</New> <Number>0987654321</Number> <Path>/download.lua?path=/data/tam/rec/rec.0.000</Path> </Message> </Root>
und so das erzeugte JSON
Result: {"Root":{"Message":[{"Index":"1","Tam":"0","Called":"01234567890","Date":"01.09.20 11:56","Duration":"0:01","Inbook":"1","Name":"Feuersturm","New":"1","Number":"0987654321","Path":"/download.lua?path=/data/tam/rec/rec.0.001"},{"Index":"0","Tam":"0","Called":"01234567890","Date":"01.09.20 11:51","Duration":"0:01","Inbook":"1","Name":"Feuersturm","New":"0","Number":"0987654321","Path":"/download.lua?path=/data/tam/rec/rec.0.000"}]}}
In der Fritzbox werden die Einträge im Anrufbeantworter wie folgt dargestellt:
-
Mach ich wo einen Fehler?
tr-064.0 2020-09-03 22:56:03.047 info (20504) {"NewURL":"http://192.168.178.1:49000/tamcalllist.lua?sid=xxxxxxxxxxxxxxxx&tamindex=0"} javascript.0 2020-09-03 22:56:03.030 error (9576) at Script.runInContext (vm.js:131:20) javascript.0 2020-09-03 22:56:03.030 error (9576) at script.js.Fritz.Anrufbeantworter:55:1 javascript.0 2020-09-03 22:56:03.030 error (9576) at Fritzbox_Anrufbeantworter_Daten_abfragen (script.js.Fritz.Anrufbeantworter:16:4) javascript.0 2020-09-03 22:56:03.030 error (9576) ReferenceError: debug is not defined javascript.0 2020-09-03 22:56:03.030 error (9576) ^ javascript.0 2020-09-03 22:56:03.029 error (9576) if(debug) console.log("commandResult: " + getState("tr-064.0.states.commandResult").val); javascript.0 2020-09-03 22:56:03.029 error (9576) script.js.Fritz.Anrufbeantworter: script.js.Fritz.Anrufbeantworter:16
-
@sigi234 Ich hab eine Zeile im Skript ganz oben vergesen. Hab sie oben im Skript gerade ergänzt.
Füge bei dir einfach folgende Zeile am Anfang des Skriptes ein:const debug = true;
Über den Debug Werte steuere ich in meinen Skripten ob eine Ausgabe im Log erfolgen soll.
-
@Feuersturm sagte in TR-064 Fritzbox Anrufbeantworter:
@sigi234 Ich hab eine Zeile im Skript ganz oben vergesen. Hab sie oben im Skript gerade ergänzt.
Füge bei dir einfach folgende Zeile am Anfang des Skriptes ein:const debug = true;
Über den Debug Werte steuere ich in meinen Skripten ob eine Ausgabe im Log erfolgen soll.
javascript.0 2020-09-03 23:08:05.950 error at Parser.exports.Parser.Parser.parseString (C:\Program Files\iobroker\Test\node_modules\iobroker.javascript\node_modules\xml2js\lib\parser.js:323:31)) javascript.0 2020-09-03 23:08:05.950 error at SAXParser.write (C:\Program Files\iobroker\Test\node_modules\iobroker.javascript\node_modules\xml2js\node_modules\sax\lib\sax.js:1436:13) javascript.0 2020-09-03 23:08:05.950 error at closeTag (C:\Program Files\iobroker\Test\node_modules\iobroker.javascript\node_modules\xml2js\node_modules\sax\lib\sax.js:889:7) javascript.0 2020-09-03 23:08:05.950 error at emitNode (C:\Program Files\iobroker\Test\node_modules\iobroker.javascript\node_modules\xml2js\node_modules\sax\lib\sax.js:629:5) javascript.0 2020-09-03 23:08:05.950 error at emit (C:\Program Files\iobroker\Test\node_modules\iobroker.javascript\node_modules\xml2js\node_modules\sax\lib\sax.js:624:35) javascript.0 2020-09-03 23:08:05.950 error at SAXParser.onclosetag (C:\Program Files\iobroker\Test\node_modules\iobroker.javascript\node_modules\xml2js\lib\parser.js:262:26) javascript.0 2020-09-03 23:08:05.950 error at Parser.EventEmitter.emit (domain.js:482:12) javascript.0 2020-09-03 23:08:05.950 error at Parser.emit (events.js:315:20) javascript.0 2020-09-03 23:08:05.950 error at Parser.<anonymous> (C:\Program Files\iobroker\Test\node_modules\iobroker.javascript\node_modules\xml2js\lib\parser.js:304:18) javascript.0 2020-09-03 23:08:05.950 error at script.js.Fritz.Anrufbeantworter:174:74 javascript.0 2020-09-03 23:08:05.950 error (9576) Error in request callback: Error [ERR_UNHANDLED_ERROR]: Unhandled error. (TypeError: Cannot read property 'Index' of undefined
-
@sigi234 Du musst in der Javascript Instanz unter "Zusätzliche NPM Module" noch "xml2js" (mit Enter bestätigen) eintragen
-
@Feuersturm sagte in TR-064 Fritzbox Anrufbeantworter:
@sigi234 Du musst in der Javascript Instanz unter "Zusätzliche NPM Module" noch "xml2js" (mit Enter bestätigen) eintragen
Habe ich.
javascript.0 2020-09-03 23:14:19.107 error at Parser.exports.Parser.Parser.parseString (C:\Program Files\iobroker\Test\node_modules\iobroker.javascript\node_modules\xml2js\lib\parser.js:323:31)) javascript.0 2020-09-03 23:14:19.107 error at SAXParser.write (C:\Program Files\iobroker\Test\node_modules\iobroker.javascript\node_modules\xml2js\node_modules\sax\lib\sax.js:1436:13) javascript.0 2020-09-03 23:14:19.107 error at closeTag (C:\Program Files\iobroker\Test\node_modules\iobroker.javascript\node_modules\xml2js\node_modules\sax\lib\sax.js:889:7) javascript.0 2020-09-03 23:14:19.107 error at emitNode (C:\Program Files\iobroker\Test\node_modules\iobroker.javascript\node_modules\xml2js\node_modules\sax\lib\sax.js:629:5) javascript.0 2020-09-03 23:14:19.107 error at emit (C:\Program Files\iobroker\Test\node_modules\iobroker.javascript\node_modules\xml2js\node_modules\sax\lib\sax.js:624:35) javascript.0 2020-09-03 23:14:19.107 error at SAXParser.onclosetag (C:\Program Files\iobroker\Test\node_modules\iobroker.javascript\node_modules\xml2js\lib\parser.js:262:26) javascript.0 2020-09-03 23:14:19.107 error at Parser.EventEmitter.emit (domain.js:482:12) javascript.0 2020-09-03 23:14:19.107 error at Parser.emit (events.js:315:20) javascript.0 2020-09-03 23:14:19.107 error at Parser.<anonymous> (C:\Program Files\iobroker\Test\node_modules\iobroker.javascript\node_modules\xml2js\lib\parser.js:304:18) javascript.0 2020-09-03 23:14:19.107 error at script.js.Fritz.Anrufbeantworter:174:74 javascript.0 2020-09-03 23:14:19.107 error (14724) Error in request callback: Error [ERR_UNHANDLED_ERROR]: Unhandled error. (TypeError: Cannot read property 'Index' of undefined javascript.0 2020-09-03 23:14:19.104 info (14724) script.js.Fritz.Anrufbeantworter: Result: {"Root":{"Message":{"Index":"0","Tam":"0","Called":"0000000000000000","Date":"25.07.20 08:48","Duration":"0:01","Inbook":"0","Name":"","New":"0","Number":"
-
@sigi234 Hmm, jetzt gehen mir langsam die Ideen aus. Welche (Adapter) Versionen setzt du ein? Ich habe folgendes im Einsatz:
js-controller: 3.1.6
node: 12.18.0
nodejs: v12.18.0
npm: 6.14.4
Admin: 4.1.7
Java Script Enging: 4.6.26Zum Zeitpunkt 2020-09-03 23:14:19.104 wird der JSON String ja ausgegeben. Der String scheint abgeschnitten zu sein. Hat er die gleiche Struktur wie in meinem Beispiel oben?
-
@Feuersturm sagte in TR-064 Fritzbox Anrufbeantworter:
Zum Zeitpunkt 2020-09-03 23:14:19.104 wird der JSON String ja ausgegeben. Der String scheint abgeschnitten zu sein. Hat er die gleiche Struktur wie in meinem Beispiel oben?
Geht jetzt, habe mal auf den AB gesprochen.
-
@sigi234 Das klingt doch gut
Ich tüftel gerade damit, dass ich den JSON String mit dem widget "basic - table" angezeigt bekomme. Wenn ich den Inhalt aus dem Datenpunkt im Widget angebe wird mir nichts ausgegeben.
Wenn man am Anfang
{"Root": {"Message":
sowie am Ende}}
vom JSON String entfernt, dann lassen sich meine beiden Einträge anzeigen
-
Da kann uns sicher ein Spezialist helfen.
Cool wäre es wenn man den AB Status in einen DP schreiben kann.@apollon77
Das haben wir schon lange gesucht, da kann man ja was machen? -
@sigi234 Die Info das eine neue Nachricht / noch nicht abgehörte Nachrichten vorhanden sind kann man mit ein paar Zeilen garantiert extrahieren.
Mit dem Ausdruckresult.Root.Message.length
kommt man an die Info wieviele Nachrichten auf dem AB sind und dann lässt man eine Schleife durchlaufen wo aufJSON.stringify(result.Root.Message[i].New
== 1 geprüft wird. Wenn mindestens einmal die Prüfung true ist, schreibt man dies in einen neuen Datenpunkt.log("Einträge / Länge: " + JSON.stringify(result.Root.Message.length)); log("Nachricht neu: " + JSON.stringify(result.Root.Message[0].New));
Das werde ich in den nächsten Tagen noch in das Skript einbauen aber nicht mehr heute
-
Ich hänge gerade noch an folgender Stelle.
Im XML ist als Kommentar folgende Information enthalten<!-- tam calls:2 -->
Diese Information wird leider nicht ins JSON Format mit überführt. Ich vermute weil es im XML nur ein Kommentar ist. Gibt es eine einfache Möglichkeit, dass diese Kommentare auch mit in der JSON Struktur landen oder muss man sich das XML zurechntschneiden und wegspeichern, um an die Information zu gelangen?
Diese Info könnte man nutzen, um einfacher die Auswertung vom JSON Format zu machen, da dies in Abhängigkeit von der Anzalhl der Nachrichten etwas anders aufgebaut ist:
Abhängig von der Anzahl der Nachrichten auf dem AB sieht die JSON Sturktur etwas anders aus
//Wenn keine Nachricht auf dem AB ist //JSON: {"Root":"\n\n\n\n"} //Wenn nur eine Nachricht vorhanden ist, dann enthält Message kein Array: //JSON: {"Root":{"Message":{"Index":"0" //Wenn mehr als eine Nachricht vorhanden sind, dann enthält Message ein Array: //JSON: {"Root":{"Message":[{"Index":"1",....},{"Index":"0",...}]}}
-
Nearly all services except GetSecurityPort from the example above need authentication. The best way to achieve this is to add an extra user with its own password.
-
@sigi234 sagte in TR-064 Fritzbox Anrufbeantworter:
Das haben wir schon lange gesucht, da kann man ja was machen?
Details als Issue oder besser ein PR wäre awesome
-
@Feuersturm
Bist du schon weiter? -
@sigi234 Noch nicht aber ich bin guter Dinge, dass ich jetzt am Wochenende abends etwas Zeit dafür habe. Die Idee dass mittelfristig in den Adapter zu überführen finde ich gut. Da ich noch keine Erfahrung mit der Adapter Programmierung habe würde ich erstmal das Skript weiter entwickeln und dann kann man die Mechanismen vermutlich relativ einfach in den Adapter überführen.
Es gibt wohl auch die Möglichkeit, dass man Nachrichten auch löschen kann und das man den Status Neu verändern kann wobei ich noch nicht weiß wofür man das effektiv gebrauchen kann.