フューチャー技術ブログ

API費用8ドル。「ホリエモンAI選挙」の分析フローをDifyで一部再現してみた ‐ マルチAIエージェント実装ガイド

はじめに

最近、「マルチAIエージェント」という言葉をよく耳にするようになりました。

単一のLLMですべてを解決するのではなく、検索が得意なAI、分析が得意なAIなどを適材適所で組み合わせる手法です。その好例として話題になったのが、2026年の第51回衆議院議員選挙に合わせて公開された「ホリエモンAI選挙」です。

このAI選挙予測サイトを見て、「面白そう」と思ったエンジニアや「ウチのビジネスでも活かせそう」と直感したビジネスパーソンは少なくないはずです。私が凄いなと思った点は、「予測の仕組み(AIエージェントの組み合わせ、プロンプト)」が公開されていることです。つまり、各サービスの契約さえあれば、誰でも同じロジックを手動で試すことができます。具体的には、以下のような役割分担です。

  • Perplexity で最新の世論調査、報道データを検索させて
  • Grok でX上の投稿を検索してSNSトレンドを分析させて
  • Gemini でそれらの情報を統合・構造化してもらう

このフローは手動で実行可能ですが、毎回3つのサービスを開いてプロンプトをコピー&ペーストし、結果を貼り合わせる単調作業を繰り返すのは「プログラマーの三大美徳」(怠惰・短気・傲慢)に反します。

そこで、この記事では、Dify Cloud と 3つのAI Agent API(Perplexity / Grok / Gemini)を組み合わせて、ボタン1つで選挙情勢分析レポートを自動生成するワークフローを構築する方法 を解説します。

「そんな複雑なシステム、個人で検証したら高いんじゃないの?」と思われがちですが、今回の検証(分析フローの一部再現)にかかったAPI費用は、たったの8ドル(約1,200円) です。本記事では、用意するソースコードはPythonが数行程度で、Dify Cloud上で動かせる(ローカル環境不要)手順に加え、「各サービスのAPIキーをどこで取るのか」「Difyのどの画面で何を設定するのか」を解説します。

本記事は「ホリエモンAI選挙」の仕組みを参考にDifyでワークフローを構築する技術解説記事であり、同サービスを完全に再現するものではありません。以下の点にご留意ください。

  • 「ホリエモンAI選挙」では「全体情勢分析」「都道府県別分析」「比例ブロック分析」の3種類の分析が行われていますが、本記事では 「全体情勢分析」のみ を対象としています
  • 本家で投入されている事前データ(選挙区の区割り情報、立候補者情報など)の投入は行っていません
  • プロンプトの一部について、公開情報から推測して埋め合わせた箇所があります

完成イメージ

まず、最終的に何ができるのかを先にお見せします。

ワークフローの全体構成は以下の通りです。

graph LR
    %% ノード定義
    Start((スタート))
    DateGen[検索日作成
Code] Perplexity[Perplexity
Web検索] Grok[Grok
SNS分析] Gemini[Gemini
データ統合] TableGen[テーブル化
Code] Result(((集計結果
End))) Error(((エラー
End))) Start --> DateGen DateGen --> Perplexity DateGen --> Grok Perplexity --> Gemini Grok --> Gemini Gemini --> TableGen TableGen --> Result Perplexity -.->|Fail Branch| Error Grok -.->|Fail Branch| Error Gemini -.->|Fail Branch| Error TableGen -.->|Fail Branch| Error %% スタイル定義 classDef default stroke:#333,stroke-width:2px; style Start fill:#155EEF,stroke:#155EEF,color:#fff style DateGen fill:#F04438,stroke:#F04438,color:#fff style Perplexity fill:#444CE7,stroke:#444CE7,color:#fff style Grok fill:#A020F0,stroke:#A020F0,color:#fff style Gemini fill:#155EEF,stroke:#155EEF,color:#fff style TableGen fill:#F04438,stroke:#F04438,color:#fff style Result fill:#333,stroke:#333,color:#fff style Error fill:#D92D20,stroke:#D92D20,color:#fff

