但是,要从头自己开始学习还是有点点困难的。所以这次记录了一下从头开始学习Lambda函+Cognito与Android配合使用的过程。
“AWS Lambda 是一项计算服务,可使您无需预配置或管理服务器即可运行代码。”
所以,我们只需要关注于代码本身,由AWS Lambda 管理提供内存、CPU、网络和其他资源均衡的计算机群。
创建一个简单的 Android 移动应用程序,通过在本地应用程序中输入字符串并提交,从 Amazon Cognito 身份池检索 AWS 凭证,并使用包含请求数据的事件,调用 Lambda 函数处理请求,记录Cloud Watch日志并向前端返回响应,返回字符串。
如果这些字连在一起就看不懂了:其实就是输入两个字符串,提交之后触发了Lambda函数;触发成功,返回toast,由Lambda函数记录调用的日志。
AWS建议 不要使用 AWS 账户根用户执行任务,而是应为需要管理员访问权限的每个人创建新的 IAM 用户。
所以,拥有IAM用户是访问AWS服务的先决条件。
想要通过控制台创建用户,可以下载AWS CLI。
a. 用户名即之后用于访问服务的用户名
b. 因为之后要通过CLI访问,所以务必要选择编程访问。选择AWS控制台访问的话,之后可以通过 IAM 控制台控制面板中找到账户的登录 URL进行登录。
https://
account-ID-or-alias
.signin.aws.amazon.com/console
筛选策略中,选择AdministratorAccess
,即赋予组用户管理员权限。
本次实验暂时用不到。接下来的步骤确认信息无误后,拥有管理员权限的IAM用户即可生成。
查看创建好的用户,记录用户ARN,用于之后身份验证服务。arn是AWS资源的唯一标识。
要让AWS Lambda对我们的资源进行访问、操作,以实现具体功能,就要授予其相对应的权限。
IAM 角色就是信任的实体授予权限的安全方法。实体可以是其他账户的IAM用户、AWS的服务、EC2上的程序代码等。
这里选择AWSLambdaBasicExecutionRole
。
lambda-android-role
接下来的操作会用到shell,编写一个简单的触发器,在本地打包创建成lambda函数并提交。
在这里,我选择用WSL 2(Windows Subsystem for Linux 2) 进行脚本编写创建,如果使用Linux/MacOS直接调用默认shell就可以了。
其实后来我看开发指南,Windows自带的Power Shell也是可以的,被我弄复杂了;但是Power Shell长得丑,算了不亏。
方法有很多,只要是shell能编写程序就OK。
Lambda支持很多语言,如:javascript, java, python, c# … 这里按照官方给的教程,我用的是node.js,很简洁。
这里我们简单定义了,调用程序中的函数,在console中记录事件并返回给客户端一个字符串的操作。
exports.handler = function(event, context, callback) {
console.log("Received event: ", event);
var data = {
"greetings": "Hello, " + event.firstName + " " + event.lastName + "."
};
callback(null, data);
}
把文件打包成压缩包,并下载aws命令行。
zip function.zip index.js
sudo apt install awscli
如果是第一次使用AWS CLI,那么就要设置用户。
aws config
按照指令,输入STEP - 1中下载好[用户]-[安全证书]中的key,以及服务的地域等信息。
create-function
命令创建 Lambda 函数AndroidBackendLambdaFunction
。aws lambda create-function --function-name [lambda函数名] \ --zip-file
fileb://[目录路径]/function.zip --handler index.handler --runtime nodejs12.x \
--role [lambda role arn]
Amazon Cognito 提供用户池和身份池。
用户池是为应用程序提供注册和登录选项的用户目录,身份池提供 AWS 凭证以向用户授予对其他 AWS 服务的访问权限。
这里用身份池,并授予其执行Lambda AndroidBackendLambdaFunction
函数的权限。
身份池名为 JavaFunctionAndroidEventHandlerPool
,并弃用未经验证的访问权限(即允许游客登录应用程序,触发该函数)。
在未经身份验证(unauthenticated identities)的角色中,输入
{
"Version":"2012-10-17",
"Statement":[
{
"Effect":"Allow",
"Action":[
"mobileanalytics:PutEvents",
"cognito-sync:*"
],
"Resource":[
"*"
]
},
{
"Effect":"Allow",
"Action":[
"lambda:invokefunction"
],
"Resource":[
"arn:aws:lambda:us-east-1:[account-id]:function:AndroidBackendLambdaFunction"
]
}
]
}
Resource对应的就是AndroidBackendLambdaFunction的arn。[account-id]即IAM用户的id。
创建成功后,系统会提示保存身份池id等信息,记录下来,下一步将会用到.
做了这么多,终于可以开始创建应用了,其实应用在这个demo里面就显得比较简单。
值得注意的是,Lambda函数最好在API 28以上运行。在新建APP的时候,注意API版本。
如果在创建工程之后,同步(sync)失败,(可能报app compact xxx29.+之类的错),有以下几种解决方法:
创建的工程名为AndroidEventGenerator
在 build.gradle (Module:app
) 文件的 dependencies
部分中添加以下内容
implementation 'com.amazonaws:aws-android-sdk-core:2.2.+'
implementation 'com.amazonaws:aws-android-sdk-lambda:2.2.+'
implementation "commons-logging:commons-logging:1.2"
AndroidManifest.xml
中,添加连接 Internet的权限<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.androideventgenerator">
...
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
manifest>
RequestClass
和ResponseClass
package com.example.androideventgenerator;
public class RequestClass {
String firstName;
String lastName;
public RequestClass(String firstName, String lastName) {
this.firstName = firstName;
this.lastName = lastName;
}
public RequestClass() {
}
public String getFirstName() {
return firstName;
}
public void setFirstName(String firstName) {
this.firstName = firstName;
}
public String getLastName() {
return lastName;
}
public void setLastName(String lastName) {
this.lastName = lastName;
}
}
package com.example.androideventgenerator;
public class ResponseClass {
String greetings;
public ResponseClass(String greetings) {
this.greetings = greetings;
}
public ResponseClass() {
}
public String getGreetings() {
return greetings;
}
public void setGreetings(String greetings) {
this.greetings = greetings;
}
}
MyInterface
用annotation@LambdaFunction
对接口函数进行映射,注意这个映射的函数名应该一致,参数是request。
package com.example.androideventgenerator;
import com.amazonaws.mobileconnectors.lambdainvoker.LambdaFunction;
public interface MyInterface {
/**
* Invoke the Lambda function "AndroidBackendLambdaFunction".
* The function name is the method name.
*/
@LambdaFunction
ResponseClass AndroidBackendLambdaFunction(RequestClass request);
}
MainActivity
代码要注意的地方我写在注释里。。好长啊
package com.example.androideventgenerator;
import android.os.AsyncTask;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.Toast;
import com.amazonaws.auth.CognitoCachingCredentialsProvider;
import com.amazonaws.mobileconnectors.lambdainvoker.LambdaFunctionException;
import com.amazonaws.mobileconnectors.lambdainvoker.LambdaInvokerFactory;
import com.amazonaws.regions.Regions;
import androidx.appcompat.app.AppCompatActivity;
public class MainActivity extends AppCompatActivity {
EditText editText1, editText2;
Button button;
RequestClass request;
MyInterface myInterface;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
// 创建提供Cognito服务的对象实例,这句在创建身份池的时候AWS已经提供了
CognitoCachingCredentialsProvider cognitoProvider = new CognitoCachingCredentialsProvider(
//把XXXXXXXXX改成创建的身份池id
this.getApplicationContext(), "us-east-1:XXXXXXXXX", Regions.US_EAST_1);
// 创建调用Lambda函数的工厂代理对象
//第三个参数是 a AWS credentials provider
LambdaInvokerFactory factory = new LambdaInvokerFactory(this.getApplicationContext(),
Regions.US_EAST_1, cognitoProvider);
//用工厂和默认的binder(第二个参数)创建一个Lambda对象实例
myInterface = factory.build(MyInterface.class);
editText1 = findViewById(R.id.editText);
editText2 = findViewById(R.id.editText2);
button = findViewById(R.id.button);
button.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
String firstName = editText1.getText().toString();
String LastName = editText2.getText().toString();
request = new RequestClass(firstName, LastName);
// The Lambda function invocation results in a network call.
// Make sure it is not called from the main thread.
//执行一个异步的请求
new myAsyncTask().execute(request);
}
});
}
//为了防止内存泄漏,不要把AsyncTask放在主线程中
class myAsyncTask extends AsyncTask<RequestClass, Void, ResponseClass> {
@Override
protected ResponseClass doInBackground(RequestClass... params) {
// invoke "echo" method. In case it fails, it will throw a
// LambdaFunctionException.
try {
//******调用Lambda函数,返回request********//
return myInterface.AndroidBackendLambdaFunction(params[0]);
} catch (LambdaFunctionException lfe) {
Log.e("Tag", "Failed to invoke echo", lfe);
return null;
}
}
@Override
protected void onPostExecute(ResponseClass result) {
if (result == null) {
return;
}
/* 执行完onBackgoundTask方法后,执行onPostExecute方法 !
** 如果onBackgoundTask return回来的结果不为空,即Lambda函数调用成功的话!
** 就do a toast !
*/
Toast.makeText(MainActivity.this, result.getGreetings(), Toast.LENGTH_LONG).show();
}
}
}
虽然我的布局有点丑,但是可以看出,点击SUBMIT之后,输入的字符串在应用内被响应了。
同时,在AWS CloudWatch或AWS Lambda监控的CloudWatch中,都可以查看到Lambda函数记录的日志内容。
补充:调用函数的收费时间等信息也是可以看到的。
可以看出,我们只写了Lambda功能代码,没有部署过物理资源,而实际调用物理资源的时间也微乎其微。
参考: AWS Lambda 开发人员指南