<p><!-- META
{
"version": "1.0",
"style": "SRE_EXPERT_DOC",
"focus": "jq_advanced_manipulation",
"robustness_level": "high"
}
-->
本記事は<strong>Geminiの出力をプロンプト工学で整理した業務ドラフト(未検証)</strong>です。</p>
<h1 class="wp-block-heading">jqを駆使したAPIレスポンスの高度な集計:group_byと<code>reduce</code>によるインベントリ解析の自動化</h1>
<h2 class="wp-block-heading">【導入と前提】</h2>
<p>本スクリプトは、クラウドインフラのAPIやマイクロサービスから取得した大量のJSON配列を、特定のキー(環境、サービス名、ステータス等)ごとにグループ化し、合計値やカウントを算出する処理を自動化・堅牢化します。</p>
<ul class="wp-block-list">
<li><p><strong>実行環境</strong>: GNU/Linux (Ubuntu/RHEL/CentOS)</p></li>
<li><p><strong>前提ツール</strong>: <code>jq</code> (v1.6以上推奨), <code>curl</code> (v7.40以上)</p></li>
<li><p><strong>対象</strong>: 構造化されたJSONログや、クラウドプロバイダーのインベントリデータ</p></li>
</ul>
<h2 class="wp-block-heading">【処理フローと設計】</h2>
<div class="wp-block-merpress-mermaidjs diagram-source-mermaid"><pre class="mermaid">
graph TD
A["API/JSON Raw Data"] --> B{"jq Process"}
B --> C["group_by: 特定キーで配列を多次元化"]
B --> D["reduce: 単一パスでの累積・集計"]
C --> E["map: 各グループの要約オブジェクト作成"]
D --> F["Final Report: カテゴリ別合計/集計値"]
E --> F
F --> G["stdout / Log File"]
</pre></div>
<p><code>group_by</code>は直感的な記述が可能ですが、内部でソートを行うため計算量は $O(N \log N)$ となります。一方、<code>reduce</code>は1スキャン($O(N)$)で処理可能なため、大規模データのストリーム処理に適しています。</p>
<h2 class="wp-block-heading">【実装:堅牢な自動化スクリプト】</h2>
<p>以下は、ダミーのインベントリデータから「ステータスごとのカウント」と「コストの合計」を算出する実戦的なスクリプトです。</p>
<div class="codehilite">
<pre data-enlighter-language="generic">#!/usr/bin/env bash
# -----------------------------------------------------------------------------
# Script: aggregate_inventory.sh
# Description: jqのgroup_byとreduceを用いた高度なJSON集計
# -----------------------------------------------------------------------------
set -euo pipefail
IFS=$'\n\t'
# 一時ファイルのクリーンアップ設定
TMP_FILE=$(mktemp)
trap 'rm -f "$TMP_FILE"' EXIT
# --- 1. データ取得 (ダミーAPIレスポンスの生成) ---
cat << 'EOF' > "$TMP_FILE"
[
{"id": "v-001", "status": "running", "cost": 150, "region": "ap-northeast-1"},
{"id": "v-002", "status": "stopped", "cost": 10, "region": "us-east-1"},
{"id": "v-003", "status": "running", "cost": 200, "region": "ap-northeast-1"},
{"id": "v-004", "status": "running", "cost": 150, "region": "us-east-1"},
{"id": "v-005", "status": "terminated", "cost": 0, "region": "ap-northeast-1"}
]
EOF
echo "=== [1] group_by によるリージョン別集計 ==="
# group_by(.key) は指定キーで配列を分割し、[ [obj, obj], [obj] ] の形にする
jq -r '
group_by(.region) | map({
region: .[0].region,
count: length,
total_cost: map(.cost) | add
})
' "$TMP_FILE"
echo -e "\n=== [2] reduce によるステータス別の高度な集計 ==="
# reduce はメモリ効率が良く、初期値 ( {} ) に対して逐次更新を行う
jq -r '
reduce .[] as $item (
{};
.[$item.status].count += 1 |
.[$item.status].total_cost += $item.cost
) | to_entries | map({
status: .key,
metrics: .value
})
' "$TMP_FILE"
# 補足: 実戦では curl を以下のように利用
# curl -sfL --retry 3 --retry-delay 5 "https://api.example.com/v1/resources" \
# -H "Authorization: Bearer ${API_TOKEN}" | jq ...
</pre>
</div>
<h3 class="wp-block-heading">systemd/Timer 設定例(定期集計用)</h3>
<p>集計処理をバックグラウンドで定期実行する場合のユニット定義例です。</p>
<div class="codehilite">
<pre data-enlighter-language="generic"># /etc/systemd/system/inventory-aggregator.service
[Unit]
Description=Daily Cloud Cost Aggregator
[Service]
Type=oneshot
ExecStart=/usr/local/bin/aggregate_inventory.sh
EnvironmentFile=/etc/default/cloud-api-creds
User=reports-user
# /etc/systemd/system/inventory-aggregator.timer
[Timer]
OnCalendar=daily
Persistent=true
[Install]
WantedBy=timers.target
</pre>
</div>
<h2 class="wp-block-heading">【検証と運用】</h2>
<h3 class="wp-block-heading">正常系の確認</h3>
<p>スクリプトを実行し、以下のJSON構造が出力されることを確認します。</p>
<div class="codehilite">
<pre data-enlighter-language="generic"># 実行
./aggregate_inventory.sh
# 期待される出力(抜粋)
# [
# { "status": "running", "metrics": { "count": 3, "total_cost": 500 } },
# ...
# ]
</pre>
</div>
<h3 class="wp-block-heading">ログ確認方法</h3>
<p>systemd経由で実行している場合は、<code>journalctl</code> を使用して、パイプラインの途中でエラーが発生していないか確認します。</p>
<div class="codehilite">
<pre data-enlighter-language="generic"># ユニットの最新ログを確認
journalctl -u inventory-aggregator.service -n 50 --no-pager
</pre>
</div>
<h2 class="wp-block-heading">【トラブルシューティングと落とし穴】</h2>
<ol class="wp-block-list">
<li><p><strong>データ型の不一致</strong>: <code>reduce</code>内で加算を行う際、<code>cost</code>が文字列として渡されるとエラーになります。<code>($item.cost | tonumber)</code> を使用して型を明示的に変換してください。</p></li>
<li><p><strong>大規模データのメモリ消費</strong>: <code>group_by</code> は全データをメモリに展開してソートします。数GB単位のJSONを扱う場合は、<code>jq -c '.[]'</code> で各行を分離し、<code>reduce</code> または外部の <code>awk</code> 等と組み合わせるストリーム処理を検討してください。</p></li>
<li><p><strong>環境変数の扱い</strong>: APIトークンなどを <code>jq</code> の引数に直接渡すと <code>ps</code> コマンドから見えてしまうため、必ず <code>--arg</code> オプション経由で渡すようにします(例: <code>jq --arg key "$VAL" '.[$key]'</code>)。</p></li>
</ol>
<h2 class="wp-block-heading">【まとめ】</h2>
<p>運用の冪等性と堅牢性を維持するための3つのポイント:</p>
<ul class="wp-block-list">
<li><p><strong>計算効率の選択</strong>: 直感的な <code>group_by</code> か、高速な <code>reduce</code> かをデータサイズに応じて使い分ける。</p></li>
<li><p><strong>エラーハンドリング</strong>: <code>set -euo pipefail</code> を徹底し、パイプライン途中の <code>jq</code> の失敗を検知できるようにする。</p></li>
<li><p><strong>型保証</strong>: <code>tonumber</code> や <code>tostring</code> を活用し、入力データがスキーマから外れている場合のクラッシュを未然に防ぐ。</p></li>
</ul>
本記事はGeminiの出力をプロンプト工学で整理した業務ドラフト(未検証)です。
jqを駆使したAPIレスポンスの高度な集計:group_byとreduceによるインベントリ解析の自動化
【導入と前提】
本スクリプトは、クラウドインフラのAPIやマイクロサービスから取得した大量のJSON配列を、特定のキー(環境、サービス名、ステータス等)ごとにグループ化し、合計値やカウントを算出する処理を自動化・堅牢化します。
実行環境: GNU/Linux (Ubuntu/RHEL/CentOS)
前提ツール: jq (v1.6以上推奨), curl (v7.40以上)
対象: 構造化されたJSONログや、クラウドプロバイダーのインベントリデータ
【処理フローと設計】
graph TD
A["API/JSON Raw Data"] --> B{"jq Process"}
B --> C["group_by: 特定キーで配列を多次元化"]
B --> D["reduce: 単一パスでの累積・集計"]
C --> E["map: 各グループの要約オブジェクト作成"]
D --> F["Final Report: カテゴリ別合計/集計値"]
E --> F
F --> G["stdout / Log File"]
group_byは直感的な記述が可能ですが、内部でソートを行うため計算量は $O(N \log N)$ となります。一方、reduceは1スキャン($O(N)$)で処理可能なため、大規模データのストリーム処理に適しています。
【実装:堅牢な自動化スクリプト】
以下は、ダミーのインベントリデータから「ステータスごとのカウント」と「コストの合計」を算出する実戦的なスクリプトです。
#!/usr/bin/env bash
# -----------------------------------------------------------------------------
# Script: aggregate_inventory.sh
# Description: jqのgroup_byとreduceを用いた高度なJSON集計
# -----------------------------------------------------------------------------
set -euo pipefail
IFS=$'\n\t'
# 一時ファイルのクリーンアップ設定
TMP_FILE=$(mktemp)
trap 'rm -f "$TMP_FILE"' EXIT
# --- 1. データ取得 (ダミーAPIレスポンスの生成) ---
cat << 'EOF' > "$TMP_FILE"
[
{"id": "v-001", "status": "running", "cost": 150, "region": "ap-northeast-1"},
{"id": "v-002", "status": "stopped", "cost": 10, "region": "us-east-1"},
{"id": "v-003", "status": "running", "cost": 200, "region": "ap-northeast-1"},
{"id": "v-004", "status": "running", "cost": 150, "region": "us-east-1"},
{"id": "v-005", "status": "terminated", "cost": 0, "region": "ap-northeast-1"}
]
EOF
echo "=== [1] group_by によるリージョン別集計 ==="
# group_by(.key) は指定キーで配列を分割し、[ [obj, obj], [obj] ] の形にする
jq -r '
group_by(.region) | map({
region: .[0].region,
count: length,
total_cost: map(.cost) | add
})
' "$TMP_FILE"
echo -e "\n=== [2] reduce によるステータス別の高度な集計 ==="
# reduce はメモリ効率が良く、初期値 ( {} ) に対して逐次更新を行う
jq -r '
reduce .[] as $item (
{};
.[$item.status].count += 1 |
.[$item.status].total_cost += $item.cost
) | to_entries | map({
status: .key,
metrics: .value
})
' "$TMP_FILE"
# 補足: 実戦では curl を以下のように利用
# curl -sfL --retry 3 --retry-delay 5 "https://api.example.com/v1/resources" \
# -H "Authorization: Bearer ${API_TOKEN}" | jq ...
systemd/Timer 設定例(定期集計用)
集計処理をバックグラウンドで定期実行する場合のユニット定義例です。
# /etc/systemd/system/inventory-aggregator.service
[Unit]
Description=Daily Cloud Cost Aggregator
[Service]
Type=oneshot
ExecStart=/usr/local/bin/aggregate_inventory.sh
EnvironmentFile=/etc/default/cloud-api-creds
User=reports-user
# /etc/systemd/system/inventory-aggregator.timer
[Timer]
OnCalendar=daily
Persistent=true
[Install]
WantedBy=timers.target
【検証と運用】
正常系の確認
スクリプトを実行し、以下のJSON構造が出力されることを確認します。
# 実行
./aggregate_inventory.sh
# 期待される出力(抜粋)
# [
# { "status": "running", "metrics": { "count": 3, "total_cost": 500 } },
# ...
# ]
ログ確認方法
systemd経由で実行している場合は、journalctl を使用して、パイプラインの途中でエラーが発生していないか確認します。
# ユニットの最新ログを確認
journalctl -u inventory-aggregator.service -n 50 --no-pager
【トラブルシューティングと落とし穴】
データ型の不一致: reduce内で加算を行う際、costが文字列として渡されるとエラーになります。($item.cost | tonumber) を使用して型を明示的に変換してください。
大規模データのメモリ消費: group_by は全データをメモリに展開してソートします。数GB単位のJSONを扱う場合は、jq -c '.[]' で各行を分離し、reduce または外部の awk 等と組み合わせるストリーム処理を検討してください。
環境変数の扱い: APIトークンなどを jq の引数に直接渡すと ps コマンドから見えてしまうため、必ず --arg オプション経由で渡すようにします(例: jq --arg key "$VAL" '.[$key]')。
【まとめ】
運用の冪等性と堅牢性を維持するための3つのポイント:
計算効率の選択: 直感的な group_by か、高速な reduce かをデータサイズに応じて使い分ける。
エラーハンドリング: set -euo pipefail を徹底し、パイプライン途中の jq の失敗を検知できるようにする。
型保証: tonumber や tostring を活用し、入力データがスキーマから外れている場合のクラッシュを未然に防ぐ。
コメント