Have you ever want to build your action with Actions on Google but got overwhelmed with several required technologies? There are various steps in different services Actions on Google, Dialogflow, Firebase in documentation and inexperienced user could be confused.
Simply follow steps below to build voice app with Actions on Google Node.js library, Dialogflow, TypeScript and Firebase Cloud Functions. The template is for both beginners and experienced users.
Using steps below you'll create simple action using Actions on Google and build a fulfillment for it. Then you can use it as the base for your Action.
- Actions on Google β developer platform that lets you extend Google Assistant with your own actions
- Dialogflow β platform for natural language understanding (NLU), which helps with building actions suitable for humans
- Firebase (Cloud Functions, Realtime Database) β development platform which gives you ability to build rich responses (besides other things) using your favorite programming language
- Node.js β runtime environment often used for building server-side applications written in JavaScript
- TypeScript β programming language which is transpiled to JavaScript (so we can use it to write apps for Node.js)
- A little bit knowledge of TypeScript and Node.js (if you don't have it, try it anyway)
- Text editor or IDE (recommended WebStorm or Visual Studio Code)
- Google account
- Git
- Install Node.js according to official site https://nodejs.org/en/ (LTS version)
- Or install Node.js using Node version manager (https://github.com/creationix/nvm#installation-and-update)
- Clone this repository using command
git clone https://github.com/novalu/actions-on-google-typescript-template.git yourproject
- Go to http://console.firebase.google.com
- Create your project
- In your terminal run
npm i -g firebase-tools
- Run command
firebase login
- This will open browser, login to your Google account and allow access for Firebase CLI
- You should see something like "Success! Logged in as youraccount@gmail.com" in the terminal
- Change working directory to
yourproject
- Run command
firebase init
- Choose
Functions
with Space and then Enter to confirm - Choose project you've created in previous step
- Choose TypeScript
- Choose (Y) to use TSLint
- Choose (N) several times not to overwrite existing files
- Choose (n) not to install dependencies
- Choose
- You should see message "Firebase initialization complete!"
Note: More detailed instructions are available as this Actions on Google guide.
- Run command
npm i --only=dev
to install neccessary development tools. - Run command
npx gulp install
to install dependencies for the whole project.
Create Actions on Google project according to instructions below. This will create your first action with simple text response using Dialogflow. Test this app in the simulator to make sure everything works.
- Use the Actions on Google Console to add a new project with a name of your choosing and click Create Project.
- Scroll down to the More Options section, and click on the Conversational card.
- On the left navigation menu under BUILD, click on Actions. Click on Add Your First Action and choose your app's language(s).
- Select Custom intent, click BUILD. This will open a Dialogflow console. Click CREATE.
- Click on the gear icon to see the project settings.
- Select "Export and Import".
- Select "Restore from zip". Follow the directions to restore from the SillyNameMaker.zip in this repo.
- On the left navigation menu click on Intents.
- Click on the make_name intent.
- Scroll down to Responses, click on Set this intent as end of conversation.
- Click Save.
- Select Integrations from the left navigation menu and open the Integration Settings menu for Actions on Google.
- Enable Auto-preview changes and Click Test. This will open the Actions on Google simulator.
- Type
Talk to my test app
in the simulator, or sayOK Google, talk to my test app
to any Actions on Google enabled device signed into your developer account.
Steps above are copied from https://github.com/actions-on-google/dialogflow-silly-name-maker.
If you want to generate dynamic responses using this project, then make this steps:
- Enable webhook for your intent:
- open Dialogflow console,
- select Intents from the left menu,
- choose
make_name
intent, - scroll to Fulfillment,
- choose Enable fulfillment,
- switch on the Enable webhook call for this intent,
- click Save.
- Deploy your functions:
- Run command
npx gulp deploy-development
.Β This will compile and upload your function to Firebase Cloud Functions. If everything goes right, you should see deployed function URL labeled asdialogflowDevelopment
(take a note*) - If you want to deploy your development and production functions side by side, you can use
npx gulp deploy-production
too.
- Run command
- Set fulfillment URL:
- Open Dialogflow console,
- select Fulfillment from the left menu,
- switch on the Webhook option,
- fill URL of your function (from step 2.1),
- click Save.
- Test in Actions on Google simulator:
- Open Dialogflow console,
- select Integrations from the left menu,
- click on Integration Settings for the Google Assistant,
- click on Test,
- confirm Auto preview.
You should see Actions on Google simulator. Start with typing Talk to my test app
in the Input and confirm. Simulator should return as a response: Alright, your silly name by Firebase functions is ... I hope you like it! See you next time!
. You can see this response is written in functions/src/fulfillments/impl/SillyNameFulfillment.ts.
Note: Silly name is created not only with number and color you've provided to Google Assistant, but with extra random fruit from FruitsStorage too. FruitsStorage is resolved to FruitsLocalStorage, which means that will be used fruits from static array. If you want to use fruits from Firebase Realtime Database, see below.
Now it's time to add fulfillment for your new action. Use Node.js with TypeScript to add your fulfillments, managers and storages.
If you make any changes to your fulfillment, you should deploy functions with npx gulp deploy
.
This project use a dependency injection pattern for inject dependencies at the runtime. Generally it is not required to construct your classes with dependencies as a parameters in constructors. You just define which dependency your class need and dependency injection framework will take care of it.
Imagine you have an interface DataStorage
and implementations LocalDataStorage
and FirebaseDataStorage
. Then you can add these to the dependency injection as follows:
- Add interface definition to functions/src/di/types.ts as shown here:
DataStorage: Symbol("DataStorage")
- Add binding to symbol from types.ts in functions/src/di/baseContainer.ts
baseContainer.bind<DataStorage>(TYPES.DataStorage)
.to(LocalDataStorage)
.inSingletonScope();
We've just defined that DataStorage
dependency will be resolved to LocalDataStorage
. If you want to use different implementation, you can change it here without the touching your business logic.
If you want to resolve class to the instance of the same class, you can safely use this:
baseContainer.bind<CustomClass>(TYPES.CustomClass)
.to(CustomClass)
.inSingletonScope();
If you want to use class from dependency injection, then define it in a constructor with @inject
annotation:
constructor(
@inject(TYPES.DataStorage) private dataStorage: DataStorage,
@inject(TYPES.CustomClass) private customClassInstance: CustomClass,
) {}
This will inject LocalDataStorage
as dataStorage
and CustomClass
as customClassInstance
member properties.
Fulfillments in this projects are meant as a logical blocks of intent responses.
- Create implementation of Fulfillment.ts in functions/src/fulfillments/impl.
- Add class to dependency injection.
- Add fulfillment symbol to functions/src/config/fulfillments.ts
Managers are meant to be helper classes which will be used by your fulfillments. Manager should contain business logic for your fulfillments. Fulfillments should be separated from business logic in managers to easily test your business logic.
- Create manager class in functions/src/managers
- Add class to dependency injection.
Sometimes you need a collection of items you want to use in your business logic. Creating a black-box storage is the best practice. You can use several storages which you can swap in the dependency injection, e.g. local storage with hardcoded collection of items (for debugging purposes) and Firebase storage to fetch data from Firebase database (for production).
You can test your app by using your managers in the src/TestApp.ts. Run npx gulp build
to transpile from source code and then npx gulp run-test
to execute the test
method.
If you would to have your app automatically transpiled and executed when source code is changed, run npx gulp build-and-run continuous
or simply shorthand npx gulp
. Whenever you make any changes in .ts
files, project will be automatically recompiled using tsc
and then executed.
You can start debugging test app by running npx gulp debug-test
. Similarly as running test continuously when source code is change, you can run continuously debugging by command npx gulp build-and-debug-continuous
.
If you want to write logging info to both your console and Firebase Cloud Functions log, you can use class src/utils/log/Logger.ts as a dependency. Then you can call methods trace
, debug
, info
, warn
, error
, fatal
. Logger use implementation depending on whether is app executed locally (library signale
) or in Firebase Cloud Functions environment (simple console logging).
If your function deployed to Firebase Cloud Functions is not working, most likely you'll find some useful info in Firebase console. Choose Develop > Functions > Log.
If you want to communicate with third-party API to retrieve or post data, you can use class src/utils/network/NetworkRequest.ts. This class use awesome superagent
library to make network requests. Define the NetworkRequest
dependency and then use it e.g. as follows:
try {
const rates = await this.request.getJson("https://api.exchangeratesapi.io/latest");
rates.body // JSON
} catch (err) {
this.logger.error(err);
}
Note: If you want to use external APIs in your Firebase Cloud Function, you must upgrade your Firebase account to Blaze plan.
If you want to use your database in your Firebase project as a storage, follow these steps:
- Enable Firebase Realtime Database by going to Firebase Console > Develop > Database, choose Create Database, select Start in Test Mode, and then Enable
- Make sure you have opened Realtime Database at the top of the screen
- Add some data to the database
- Go to Project settings > Service accounts
- In the left pane select Firebase Admin SDK, then Node.js, click on Generate new private key > Generate key and download JSON file
- Rename file from previous step to service-account.json and copy it to /functions
- Go to Database section and copy URL to Firebase Realtime Database to DATABASE_URL constant in functions/src/helpers/FirebaseHelper.ts
Now you can use SillyName example with prepared FruitsFirebaseStorage, which is storage for fruits obtained from Firebase Realtime Database. Follow these steps to fill database with fruits and change storage source:
- Fill database with these data in the root:
- fruits
- 0: "banana"
- 1: "orange"
- 2: "lemon"
- fruits
- Change resolved value of FruitsStorage to FruitsFirebaseStorage in functions/src/di/baseContainer.ts (see Dependency Injection section above for more info about this step).
If you want to send message to your Slack channel as a part of your fulfillment, then create an Incoming Webhook as instructed here: https://api.slack.com/incoming-webhooks. Then fill the WEBHOOK_URL
with your webhook URL in functions/src/helpers/SlackHelper.ts. Then use SlackHelper#sendMessage
method.
If you want to add package dependency from npmjs.com, the run command npm i packagename
in the directory functions. Make sure the dependency is listed in inner package.json file, so it will be used both locally and by Firebase Cloud Functions.
app
ββββfunctions ... this functions project will be deployed to Firebase Cloud Functions
β ββββsrc
β β ββββconfig ... configuration files
β β ββββdi ... dependency injection configuration for the functions project
β β ββββfulfillments ... fulfillments logic
β β ββββhelpers ... classes you can use when using Firebase or Slack integration
β β ββββmanagers ... classes with business logic for your fulfillments
β β ββββmodel ... models classes which is used by your managers
β β ββββstorages ... storages for the models
β β ββββutils ... utilities for logging, networking, ...
β β β FunctionsApp.ts
β β β main.ts ... entry point for executing FunctionsApp in the cloud function
β β package.json ... dependencies for the functions project
β β tsconfig.json ... TypeScript configuration for the functions project
ββββsrc ... main project for testing locally
β ββββdi ... dependency injection for the main project
β β β container.ts
β β β types.ts
β β main.ts ... entry point for executing TestApp
β β TestApp.ts ... main class for testing locally
| firebase.json ... configuration file with location of the functions project
β gulpfile.js ... configuration file for the Gulp build system
β nodemon.js ... configuration file for the Nodemon
β package.json ... npm configuration file with scripts, dependencies, ... definition
β README.md ... this readme
β SillyNameMaker.zip ... archive containing Dialogflow agent for example action
β tsconfig.json ... TypeScript configuration for the main project
- Weather in Czech Republic (App Directory)
Feel free to use this project for building your actions. Pull request welcome. If you like the template, don't forget to leave a star!
If you like to support me, buy me a beer using PayPal (paypal.me/novalu). Thank you!
- Authors of libraries used in this template.
- Authors of SillyNameMaker example action (https://github.com/actions-on-google/dialogflow-silly-name-maker)
- Betatesters: Attendees of Voice Hackathon 2019 in Brno
This project was created as a base for our project at "Ok, Google, do a hackathon" which was held in December 2018 in coworking center Vault 42, Olomouc, Czech Republic. Our team built voice receptionist which is now in use by the coworking center.