LoginSignup
2
0

More than 3 years have passed since last update.

CloudWatchイベントでスケジュールされたECSタスクをAWS SDK for Javaで直接起動する

Posted at

TL; DR

CloudWatchイベントでスケジュールされたECSタスクを、AWS SDK for Javaを利用して直接起動する機会があったので、備忘録がてら知見としてまとめます。

環境

  • AWS CLI: 2.1.16
  • Kotlin: 1.3.50
  • AWS SDK for Java: 2.8.7

AWS CLI

やること自体はとても単純ですので、簡単にAWS CLIで動作確認していきます。

まず、list-targets-by-ruleでCloudWatchイベントでスケジュールされたECSタスクのパラメータを取得します。

$ aws events list-targets-by-rule --rule "batch-schedule"

レスポンスを確認すると、Arnにクラスター名、EcsParameters以下にECSタスクの起動に必要なパラメータが色々設定されているのが分かります。

{
    "Targets": [
        {
            "Id": "batch-schedule",
            "Arn": "arn:aws:ecs:ap-northeast-1:{アカウントID}:cluster/{クラスター名}",
            "RoleArn": "arn:aws:iam::{アカウントID}:role/{ロール名}",
            "EcsParameters": {
                "TaskDefinitionArn": "arn:aws:ecs:ap-northeast-1:{アカウントID}:task-definition/{ファミリー名}:{リビジョン番号}",
                "TaskCount": 1,
                "LaunchType": "FARGATE",
                "NetworkConfiguration": {
                    "awsvpcConfiguration": {
                        "Subnets": [
                            "{サブネットID}"
                        ],
                        "SecurityGroups": [
                            "{セキュリティグループID}"
                        ],
                        "AssignPublicIp": "DISABLED"
                    }
                },
                "PlatformVersion": "1.3.0"
            }
        }
    ]
}

後は上記で取得したパラメータを使ってrun-taskを実行するだけです。

$ aws ecs run-task \
--cluster "arn:aws:ecs:ap-northeast-1:{アカウントID}:cluster/{クラスター名}" \
--task-definition "arn:aws:ecs:ap-northeast-1:{アカウントID}:task-definition/{ファミリー名}:{リビジョン番号}" \
--launch-type "FARGATE" \
--network-configuration "awsvpcConfiguration={subnets=[{サブネットID}],securityGroups=[{セキュリティグループID}],assignPublicIp=DISABLED}"

AWS SDK for Java

上記と同じことをAWS SDKで実行すると下記のようになります。

build.gradle.kts
dependencies {
    ...

    implementation("software.amazon.awssdk:ecs")
    implementation("software.amazon.awssdk:cloudwatchevents")
}
import software.amazon.awssdk.services.cloudwatchevents.CloudWatchEventsClient
import software.amazon.awssdk.services.cloudwatchevents.model.ListTargetsByRuleRequest
import software.amazon.awssdk.services.ecs.EcsClient
import software.amazon.awssdk.services.ecs.model.AwsVpcConfiguration
import software.amazon.awssdk.services.ecs.model.NetworkConfiguration
import software.amazon.awssdk.services.ecs.model.RunTaskRequest
import software.amazon.awssdk.services.ecs.model.RunTaskResponse


fun runTask(): RunTaskResponse {
    // CloudWatchイベントでスケジュールされたECSタスクのパラメータを取得
    val cloudWatchEvents = CloudWatchEventsClient.builder().build()
    val listTargetsByRuleRequest = ListTargetsByRuleRequest.builder()
        .rule("batch-schedule")
        .build()
    val listTargetsByRuleResponse = cloudWatchEvents.listTargetsByRule(listTargetsByRuleRequest)
    val eventRule = listTargetsByRuleResponse.targets().first()
    val ecsParameters = eventRule.ecsParameters()

    // CloudWatch Eventsのパッケージで定義されたNetworkConfigurationを元に
    // ECSのパッケージで定義されたNetworkConfigurationを作成する
    val networkConfiguration = ecsParameters.networkConfiguration().awsvpcConfiguration().let {
        NetworkConfiguration.builder().awsvpcConfiguration(
            AwsVpcConfiguration.builder()
                .subnets(it.subnets())
                .securityGroups(it.securityGroups())
                .assignPublicIp(it.assignPublicIp().name)
                .build()
        ).build()
    }

    // ECSタスクを実行する
    val ecs = EcsClient.builder().build()
    val runTaskRequest = RunTaskRequest.builder()
        .cluster(eventRule.arn())
        .taskDefinition(ecsParameters.taskDefinitionArn())
        .launchType(ecsParameters.launchType().name)
        .networkConfiguration(networkConfiguration)
        .build()
    return ecs.runTask(runTaskRequest)
}

NetworkConfigurationの詰め替えが若干面倒ではありますが、ListTargetsByRuleで取得したパラメータを元にRunTaskを実行するだけなのは特に変わらないですね。

まとめ

実際はClientをシングルトンにしたり色々やると思いますが、記事にするにあたって諸々簡略化しました。
AWS CLIもAWS SDKも結局はAWS APIをコールしてるだけなので、AWS APIにもっと慣れ親しんだ方が理解が早そうだなぁと思う今日この頃です。

2
0
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
2
0