2
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

Amazon CloudWatch ダッシュボードの画像にClaude3でコメント付けて返すSlackBot

Last updated at Posted at 2024-04-23

長いタイトルですが、タイトルそのまんまです。
先日、「Amazon Bedrock」で「Claude3 Opus」が使えるようになったので何かやってみたくなりました。

そこで、作ったはいいけどあまり見られる事が無い...CloudWatchダッシュボードの画像にClaude3で解析コメントを付けてSlackに送ったら、多少皆が気を留めてくれるかもと思いました。

Opusの性能を試す題材として適切かは微妙ですが、せっかくやってみたので備忘録も兼ねて手順を残します。

Amazon CloudWatch ダッシュボード

今回はサンプルとして、個人で動かしているLambdaでダッシュボードを作りました。
image.png

右上の「アクション」から「ソースの表示/編集」を開くと、ダッシュボードの内容をJson形式で取得できます。
image.png

以下のような感じでコピーできるので、source.jsonとして保存します。

source.json
{
    "widgets": [
        {
            "height": 6,
            "width": 6,
            "y": 12,
            "x": 6,
            "type": "metric",
            "properties": {
                "metrics": [
                    [ "AWS/Lambda", "Errors", "FunctionName", "test-function", { "region": "ap-northeast-1" } ]
                ],
                "view": "timeSeries",
                "stacked": false,
                "region": "ap-northeast-1",
                "period": 300,
                "stat": "Sum"
            }
        },
        {
            "height": 6,
            "width": 6,
            "y": 0,
            "x": 0,
            "type": "metric",
            "properties": {
                "metrics": [
                    [ "AWS/Lambda", "Invocations", "FunctionName", "test-function" ]
                ],
                "view": "timeSeries",
                "stacked": false,
                "region": "ap-northeast-1",
                "stat": "Sum",
                "period": 300
            }
        },
        ...
    ]
}

以下のAPIを使ってこのJsonを投げていくと画像が取得できるようです。

というわけで、ここからはLambdaに移ります。

Lambda

取得した画像をどうやって1枚の画像にしようか...と試行錯誤しつつ、Clauda君に聞いてみたらsharpというライブラリを教えてもらえたのでそれでやってみる事にしました。
最終的にDockerコンテナにして動かしてます。

Dockerfile
FROM public.ecr.aws/lambda/nodejs:18

WORKDIR /var/task

RUN yum -y update && \
    yum install -y gcc-c++ make libpng-devel libjpeg-devel libwebp-devel giflib-devel libtiff-devel libheif-devel librsvg2-devel

ENV npm_config_arch=x64 npm_config_platform=linux npm_config_sharp_binary_host="none"

COPY package.json ./

RUN npm install
RUN npm install -g typescript

COPY . .

RUN npm run build

CMD ["dist/index.handler"]

Lambdaのコードは以下のようになりました。
画像を連結して、それをClaude3 Oupsで解析してもらっています。
色々思うところはあるものの一旦動いたからヨシとしよう!

index.ts
import * as AWS from 'aws-sdk';
import sharp from 'sharp';
import { WebClient } from '@slack/web-api';
import { BedrockRuntimeClient, InvokeModelCommand } from '@aws-sdk/client-bedrock-runtime';
import source from './source.json'; //先程のjsonを読み込む

const slackToken = process.env.SLACK_APP_TOKEN;
const slackChannelId = process.env.SLACK_CHANNEL_ID;
const dashboardUrl = process.env.CLOUDWATCH_DASHBOARD_URL;
const modelId = process.env.MODEL_ID; //'anthropic.claude-3-opus-20240229-v1:0';
const prompt = 'この画像は3日間のメトリクスです。要点を簡潔に教えて下さい。優しく丁寧な言葉遣いで語尾にはワンとつけてください';

const cloudwatch = new AWS.CloudWatch();
const slackWeb = new WebClient(slackToken);
const client = new BedrockRuntimeClient({ region: 'us-west-2' });

