4
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?

Copilot in Azureを使ってみた:タグのつけ忘れを検出せよ!

Posted at

この記事では、Azureポータルから利用できる機能であるCopilot in Azureを試しに使ってみた結果を紹介します。今回は手始めに、「タグをつけ忘れたAzureサービスの検索」を手伝ってもらいました。

想定シナリオ:タグでAzureサービス利用者を管理

筆者は仕事柄、新しいAzureサービスの検証や、Azureサービスを用いたPoC開発に取り組んでいます。同様の業務に取り組んでいる同僚と、検証用のAzure環境を共有して、各自がいろいろなAzureサービスを使って検証・開発を進めています。

複数の開発者でAzure環境を共有している場合、Azure環境の運用管理者としては、Azureサービスについて利用者に問い合わせが必要になることがあります(例:使用料が異様に高い場合)。Azureサービスのタグとして利用者を明記する運用ルールにしておけば、誰が使っているAzureサービスなのか一目瞭然となり、便利です。

一方、Azureサービス作成時にタグをつけ忘れてしまい、そのままタグが存在しないままのAzureサービスが残ることがあります。時間が経つと、Azureサービスを作った本人すら作ったことを忘れてしまい、誰が作ったのかわからないAzureサービスが出てくる可能性もあります。

そのため、タグがついていないAzureサービスは、なるべく早めに把握し、Azure環境の利用者達に「このAzureサービスを作ったのは誰ですか?」と確認したいところです。Copilot in Azureには、Azureサービスの検索に活用できるAzure Resource Graphが組み込まれているので、サービス検索をCopilot in Azureに助けてもらうことにしました。

関連するAzure機能の紹介

Azure Resource Managerとは

Azure Resource Managerは、Azureサービスの管理(作成、変更、削除など)に利用される機能です。Azure Resource Managerでは、各Azureサービスは「リソース」と呼ばれる要素から構成されます。

Azure Resource Managerは各リソースに関して、設定情報(リソース名、リージョン、料金プランなどの設定値)を取得・操作できるREST APIを提供します。例として、WebサイトをホスティングするサービスであるAzure Web Appsの設定情報をJSON形式で出力する際のイメージは以下のようになります(一部の設定情報だけ抜き出しています)。

{
    "id": "/subscriptions/<サブスクリプションID>/resourceGroups/<リソースグループ名>/providers/Microsoft.Web/sites/<リソース名>",
    "name": "<リソース名>",
    "type": "Microsoft.Web/sites",
    "kind": "app,linux",
    "location": "Japan East",
    "properties": {
        "state": "Running",
        (※省略)
    }
}

Azureポータル、Azure CLI、Azure PowerShellなどでリソースを操作するときも、裏ではこのREST APIが呼ばれているようです。Azure Resource Managerに関する詳細は以下のリンク先をご参照ください。

Azure Resource Graphとは

Azure Resource Graphは、Azure Resource Managerから提供される設定情報の検索・活用に利用される機能です。クエリを実行することで、条件に基づいてリソースや設定情報を検索する際に活用できます。本記事では触れませんが、リソースの設定情報の変更履歴を検出する機能もあります。詳細は以下のリンク先をご参照ください。

Azure Resource Graphはリソースや設定情報の検索に便利な一方、Kusto Query Language (KQL)というクエリ言語でクエリを記述する必要があります。筆者はこのクエリ言語に馴染みがなかったので、クエリ作成をCopilot in Azureに助けてもらおうと考えました。

Copilot in Azureとは

Copilot in Azureは、AzureポータルまたはAzure mobile appを介して、Azureの運用管理を支援する機能です(Copilot for Azureと表記されていることもあります)。ユーザからのリクエストに応じて、Azure Resource Graphのクエリを生成してリソースを検索したり、そのリソースに対する操作(例:仮想マシンの電源オフ)をしたりなど、様々なタスクを実行できるようです。詳細は以下のリンク先をご参照ください。

Copilot in Azureは、2024年8月30日現在ではプレビュー段階です。少なくとも現状ではいくつかの制限があり、こちらのリンク先にまとめられています。例えば以下のような制限があります。

  • 現状では英語のみに対応しています。チャットの質問・回答がいずれも英語のみである上に、Azureポータルの言語設定自体を英語に設定する必要があります。
  • 投稿できるリクエストの数に制限があります。具体的には、各チャット(ユーザのリクエストと応答の一連のやり取り)で行える投稿は15件までです。また、24時間中に作成できるチャットは10件のみです。

