ブロックチェーンとAIを組み合わせた役員動画の真贋証明システム

【Python×Solidity】役員動画の真贋を証明するブロックチェーン認証システム実装ハンズオン

約9分で読めます
文字サイズ:
【Python×Solidity】役員動画の真贋を証明するブロックチェーン認証システム実装ハンズオン
目次

この記事の要点

  • ディープフェイク動画による企業リスクへの対抗策
  • ブロックチェーンによる動画の改ざん防止と来歴証明
  • AIによる動画コンテンツの真贋高精度判定

生成AIの進化は目覚ましく、創造性を拡張する一方で、新たな社会的脅威も突きつけています。データ分析やマーケティング効果測定においても、扱うデータの信頼性は極めて重要です。特に企業ガバナンスの観点から看過できないのが、経営トップの「なりすまし動画(ディープフェイク)」による詐欺や風評被害のリスクです。

本記事では、概念論だけでなく、実際に手を動かしてコードを書くことで、ブロックチェーンがどのように動画データの「真正性(Authenticity)」を担保できるのか、そのメカニズムを解説します。データの信頼性を確保する技術的基盤を理解することは、正確な分析やUI/UX改善の第一歩となります。

イントロダクション:なぜ今、動画の「真正性」をコードで担保するのか

「百聞は一見に如かず」という言葉がありますが、現代のデジタル空間において、その前提は崩れ去ろうとしています。CEOが重大な発表を行う動画が、実はAIによって生成された偽物だったとしたら、株価操作や社会的信用の失墜は避けられません。

生成AIによる「偽のCEO動画」リスクの現実

従来のサイバー攻撃はシステムへの侵入が主でしたが、これからは「情報の信頼性」そのものが攻撃対象になります。特に役員の発言は市場への影響力が大きく、攻撃者にとって格好のターゲットです。

ここで重要なのが、動画データそのものが「改ざんされていない」ことを証明する技術的手段です。従来の中央集権的なサーバー管理では、管理者権限を持つ内部犯による改ざんや、サーバー自体のハッキングリスクを完全に排除することは困難でした。

従来の電子署名とブロックチェーン活用の決定的な違い

そこで注目されるのがブロックチェーンです。この技術の核心は「分散型台帳による耐改ざん性」にあります。一度ブロックチェーンに刻まれたデータは、ネットワーク参加者全員の合意なしには書き換えることができません。これは、デジタルデータに物理的な石碑のような「不変性(Immutability)」を与えることに等しいと言えます。

本チュートリアルのゴール:ハッシュ値を用いた簡易認証基盤の構築

今回は、以下のアーキテクチャを持つプロトタイプシステム「TrustVideo-PoC」を構築します。

  1. ハッシュ生成: Pythonを使って動画ファイルのデジタル指紋(ハッシュ値)を生成。
  2. オンチェーン記録: Solidityで書かれたスマートコントラクトを通じて、ハッシュ値をイーサリアム(テストネット)に記録。
  3. 検証プロセス: 検証用スクリプトで、手元の動画とブロックチェーン上の記録を照合。

技術的なブラックボックスをなくし、コードレベルで信頼の構造を理解していきましょう。

Step 1:開発環境のセットアップとツール選定

まずは、開発環境を整えます。Web3開発はツールの進化が早いため、安定したバージョンと構成を選ぶことが重要です。

Python環境とWeb3.pyのインストール

今回は汎用性の高いPythonを使用し、イーサリアム・ネットワークと対話するためのライブラリ web3.py を利用します。また、スマートコントラクト言語であるSolidityのコンパイルには py-solc-x を使います。

ターミナルを開き、プロジェクト用ディレクトリを作成して必要なライブラリをインストールしてください。

# プロジェクトディレクトリの作成
mkdir trust_video_poc
cd trust_video_poc

# 仮想環境の作成(推奨)
python -m venv venv
source venv/bin/activate  # Windowsの場合は venv\Scripts\activate

# ライブラリのインストール
pip install web3 py-solc-x

