alexa-remote2
Advanced tools
Comparing version
@@ -37,3 +37,3 @@ /* jshint -W097 */ | ||
if (cookie) this.setCookie(cookie); | ||
let baseUrl = 'layla.amazon.de'; | ||
let baseUrl = 'alexa.amazon.de'; //'layla.amazon.de'; | ||
let self = this; | ||
@@ -58,2 +58,3 @@ let opts = {}; | ||
self._options.amazonPage = self._options.amazonPage || 'amazon.de'; | ||
baseUrl = 'alexa.' + self._options.amazonPage; | ||
@@ -115,3 +116,3 @@ cookie = opts.cookie; | ||
if (!this.commsId) this.commsId = account.commsId; | ||
if (!this.directId) this.directId = account.directId; | ||
//if (!this.directedId) this.directedId = account.directedId; | ||
}); | ||
@@ -128,117 +129,123 @@ } | ||
if (!err && wakeWords) wakeWords = wakeWords.wakeWords; | ||
this.getAutomationRoutines ((err, routines) => { | ||
this.routines = []; | ||
if (!err && routines) { | ||
for (let i = 0; i < routines.length; i++) { | ||
let routine = routines[i]; | ||
if (routine['@type'] !== 'com.amazon.alexa.behaviors.model.Automation') { | ||
self._options.logger && self._options.logger('Ignore unknown type of Automation Routine ' + routine['@type']); | ||
continue; | ||
this.getMusicProviders((err, providers) => { | ||
this.musicProviders = []; | ||
if (!err && providers) { | ||
this.musicProviders = providers; | ||
} | ||
this.getAutomationRoutines ((err, routines) => { | ||
this.routines = []; | ||
if (!err && routines) { | ||
for (let i = 0; i < routines.length; i++) { | ||
let routine = routines[i]; | ||
if (routine['@type'] !== 'com.amazon.alexa.behaviors.model.Automation') { | ||
self._options.logger && self._options.logger('Ignore unknown type of Automation Routine ' + routine['@type']); | ||
continue; | ||
} | ||
if (!routine.sequence) { | ||
self._options.logger && self._options.logger('Automation Routine has no sequence ' + JSON.stringify(routine)); | ||
continue; | ||
} | ||
let name = routine.name; | ||
if (!name && routine.triggers && routine.triggers[0].payload && routine.triggers[0].payload.utterance) { | ||
name = routine.triggers[0].payload.utterance; | ||
} | ||
else if (!name && routine.triggers && routine.triggers[0].payload && routine.triggers[0].payload.schedule && routine.triggers[0].payload.schedule.triggerTime) { | ||
name = routine.triggers[0].payload.schedule.triggerTime; | ||
if (name.length === 6) name = name.replace(/^({0-9}{2})({0-9}{2})({0-9}{2})$/, '$1:$2:$3'); | ||
if (routine.triggers[0].payload.schedule.recurrence) name += ` ${routine.triggers[0].payload.schedule.recurrence}`; | ||
} | ||
else { | ||
self._options.logger && self._options.logger('Ignore unknown type of Automation Routine Trigger' + JSON.stringify(routine.triggers.payload)); | ||
name = 'Unknown'; | ||
} | ||
routine.friendlyName = name; | ||
let idSplit = routine.automationId.split('.'); | ||
routine.friendlyAutomationId = idSplit[idSplit.length - 1]; | ||
this.routines.push(routine); | ||
} | ||
if (!routine.sequence) { | ||
self._options.logger && self._options.logger('Automation Routine has no sequence ' + JSON.stringify(routine)); | ||
continue; | ||
} | ||
let name = routine.name; | ||
if (!name && routine.triggers && routine.triggers[0].payload && routine.triggers[0].payload.utterance) { | ||
name = routine.triggers[0].payload.utterance; | ||
} | ||
else if (!name && routine.triggers && routine.triggers[0].payload && routine.triggers[0].payload.schedule && routine.triggers[0].payload.schedule.triggerTime) { | ||
name = routine.triggers[0].payload.schedule.triggerTime; | ||
if (name.length === 6) name = name.replace(/^({0-9}{2})({0-9}{2})({0-9}{2})$/, '$1:$2:$3'); | ||
if (routine.triggers[0].payload.schedule.recurrence) name += ` ${routine.triggers[0].payload.schedule.recurrence}`; | ||
} | ||
else { | ||
self._options.logger && self._options.logger('Ignore unknown type of Automation Routine Trigger' + JSON.stringify(routine.triggers.payload)); | ||
name = 'Unknown'; | ||
} | ||
routine.friendlyName = name; | ||
let idSplit = routine.automationId.split('.'); | ||
routine.friendlyAutomationId = idSplit[idSplit.length - 1]; | ||
this.routines.push(routine); | ||
} | ||
} | ||
this.getDevices((err, result) => { | ||
if (!err && result && Array.isArray(result.devices)) { | ||
let customerIds = {}; | ||
this.devices = result.devices; | ||
result.devices.forEach((device) => { | ||
this.serialNumbers [device.serialNumber] = device; | ||
let name = device.accountName; | ||
this.names [name] = device; | ||
this.names [name.toLowerCase()] = device; | ||
if (device.deviceTypeFriendlyName) { | ||
name += ' (' + device.deviceTypeFriendlyName + ')'; | ||
this.getDevices((err, result) => { | ||
if (!err && result && Array.isArray(result.devices)) { | ||
let customerIds = {}; | ||
this.devices = result.devices; | ||
result.devices.forEach((device) => { | ||
this.serialNumbers [device.serialNumber] = device; | ||
let name = device.accountName; | ||
this.names [name] = device; | ||
this.names [name.toLowerCase()] = device; | ||
} | ||
device._orig = JSON.parse(JSON.stringify(device)); | ||
device._name = name; | ||
device.sendCommand = this.sendCommand.bind(this, device); | ||
device.setTunein = this.setTunein.bind(this, device); | ||
device.rename = this.renameDevice.bind(this, device); | ||
device.setDoNotDisturb = this.setDoNotDisturb.bind(this, device); | ||
device.delete = this.deleteDevice.bind(this, device); | ||
if (device.deviceTypeFriendlyName) this.friendlyNames[device.deviceTypeFriendlyName] = device; | ||
if (customerIds[device.deviceOwnerCustomerId] === undefined) customerIds[device.deviceOwnerCustomerId] = 0; | ||
customerIds[device.deviceOwnerCustomerId] += 1; | ||
if (this.version === undefined) this.version = device.softwareVersion; | ||
if (this.customer === undefined) this.customer = device.deviceOwnerCustomerId; | ||
device.isControllable = ( | ||
device.capabilities.includes('AUDIO_PLAYER') || | ||
device.capabilities.includes('AMAZON_MUSIC') || | ||
device.capabilities.includes('TUNE_IN') | ||
); | ||
device.hasMusicPlayer = ( | ||
device.capabilities.includes('AUDIO_PLAYER') || | ||
device.capabilities.includes('AMAZON_MUSIC') | ||
); | ||
device.isMultiroomDevice = (device.clusterMembers.length > 0); | ||
device.isMultiroomMember = (device.parentClusters.length > 0); | ||
if (device.deviceTypeFriendlyName) { | ||
name += ' (' + device.deviceTypeFriendlyName + ')'; | ||
this.names [name] = device; | ||
this.names [name.toLowerCase()] = device; | ||
} | ||
device._orig = JSON.parse(JSON.stringify(device)); | ||
device._name = name; | ||
device.sendCommand = this.sendCommand.bind(this, device); | ||
device.setTunein = this.setTunein.bind(this, device); | ||
device.rename = this.renameDevice.bind(this, device); | ||
device.setDoNotDisturb = this.setDoNotDisturb.bind(this, device); | ||
device.delete = this.deleteDevice.bind(this, device); | ||
if (device.deviceTypeFriendlyName) this.friendlyNames[device.deviceTypeFriendlyName] = device; | ||
if (customerIds[device.deviceOwnerCustomerId] === undefined) customerIds[device.deviceOwnerCustomerId] = 0; | ||
customerIds[device.deviceOwnerCustomerId] += 1; | ||
if (this.version === undefined) this.version = device.softwareVersion; | ||
if (this.customer === undefined) this.customer = device.deviceOwnerCustomerId; | ||
device.isControllable = ( | ||
device.capabilities.includes('AUDIO_PLAYER') || | ||
device.capabilities.includes('AMAZON_MUSIC') || | ||
device.capabilities.includes('TUNE_IN') | ||
); | ||
device.hasMusicPlayer = ( | ||
device.capabilities.includes('AUDIO_PLAYER') || | ||
device.capabilities.includes('AMAZON_MUSIC') | ||
); | ||
device.isMultiroomDevice = (device.clusterMembers.length > 0); | ||
device.isMultiroomMember = (device.parentClusters.length > 0); | ||
if (notifications && Array.isArray(notifications)) { | ||
notifications.forEach((noti) => { | ||
if (noti.deviceSerialNumber === device.serialNumber) { | ||
if (device.notifications === undefined) device.notifications = []; | ||
noti.set = this.changeNotification.bind(this, noti); | ||
device.notifications.push(noti); | ||
if (notifications && Array.isArray(notifications)) { | ||
notifications.forEach((noti) => { | ||
if (noti.deviceSerialNumber === device.serialNumber) { | ||
if (device.notifications === undefined) device.notifications = []; | ||
noti.set = this.changeNotification.bind(this, noti); | ||
device.notifications.push(noti); | ||
} | ||
}); | ||
} | ||
if (Array.isArray (wakeWords)) wakeWords.forEach ((o) => { | ||
if (o.deviceSerialNumber === device.serialNumber && typeof o.wakeWord === 'string') { | ||
device.wakeWord = o.wakeWord.toLowerCase(); | ||
} | ||
}); | ||
} | ||
if (Array.isArray (wakeWords)) wakeWords.forEach ((o) => { | ||
if (o.deviceSerialNumber === device.serialNumber && typeof o.wakeWord === 'string') { | ||
device.wakeWord = o.wakeWord.toLowerCase(); | ||
} | ||
}); | ||
}); | ||
this.ownerCustomerId = Object.keys(customerIds)[0]; | ||
} | ||
if (opts.bluetooth) { | ||
this.getBluetooth((err, res) => { | ||
if (err || !res || !Array.isArray(res.bluetoothStates)) { | ||
opts.bluetooth = false; | ||
return callback && callback (); | ||
} | ||
res.bluetoothStates.forEach((bt) => { | ||
if (bt.pairedDeviceList && this.serialNumbers[bt.deviceSerialNumber]) { | ||
this.serialNumbers[bt.deviceSerialNumber].bluetoothState = bt; | ||
bt.pairedDeviceList.forEach((d) => { | ||
bt[d.address] = d; | ||
d.connect = function (on, cb) { | ||
self[on ? 'connectBluetooth' : 'disconnectBluetooth'] (self.serialNumbers[bt.deviceSerialNumber], d.address, cb); | ||
}; | ||
d.unpaire = function (val, cb) { | ||
self.unpaireBluetooth (self.serialNumbers[bt.deviceSerialNumber], d.address, cb); | ||
}; | ||
}); | ||
this.ownerCustomerId = Object.keys(customerIds)[0]; | ||
} | ||
if (opts.bluetooth) { | ||
this.getBluetooth((err, res) => { | ||
if (err || !res || !Array.isArray(res.bluetoothStates)) { | ||
opts.bluetooth = false; | ||
return callback && callback (); | ||
} | ||
res.bluetoothStates.forEach((bt) => { | ||
if (bt.pairedDeviceList && this.serialNumbers[bt.deviceSerialNumber]) { | ||
this.serialNumbers[bt.deviceSerialNumber].bluetoothState = bt; | ||
bt.pairedDeviceList.forEach((d) => { | ||
bt[d.address] = d; | ||
d.connect = function (on, cb) { | ||
self[on ? 'connectBluetooth' : 'disconnectBluetooth'] (self.serialNumbers[bt.deviceSerialNumber], d.address, cb); | ||
}; | ||
d.unpaire = function (val, cb) { | ||
self.unpaireBluetooth (self.serialNumbers[bt.deviceSerialNumber], d.address, cb); | ||
}; | ||
}); | ||
} | ||
}); | ||
callback && callback(); | ||
}); | ||
callback && callback(); | ||
}); | ||
} else { | ||
callback && callback (); | ||
} | ||
} else { | ||
callback && callback (); | ||
} | ||
}); | ||
}); | ||
@@ -744,3 +751,3 @@ }); | ||
if (typeof command === 'object') { | ||
seqCommandObj = command.sequence; | ||
seqCommandObj = command.sequence || command; | ||
} | ||
@@ -779,2 +786,10 @@ else { | ||
break; | ||
case 'volume': | ||
seqCommandObj.startNode.type = 'Alexa.DeviceControls.Volume'; | ||
value = ~~value; | ||
if (value < 0 || value > 100) { | ||
return callback(new Error('Volume needs to be between 0 and 100')); | ||
} | ||
seqCommandObj.startNode.operationPayload.value = value; | ||
break; | ||
case 'speak': | ||
@@ -785,15 +800,29 @@ seqCommandObj.startNode.type = 'Alexa.Speak'; | ||
} | ||
value = value.replace(/ä/g,'ae'); | ||
value = value.replace(/ä/g,'Ae'); | ||
value = value.replace(/ö/g,'oe'); | ||
value = value.replace(/Ö/g,'Oe'); | ||
value = value.replace(/ü/g,'ue'); | ||
value = value.replace(/Ü/g,'Ue'); | ||
value = value.replace(/ß/g,'ss'); | ||
value = value.replace(/&/g,'und'); | ||
value = value.replace(/é/g,'e'); | ||
value = value.replace(/á/g,'a'); | ||
value = value.replace(/ó/g,'o'); | ||
value = value.replace(/[^-a-zA-Z0-9_,.?! ]/g,''); | ||
value = value.replace(/ /g,'_'); | ||
value = value | ||
.replace(/Â|À|Å|Ã/g, "A") | ||
.replace(/á|â|à|å|ã/g, "a") | ||
.replace(/Ä/g, "Ae") | ||
.replace(/ä/g, "ae") | ||
.replace(/Ç/g, "C") | ||
.replace(/ç/g, "c") | ||
.replace(/É|Ê|È|Ë/g, "E") | ||
.replace(/é|ê|è|ë/g, "e") | ||
.replace(/Ó|Ô|Ò|Õ|Ø/g, "O") | ||
.replace(/ó|ô|ò|õ/g, "o") | ||
.replace(/Ö/g, "Oe") | ||
.replace(/ö/g, "oe") | ||
.replace(/Š/g, "S") | ||
.replace(/š/g, "s") | ||
.replace(/ß/g, "ss") | ||
.replace(/Ú|Û|Ù/g, "U") | ||
.replace(/ú|û|ù/g, "u") | ||
.replace(/Ü/g, "Ue") | ||
.replace(/ü/g, "ue") | ||
.replace(/Ý|Ÿ/g, "Y") | ||
.replace(/ý|ÿ/g, "y") | ||
.replace(/Ž/g, "Z") | ||
.replace(/ž/, "z") | ||
.replace(/&/, "und") | ||
.replace(/[^-a-zA-Z0-9_,.?! ]/g,'') | ||
.replace(/ /g,'_'); | ||
if (value.length === 0) { | ||
@@ -843,2 +872,58 @@ return callback && callback(new Error('Can not speak empty string', null)); | ||
AlexaRemote.prototype.getMusicProviders = function (callback) { | ||
this.httpsGet ('/api/behaviors/entities?skillId=amzn1.ask.1p.music', | ||
callback, | ||
{ | ||
headers: { | ||
'Routines-Version': '1.1.201102' | ||
} | ||
} | ||
); | ||
}; | ||
AlexaRemote.prototype.playMusicProvider = function (serialOrName, providerId, searchPhrase, callback) { | ||
let dev = this.find(serialOrName, callback); | ||
if (!dev) return; | ||
if (searchPhrase === '') return; | ||
const operationPayload = { | ||
'deviceType': dev.deviceType, | ||
'deviceSerialNumber': dev.serialNumber, | ||
'locale': 'de-DE', // TODO!! | ||
'customerId': this.ownerCustomerId, | ||
'musicProviderId': providerId, | ||
'searchPhrase': searchPhrase | ||
}; | ||
const validateObj = { | ||
'type': 'Alexa.Music.PlaySearchPhrase', | ||
'operationPayload': JSON.stringify(operationPayload) | ||
}; | ||
this.httpsGet (`/api/behaviors/operation/validate`, | ||
(err, res) => { | ||
if (err) { | ||
return callback && callback(err, res); | ||
} | ||
if (res.result !== 'VALID') { | ||
return callback && callback(new Error('Request invalid'), res); | ||
} | ||
validateObj.operationPayload = res.operationPayload; | ||
const seqCommandObj = { | ||
'@type': 'com.amazon.alexa.behaviors.model.Sequence', | ||
'startNode': validateObj | ||
}; | ||
seqCommandObj.startNode['@type'] = 'com.amazon.alexa.behaviors.model.OpaquePayloadOperationNode'; | ||
return this.sendSequenceCommand(serialOrName, seqCommandObj, callback); | ||
}, | ||
{ | ||
method: 'POST', | ||
data: JSON.stringify(validateObj) | ||
} | ||
); | ||
}; | ||
AlexaRemote.prototype.sendTextMessage = function (conversationId, text, callback) { | ||
@@ -845,0 +930,0 @@ let o = { |
{ | ||
"name": "alexa-remote2", | ||
"version": "0.2.4", | ||
"version": "0.2.5", | ||
"description": "Remote Control for amazon echo devices", | ||
@@ -5,0 +5,0 @@ "author": { |
@@ -47,4 +47,9 @@ | ||
## Thanks: | ||
Partly based on [Amazon Alexa Remote Control](http://blog.loetzimmer.de/2017/10/amazon-alexa-hort-auf-die-shell-echo.html) (PLAIN shell) and [alexa-remote-control](https://github.com/thorsten-gehrig/alexa-remote-control) and [OpenHab-Addon](https://github.com/openhab/openhab2-addons/blob/f54c9b85016758ff6d271b62d255bbe41a027928/addons/binding/org.openhab.binding.amazonechocontrol) | ||
Thank you for that work. | ||
## Known issues/Todos | ||
* reading notifications works, but changing NOT! | ||
* getNotification works, changeNotification not ... maybe change is DELETE +Create :-) (+ source for createNotification: https://github.com/noelportugal/alexa-reminders/blob/master/alexa-reminders.js#L75, and Delete/create: https://github.com/openhab/openhab2-addons/blob/f54c9b85016758ff6d271b62d255bbe41a027928/addons/binding/org.openhab.binding.amazonechocontrol/src/main/java/org/openhab/binding/amazonechocontrol/internal/Connection.java#L829) | ||
* | ||
@@ -54,2 +59,4 @@ ## Changelog: | ||
### 0.2.x | ||
* (Apollon77) 0.2.5: new functions to read musicproviders and send searchphrases for them | ||
* (Apollon77) 0.2.5: by default direct all calls to "alexa."+amazonPage to be more generic, overwritable | ||
* (Apollon77) 0.2.4: several smaller bugfixes | ||
@@ -56,0 +63,0 @@ * (Apollon77) 0.2.4: an speak call with empty string will return an error |
54984
8.94%1054
7.88%82
9.33%