本題:タグをつけ忘れたAzureサービスを探せ!

前提とする環境

今回は、検証用にAzure Web Appsを4個用意しました。例えばAzureポータルでApp Serviceを一覧表示して、タグを表示させることもできますが、目視でタグの有無を確認すると見落としが発生しやすくなります。複数人でAzure環境を共有していると、リソースの数は4個よりかなり多くなるので、管理がさらに大変になります。

各リソースに付与したタグは以下の通りです。リソースのうち、TestWebApps02というリソースはタグをつけ忘れた想定なので、タグを何も付与していません。他の3個には、利用者名を表すためにuserをキーとするタグを付与してあります。

リソース名 タグ
TestWebApps01 user: mitsu
TestWebApps02 ※タグなし
TestWebApps03 user: mitsu
WebAppDevelABC user: bishi

Copilot in Azureは生成AIらしく、同じ質問でも回答が毎回変わったり、質問に対して誤った回答をしたりします。そのため今回は、正解となる検索対象(タグ未付与のリソース)が明らかになっている状態で、クエリ作成を進めました。

対象のAzureサービスを検索できるまでの軌跡

それでは、Copilot in Azureを使って、タグをつけ忘れたリソースを探すクエリを作成した流れを紹介します。

準備:リソース一覧を取得する

まだリソース検索用のクエリを一度も作ったことがなかったので、まずは試しに、Azure環境にあるAzure Web Appsを一覧化するクエリを作ってもらいました。生成されたクエリは以下の通りです。2行目でAzure Web Appsを絞り込み、3行目でリソース名とタグを抽出してくれそうな記述になっています。

上図にRunというボタンがあり、このボタンをクリックすると、Azureポータル上でAzure Resource Graph エクスプローラーという機能が起動するので、クエリの編集や実行ができます。ただ、Copilot in Azureもクエリ実行・結果表示をしてくれます。Copilot in Azureにクエリを実行してもらった結果は以下の通りです。クエリをもとに、4個のAzure Web Appsの名前とタグを検索できました。

今回作ってもらったクエリは非常にシンプルでしたが、=~という演算子の意味を知らなかったので、聞いてみました。以下の通り、参考文献つきで詳しく教えてくれました。便利ですね。回答内容は誤りを含んでいるかもしれないので、参考文献を合わせて確認すると確実ですね(もしかしたら、参考文献へのリンクが誤っている可能性もあるかもしれませんが)。

タグ未付与リソースの検索クエリを作る

リソースの数が少なければ、先ほど作成したクエリで表示されるタグ一覧からタグが付いていないリソースを探すこともできます。ただ、リソースの数が多くなると少し面倒なので、本題の「タグが付与されていないリソース」を検索するクエリを作ることにしました。

Copilot in Azureでは1つのチャットにつき、リクエストを15回までしか送れないので、一応新しいチャットを立ち上げます。以下のようにNew chatを選べば新しいチャットを開始できます。(※ここで以前のチャットは削除されますので、必要なチャット履歴は控えておくとよいと思います)

最初に生成したクエリは以下の通りです。クエリの知識がない状態の筆者は、isnull(tags)isempty(tags)という記述によって、tagsとしてタグが取得されており、タグがnullもしくは空ということを判別できるのかな...?と考えていました。

クエリを実行してもらった結果がこちらです。回答としては「タグがついていないAzure Web Appsはありません」とのこと。本当は、タグが付いていないリソースが1個存在するはずなので、クエリに誤りがありそうです。

タグ未付与リソースの検索クエリを改善

生成されたクエリにはwhere isnull(tags) or isempty(tags)とあり、タグがnullか空の場合に条件に合致しそうに見えます。ここで気になるのは、tagsには実際に何が設定されているのか?ということです。

最初に作成したリソース一覧取得のクエリの出力結果を再掲すると、以下の通りです。実は、タグ未付与のリソースのTags列には{}と表示されています。

ここが怪しいような気がしたので、以下のようなリクエストを作ってみました。

以下の回答があり、タグの絞り込みの条件がtags == '{}'に修正されました。

