Learn more about how to let your users do In-Skill Purchases (ISP) with your Alexa Skill. You can also use the following template to get started: Jovo Template for Alexa ISP.
- Introduction to In-Skill Purchases
- Manage Products with ASK CLI
- Update the Language Model
- Implement Purchasing with Jovo
In May 2018, Amazon introduced the ability for Skill developers to make money through in-skill purchasing. It allows you to sell items either through one-time purchases or subscriptions. Take a look at the official reference by Amazon: In-Skill Purchasing Overview.
There are three things that need to be done to successfully implement in-skill purchasing for your Alexa Skill with Jovo:
Purchasable products can be added and managed with ASK CLI, the command line interface for the Alexa Skills Kit.
If you're using the common Jovo project structure, go into your platforms/alexaSkill
folder and then tell the ASK ClI to add isp
. After going through the process, it will add a new folder isps
to your project files.
# Go into alexaSkill folder
$ cd platforms/alexaSkill
$ ask add isp
? List of in-skill product types you can choose (Use arrow keys)
❯ Entitlement
Subscription
? List of in-skill product templates you can choose (Use arrow keys)
> Entitlement_Template
? Please type in your new in-skill product name:
frozen_sword
In-skill product frozen_sword.json is saved to ./isps/entitlement/frozen_sword.json
After that, there should be a frozen_sword.json
file in your alexaSkill/isps
folder. The content of that json
file is used to define the price, release date, description as well as the prompts Alexa will use when she handles the transaction and much much more. You can find examples and a small description for each field here.
Here's how our product's file looks like:
{
"version": "1.0",
"type": "ENTITLEMENT",
"referenceName": "frozen_sword",
"publishingInformation": {
"locales": {
"en-US": {
"name": "Frozen Sword",
"smallIconUri": "https://s3.amazonaws.com/jovocards/logo108.png",
"largeIconUri": "https://s3.amazonaws.com/jovocards/logo512.png",
"summary": "A sword once used by Arthas.",
"description": "Use the overpowered Frozen Sword",
"examplePhrases": [
"Alexa, buy the frozen sword"
],
"keywords": [
"frozen sword",
"sword",
"frozen"
],
"customProductPrompts": {
"purchasePromptDescription": "Do you want to buy the Frozen Sword?",
"boughtCardDescription": "Congrats. You successfully purchased the Frozen Sword!"
}
}
},
"distributionCountries": [
"US"
],
"#": {
"amazon.com": {
"releaseDate": "2018-05-14",
"defaultPriceListing": {
"price": 0.99,
"currency": "USD"
}
}
},
"taxInformation": {
"category": "SOFTWARE"
}
},
"privacyAndCompliance": {
"locales": {
"en-US": {
"privacyPolicyUrl": "https://www.yourcompany.com/privacy-policy"
}
}
},
"testingInstructions": "This is an example product.",
"purchasableState": "PURCHASABLE"
}
Here is an example language model you can for purchasing and refunding products. It is recommended to use the reference name as an ID
when creating the input/slot values:
"intents": [
{
"name": "BuySkillItemIntent",
"phrases": [
"buy",
"shop",
"buy {ProductName}",
"purchase {ProductName}",
"want {ProductName}",
"would like {ProductName}"
],
"inputs": [
{
"name": "ProductName",
"type": "LIST_OF_PRODUCT_NAMES"
}
]
},
{
"name": "RefundSkillItemIntent",
"phrases": [
"return {ProductName}",
"refund {ProductName}",
"want a refund for {ProductName}",
"would like to return {ProductName}"
],
"inputs": [
{
"name": "ProductName",
"type": "LIST_OF_PRODUCT_NAMES"
}
]
}
],
"inputTypes": [
{
"name": "LIST_OF_PRODUCT_NAMES",
"values": [
{
"value": "Frozen Sword",
"id": "frozen_sword"
}
]
}
],
In general, you can access the in-skill purchasing interface like this:
// @language=javascript
this.$alexaSkill.$inSkillPurchase
// @language=typescript
this.$alexaSkill!.$inSkillPurchase
Before you start any kind of transaction or refund process, always retrieve the product first:
// @language=javascript
let productReferenceName = 'frozen_sword';
try {
const product = await this.$alexaSkill.$inSkillPurchase.getProductByReferenceName(productReferenceName);
} catch (error) {
}
// @language=typescript
let productReferenceName = 'frozen_sword';
try {
const product = await this.$alexaSkill!.$inSkillPurchase.getProductByReferenceName(productReferenceName);
} catch (error) {
}
The data you get looks like this:
{ productId: 'amzn1.adg.product.994e8ec0-2f73-4130-837a-0bb06abe1ded',
referenceName: 'frozen_sword',
type: 'ENTITLEMENT',
name: 'Frozen Sword',
summary: 'A sword once used by Arthas.',
entitled: 'NOT_ENTITLED',
entitlementReason: 'NOT_PURCHASED',
purchasable: 'PURCHASABLE',
activeEntitlementCount: 0,
purchaseMode: 'TEST'
}
Using that data we can check if the user already owns the product:
// @language=javascript
try {
const product = await this.$alexaSkill.$inSkillPurchase.getProductByReferenceName(productReferenceName);
if (product.entitled === 'ENTITLED') {
// user already owns it
} else {
// user does not own it
}
} catch (error) {
}
// @language=typescript
try {
const product = await this.$alexaSkill!.$inSkillPurchase.getProductByReferenceName(productReferenceName);
if (product.entitled === 'ENTITLED') {
// user already owns it
} else {
// user does not own it
}
} catch (error) {
}
The upsell()
method is used to proactively offer the user your products. To send out the request, you need three things:
Name | Description | Value | Required
:--- | :--- | :--- | :--- | :---
productId
| ID used to determine the correct product | String
| Yes
prompt
| Prompt Alexa will read to ask the user whether they are interested | String
| Yes
token
| Token you use to help you resume the Skill after the transaction finished | String
| Yes
// @language=javascript
async UpsellIntent() {
let productReferenceName = 'frozen_sword';
try {
const product = await this.$alexaSkill.$inSkillPurchase.getProductByReferenceName(productReferenceName);
if (product.entitled === 'ENTITLED') {
return this.tell('You have already bought this item.');
} else {
let prompt = 'The frozen sword will help you on your journey. Are you interested?';
let token = 'testToken';
this.$alexaSkill.$inSkillPurchase.upsell(product.productId, prompt, token);
}
} catch (error) {
}
},
// @language=typescript
async UpsellIntent() {
let productReferenceName = 'frozen_sword';
try {
const product = await this.$alexaSkill!.$inSkillPurchase.getProductByReferenceName(productReferenceName);
if (product.entitled === 'ENTITLED') {
return this.tell('You have already bought this item.');
} else {
let prompt = 'The frozen sword will help you on your journey. Are you interested?';
let token = 'testToken';
this.$alexaSkill!.$inSkillPurchase.upsell(product.productId, prompt, token);
}
} catch (error) {
}
},
The buy()
method is used to start the transaction after the user requested the purchase.
Name | Description | Value | Required
:--- | :--- | :--- | :--- | :---
productId
| ID used to determine the correct product | String
| Yes
token
| Token you use to help you resume the Skill after the transaction finished | String
| Yes
// @language=javascript
async BuySkillItemIntent() {
let productReferenceName = this.$inputs.productName.id;
try {
const product = await this.$alexaSkill.$inSkillPurchase.getProductByReferenceName(productReferenceName);
if (product.entitled === 'ENTITLED') {
return this.tell('You have already bought this item.');
} else {
let token = 'testToken';
this.$alexaSkill.$inSkillPurchase.buy(product.productId, token);
}
} catch (error) {
}
},
// @language=typescript
async BuySkillItemIntent() {
let productReferenceName = this.$inputs.productName.id;
try {
const product = await this.$alexaSkill!.$inSkillPurchase.getProductByReferenceName(productReferenceName);
if (product.entitled === 'ENTITLED') {
return this.tell('You have already bought this item.');
} else {
let token = 'testToken';
this.$alexaSkill!.$inSkillPurchase.buy(product.productId, token);
}
} catch (error) {
}
},
The cancel()
method is used to start the refund process after the user asked for it.
Name | Description | Value | Required
:--- | :--- | :--- | :--- | :---
productId
| ID used to determine the correct product | String
| Yes
token
| Token you use to help you resume the Skill after the transaction finished | String
| Yes
// @language=javascript
async RefundSkillItemIntent() {
let productReferenceName = this.$inputs.productName.id;
try {
const product = await this.$alexaSkill.$inSkillPurchase.getProductByReferenceName(productReferenceName);
if (product.entitled === 'ENTITLED') {
return this.tell('You have already bought this item.');
} else {
let token = 'testToken';
this.$alexaSkill.$inSkillPurchase.cancel(product.productId, token);
}
} catch (error) {
}
},
// @language=typescript
async RefundSkillItemIntent() {
let productReferenceName = this.$inputs.productName.id;
try {
const product = await this.$alexaSkill!.$inSkillPurchase.getProductByReferenceName(productReferenceName);
if (product.entitled === 'ENTITLED') {
return this.tell('You have already bought this item.');
} else {
let token = 'testToken';
this.$alexaSkill!.$inSkillPurchase.cancel(product.productId, token);
}
} catch (error) {
}
},
After successfully going through the process of purchasing or refunding a product, your Skill will receive a request notifying you about the result:
{
"type": "Connections.Response",
"requestId": "string",
"timestamp": "string",
"name": "Upsell",
"status": {
"code": "string",
"message": "string"
},
"payload": {
"purchaseResult":"ACCEPTED",
"productId":"string",
"message":"optional additional message"
},
"token": "string"
}
The important parts of that request are:
Name | Description |
---|---|
name |
Either Upsell , Buy or Cancel . Used to determine which kind of transaction took place |
payload.purchaseResult |
Either ACCEPTED , DECLINED , ALREADY_PURCHASED or ERROR . Used to determine the outcome of the transaction |
payload.productId |
The product in question |
token |
The token used to resume the skill where it left off |
That request will be mapped to the built-in ON_PURCHASE
intent:
// @language=javascript
ON_PURCHASE() {
const name = this.$request.name;
const productId = this.$alexaSkill.$inSkillPurchase.getProductId();
const purchaseResult = this.$alexaSkill.$inSkillPurchase.getPurchaseResult();
const token = this.$request.token;
if (purchaseResult === 'ACCEPTED') {
this.tell('Great! Let\'s use your new item');
} else {
this.tell('Okay. Let\'s continue where you left off.');
}
},
// @language=typescript
ON_PURCHASE() {
const name = this.$request.name;
const productId = this.$alexaSkill!.$inSkillPurchase.getProductId();
const purchaseResult = this.$alexaSkill!.$inSkillPurchase.getPurchaseResult();
const token = this.$request.token;
if (purchaseResult === 'ACCEPTED') {
this.tell('Great! Let\'s use your new item');
} else {
this.tell('Okay. Let\'s continue where you left off.');
}
},