天猫精灵开放平台实验—基于模板创建开发屏显页面技能
一、前提准备
使用本地 VSCode 快速开发一个技能屏显界面,并通过控制台的操作连接猫精设备进行真机测试。
- 1、本实验目前支持CentOS、Ubuntu、MacOS上支持。本实验采用Ubuntu 20.4系统。
- 2、在Ubuntu系统上安装VSCode开发软件。# 系列文章目录
data:image/s3,"s3://crabby-images/09cbd/09cbd0cb9827946173eda7c2788f246b418b029e" alt="image-20211110140024035"
二、登录天猫精灵开放平台
1、登录天猫精灵开放平台 ,登陆后点击控制台,选择技能应用平台并点击进入。 data:image/s3,"s3://crabby-images/f1559/f15593b3fc56549c67ab31f891c5509e3bbeb918" alt="image-20211110141145808"
(一)创建新技能
1、创建语音技能
data:image/s3,"s3://crabby-images/7432d/7432d8b8655c2ec87ae530eeeb9ef5002d836941" alt="image-20211110141220240"
2、填写基本信息
data:image/s3,"s3://crabby-images/54c2d/54c2d3eceb88835a65639c53fbe56e3c2dbfe082" alt="image-20211110141508919"
(二)创建后端服务
1、创建后端服务
data:image/s3,"s3://crabby-images/906ee/906ee5b072c96f41a69dbd46d325c5164a6d88ba" alt="image-20211110141713310"
2、关联阿里云账号
data:image/s3,"s3://crabby-images/dd3ad/dd3ade6d41f2ed8f09f94f71ae506a26efcd1615" alt="image-20211110141845503" 登录阿里云账号
data:image/s3,"s3://crabby-images/42d27/42d27d2bea2d58d7a7624269833d99e1d58b5b6f" alt="image-20211110141919808" 完成授权,点击下一步。 data:image/s3,"s3://crabby-images/50a50/50a500df0b550c8ffd169fe3dea79486295bee96" alt="image-20211110142008591" 开通相关服务。 data:image/s3,"s3://crabby-images/274b6/274b673a62469e59174f3b9ac148a094b07a1268" alt="image-20211110142056092"
3、创建技能应用
data:image/s3,"s3://crabby-images/c2086/c2086ad018e10eab643c84307d06aed51a3b9466" alt="image-20211110142157712" 创建应用 data:image/s3,"s3://crabby-images/76590/76590a39b4c68f536b6883ec6496a327073b2b24" alt="image-20211110142248109" 保存创建的服务 data:image/s3,"s3://crabby-images/c49d8/c49d81ce752d69c6230fc97af9f58ad83f68b06d" alt="image-20211110142413385" 点击“前往开发”。 data:image/s3,"s3://crabby-images/da1f1/da1f1685f3af7539ca7752666db1965758c73c6a" alt="image-20211110142545487" 查看语音交互模型已经自动完成创建 data:image/s3,"s3://crabby-images/8bb88/8bb883ce9c915c3a4fe1c41437638eddd814e5ff" alt="image-20211110145004762"
4、调试代码
data:image/s3,"s3://crabby-images/d80f9/d80f90f984b1852281c5581e3116ac3fe21cfade" alt="image-20211110143837044" 示例代码如下
package com.alibaba.ailabs;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.stream.Collectors;
import com.alibaba.ailabs.common.AbstractEntry;
import com.alibaba.da.coin.ide.spi.meta.AskedInfoMsg;
import com.alibaba.da.coin.ide.spi.meta.ExecuteCode;
import com.alibaba.da.coin.ide.spi.meta.GwCommand;
import com.alibaba.da.coin.ide.spi.meta.ResultType;
import com.alibaba.da.coin.ide.spi.standard.ResultModel;
import com.alibaba.da.coin.ide.spi.standard.TaskQuery;
import com.alibaba.da.coin.ide.spi.standard.TaskResult;
import com.alibaba.fastjson.JSON;
import com.aliyun.fc.runtime.Context;
import com.google.common.collect.Lists;
public class GenieEntry extends AbstractEntry {
@Override
public ResultModel<TaskResult> execute(TaskQuery taskQuery, Context context) {
context.getLogger().info("taskQuery: " + JSON.toJSONString(taskQuery));
String screenStatus = "online";
Map<String, String> paramMap = taskQuery.getSlotEntities().stream().collect(
Collectors.toMap(slotItem -> slotItem.getIntentParameterName(), slotItem -> slotItem.getOriginalValue()));
String reply;
String city = paramMap.get("city");
String date = paramMap.get("date");
if ("weather".equals(taskQuery.getIntentName())) {
if (city == null) {
reply = "您要查询哪个城市的天气?";
return askReply(reply, "city", taskQuery.getIntentId());
}
reply = city + date + "天气 晴。";
if ("online".equals(screenStatus)) {
return weatherTplReply(reply, city);
} else {
return reply(reply);
}
} else if ("air_quality".equals(taskQuery.getIntentName())) {
if (city == null) {
reply = "您要查询哪个城市的空气质量?";
return askReply(reply, "city", taskQuery.getIntentId());
}
reply = city + date + "空气质量46 优。";
if ("online".equals(screenStatus)) {
return airTplReply(reply, city);
} else {
return reply(reply);
}
}
reply = "请检查意图名称是否正确,或者新增的意图没有在代码里添加对应的处理分支。";
return reply(reply);
}
private ResultModel<TaskResult> reply(String reply) {
ResultModel<TaskResult> res = new ResultModel<>();
TaskResult taskResult = new TaskResult();
taskResult.setReply(reply);
taskResult.setExecuteCode(ExecuteCode.SUCCESS);
taskResult.setResultType(ResultType.RESULT);
res.setReturnCode("0");
res.setReturnValue(taskResult);
return res;
}
private ResultModel<TaskResult> askReply(String reply, String parameterName, Long intentId) {
ResultModel<TaskResult> res = new ResultModel<>();
TaskResult taskResult = new TaskResult();
taskResult.setReply(reply);
taskResult.setExecuteCode(ExecuteCode.SUCCESS);
taskResult.setResultType(ResultType.ASK_INF);
AskedInfoMsg askedInfoMsg = new AskedInfoMsg(parameterName, intentId);
taskResult.setAskedInfos(Lists.newArrayList(askedInfoMsg));
res.setReturnCode("0");
res.setReturnValue(taskResult);
return res;
}
private ResultModel<TaskResult> weatherTplReply(String reply, String city) {
return commonReply(reply, city, reply);
}
private ResultModel<TaskResult> airTplReply(String reply, String city) {
return commonReply(reply, city, reply);
}
private ResultModel<TaskResult> commonReply(String reply, String city, String detail) {
GwCommand speakGwCommand = new GwCommand("AliGenie.Speaker", "Speak");
HashMap<String, Object> payload = new HashMap<>();
payload.put("type", "text");
payload.put("text", reply);
speakGwCommand.setPayload(payload);
GwCommand tplGwCommand = new GwCommand("AliGenie.Screen", "Render");
HashMap<String, Object> payload1 = new HashMap<>();
payload1.put("pageType", "TPL.RenderTemplate");
payload1.put("pageTitle", "天气小助手");
HashMap<String, Object> data = new HashMap<>();
data.put("template", "aligenie_weather_tpl");
HashMap<String, String> dataSource = new HashMap<>();
dataSource.put("imageUrl", "https://ailabs-iot.aligenie.com/iap/platform3.0/weather-banner.png");
dataSource.put("city", city);
dataSource.put("minTemperature", "36°");
dataSource.put("maxTemperature", "38°");
dataSource.put("detail", detail);
data.put("dataSource", dataSource);
payload1.put("data", data);
tplGwCommand.setPayload(payload1);
return tplReply(Lists.newArrayList(speakGwCommand, tplGwCommand));
}
private ResultModel<TaskResult> tplReply(List<GwCommand> gwCommands) {
ResultModel<TaskResult> res = new ResultModel<>();
TaskResult taskResult = new TaskResult();
taskResult.setExecuteCode(ExecuteCode.SUCCESS);
taskResult.setResultType(ResultType.RESULT);
taskResult.setGwCommands(gwCommands);
res.setReturnCode("0");
res.setReturnValue(taskResult);
return res;
}
}
5、推送代码至远程分支
为防止您的代码丢失,请务必提交并推送代码到远程分支
[admin@f99f9e13-a8b9-4189-bada-d24034966e0e-77d8d5765d-h8vtm /home/admin/workspace/codeup.aliyun.com/618249ae404574409feabd87/workbench/repo_2021-11-10_2021111001491236]
$git status
On branch master
Your branch is up to date with 'origin/master'.
Changes not staged for commit:
(use "git add <file>..." to update what will be committed)
(use "git checkout -- <file>..." to discard changes in working directory)
modified: src/main/java/com/alibaba/ailabs/GenieEntry.java
no changes added to commit (use "git add" and/or "git commit -a")
[admin@f99f9e13-a8b9-4189-bada-d24034966e0e-77d8d5765d-h8vtm /home/admin/workspace/codeup.aliyun.com/618249ae404574409feabd87/workbench/repo_2021-11-10_2021111001491236]
$git add src
[admin@f99f9e13-a8b9-4189-bada-d24034966e0e-77d8d5765d-h8vtm /home/admin/workspace/codeup.aliyun.com/618249ae404574409feabd87/workbench/repo_2021-11-10_2021111001491236]
$git commit -m "weather_tpl"
[master 4bf99d6] weather_tpl
1 file changed, 1 insertion(+), 1 deletion(-)
[admin@f99f9e13-a8b9-4189-bada-d24034966e0e-77d8d5765d-h8vtm /home/admin/workspace/codeup.aliyun.com/618249ae404574409feabd87/workbench/repo_2021-11-10_2021111001491236]
$git push
Counting objects: 9, done.
Delta compression using up to 5 threads.
Compressing objects: 100% (5/5), done.
Writing objects: 100% (9/9), 632 bytes | 42.00 KiB/s, done.
Total 9 (delta 2), reused 0 (delta 0)
To https://codeup.aliyun.com/618249ae404574409feabd87/workbench/repo_2021-11-10_2021111001491236.git
4235ca4..4bf99d6 master -> master
[admin@f99f9e13-a8b9-4189-bada-d24034966e0e-77d8d5765d-h8vtm /home/admin/workspace/codeup.aliyun.com/618249ae404574409feabd87/workbench/repo_2021-11-10_2021111001491236]
$
6、部署代码
data:image/s3,"s3://crabby-images/83967/83967e3bb2bfe00ae1905ab1a635f7ef42eaa4ca" alt="image-20211110144052707" data:image/s3,"s3://crabby-images/40ab5/40ab580981c4a5fec4de9fadfda900db94a27748" alt="image-20211110144350443" data:image/s3,"s3://crabby-images/31438/314383f04570b497e4af62d0ffe10e9bf6967dca" alt="image-20211110144746354"
(三)创建屏显模板
1、新建屏显模板
data:image/s3,"s3://crabby-images/01283/012837376b4247a166679aaa1ab51f68dfb39d88" alt="image-20211110143332578"
2、填写模板名称、标识、描述
data:image/s3,"s3://crabby-images/d9e01/d9e01e0be1beccc94ee90f6be04aa532d750ca0f" alt="image-20211110143558959" data:image/s3,"s3://crabby-images/1d276/1d2762bd60943fd7262f2fa454ed84a6dff4c203" alt="image-20211110143659486"
(四)在线测试
data:image/s3,"s3://crabby-images/0333f/0333f3ee8df137678f13909f0978c410ddd25630" alt="image-20211110145217543"
(五)真机测试
参考链接:快速开始 · 语雀 (yuque.com)
1、安装waft-cli 工具
在Ubuntu系统上使用VSCode软件安装waft-cli 工具
PS: 权限问题使用sudo执行即可
npm i waft-cli -g
data:image/s3,"s3://crabby-images/8cd61/8cd61ca2a4aba763f695c61e10edb39564d7fec7" alt="image-20211110151924658"
xyb@xyb-virtual-machine:~/waft$ sudo npm i waft-cli -g
[sudo] xyb 的密码:
npm WARN deprecated uuid@3.4.0: Please upgrade to version 7 or higher. Older versions may use Math.random() in certain circumstances, which is known to be problematic. See https://v8.dev/blog/math-random for details.
npm WARN deprecated @types/ora@3.2.0: This is a stub types definition. ora provides its own type definitions, so you do not need this installed.
npm WARN deprecated request@2.88.2: request has been deprecated, see https://github.com/request/request/issues/3142
npm WARN deprecated har-validator@5.1.5: this library is no longer supported
npm WARN deprecated loadsh@0.0.4: This is a typosquat on the popular Lodash package. This is not maintained nor is the original Lodash package.
npm WARN deprecated @types/colors@1.2.1: This is a stub types definition. colors provides its own type definitions, so you don't need this installed.
npm WARN deprecated @types/table@6.3.2: This is a stub types definition. table provides its own type definitions, so you do not need this installed.
npm WARN deprecated @stylelint/postcss-markdown@0.36.2: Use the original unforked package instead: postcss-markdown
/usr/local/bin/waft -> /usr/local/lib/node_modules/waft-cli/bin/waft
/usr/local/bin/asc -> /usr/local/lib/node_modules/waft-cli/bin/asc
> core-js-pure@3.19.1 postinstall /usr/local/lib/node_modules/waft-cli/node_modules/core-js-pure
> node -e "try{require('./postinstall')}catch(e){}"
Thank you for using core-js ( https://github.com/zloirock/core-js ) for polyfilling JavaScript standard library!
The project needs your help! Please consider supporting of core-js:
> https://opencollective.com/core-js
> https://patreon.com/zloirock
> https://paypal.me/zloirock
> bitcoin: bc1qlea7544qtsmj2rayg0lthvza9fau63ux0fstcz
Also, the author of core-js ( https://github.com/zloirock ) is looking for a good job -)
npm WARN notsup Unsupported engine for is-root@3.0.0: wanted: {"node":">=12"} (current: {"node":"10.19.0","npm":"6.14.4"})
npm WARN notsup Not compatible with your version of node/npm: is-root@3.0.0
+ waft-cli@0.6.50
added 963 packages from 553 contributors in 202.846s
xyb@xyb-virtual-machine:~/waft$
2、创建项目
通过waft脚手架初始化项目,会引导您进行项目命名等,您将得到一个新project
sudo waft init
data:image/s3,"s3://crabby-images/44e90/44e905d3e6aadddc3d1509733d58e46f5e486e5e" alt="image-20211110152341414" data:image/s3,"s3://crabby-images/a75e4/a75e4036a121096825dd33c33f3baf0c87df4d1e" alt="image-20211110152419947" data:image/s3,"s3://crabby-images/1bc37/1bc3722389fff4447cf635f0f6cdb9f754479cbe" alt="image-20211110152550536"
xyb@xyb-virtual-machine:~/waft$ sudo waft init
? What's project name? my-demo
? Which template package to from? waft-empty-template
💻 npm install
init done
xyb@xyb-virtual-machine:~/waft$
3、启动调试服务
xyb@xyb-virtual-machine:~/waft$ cd my-weather/
xyb@xyb-virtual-machine:~/waft/my-weather$ sudo npm run start
> my-weather@0.6.0 start /home/xyb/waft/my-weather
> waft start
🖥 waft工作台已启动
http://127.0.0.1:8080/ui
真机调试可连接到本机ip: 192.168.1.107
🌐 web预览服务已启动
http://127.0.0.1:8080/web/preview.html?action=preview&path=pages/index/index
😊 start success
Welcome to waft development.
Type .help for more information.
waft:0.6.16
waft-cli:0.6.50
waft> /bin/sh: 1: open: not found
waft>
waft> Device connection received 1
[Devtools Server] New Device connected 1
waft> [Workbench] connected 1
waft> [Inspector] panel connected
waft工作台 data:image/s3,"s3://crabby-images/48049/48049b03836b7ed08f400d1adf01c5b038de6c15" alt="image-20211110171641686"
4、打开天猫精灵开发者模式
注意事项:测试环境天猫精灵所连接的网络需跟本地端在同一局域网内。
(1)在天猫精灵上,点击“更多设置”。
data:image/s3,"s3://crabby-images/70647/7064765f5bc3d45a9de468014d20dbc6b1f0b4e1" alt="在这里插入图片描述"
(2)点击“升级更新”。
data:image/s3,"s3://crabby-images/29afb/29afbf4bdefd71af1adb71aa723d0d1448e3b970" alt="在这里插入图片描述"
(3)连续点击“系统版本”五下,打开“开发者模式”。
data:image/s3,"s3://crabby-images/25839/25839eb4ffc17ac0fdbe9b126387f741fb9b39db" alt="在这里插入图片描述"
(4)开启WAFT调试
data:image/s3,"s3://crabby-images/e6b9e/e6b9e4c5dbbb83b993469ebf966a58abf355cd47" alt="在这里插入图片描述"
(5)打开调试面板,输入本地端的IP地址
data:image/s3,"s3://crabby-images/881af/881af6ee15f9c063a2aa4ffc1f4ce84893b35414" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/a7155/a7155f9b620ad42edd30a018b1198af751770fbc" alt="在这里插入图片描述"
REPL命令行模式
启动后自动开启REPL模式,可以通过.help查询支持哪些命令列表
4、打包
waft> .build
waft>
? start build app
I/O Read : 22.015 ms n=796
I/O Write : 98.799 ms n=3
Parse : 510.609 ms n=250
Initialize : 51.435 ms n=1
Compile : 1159.436 ms n=1
Emit : 2273.466 ms n=2
Validate : 294.048 ms n=1
Optimize : 1.126 ms n=1
Transform : n/a n=0
? build app done
? compiler success output files to ./build/
waft>
data:image/s3,"s3://crabby-images/9f411/9f41162406de5cc880599efd0bc965d852e63721" alt="image-20211110171905402"
5、上传app.wasm
data:image/s3,"s3://crabby-images/97816/97816cdce62993e101786880e763078a9ee64c15" alt="image-20211110172203218" data:image/s3,"s3://crabby-images/1bca9/1bca90f00533f0958606579124d2ed7d20c9a2ee" alt="image-20211110172245396"
6、推送
waft> .push
push end md5:8ea41e2c19ca3ee39268a504561c123f file:/home/xyb/waft/my-weather/build/app.wasm
waft>
7、添加测试设备
data:image/s3,"s3://crabby-images/ef869/ef869f394f61a94f212547f8ba9fedcacf70ffaa" alt="image-20211110172654016" data:image/s3,"s3://crabby-images/3a80d/3a80d9c3b3efb1d551226f5add51ad68b6bf2901" alt="image-20211110172717662"
data:image/s3,"s3://crabby-images/32606/326064b7a7790ea114fa779f418f6623a2db95fa" alt="image-20211110172740959"
8、测试
data:image/s3,"s3://crabby-images/1439a/1439afb149eae0fd9f2e87834ef9dd73a74d5144" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/b063d/b063d153112fb240e36bcd1582d1a9657ff0cc58" alt="在这里插入图片描述"
|