/*
* Copyright 2015-2016 the original author or authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package jp.classmethod.aws.gradle.lambda;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.RandomAccessFile;
import java.nio.MappedByteBuffer;
import java.nio.channels.FileChannel;
import java.util.Map;
import lombok.Getter;
import lombok.Setter;
import org.gradle.api.GradleException;
import org.gradle.api.internal.ConventionTask;
import org.gradle.api.tasks.TaskAction;
import com.amazonaws.services.lambda.AWSLambda;
import com.amazonaws.services.lambda.model.CreateFunctionRequest;
import com.amazonaws.services.lambda.model.CreateFunctionResult;
import com.amazonaws.services.lambda.model.Environment;
import com.amazonaws.services.lambda.model.FunctionCode;
import com.amazonaws.services.lambda.model.FunctionConfiguration;
import com.amazonaws.services.lambda.model.GetFunctionRequest;
import com.amazonaws.services.lambda.model.GetFunctionResult;
import com.amazonaws.services.lambda.model.ResourceNotFoundException;
import com.amazonaws.services.lambda.model.Runtime;
import com.amazonaws.services.lambda.model.UpdateFunctionCodeRequest;
import com.amazonaws.services.lambda.model.UpdateFunctionCodeResult;
import com.amazonaws.services.lambda.model.UpdateFunctionConfigurationRequest;
import com.amazonaws.services.lambda.model.UpdateFunctionConfigurationResult;
import com.amazonaws.services.lambda.model.VpcConfig;
public class AWSLambdaMigrateFunctionTask extends ConventionTask {
@Getter
@Setter
private String functionName;
@Getter
@Setter
private String role;
@Getter
@Setter
private Runtime runtime;
@Getter
@Setter
private String handler;
@Getter
@Setter
private String functionDescription;
@Getter
@Setter
private Integer timeout;
@Getter
@Setter
private Integer memorySize;
@Getter
@Setter
private File zipFile;
@Getter
@Setter
private S3File s3File;
@Getter
@Setter
private VpcConfigWrapper vpc;
@Getter
@Setter
private Map<String, String> environment;
@Getter
@Setter
private Boolean publish;
@Getter
private CreateFunctionResult createFunctionResult;
public AWSLambdaMigrateFunctionTask() {
setDescription("Create / Update Lambda function.");
setGroup("AWS");
}
@TaskAction
public void createOrUpdateFunction() throws FileNotFoundException, IOException {
// to enable conventionMappings feature
String functionName = getFunctionName();
File zipFile = getZipFile();
S3File s3File = getS3File();
if (functionName == null) {
throw new GradleException("functionName is required");
}
if ((zipFile == null && s3File == null) || (zipFile != null && s3File != null)) {
throw new GradleException("exactly one of zipFile or s3File is required");
}
if (s3File != null) {
s3File.validate();
}
AWSLambdaPluginExtension ext = getProject().getExtensions().getByType(AWSLambdaPluginExtension.class);
AWSLambda lambda = ext.getClient();
try {
GetFunctionResult getFunctionResult =
lambda.getFunction(new GetFunctionRequest().withFunctionName(functionName));
FunctionConfiguration config = getFunctionResult.getConfiguration();
if (config == null) {
config = new FunctionConfiguration().withRuntime(Runtime.Nodejs);
}
updateFunctionCode(lambda);
updateFunctionConfiguration(lambda, config);
} catch (ResourceNotFoundException e) {
getLogger().warn(e.getMessage());
getLogger().warn("Creating function... {}", functionName);
createFunction(lambda);
}
}
private void createFunction(AWSLambda lambda) throws IOException {
// to enable conventionMappings feature
File zipFile = getZipFile();
S3File s3File = getS3File();
FunctionCode functionCode;
if (zipFile != null) {
try (RandomAccessFile raf = new RandomAccessFile(getZipFile(), "r");
FileChannel channel = raf.getChannel()) {
MappedByteBuffer buffer = channel.map(FileChannel.MapMode.READ_ONLY, 0, channel.size());
buffer.load();
functionCode = new FunctionCode().withZipFile(buffer);
}
} else {
// assume s3File is not null
functionCode = new FunctionCode()
.withS3Bucket(s3File.getBucketName())
.withS3Key(s3File.getKey())
.withS3ObjectVersion(s3File.getObjectVersion());
}
CreateFunctionRequest request = new CreateFunctionRequest()
.withFunctionName(getFunctionName())
.withRuntime(getRuntime())
.withRole(getRole())
.withHandler(getHandler())
.withDescription(getFunctionDescription())
.withTimeout(getTimeout())
.withMemorySize(getMemorySize())
.withPublish(getPublish())
.withVpcConfig(getVpcConfig())
.withEnvironment(new Environment().withVariables(getEnvironment()))
.withCode(functionCode);
createFunctionResult = lambda.createFunction(request);
getLogger().info("Create Lambda function requested: {}", createFunctionResult.getFunctionArn());
}
private void updateFunctionCode(AWSLambda lambda) throws IOException {
// to enable conventionMappings feature
File zipFile = getZipFile();
S3File s3File = getS3File();
UpdateFunctionCodeRequest request = new UpdateFunctionCodeRequest()
.withFunctionName(getFunctionName());
if (zipFile != null) {
try (RandomAccessFile raf = new RandomAccessFile(getZipFile(), "r");
FileChannel channel = raf.getChannel()) {
MappedByteBuffer buffer = channel.map(FileChannel.MapMode.READ_ONLY, 0, channel.size());
buffer.load();
request = request.withZipFile(buffer);
}
} else {
// assume s3File is not null
request = request
.withS3Bucket(s3File.getBucketName())
.withS3Key(s3File.getKey())
.withS3ObjectVersion(s3File.getObjectVersion());
}
UpdateFunctionCodeResult updateFunctionCode = lambda.updateFunctionCode(request);
getLogger().info("Update Lambda function requested: {}", updateFunctionCode.getFunctionArn());
}
private void updateFunctionConfiguration(AWSLambda lambda, FunctionConfiguration config) {
String updateFunctionName = getFunctionName();
if (updateFunctionName == null) {
updateFunctionName = config.getFunctionName();
}
String updateRole = getRole();
if (updateRole == null) {
updateRole = config.getRole();
}
Runtime updateRuntime = getRuntime();
if (updateRuntime == null) {
updateRuntime = Runtime.fromValue(config.getRuntime());
}
String updateHandler = getHandler();
if (updateHandler == null) {
updateHandler = config.getHandler();
}
String updateDescription = getFunctionDescription();
if (updateDescription == null) {
updateDescription = config.getDescription();
}
Integer updateTimeout = getTimeout();
if (updateTimeout == null) {
updateTimeout = config.getTimeout();
}
Integer updateMemorySize = getMemorySize();
if (updateMemorySize == null) {
updateMemorySize = config.getMemorySize();
}
UpdateFunctionConfigurationRequest request = new UpdateFunctionConfigurationRequest()
.withFunctionName(updateFunctionName)
.withRole(updateRole)
.withRuntime(updateRuntime)
.withHandler(updateHandler)
.withDescription(updateDescription)
.withTimeout(updateTimeout)
.withVpcConfig(getVpcConfig())
.withEnvironment(new Environment().withVariables(getEnvironment()))
.withMemorySize(updateMemorySize);
UpdateFunctionConfigurationResult updateFunctionConfiguration = lambda.updateFunctionConfiguration(request);
getLogger().info("Update Lambda function configuration requested: {}",
updateFunctionConfiguration.getFunctionArn());
}
private VpcConfig getVpcConfig() {
if (getVpc() != null) {
return getVpc().toVpcConfig();
}
return null;
}
}