ローカルブロックチェーン「Ganache」の準備

開発中にメインネット(本番環境)を使うと、テストのたびに実際の暗号資産(Gas代)を消費してしまいます。そこで、ローカル環境で動作するパーソナル・ブロックチェーン「Ganache」を使用します。

Ganache公式サイトからGUI版をダウンロードしてインストールするか、Node.js環境がある方はCLIツールをインストールしてください。ここでは手軽なGUI版を想定します。

Ganacheを起動し、「Quickstart」を選択すると、テスト用の10個のアカウント(それぞれ100 ETH保有)と秘密鍵が生成されます。RPC Serverのアドレス(通常 http://127.0.0.1:7545)をメモしておいてください。

開発ディレクトリの構成

最終的に以下のようなファイル構成を目指します。

trust_video_poc/
├── contracts/
│   └── ProofOfOrigin.sol  # スマートコントラクト
├── scripts/
│   ├── deploy.py          # デプロイ用スクリプト
│   ├── hash_video.py      # 動画ハッシュ化スクリプト
│   └── verify.py          # 検証用スクリプト
└── assets/
    └── sample_ceo_msg.mp4 # テスト用動画ファイル(適当な動画を用意してください)

準備が整いましたら、信頼の基点となる「ハッシュ」の生成から始めます。

Step 2:動画データの指紋「ハッシュ値」を生成する

Step 1:開発環境のセットアップとツール選定 - Section Image

ブロックチェーンはデータの保存には向いていません。動画のような大容量データを直接チェーンに書き込むと、莫大なコスト(Gas代)がかかり、ネットワークを圧迫します。そこで登場するのが「ハッシュ関数」です。

動画ファイル自体をブロックチェーンに乗せない理由

本システムでは動画そのものではなく、動画から生成された固定長の文字列(ハッシュ値)のみを記録します。これを「アンカリング」と呼びます。動画ファイルが1バイトでも変更されれば、生成されるハッシュ値は全く異なるものになります。つまり、ハッシュ値さえ安全に保管されていれば、元の動画が改ざんされていないことを数学的に証明できるのです。

SHA-256アルゴリズムによる動画ファイルのハッシュ化

Pythonの標準ライブラリ hashlib を使って、SHA-256アルゴリズムで動画のハッシュを生成するスクリプト scripts/hash_video.py を作成します。

# scripts/hash_video.py
import hashlib

def generate_video_hash(file_path):
    """
    動画ファイルを読み込み、SHA-256ハッシュ値を生成して返す関数
    """
    sha256_hash = hashlib.sha256()
    
    try:
        # バイナリモードでファイルを開く
        with open(file_path, "rb") as f:
            # メモリ節約のため、少しずつ読み込んでハッシュを更新
            for byte_block in iter(lambda: f.read(4096), b""):
                sha256_hash.update(byte_block)
        
        return sha256_hash.hexdigest()
        
    except FileNotFoundError:
        print(f"エラー: ファイルが見つかりません - {file_path}")
        return None

if __name__ == "__main__":
    # テスト用動画ファイルのパスを指定
    video_path = "assets/sample_ceo_msg.mp4"
    
    # ダミーファイルがない場合は作成(テスト用)
    import os
    if not os.path.exists(video_path):
        os.makedirs("assets", exist_ok=True)
        with open(video_path, "w") as f:
            f.write("This is a test video content for CEO message.")
            
    video_hash = generate_video_hash(video_path)
    if video_hash:
        print(f"動画ファイル: {video_path}")
        print(f"生成されたハッシュ値: 0x{video_hash}")

実行と確認

このスクリプトを実行すると、0x で始まる64文字の16進数文字列が表示されます。これが、対象となる動画データの「デジタル指紋」です。この指紋をブロックチェーンという「永遠の台帳」に刻み込む準備ができました。

Step 3:真贋情報を刻むスマートコントラクトの実装

ここからがWeb3開発の本丸です。Solidityを使って、ハッシュ値を記録・管理するスマートコントラクト ProofOfOrigin を作成します。

Solidityによる「ProofOfOrigin」コントラクトの記述

contracts/ProofOfOrigin.sol を作成し、以下のコードを記述してください。シンプルですが、ガバナンスの観点から重要な機能を備えています。

// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;

contract ProofOfOrigin {
    // 動画ハッシュと登録者アドレスを紐付けるマッピング
    // key: 動画のハッシュ値 (bytes32)
    // value: 登録者のウォレットアドレス
    mapping(bytes32 => address) public videoRegistry;

    // 動画が登録されたタイムスタンプを記録
    mapping(bytes32 => uint256) public registrationTime;

    // イベント定義:登録時にログを残す
    event VideoRegistered(bytes32 indexed videoHash, address indexed registrant, uint256 timestamp);

    // 動画ハッシュを登録する関数
    function registerVideo(bytes32 _videoHash) public {
        // 既に登録されているかチェック(二重登録防止)
        require(videoRegistry[_videoHash] == address(0), "Error: Video already registered.");

        // 送信者のアドレスと現在時刻を記録
        videoRegistry[_videoHash] = msg.sender;
        registrationTime[_videoHash] = block.timestamp;

        // イベント発行
        emit VideoRegistered(_videoHash, msg.sender, block.timestamp);
    }

    // 検証用関数:ハッシュが登録済みか確認
    function verifyVideo(bytes32 _videoHash) public view returns (bool, address, uint256) {
        address registrant = videoRegistry[_videoHash];
        if (registrant == address(0)) {
            return (false, address(0), 0);
        }
        return (true, registrant, registrationTime[_videoHash]);
    }
}

コントラクトのコンパイルとローカルチェーンへのデプロイ

次に、Pythonを使ってこのコントラクトをコンパイルし、Ganache上にデプロイします。scripts/deploy.py を作成します。

注意: GanacheのRPC URL(http://127.0.0.1:7545)や秘密鍵は、ご自身の環境に合わせて変更してください。

# scripts/deploy.py
from web3 import Web3
from solcx import compile_standard, install_solc
import json
import os

# Solidityコンパイラのインストール(初回のみ)
install_solc('0.8.0')

# Ganacheへの接続
w3 = Web3(Web3.HTTPProvider("http://127.0.0.1:7545"))
chain_id = 1337  # GanacheのデフォルトChain ID

# デプロイ用アカウント(Ganacheの一番上のアカウント)
my_address = "0xYOUR_GANACHE_ADDRESS_HERE" # ← Ganacheのアドレスに書き換え
private_key = "YOUR_PRIVATE_KEY_HERE"      # ← Ganacheの秘密鍵に書き換え

def deploy_contract():
    # Solidityファイルの読み込み
    with open("contracts/ProofOfOrigin.sol", "r") as file:
        proof_of_origin_file = file.read()

    # コンパイル
    compiled_sol = compile_standard(
        {
            "language": "Solidity",
            "sources": {"ProofOfOrigin.sol": {"content": proof_of_origin_file}},
            "settings": {
                "outputSelection": {
                    "*": {"*": ["abi", "metadata", "evm.bytecode", "evm.sourceMap"]}
                }
            },
        },
        solc_version="0.8.0",
    )

    bytecode = compiled_sol["contracts"]["ProofOfOrigin.sol"]["ProofOfOrigin"]["evm"]["bytecode"]["object"]
    abi = compiled_sol["contracts"]["ProofOfOrigin.sol"]["ProofOfOrigin"]["abi"]

    # コントラクトオブジェクトの作成
    ProofOfOrigin = w3.eth.contract(abi=abi, bytecode=bytecode)

    # トランザクションの構築
    nonce = w3.eth.get_transaction_count(my_address)
    transaction = ProofOfOrigin.constructor().build_transaction({
        "chainId": chain_id,
        "gasPrice": w3.eth.gas_price,
        "from": my_address,
        "nonce": nonce,
    })

    # トランザクションの署名
    signed_txn = w3.eth.account.sign_transaction(transaction, private_key=private_key)

    # 送信と完了待ち
    print("コントラクトをデプロイ中...")
    tx_hash = w3.eth.send_raw_transaction(signed_txn.rawTransaction)
    tx_receipt = w3.eth.wait_for_transaction_receipt(tx_hash)

    print(f"デプロイ完了! コントラクトアドレス: {tx_receipt.contractAddress}")
    
    # アドレスとABIを保存(後のステップで使用)
    data = {"address": tx_receipt.contractAddress, "abi": abi}
    with open("contract_data.json", "w") as outfile:
        json.dump(data, outfile)

if __name__ == "__main__":
    deploy_contract()

このスクリプトを実行すると、Ganache上のブロックチェーンにスマートコントラクトが配置され、その住所(アドレス)が contract_data.json に保存されます。これで、システムは稼働状態になりました。

Step 4:検証システムの構築と動作テスト

Step 3:真贋情報を刻むスマートコントラクトの実装 - Section Image

最後に、ユーザーが動画をアップロードした際に、それが「本物」かどうかを判定する検証システムを作ります。これは、先ほどデプロイしたコントラクトに問い合わせを行うクライアントアプリの役割を果たします。

検証用Pythonスクリプトの実装

scripts/verify.py を作成し、動画ハッシュの生成からブロックチェーン照合までを一気通貫で行います。

# scripts/verify.py
from web3 import Web3
import json
import hashlib

# Ganache接続設定
w3 = Web3(Web3.HTTPProvider("http://127.0.0.1:7545"))
chain_id = 1337

# アカウント設定(登録用)
my_address = "0xYOUR_GANACHE_ADDRESS_HERE" # deploy.pyと同じもの
private_key = "YOUR_PRIVATE_KEY_HERE"

# デプロイ情報の読み込み
with open("contract_data.json", "r") as f:
    contract_data = json.load(f)
    contract_address = contract_data["address"]
    contract_abi = contract_data["abi"]

contract = w3.eth.contract(address=contract_address, abi=contract_abi)

def get_video_hash_bytes(file_path):
    sha256_hash = hashlib.sha256()
    try:
        with open(file_path, "rb") as f:
            for byte_block in iter(lambda: f.read(4096), b""):
                sha256_hash.update(byte_block)
        # Solidityのbytes32型に合わせるため、0x付きの16進数文字列をバイト列に変換
        return bytes.fromhex(sha256_hash.hexdigest())
    except FileNotFoundError:
        return None

def register_video(file_path):
    video_hash = get_video_hash_bytes(file_path)
    print(f"登録中... ハッシュ: {video_hash.hex()}")
    
    nonce = w3.eth.get_transaction_count(my_address)
    txn = contract.functions.registerVideo(video_hash).build_transaction({
        "chainId": chain_id,
        "gasPrice": w3.eth.gas_price,
        "from": my_address,
        "nonce": nonce,
    })
    signed_txn = w3.eth.account.sign_transaction(txn, private_key=private_key)
    tx_hash = w3.eth.send_raw_transaction(signed_txn.rawTransaction)
    w3.eth.wait_for_transaction_receipt(tx_hash)
    print("登録完了!")

def verify_video(file_path):
    print(f"検証中: {file_path}")
    video_hash = get_video_hash_bytes(file_path)
    
    # コントラクトのverifyVideo関数を呼び出し(callなのでガス代不要)
    is_registered, registrant, timestamp = contract.functions.verifyVideo(video_hash).call()
    
    if is_registered:
        print("✅ 認証成功: この動画はブロックチェーンに登録されています。")
        print(f"   - 登録者: {registrant}")
        print(f"   - 登録時刻: {timestamp}")
    else:
        print("❌ 認証失敗: この動画の記録は見つかりませんでした。改ざんの可能性があります。")

if __name__ == "__main__":
    video_path = "assets/sample_ceo_msg.mp4"
    
    # 1. まず動画をブロックチェーンに登録
    register_video(video_path)
    
    # 2. 同じ動画を検証(成功するはず)
    verify_video(video_path)
    
    # 3. 改ざんシミュレーション
    # 動画の内容を少し書き換えた偽ファイルを作成
    fake_video_path = "assets/fake_ceo_msg.mp4"
    with open(video_path, "rb") as f_orig:
        content = f_orig.read()
    
    # 末尾に1バイト追加するだけの微細な改ざん
    with open(fake_video_path, "wb") as f_fake:
        f_fake.write(content + b"0")
        
    # 4. 偽動画を検証(失敗するはず)
    print("\n--- 改ざんされた動画を検証 ---")
    verify_video(fake_video_path)

動作結果の確認

このスクリプトを実行すると、以下のような結果が得られるはずです。

  1. 正規の動画は「✅ 認証成功」となり、登録者のアドレスが表示されます。
  2. わずか1バイト改ざんしただけの動画は、ハッシュ値が全く異なるため「❌ 認証失敗」となります。

これが、コードによって担保された「真正性」の姿です。人間の目には区別がつかない微細な改ざんであっても、数学と暗号技術は決して見逃しません。

応用と課題:実運用に向けたセキュリティとUX

プロトタイプは完成しましたが、これを企業のガバナンスシステムとして実運用するには、いくつかの壁を乗り越える必要があります。

秘密鍵の管理問題とマルチシグの必要性

今回のコードでは秘密鍵をスクリプト内にベタ書きしていますが、本番環境では絶対にNGです。秘密鍵が流出すれば、攻撃者が「正規の署名者」になりすまして偽動画を登録できてしまいます。

解決策として、マルチシグ(Multi-Signature)の導入が必須です。例えば、「CEO、CTO、広報部長のうち2名の署名がないと動画を登録できない」というルールをスマートコントラクトに記述することで、単一障害点を排除し、ガバナンスを強化できます。

動画撮影時の「Origin(発生源)」証明の難しさ

ブロックチェーンは「記録された後の改ざん」は防げますが、「記録する前のデータが本物か」までは関知しません(Oracle問題)。もし、撮影段階でディープフェイクが作られ、それを担当者が正規の手順で登録してしまったらどうなるでしょうか。

ここには、C2PA(Coalition for Content Provenance and Authenticity)のような技術規格との連携が必要です。カメラハードウェア自体が撮影時に署名を付与し、その署名データを含めてハッシュ化・ブロックチェーン登録を行うことで、「撮影から公開まで」の完全なChain of Custody(管理の連鎖)を構築することが、次世代のスタンダードになるでしょう。

まとめ:信頼は技術と運用の融合から生まれる

デプロイ用アカウント(Ganacheの一番上のアカウント) - Section Image 3

今回構築したシステムは、数十行のコードですが、デジタル社会におけるデータの信頼性を守るための強力な防壁の第一歩です。

  • 不変性: ブロックチェーンに記録されたハッシュは、何人たりとも改ざんできない。
  • 透明性: 誰がいつ登録したかが、パブリックに検証可能である。
  • 検証可能性: 誰でも手元のファイルと照合し、真贋を判定できる。

しかし、技術はあくまでツールです。真に強固なガバナンスを実現するには、このシステムを運用する組織のルール設計や、倫理観が不可欠です。データ分析やマーケティング施策においても、基盤となるデータの信頼性が確保されて初めて、正確な効果測定やUI/UXの改善が可能になります。技術と人が調和した運用体制を築くことが、データドリブンな意思決定を支える鍵となります。

より実践的なブロックチェーン導入事例やセキュリティ体制の構築については、専門的な知見を参考にしながら、組織への定着までを含めたロードマップを検討することをおすすめします。

【Python×Solidity】役員動画の真贋を証明するブロックチェーン認証システム実装ハンズオン - Conclusion Image

コメント

コメントは1週間で消えます
コメントを読み込み中...