export async function handler(event: any) {
  // CloudWatchから画像データを取得する
  const promises = source.widgets.map(async (widget) => {
    const metricWidget = {
      ...widget.properties,
      start: '-P3D', //範囲3日
      timezone: '+0900',
    };

    return cloudwatch
      .getMetricWidgetImage({
        OutputFormat: 'png',
        MetricWidget: JSON.stringify(metricWidget),
      })
      .promise();
  });
  //有効な画像データのみをフィルタリング
  const imageBuffers = (await Promise.all(promises)).map((result) => result.MetricWidgetImage).filter((buffer): buffer is Buffer => buffer !== undefined);
  if (imageBuffers.length === 0) return;
  // 画像のマージに必要な設定
  const imageWidth = 600;
  const imageHeight = 500;
  const columns = 3;
  const rows = Math.ceil(imageBuffers.length / columns);
  let mergedImage = sharp({
    create: {
      width: imageWidth * columns,
      height: imageHeight * rows,
      channels: 4,
      background: { r: 255, g: 255, b: 255, alpha: 0 },
    },
  }).png();
  // 複数の画像を一つに合成
  const compositeOptions = imageBuffers.map((buffer, index) => ({
    input: buffer,
    left: (index % columns) * imageWidth,
    top: Math.floor(index / columns) * imageHeight,
  }));
  mergedImage = mergedImage.composite(compositeOptions);
  const finalBuffer = await mergedImage.toBuffer();
  const base64String = finalBuffer.toString('base64');
  //Claude3を使用して画像分析
  const aiResponse = await claudeCompletion(base64String);
  const initialComment = `OPSちゃんからのメッセージ:\n\n${aiResponse}\n\n詳しくはこちらから:\n\n ${dashboardUrl}`;
  // Slackにファイルとして画像をアップロード
  await slackWeb.files.upload({
    channels: slackChannelId,
    file: finalBuffer,
    filename: new Date().toLocaleTimeString(),
    title: new Date().toLocaleTimeString(),
    filetype: 'image/png',
    initial_comment: initialComment,
  });

  return {
    statusCode: 200,
    body: JSON.stringify('succeed'),
  };
}

// Bedrock Claude3 Opus
const claudeCompletion = async (base64String: string) => {
  const payload = {
    anthropic_version: 'bedrock-2023-05-31',
    max_tokens: 4096,
    messages: [
      {
        role: 'user',
        content: [
          { type: 'text', text: prompt },
          {
            type: 'image',
            source: {
              type: 'base64',
              data: base64String,
              media_type: 'image/png',
            },
          },
        ],
      },
    ],
  };

  const command = new InvokeModelCommand({
    contentType: 'application/json',
    body: JSON.stringify(payload),
    modelId: modelId,
  });

  const apiResponse = await client.send(command);
  const decodedResponseBody = new TextDecoder().decode(apiResponse.body);
  const responseBody = JSON.parse(decodedResponseBody);
  const text = responseBody.content[0].text;
  console.log(text);
  return text;
};

Claude3 Opusのモデルアクセスをリクエストしておきます。
現在、米国西部 (us-west-2) で利用可能なようです。

image.png

Slackでアプリの作成、トークンの取得などは下記参照。

権限はfiles:write chat:write が必要です。
image.png

結果

LambdaをデプロイしEventBridgeでスケジュール設定して実行してみます。
すると以下のようにSlackに画像 & コメント付きでメッセージが...

image.png

か..かわいい!!!

何度か試していて時々間違った事を言う事もありましたが、気を引くくらいの効果はありそうなので一旦はヨシとしましょう。

やっぱりただ画像を送るより明るい感じにはなりますね!

ただ、Claude3 Opusの性能を測る題材ではないですよねえ...🙇‍♂️
使い所は今後も色々と探していこうと思います。

2
1
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
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?