The purpose of this repository is to reduce the initial effort of creating a good and reliable Alexa Smart Home Skills.
On the other hand, this code was built to provide a basic Alexa integration with 3rd-party Arduino-like controllers and different sensors.
The main idea is about creating so-called virtual devices, which could be then mapped with real micro-controllers via a custom middleware.
You may just wondering why do we need a middleware service? That's due to Smart Home Skills' limitation in terms of a backend. Unfortunately, Amazon provides only a single option - AWS Lambda. That's a pretty straightforward business strategy. However, from developers' perspective it might be really painful to deal with Lambdas while Smart Home Skills development. Here's why:
- When lambda is cold started, it causes a several secs delay, until the device reacts to the voice command. It's very annoying, especially when you need to e.g. turn on the light while opening the front door. With such a huge delay, it's much easier to press the button on the switch instead of a voice command.
- Cold start also causes devices' discovery issues. It may take some time until a round trip between Lambda and an external cloud provider is over (including devices states fetching). So Alexa sometimes fails to find new devices.
- It's impossible to perform a direct MQTT integration into Alexa skill backend, as Lambda is not intended for long-running operations. We can't just subscribe to some topics and keep the connection open. And when Lambda is timed-out, we have to establish it once again, which causes sad delays and timeouts.
So this API was built keeping in mind all the above factors. Smart Home Skill will interact with a custom middleware, which knows everything about your devices, supplies requested information to Alexa, and proxies commands to micro-controllers.
An API style for building a Smart Home Skill is pretty similar to common skills' SDK.
It's enough to call SkillBuilders.smartHome
to drill into a chain of actions you may use to define your skill's handlers.
exports.handler = SkillBuilders.smartHome(new Middleware())
.addRequestHandlers(
AuthorizationHandler,
PowerHandler,
BrightnessHandler,
ChannelHandler,
SpeakerHandler,
DiscoveryHandler,
StateHandler
)
.lambda()
To simplify the overall development effort there was created a basic set of types required for working with Alexa Smart Home Skills. You can find their declarations in ./typings/Alexa.d.ts. Some other useful structures are located in ./src/model folder.
It might be a custom https server, which routes requests between Alexa and micro-controllers.
On the API level you can either reuse a built-in Middleware
class or create your own by implementing MiddlewareService
interface.
You can find a simple implementation in the following repository.
There're several built-in handlers implemented:
- AuthorizationHandler: triggered while skill's activation after linking your account. That's just a basic implementation. For more advanced scenarios check Authorization docs. You may also want to check json samples to understand the required format.
- DiscoveryHandler: triggered while asking Alexa to discover devices (or via similar feature in Alexa app). Note that it's a key handler for sending required info from micro-controllers to Alexa (through Middleware). Check json samples for more details.
- StateHandler: key handler for devices' state notifications. If you open Alexa mobile app and check your devices, the app will continuously poll this handler for state updates. Check state reporting and json samples for details.
- BrightnessHandler, ChannelHandler, PowerHandler and SpeakerHandler: triggered while asking Alexa to perform some common actions, e.g. "turn on / off" something.
You can create your own handlers by implementing RequestHandler
interface. I'd recommend to use existing handlers' implementations as a baseline.
Also note that you have to follow an official json schema. As you may already noticed, this repository also contains a set a sample json messages for a quick start.
If you have some comments, thoughts or improvement ideas, feel free to create issues or PRs.
To set everything up, install the required dependencies first:
npm install
Create a .env file in the root of the project with the following content:
HOST=
PORT=
TIMEZONE=Europe/Kiev
API_VERSION=3
HOST / PORT refers to your Middleware service address. Note that you have to provide these environment variables on AWS Lambda as well.
There're several useful scripts for getting latest schema and generating typings:
npm run getSchema
npm run generateInterfaces
npm run generateMessages
You can check ./generateTypings.sh, which simplifies this process.
When you're ready for publishing just run the following command:
npm run build
That will create an ./archive.zip which could be uploaded to Amazon Lambda then.
To get more details on how to create a Smart Home skill from scratch, use the following instruction.
Let's consider the following scenario: user wants to turn on the light.
Light bubble might be controlled by NodeMCU board via relay or RF transmitter.
If micro-controller sends the following json to home/devices topic, Middleware will put it into in-memory storage for further usage by Alexa Smart Home Skill.
[
{
"endpointId": "lobby_lamp_1",
"friendlyName": "light",
"description": "Lobby Lamp 1",
"manufacturerName": "Home",
"cookie": {},
"displayCategories": [
"LIGHT"
],
"capabilities": [
{
"type": "AlexaInterface",
"interface": "Alexa.PowerController",
"version": "3",
"properties": {
"supported": [
{
"name": "powerState"
}
],
"proactivelyReported": true,
"retrievable": true
}
},
{
"type": "AlexaInterface",
"interface": "Alexa",
"version": "3"
},
{
"type": "AlexaInterface",
"interface": "Alexa.EndpointHealth",
"version": "3",
"properties": {
"supported": [
{
"name": "connectivity"
}
],
"proactivelyReported": true,
"retrievable": true
}
}
]
}
]
When user first activates a skill and run devices' discovery, Smart Home Skill calls DiscoveryHandler, which then requests devices from our Middleware /devices endpoint.
When user says Alexa, light on (assuming the mentioned above friendly name in json), Smart Home Skill calls PowerHandler, which then sends a power control command, e.g.
[
{
"command":"TurnOn",
"state":true
}
]
to our Middleware -> /device/lobby_lamp_1 endpoint.
This command is published to individual home/device/lobby_lamp_1 topic, which our NodeMCU board is subscribed to.
Micro-controller parses json message and executes the requested command via relay or RF transmitter.