Dify上で「実行」ボタンを押すと、以下のフローが自動で処理されます。

  1. PerplexityがWeb上の世論調査・ニュースを検索して分析
  2. GrokがX上の投稿を分析してSNS世論を分析
  3. 上記2つの結果をGeminiが統合、構造化されたJSONを出力
  4. PythonコードがJSONをMarkdownテーブルに整形

最終出力には、以下のように、内閣支持率、各政党それぞれの支持率・トレンド・分析コメント、データソースリストが含まれます。

(出力フォーマット以外の最終的な出力は黒塗りさせていただきました。皆さんも手元で試してみて、どんな結果が得られるのかを検証してみて下さい!)

スクリーンショット_2026-02-09_1.53.47.png

検証に要した費用

今回のワークフローには、Dify・Perplexity・Grok・Gemini を利用しています。動作検証にあたり、初回費用枠や最低購入額等を利用することで、8ドル(1,200円 = 150円×8ドル)となりました。(※為替レートは執筆時点の概算)

サービス名 検証に要した費用 備考
Dify Cloud $0 今回のワークフローは「無料枠」で実現可能
Perplexity $3 APIクレジットの最低購入額が $3 のため
Grok(xAI) $5 APIクレジットの最低購入額が $5 のため
Gemini(Google AI Studio) $0 無料トライアル分の「91日間可能な $300 のウェルカムクレジット」を利用した

事前準備1:Perplexity API

APIキーの取得手順

  1. Perplexity にアクセスして、アカウントを作成します
  2. ログイン後、左下のアイコン → 「API」を選択します
  3. APIグループ名を入力して「保存」を選択します
スクリーンショット_2026-02-09_2.14.00.png
  1. 「API請求」に移動して、支払い方法(クレジットカード)を登録します
スクリーンショット_2026-02-09_2.15.41.png
  1. 「クレジットをもっと購入」をクリックして、最低購入額の $3 を購入します
スクリーンショット_2026-02-09_2.19.08.png
  1. 「APIキー」に移動後、「APIキーを生成する」をクリックしてキーを発行します
スクリーンショット_2026-02-08_19.12.31.png
  1. 発行されたキーを控えておきます(後でDifyに登録します)

事前準備2:xAI(Grok)API

APIキーの取得手順

  1. xAI Console にアクセスし、アカウントを作成します

  2. ダッシュボードから「API Keys」に移動します

  3. 「Create API Key」でキーを発行します

    設定パラメータ

    カテゴリ 設定項目
    Restrict access Models grok-4-1-fast-reasoning
    - Endpoints Chat
    Rate limits TPM 50000
    - RPM 20
スクリーンショット_2026-02-08_19.26.37.png
  1. 発行されたキーを控えておきます
スクリーンショット_2026-02-08_19.27.54.png
  1. Billingから「Purchase credits」を選択します
スクリーンショット_2026-02-09_5.00.31.png
  1. 最低購入額の $5 を入力し、クレジットカードを登録して購入します
スクリーンショット_2026-02-09_5.01.14.png

事前準備3:Google AI Studio(Gemini API)

APIキーの取得手順

  1. Google AI Studio にアクセスし、右上の Get started からGoogleアカウントでログインします
スクリーンショット_2026-02-08_19.37.08.png
  1. 左下の「Get API Key」を選択し、続いて右上の「APIキーを作成」をクリックします
スクリーンショット_2026-02-08_19.38.50.png
  1. 「プロジェクトを作成」から新しいプロジェクトを作成します
スクリーンショット_2026-02-08_19.43.06.png スクリーンショット_2026-02-08_19.43.17.png
  1. 「Copy API key」で取得したキーを控えておきます