クエリを実行してもらうと、以下の通り、タグ未付与のリソースを正しく抽出できるようになりました。

前提とする環境(その2)

先ほどの環境では、Azure Web Appsに付与されているタグのキーはuserのみでした。しかし、実際の環境ではuser以外にもタグが付与されているかもしれません。ということで、Azure Web Appsの想定用途を表すenvというキーを追加してみました。開発環境であれば値はdevel、試験環境であれば値はtestと設定します。

リソース名 タグ(キー:env) タグ(キー:user)
TestWebApps01 env: devel user: mitsu
TestWebApps02 env: test ※なし
TestWebApps03 env: devel user: mitsu
WebAppDevelABC env: devel user: bishi

この前提で何が問題になるかというと、TestWebApps02は依然としてuserタグをつけ忘れているのですが、envタグはついています。つまり、先ほど取り組んでいた「タグが付与されていない(空の)リソースを探す」というクエリではなく、「userタグがないリソースを探す」というクエリにしなければなりません。

リソース一覧の取得とタグの確認

念のため、再度各リソースの設定状況を確認するクエリを実行します。ここでは、Can you generate and run...というリクエストに変えて、クエリを作ったらそのまま実行してもらいました。実行結果は以下の通りで、envタグが正しく追加されています。

ちなみに、先ほどのクエリでは見られませんでしたが、今回のクエリではsubscriptionIdに基づいた絞り込みも行っています。やはり生成AIらしく、似たようなリクエストでも出力が若干異なるのでしょう。

タグ追加後、先ほどの前提で作ったタグ未付与リソース検索クエリを、Azure Resource Graphエクスプローラーで実行してみました。すると、検索対象のTestWebApps02は抽出されませんでした。もはやTestWebApps02のタグは{"env":"test"}になっているので、絞り込みの条件が合致しなくなってしまったのでしょう。

新しい前提向けのクエリ作成

新しい前提に合わせたクエリを作るために、Copilot in Azureにリクエストを投げてみました。すると、「今のAzure Resource Graphでは所望のクエリは作成できない」というような回答が返ってきました...。

生成AIでは一発で所望の結果が得られることは期待できないので、ちょっとアプローチを変えてみました。ひとまず「リソースのタグのキーだけを抽出する」ことを目的とし、以下のようなリクエストを作り、回答を得ました。

何やらmv-expandという見慣れない記述はありますが、クエリを実行してみると、以下のようにキーごとに分解した一覧表示が得られました。

今はタグのキーだけに着目したいので、以下のようなリクエストを送ると、クエリを改善してくれました。

bag_keysという記述がよくわかりませんが、クエリを実行してみると、以下のようにキーだけが抽出されました。これを見ると、各リソースに紐づいているタグのキーがわかるので、userタグがついていないリソースの検索は可能なのではないか?と期待できますね。

ちなみにmv-expandbag_keysについて意味を質問すると、以下のように教えてくれました。やはり便利ですね。ただし、今回は参考文献へのリンクは付与されていませんでした。

今までのリクエストと回答のやり取りで絞り込みの情報が得られているので、今回はクエリを作ってくれると期待し、改めて以下のようなリクエストを送ってみました。回答結果は...、クエリを作ってくれました!しかし、このクエリには先ほど登場したmv-expandbag_keysは両方とも使われていませんでした。

Azure Resource Graphエクスプローラーでクエリを実行してみましたが、以下のような結果となり、このクエリでuserタグが存在しないリソースを抽出できることがわかりました。途中のmv-expandbag_keysのやり取りの効果があったのかはわかりませんが...、所望のクエリができたので良しとしましょう。

補足:GPT-4oにも聞いてみた

Copilot in Azure以外に聞いてみるとどうなるのか気になったので、Azure OpenAIのGPT-4oにもAzure Resouce Graphのクエリを作ってもらいました。

今回はGPT-4oのバージョン2024-05-13を使用しました。システムプロンプトとしては以下を使用しました(これ以外のシステムプロンプトでどのように生成結果が変わるのかは確認していません)。

You are an expert in Azure Resource Graph. You help engineers understand and create queries for Azure Resource Graph effectively. Please check the accuracy of the information that you provide.

前提1

最初のプロンプトと回答

まずは、Copilot in Azureに最初に質問したものとほぼ同じ質問をしてみました。入力したプロンプトは以下の通りです。

