See it live in action: Demo video
Here you can find the sources of a custom Alexa skill that controls an LED strand on a Christmas tree. This repo contains:
- Java code for an AWS Lambda function which is the endpoint for the Alexa skill
- An Arduino sketch which sits between AWS IoT and a WS2811 LED strand. The sketch is optimized for running on an Arduino Yun.
If you want to build your own Alexa-controlled christmas tree with help of these sources you need specific hardware. This project used the following components:
- 1 x Arduino Yun with Linino OS
- 2 x WS2811 LED Strand with 50 LEDs each
- 3 x Jumper wires
- 1 x Power Supply Adapter and 2.1mm x 5.5mm DC Connector
- 1 x Micro-USB to USB Cable
- 1 x Amazon Echo or Amazon Dot or Amazon Tap
The following image illustrates a typical roundtrip to handle a voice user request.
The solution leverages a bunch of AWS cloud services to communicate with the hardware backend - the christmas tree. The only things you really need to set up is the Lambda function, an S3 bucket containing the MP3 files and an IAM role with AWS IoT and Dynamo permissions. The table in Dynamo as well as the thing shadow in AWS IoT will be created on the first skill invocation on the fly.
Understand what happens on a voice user request given to an Alexa device:
-
User speaks to Alexa to "open the christmas tree". ASR and NLU magic happens in the Alexa cloud service.
-
An intent is given to the skill code hosted in AWS Lambda. You can find the code in this repo.
-
If the user just desires an action like "turn on the tree" or "start the show" without giving this skill a color for the tree it looks up the last set color in Dynamo DB. If there's a color given the skill will persist the information in the same table. This is how Alexa keeps in mind the last set color of the tree. Secondly, the action and the color command is written to a thing shadow in AWS IoT.
-
If the shadow is updated an MQTT message is exposed to the delta topic of the corresponding thing. The Arduino Yun is subscribed to that topic. Side note: The name of the thing being created by the skill code is equal to the skill-id coming in (all dots replaced with a dash). This might help you if you want to rebuild the project.
-
The Arduino is polling on the Delta topic so it receives the commands as an MQTT message in JSON format. The information is extracted and the Arduino sketch performs an action with the LED strand according to what is given in the message (new color, christmas show, on, off).
-
Finally, the Arduino sends an MQTT message to the Update topic of the AWS IoT thing in order to let the world know that the action was performed.
-
The message is consumed by the AWS IoT service and the contained state information is written back to the thing shadow as a reported state. It would be possible to also have the skill read the last tree state from the thing shadow instead of looking it up in Dynamo DB. The reason for this fallback approach is MQTT is asynchronous and we cannot rely on the Arduino to give an immediate response.
-
Actually this step happens right after step 3) as the skill is decoupled from the hardware backend on purpose. So right after updating the thing shadow in AWS IoT the skill code returns output speech text and optionally an SSML tag with audio contents. The MP3s which are part of Alexa's playback (christmas sounds) are stored in an AWS S3 bucket.
-
Alexa reads out the text returned by the skill and plays back the audio in the response.
While Arduino does its work it lets you know of its current state over the first LED in the strand.
- a one time red blinking light indicates a AWS IoT connection setup failure
- a two times red blinking light indicates a failed AWS IoT connection attempt
- a three times red blinking light indicates a failed AWS IoT connection configuration
- a green flashlight indicates a successful connection to AWS IoT
- a blue flashlight indicates constant polling to the AWS IoT topic
- a yellow flashlight indicates an error while polling the AWS IoT topic
On startup you might see red flashlights for the period of time it takes for the Arduino to connect to the Wifi. If Wifi is connected there's the green flashlight followed by a constantly blinking blue light to indicate the tree is ready for commands.
If yellow is blinking the AWS IoT topic could not be reached. If that happens (e.g. Arduino lost Wifi connection) it keeps trying for nine more times until it automatically tries to reconnect. That said, after ten times yellow flashlight there should be red / green flashlight for reconnection progress. Once the Arduino reconnects to the Wifi and AWS IoT is reached again, the blue flashlights come up.
If you have any question reach out to the author via
These are the requirements in order to be able to rebuild the solution
- All the hardware listed in above section
- An AWS account for the backend resources
- An Amazon developer account
-
Check out this repository using git
-
In Amazon developer account after you logged in go to Alexa > Alexa Skills Kit > Add a new skill
-
Set up the Skill information with
- Skill type: Custom interaction model
- Language: Of your choice (you could add more later on)
- Name: Christmas Tree (but feel free to choose another name)
- Invocation Name: christmas tree (for English) or weihnachtsbaum (for German)
-
Click Next and go back to Skill information to copy the ApplicationId. You will need it in the next steps
-
In your AWS account go to S3 and create a new bucket and name it as you like
-
Upload all the MP3 files you can find in the audio folder to the newly created bucket. Secondly, select the freshly uploaded files in the S3 web interface, go to Actions and Make Public. Finally, in Properties of your bucket go to Permissions and Add CORS configuration where you would paste and save this xml portion:
<CORSConfiguration xmlns="http://s3.amazonaws.com/doc/2006-03-01/"> <CORSRule> <AllowedOrigin>http://ask-ifr-download.s3.amazonaws.com</AllowedOrigin> <AllowedMethod>GET</AllowedMethod> </CORSRule> </CORSConfiguration>
-
Next, in AWS go to IAM and create a new role called something like "Xmas-Lambda-Execution". To keep things simple just add these three managed policies to it (later you would go for the least privileges approach and adjust permission as needed)
- AmazonS3FullAccess
- AmazonDynamoDBFullAccess
- AWSIoTFullAccess
-
Create a group in IAM with name "Xmas-Arduino-Access" and add AWSIoTFullAccess as the managed policy.
-
Create a user in IAM for programmatic access and add it to the "Xmas-Arduino-Access" group. Save the credentials (AccessId and AccessSecret) provided at the end of the creation process.
-
Now edit the first two configuration items in the app.properties file in your local repo.
- AlexaAppId should be the the applicationId you got in step 4.
- S3BucketUrl should be the url to the bucket you created in step 5.
-
Now mvn package the project and you should get a JAR file.
-
In AWS create a new Lambda function for Java8 and choose the existing execution role you created in step 7. Upload the JAR file to this function. The Lambda event trigger must be set to Alexa skills Kit. If you don't see it in the list you are in the wrong AWS region. Make sure to create the Lambda function in either Ireland or N. Virginia. If you're using my code the handler must be set to io.klerch.alexa.xmastree.skill.XmasSpeechletHandler.
-
Go back to your skill in the Amazon developer console and proceed with the skill configuration.
- The Interaction model section you set up just be copy and paste what you can find in the resources/in-folder in this repo. Don't forget to create a custom slot with name TreeColors and add values of one of the customSlot-TreeColors.txt files.
- In the Configuration section you point to your Lambda function you created in step 12. The ARN can be obtained in the top right corner in AWS Lambda details page.
- In Test section just make sure test is enabled
- Execute a test in the Service simulator in the Test section. Enter the utterance Start christmas tree (Starte Weihnachtsbaum for German). The first invocation takes its time because what happenes in the background is an AWS Dynamo-table is built and an AWS IoT thing is created as well. This initial test is mandatory for the last step because the Arduino needs to have a connection to an MQTT topic which is created with the IoT thing.
-
Next, set up your Arduino Yun and connect it to a Wifi (Howto: https://www.arduino.cc/en/Guide/ArduinoYun)
-
Connect your WS2811 LED strand to the Arduino. Make sure you use the data-pin 6. If you have any question on this please contact me.
-
Follow the installation instructions at https://github.com/aws/aws-iot-device-sdk-arduino-yun. This project uses MQTT over WebSockets. Keep that in mind when you are reading and doing the instructions. The Access key and secret they are talking about are the ones you got in step 9.
-
Edit the aws_iot_config.h in your local repo and adjust values for:
- AWS_IOT_MQTT_HOST is specific to your account. The easiest way to obtain your endpoint is to use the following command in AWS CLI:
$ aws iot describe-endpoint
- AWS_IOT_MY_THING_NAME needs to be the applicationId you got in step 4. Make sure you replace all dots in that id with dashes.
- AWS_IOT_ROOT_CA_FILENAME name of the root certificate you uploaded to the Arduino as instructed in step 14.
-
With Arduino IDE upload the sketch you can find in the arduino-folder in this repo.
- You should see progress and status looking at the first LED on the strand. If your Arduino is still connected to your computer you could also use the Service Monitor tool to get detailed information and serial output.