スクリーンショット_2026-02-08_19.43.40.png
  1. 続いて、「お支払い情報を設定」をクリックします
スクリーンショット_2026-02-08_21.29.42.png
  1. $300分の無料クレジット自動請求なしが表示されていることを確認してから「同意して続行」を選択します
スクリーンショット_2026-02-08_21.30.42.png
  1. 「連絡先情報」と「お支払い方法(無料枠の場合も必須)」を入力して「無料で利用開始」を選択します
スクリーンショット_2026-02-08_21.32.29.png
  1. Google AI Studioのコンソール画面に戻り、割り当てティアが「お支払い情報を有効にする」と表示されていれば設定完了です

    注意点: 「お支払い情報を有効にする」をクリックすると以下画面に遷移しますが、課金が始まる「有効化」は不要です

スクリーンショット_2026-02-08_21.44.09.png

Dify Cloud のセットアップ

アカウント作成

Dify Cloud にアクセスして、アカウントを作成します。

スクリーンショット_2026-02-08_21.48.38.png

プラグインのインストール

Dify Cloudでは、外部LLMとの接続に「プラグイン」を使用します。今回必要なプラグインは以下の2つです。

プラグイン 用途
langgenius/perplexity PerplexityをToolノードとして使用するため
langgenius/gemini GeminiをLLMノードとして使用するため

補足: xAI(Grok)のlanggenius/x は、本ブログ執筆時点(2026.02.08)では x_search(X/Twitterを検索するオプション機能) に対応していないため、今回は利用しません。

インストール手順:

  1. Dify画面右上の「プラグイン」からメニューを開きます
  2. 「プラグインをインストールする」から「マーケットプレイス」を選択します
  3. PerplexityとGeminiを検索して、インストールします
スクリーンショット_2026-02-08_21.58.33.png
  1. 右上のアイコンをクリックして「設定」に移動して、「モデルプロバイダー」を表示します
  2. Geminiの「セットアップ」をクリックして、API Keyに Google AI Studioで発行したAPIキー を入力して保存します
スクリーンショット_2026-02-08_22.04.35.png スクリーンショット_2026-02-08_22.05.22.png
  1. トークン名の右側に緑色のランプが点灯していれば、Geminiの設定は完了です
スクリーンショット_2026-02-08_22.06.21.png

ワークフローの準備

  1. 画面上部の「スタジオ」に移動して「最初から作成」を選択します
スクリーンショット_2026-02-08_22.16.10.png
  1. 「ワークフロー」を選択して、名前と説明文を入力したら「作成する」を選択します
スクリーンショット_2026-02-08_22.16.48.png
  1. 開始ノードには「ユーザ入力」を利用します
スクリーンショット_2026-02-08_22.18.13.png

環境変数の設定

ワークフローの編集画面で、環境変数を2つ設定します。

変数名 用途
xai_api_key Secret GrokのAPIトークン
CHUDOU_DESCRIPTION String プロンプト内で参照する政党情報(後述)