Can you generate a query to list Azure Web Apps without tags?

回答には以下の2つのクエリが含まれていましたが、いずれも検索対象を正しく抽出できないものでした。クエリ1は、3行目がtags == '{}'なら正しいですが、そうなっていません。クエリ2は、Copilot in Azureで最初に得られたクエリと同じであり、検索対象を正しく抽出できません。

  • クエリ1

    Resources  
    | where type == "microsoft.web/sites"  
    | where tags == ''  
    | project name, resourceGroup, subscriptionId 
    
  • クエリ2

    Resources  
    | where type == "microsoft.web/sites"  
    | where isnull(tags) or isempty(tags)  
    | project name, resourceGroup, subscriptionId 
    

クエリの改善

Copilot in Azureに指摘したのと同様に、tagsが空の場合は{}という値になることを指摘してみました。入力したプロンプトは以下の通りです。

These commands will return a list of Azure Web Apps without tags.
I think the property 'tags' is set to '{}' if it is empty. Can you update the query?

回答は以下の通りでした。Copilot in Azureの回答では見られなかったarray_length(bag_keys(tags)) == 0という表現が追記されました。

resources  
| where type == 'microsoft.web/sites'  
| where isnull(tags) or tags == '{}' or array_length(bag_keys(tags)) == 0  
| project name, resourceGroup, location, tags  

今回のケースでは、3行目にあるtags == '{}'array_length(bag_keys(tags)) == 0は機能的には同じようですし、どちらかを選べばよさそうです。isnull(tags)も、Copilot in Azureのときに試した通り無意味なので、省略できますね。

Copilot in Azureで生成したものと結果は同じですが、違う書き方のクエリとして、最終的に以下を得ました。Azure Resource Graphエクスプローラーで実行したところ、正常に動作しました。

resources  
| where type == 'microsoft.web/sites'  
| where array_length(bag_keys(tags)) == 0  
| project name, resourceGroup, location, tags  

前提2

envタグを追加した場合も試してみました。入力したプロンプトは以下の通りです。

Can you generate a query to list Azure Web Apps missing the key 'user' in their tags?

実はこちらについては、一発で所望のクエリを得ることができました。生成されたクエリは、Copilot in Azureで最終的に得られたものと同じで、正常に動作します。

Certainly! To list Azure Web Apps that are missing the key 'user' in their tags using Azure Resource Graph, you can use the following Kusto Query Language (KQL) query:
Resources  
| where type == 'microsoft.web/sites'  
| where isnull(tags['user'])  
| project id, name, location, tags  

ちなみにお気づきの方がいるかもしれませんが、Azure OpenAIが作った方のクエリではtypeの絞り込みがtype == 'microsoft.web/sites'となっています。Copilot in Azureが生成したクエリではtype =~ 'microsoft.web/sites'になっており、演算子が違います。結局どちらも動作したのですが、どちらの演算子の方がよいかは別途調べたいと思います。

おわりに

今回は、Copilot in Azureを使ったAzure運用支援の一例として、タグをつけ忘れたリソースを探すためのAzure Resource Graphのクエリ作成を支援してもらいました。筆者はAzure Resource Graphのクエリについてほぼ知見がない状態でしたが、Copilot for Azureの支援のおかげで効率的にクエリを作成できました。

Copilot in Azureが生成するクエリの精度が高まってくると、さらに使い勝手が良くなりそうです。参考までにGPT-4oでもクエリを生成してみましたが、正直今回の結果だけでは、Copilot in AzureとGPT-4oで生成結果の違いはわかりませんでした。Azureポータルに統合されている分、今回のようなリソース検索向けにはCopilot in Azureの方が便利に使えそうです。

このクエリを定期的に自動実行するような仕組みを作れば、誰かがAzureサービスにタグをつけ忘れたとしても、そのAzureサービスを自動で検知できるようになりますね。例えば、Logic Appsを使って自動化する場合、以下のリンク先が参考になりそうです。

今回はちょっとした運用管理の困りごとの解決をCopilot in Azureに手伝ってもらいましたが、Copilot in Azureのとてもシンプルな適用例に過ぎません。今後は、もう少し複雑な運用管理のタスクを支援してもらえないか検証してみます。

4
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
4
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?