xai_api_keyをSecret型にすることで、Dify上のUIでマスク表示されます。もう一つのCHUDOU_DESCRIPTIONは政党の説明文を格納する変数で、プロンプト内から{{#env.CHUDOU_DESCRIPTION#}}で参照します。

  1. 右上の ENV をクリックします
  2. 「環境変数を追加」のポップアップに適切な「タイプ・変数名・値」を入力します
    • xai_api_key の値には、xAIで取得したAPIキーを入力します
    • CHUDOU_DESCRIPTION には政党の説明文を入力します。今回、登録する説明文はAIに生成させました
スクリーンショット_2026-02-08_22.23.26.png
  1. 環境変数を登録すると、以下のように表示されます。(CHUDOU_DESCRIPTIONの説明文はAIに生成させてみてください)
スクリーンショット_2026-02-08_22.25.24.png

ワークフロー構築

今回のワークフローは、最終的に以下のような形になります。

スクリーンショット_2026-02-09_4.49.32.png

ここからは、Difyのワークフローエディタ上で構築する各ノードを順番に解説します。

スタートノード

入力変数は何も設定しません。今回のワークフローは「完全手動実行」です。画面右上の「テスト実行」ボタンを押すことで全体のワークフローが動きます。本ブログでは、動作確認のしやすさから「手動実行」としていますが、スケジュールノードに変更すれば「定期自動実行」への切り替えも可能です。

検索日作成(Codeノード / Python)

Grokのx_searchや後続のノードに渡す日付範囲を動的生成するためのノードです。

ノードの追加

スクリーンショット_2026-02-08_22.35.13.png

Pythonの処理には以下コードを登録します。

from datetime import datetime, timedelta, timezone

def main() -> dict:
now = datetime.now(timezone.utc)

from_dt = now - timedelta(hours=24)

to_date = now.strftime("%Y-%m-%d")
from_date = from_dt.strftime("%Y-%m-%d")

return {
"from_date": from_date,
"to_date": to_date,
}

出力はfrom_dateto_dateの2つの文字列です。UTCベースで生成しているのは、Dify Cloudのサーバのタイムゾーンに依存しないようにするためです。

  • 入力変数
    • (なし)
  • 出力変数
    • from_date: String
    • to_date: String

ノードの完成形

スクリーンショット_2026-02-08_22.38.08.png

Grok(HTTP Requestノード)

Difyの xAI(Grok) プラグインには langgenius/x があります。ただし、このプラグインでは、X/Twitterを検索する x_search が本ブログ執筆時点(2026.02.08)では利用できないため、今回は HTTP Requestノードで直接APIを叩きます。

ノードの追加

スクリーンショット_2026-02-08_22.39.47.png

認証設定

「認証なし」と表示されているボタンを選択して、環境変数に登録したGrokのAPIキーを登録します

  • 認証タイプ: APIキー
  • API認証タイプ: Bearer
  • APIキー: {{#env.xai_api_key#}}(「環境変数」として登録した値)
スクリーンショット_2026-02-08_22.44.05.png

パラメータ設定内容

カテゴリ 設定項目
API メソッド POST
- URL https://api.x.ai/v1/responses
ヘッダー キー Content-Type
- application/json
タイムアウト設定 接続タイムアウト 10秒
- 読み取りタイムアウト 240秒
- 書き込みタイムアウト 30秒
  • 失敗時再試行
    • 最大試行回数: 3回
    • 再試行間隔: 1000ミリ秒
  • 例外処理
    • 例外分岐

リクエストボディ(JSON)

{
"model": "grok-4-1-fast-reasoning",
"input": [
{
"role": "user",
"content": "【日付】{{#検索日作成.to_date#}}\n\nX上での2026年衆院選に関するSNS動向を分析してください。..."
}
],
"tools": [
{
"type": "x_search",
"from_date": "{{#検索日作成.from_date#}}",
"to_date": "{{#検索日作成.to_date#}}"
}
],
"temperature": 0.7,
"top_p": 0.95,
"max_output_tokens": 30000
}

tools配列にx_searchを指定するのがポイントです。これにより、GrokがX上の投稿を実際に検索した上で回答を生成します。from_dateto_dateは前段のCodeノードの出力を変数展開しています。

プロンプトは「ホリエモンAI選挙 予測の仕組みを公開」で公開された文面を利用しています。

ノードの完成形

スクリーンショット_2026-02-08_22.59.16.png

Perplexity(Toolノード)

DifyのPerplexityプラグインを「ツール」ノードとして配置します。

ノードの追加

スクリーンショット_2026-02-08_22.57.42.png

認証設定

  1. 「APIキー認証設定」をクリックする
スクリーンショット_2026-02-08_23.01.17.png
  1. Perplexity APIで発行したAPIキーを「Perplexity API key」に入力して保存する
スクリーンショット_2026-02-08_23.02.14.png

「Query」に登録するプロンプト

ホリエモンAI選挙 予測の仕組みを公開」のプロンプトに記載された以下の変数に対して、本ブログでは以下の対応を行なっています。

  • {TODAY}: 「検索日作成ノード」で作成した値の to_date を代入
  • {CHUDOU_DESCRIPTION}: 環境変数の {CHUDOU_DESCRIPTION} を代入
  • {RECENT_PARTIES_DETAIL_BLOCK}: プロンプトから削除

パラメータ設定内容

パラメータ 設定値
Model Name sonar-pro
Max Tokens 20000
Temperature 0.6
Top K 40
Top P 0.9
Presence Penalty 0
Frequency Penalty 0.4
Return Images False
Return Related Questions False
Search Recency Filter Week
Search Context Size High
  • 失敗時再試行
    • 最大試行回数: 3回
    • 再試行間隔: 1000ミリ秒
  • 例外処理
    • 例外分岐

ノードの完成形

スクリーンショット_2026-02-08_23.17.00.png

Gemini(LLMノード)

PerplexityとGrokの出力を受け取り、構造化されたJSONとして統合するGeminiノードを追加します。

ノードの追加

スクリーンショット_2026-02-08_23.22.19.png

LLMの設定

  • モデル: gemini-3-pro-preview
  • Temperature: 0.3(構造化出力の安定性を重視)
スクリーンショット_2026-02-08_23.26.43.png

プロンプト

プロンプトの中でPerplexityとGrokの出力結果を明示的に渡します。

...
【ニュース・調査データ(Perplexity)】
-> Perplexity / {x} text

【X/Twitter世論分析(Grok)】
-> Grok / {x} body
...
スクリーンショット_2026-02-08_23.31.40.png

構造化出力(Structured Output)の活用

Difyの「構造化出力」機能を有効にし、JSONスキーマを定義しています。これにより、Geminiの出力を指定したスキーマに従わせます。ただし、このスキーマを指定したとしても、本来ならば数字が入るべきところに文字列が入ってしまうケースなどが見られたので、出力の完全保証とはなりません。

「出力変数」の structured_output から、出力スキーマを指定します。

インポートするJSON全文(クリックで展開)
{
"type": "object",
"properties": {
"analysis_date": {
"type": "string"
},
"cabinet_approval": {
"type": "object",
"properties": {
"rate": {
"type": "number"
},
"trend": {
"type": "string"
},
"source": {
"type": "string"
},
"confidence": {
"type": "number"
},
"analysis": {
"type": "string"
}
},
"required": [
"rate",
"trend",
"source",
"confidence",
"analysis"
],
"additionalProperties": false
},
"party_momentum": {
"type": "object",
"properties": {
"ldp": {
"type": "object",
"properties": {
"support_rate": {
"type": "number"
},
"trend": {
"type": "string"
},
"analysis": {
"type": "string"
}
},
"required": [
"support_rate",
"trend",
"analysis"
],
"additionalProperties": false
},
"chudou": {
"type": "object",
"properties": {
"support_rate": {
"type": "number"
},
"trend": {
"type": "string"
},
"analysis": {
"type": "string"
}
},
"required": [
"support_rate",
"trend",
"analysis"
],
"additionalProperties": false
},
"ishin": {
"type": "object",
"properties": {
"support_rate": {
"type": "number"
},
"trend": {
"type": "string"
},
"analysis": {
"type": "string"
}
},
"required": [
"support_rate",
"trend",
"analysis"
],
"additionalProperties": false
},
"dpfp": {
"type": "object",
"properties": {
"support_rate": {
"type": "number"
},
"trend": {
"type": "string"
},
"analysis": {
"type": "string"
}
},
"required": [
"support_rate",
"trend",
"analysis"
],
"additionalProperties": false
},
"jcp": {
"type": "object",
"properties": {
"support_rate": {
"type": "number"
},
"trend": {
"type": "string"
},
"analysis": {
"type": "string"
}
},
"required": [
"support_rate",
"trend",
"analysis"
],
"additionalProperties": false
},
"reiwa": {
"type": "object",
"properties": {
"support_rate": {
"type": "number"
},
"trend": {
"type": "string"
},
"analysis": {
"type": "string"
}
},
"required": [
"support_rate",
"trend",
"analysis"
],
"additionalProperties": false
},
"sansei": {
"type": "object",
"properties": {
"support_rate": {
"type": "number"
},
"trend": {
"type": "string"
},
"analysis": {
"type": "string"
}
},
"required": [
"support_rate",
"trend",
"analysis"
],
"additionalProperties": false
},
"team_mirai": {
"type": "object",
"properties": {
"support_rate": {
"type": "number"
},
"trend": {
"type": "string"
},
"analysis": {
"type": "string"
}
},
"required": [
"support_rate",
"trend",
"analysis"
],
"additionalProperties": false
},
"hoshu": {
"type": "object",
"properties": {
"support_rate": {
"type": "number"
},
"trend": {
"type": "string"
},
"analysis": {
"type": "string"
}
},
"required": [
"support_rate",
"trend",
"analysis"
],
"additionalProperties": false
},
"genzei_yukoku": {
"type": "object",
"properties": {
"support_rate": {
"type": "number"
},
"trend": {
"type": "string"
},
"analysis": {
"type": "string"
}
},
"required": [
"support_rate",
"trend",
"analysis"
],
"additionalProperties": false
},
"other": {
"type": "object",
"properties": {
"support_rate": {
"type": "number"
},
"trend": {
"type": "string"
},
"analysis": {
"type": "string"
}
},
"required": [
"support_rate",
"trend",
"analysis"
],
"additionalProperties": false
}
},
"required": [
"ldp",
"chudou",
"ishin",
"dpfp",
"jcp",
"reiwa",
"sansei",
"team_mirai",
"hoshu",
"genzei_yukoku",
"other"
],
"additionalProperties": false
},
"key_issues": {
"type": "array",
"items": {
"type": "string"
}
},
"key_issues_confidence": {
"type": "number"
},
"regional_trends": {
"type": "object",
"properties": {
"urban": {
"type": "string"
},
"urban_summary": {
"type": "string"
},
"urban_confidence": {
"type": "number"
},
"rural": {
"type": "string"
},
"rural_summary": {
"type": "string"
},
"rural_confidence": {
"type": "number"
}
},
"required": [
"urban",
"urban_summary",
"urban_confidence",
"rural",
"rural_summary",
"rural_confidence"
],
"additionalProperties": false
},
"sources": {
"type": "array",
"items": {
"type": "string"
}
}
},
"required": [
"analysis_date",
"cabinet_approval",
"party_momentum",
"key_issues",
"key_issues_confidence",
"regional_trends",
"sources"
],
"additionalProperties": false
}
スクリーンショット_2026-02-09_1.35.12.png

スキーマの主要構成:

{
"analysis_date": "分析日",
"cabinet_approval": {
"rate": "支持率(数値)",
"trend": "rising | stable | declining",
"source": "情報源",
"confidence": "信頼度(0-100)",
"analysis": "分析コメント"
},
"party_momentum": {
"ldp": { "support_rate": "...", "trend": "...", "analysis": "..." },
"chudou": { ... },
// ... 各政党
},
"key_issues": ["争点1", "争点2", "争点3"],
"regional_trends": {
"urban": "都市部の傾向",
"rural": "地方の傾向"
},
"sources": ["ソース1", "ソース2", ...]
}
  • 失敗時再試行
    • 最大試行回数: 3回
    • 再試行間隔: 1000ミリ秒
  • 例外処理
    • 例外分岐

ノードの完成形

スクリーンショット_2026-02-08_23.38.11.png

テーブル化(Codeノード / Python)

GeminiのJSON出力を、人間が読みやすいMarkdownテーブルに変換するノードです。

  • 入力変数
    • gemini: Gemini/{x} text String
  • 出力変数
    • markdown_table: String
テーブル化のPythonコード全文(クリックで展開)
import json

def main(gemini: str) -> dict:
try:
data = json.loads(gemini)
except json.JSONDecodeError:
return {"markdown_table": "エラー: 有効なJSONデータではありません。"}
except Exception as e:
return {"markdown_table": f"エラーが発生しました: {str(e)}"}

date = data.get("analysis_date", "")
cab = data.get("cabinet_approval", {})

trend_map = {
"rising": "上昇 ↗",
"declining": "下落 ↘",
"stable": "横ばい →"
}

cab_trend = trend_map.get(cab.get("trend"), cab.get("trend"))

output = []
output.append(f"### 分析日: {date}")
output.append("")
output.append("### 1. 内閣支持率・重要課題")
output.append("| 項目 | 数値/ステータス | 分析・詳細 |")
output.append("|---|---|---|")
output.append(f"| **内閣支持率** | {cab.get('rate', '-')}% ({cab_trend}) | {cab.get('analysis', '').replace('\n', '')} |")

issues = data.get("key_issues", [])
issues_str = "、".join(issues)
output.append(f"| **重要課題** | 信頼度: {data.get('key_issues_confidence')}% | {issues_str} |")
output.append("")

parties = data.get("party_momentum", {})

name_map = {
"ldp": "自民党",
"chudou": "中道改革連合",
"ishin": "日本維新の会",
"dpfp": "国民民主党",
"jcp": "日本共産党",
"reiwa": "れいわ新選組",
"sansei": "参政党",
"team_mirai": "チームみらい",
"hoshu": "日本保守党",
"genzei_yukoku": "減税日本・ゆうこく連合",
"other": "その他"
}

output.append("### 2. 政党別情勢")
output.append("| 政党 | 支持率 | トレンド | 分析 |")
output.append("|---|---|---|---|")

for key, info in parties.items():
name = name_map.get(key, key)
rate = f"{info.get('support_rate', '')}%"
trend_val = info.get("trend", "")
trend = trend_map.get(trend_val, trend_val)
analysis = info.get("analysis", "").replace("\n", " ")

output.append(f"| {name} | {rate} | {trend} | {analysis} |")

output.append("")

reg = data.get("regional_trends", {})

output.append("### 3. 地域別情勢")
output.append("| 地域 | 概況 | 詳細 | 信頼度 |")
output.append("|---|---|---|---|")
output.append(f"| **都市部** | {reg.get('urban_summary', '')} | {reg.get('urban', '')} | {reg.get('urban_confidence', '')}% |")
output.append(f"| **地方** | {reg.get('rural_summary', '')} | {reg.get('rural', '')} | {reg.get('rural_confidence', '')}% |")
output.append("")

sources = data.get("sources", [])
if sources:
output.append("---")
output.append("**データソース:**")
for src in sources:
output.append(f"- {src}")

final_markdown = "\n".join(output)

return {
"markdown_table": final_markdown
}

主な処理:

  • JSONパース処理
  • トレンド値の日本語変換
    • rising上昇 ↗
    • declining下落 ↘
    • stable横ばい →
  • 政党キーの日本語マッピング
    • ldp自民党
    • chudou中道改革連合
    • etc.
  • 各セクション(内閣支持率、政党別情勢、地域別情勢、データソース)をMarkdown表に整形
  • JSONパースに失敗した場合のエラーメッセージ出力

出力は1つのmarkdown_table文字列です。この文字列がワークフローの最終出力としてEndノードに渡されます。

ノードの完成形

スクリーンショット_2026-02-08_23.48.04.png

集計結果(出力ノード)

「テーブル化」された集計結果を表示します。

ノードの追加

スクリーンショット_2026-02-08_23.48.46.png

出力変数:

変数名 内容
markdown_table テーブル化 / {x} markdown_table String テーブル形式に整形されたGeminiの集計結果

ノードの完成形

スクリーンショット_2026-02-09_1.43.51.png

エラーハンドリング(出力ノード)

各ノード(Perplexity、Grok、Gemini、テーブル化)にはfail-branch(失敗時分岐)を設定しています。いずれかのノードが失敗した場合、専用のエラーEndノードに遷移し、どのノードで失敗したかを個別の変数として表示します。

スクリーンショット_2026-02-08_23.58.51.png

出力変数:

変数名 内容
perplexity_error Perplexity / {x} error_message String Perplexityノードのエラーメッセージ
grok_error Grok / {x} error_message String Grokノードのエラーメッセージ
gemini_error Gemini / {x} error_message String Geminiノードのエラーメッセージ
table_error テーブル化 / {x} error_message String テーブル化ノードのエラーメッセージ

これにより、外部APIや内部処理でエラーが起こった場合に、どれが落ちたのかを特定できます。

ノードの完成形

スクリーンショット_2026-02-09_1.44.31.png

ワークフローの実行

すべてのノードの配置と設定が完了したら、いよいよワークフローを実行します。

  1. 右上の「テスト実行」をクリックする
  2. ワークフローの実行完了を待つ
スクリーンショット_2026-02-09_3.41.38.png
  1. Test Run の「結果」で集計結果を確認する
スクリーンショット_2026-02-09_3.43.52.png

おわりに

本記事では、「ホリエモンAI選挙」の仕組みを参考に、Dify Cloud上でPerplexity・Grok・Geminiを組み合わせた選挙情勢分析ワークフローを構築しました。

今回改めて感じたのは、ホリエモンAI選挙が 「予測の仕組み」を公開している ことの価値です。「Web検索 + SNS検索 + データ統合・構造化」というアーキテクチャだけでなく、具体的にどのAPIを使い、どんなプロンプトを投げているのかまで公開されていたことで、本記事のような再現検証が可能になりました。この「情報収集→分析→構造化」という役割分担のパターンは、選挙分析に限らず、市場調査やブランドモニタリングなど、幅広い領域に応用できるのではないかと感じています。

また、普段の私はインフラエンジニアとして活動していますが、今回はDify Cloudを利用したことで、インフラの構築やメンテナンスについてほとんど考える必要がありませんでした。ワークフローの設計とプロンプトの調整に集中でき、使い方さえ分かっていれば、アイデアを形にするまでの時間が大幅に短縮されることを実感しました。

なお、今回の実装にあたっては、各種調査にPerplexityを多用しました。達成したい目的が明確な状況では、各情報に引用元のリンクが付いた状態で回答が得られるため、情報の裏取りがしやすく非常に便利でした。

一方で課題もあります。実行するたびに支持率などの数値が大きく変動し、出力が安定しない事象が見られました。出力の安定性を高めるには、少なくとも以下の4点の調整が必要だと考えています。

  • 質の高い事前投入データ: 本家で投入されている選挙区の区割り情報や立候補者情報を、今回は一切投入していない。正確なベースデータがなければ、AIの出力はその都度の検索結果に左右されやすくなる
  • 途中でのデータ加工処理: 各AIの出力をそのまま次のノードに渡すのではなく、中間で不要な情報の除去やフォーマットの正規化を行うことで、後段のAIへの入力品質を上げられる
  • 出力のバリデーション処理: Geminiの構造化出力後に、支持率の合計値が妥当な範囲か、数値フィールドに文字列が混入していないか等をCodeノードで検証し、異常値があれば再実行やフォールバックを行う仕組みを入れる
  • LLMに与えるパラメータの最適化: Temperature等のパラメータを用途に応じて細かくチューニングすることで、出力のばらつきを抑えられる

これらは今後の改善ポイントです。