SFTPのロックダウン:LinuxにおけるChroot Jail構築の実践ガイド

Linux tutorial - IT technology blog
Linux tutorial - IT technology blog

ファイル転送のセキュリティ:なぜSFTP Chroot Jailが必要なのか

外部業者やジュニア開発者にサーバーアクセスを許可することは、予備の部屋を塗装してもらうためだけに自宅の鍵を渡すようなものです。標準のSSHアクセスは権限が強すぎます。デフォルトでは、新規ユーザーは /etc を閲覧したり、 /var/log/auth.log を覗き見したり、内部ネットワークサービスをスキャンしたりできてしまいます。このような「ラテラルムーブメント(横展開)」は、直ちに対処すべき重大なセキュリティリスクです。

私は、高トラフィックなメディアエージェンシーの14台の本番ノードにこの特定のChroot Jail構成をデプロイしました。6ヶ月間で5万件以上の自動アップロードを処理しましたが、セキュリティ侵害やパーミッションエラーは一度も発生していません。ロジックは単純です。ユーザーを「Jail(監獄)」ディレクトリに閉じ込めるのです。ユーザーからは、そのフォルダがシステム全体のルートのように見えます。ファイル移動に必要なツールは提供されますが、割り当てられたスペース以外のデータは1バイトたりとも見ることはできません。

3年以上にわたりLinuxフリートを管理してきた経験から、 sshd_config のわずかなタイプミスが原因で、管理者が自分のサーバーからロックアウトされる場面を何度も見てきました。設定を変更する際は、必ず予備のSSHセッションを別に開いておいてください。万が一問題が発生しても、接続中のセッションがあれば設定を修正できます。

ステップ1:環境の準備

Ubuntu 22.04、Debian 12、AlmaLinux 9のいずれを使用している場合でも、プロセスはグループ管理から始まります。ユーザーを個別に管理すると設定の不整合(設定ドリフト)を招きます。代わりに、専用のグループを作成しましょう。これにより、作成するすべてのSFTPユーザーに単一のセキュリティルールセットを適用できます。

# SFTPユーザー専用のグループを作成
sudo groupadd sftp_users

次に、 upload_user という名前のユーザーを追加します。意図的にシェルアクセスを剥奪します。シェルを /bin/false に設定することで、他の制限をバイパスされたとしても、対話的なSSHセッションを絶対に開始できないようにします。

# ログインシェルなしでユーザーを作成
sudo useradd -m -g sftp_users -s /bin/false upload_user

# 複雑なパスワードを設定
sudo passwd upload_user

ステップ2:Jailの設計(パーミッションの罠)

パーミッションは、多くのシステム管理者が行き詰まるポイントです。OpenSSHには譲れないルールがあります。Chrootディレクトリの所有者は root でなければならず、他のユーザーやグループに書き込み権限を与えてはいけません。もし upload_user が自身のホームディレクトリの所有権を持っていると、接続は即座に「broken pipe」エラーで切断されます。

ユーザーが実際に作業できる場所を提供するために、サブディレクトリを作成します。ホームフォルダを「ロビー」(読み取り専用)、サブフォルダを「オフィス」(読み書き可能)と考えると分かりやすいでしょう。

# ロビーを保護:所有者はrootである必要がある
sudo chown root:root /home/upload_user
sudo chmod 755 /home/upload_user

# ワークスペースを作成:所有者はユーザーにする
sudo mkdir /home/upload_user/uploads
sudo chown upload_user:sftp_users /home/upload_user/uploads

この設定では、ユーザーがログインすると /uploads が見えます。そこでファイルのアップロード、削除、名前変更ができますが、物理的に cd .. を実行して実際のシステムルートに移動することはできません。

ステップ3:OpenSSH設定の強化

sftp_users グループを特別に扱うよう、SSHデーモン(sshd)に指示する必要があります。設定ファイルを編集します:

sudo nano /etc/ssh/sshd_config

ファイルの最下部まで移動してください。 Match ディレクティブは「強欲(greedy)」で、それ以降のすべての行に適用されます。末尾に配置することで、これらの制限ルールが誤って root や管理者アカウントに適用されるのを防ぎます。

# SFTP Chroot 設定
Match Group sftp_users
    ChrootDirectory %h
    ForceCommand internal-sftp
    AllowTcpForwarding no
    X11Forwarding no
    PasswordAuthentication yes

これらの設定が重要な役割を果たします。 ForceCommand internal-sftp はシェルの要求を無視し、組み込みのSFTPエンジンを使用させます。また、TCPおよびX11フォワーディングを無効にすることで、ユーザーがサーバーをプロキシとして利用し、ローカルネットワーク上の他のマシンを攻撃するのを防ぎます。

構文チェックを省略してはいけません。スペースが一つ欠けているだけで、再起動時にSSHサービスがクラッシュする可能性があります。

sudo sshd -t

ターミナルに何も表示されなければ、安全に再起動できます:

sudo systemctl restart ssh

ステップ4:テストとトラブルシューティング

ローカルマシンから接続して隔離状態を確認します。 uploads フォルダには入れるはずですが、より上位のディレクトリを探索しようとすると閉じ込められていることが分かります。

sftp upload_user@your_server_ip
# 接続後:
pwd
# 結果は / になるはず
cd /etc
# 結果は "permission denied" または "not found" になるはず

実環境でのモニタリング

ユーザーから接続できないと報告があった場合、推測で動かないでください。 auth.log (RHEL系システムでは journalctl )が正確な理由を教えてくれます。最近の監査では、接続失敗の90%が、親ディレクトリのパーミッションが誤って755ではなく775になっていたことが原因でした。

# 認証試行をリアルタイムで監視
sudo tail -f /var/log/auth.log | grep sshd

本番環境スケールのためのプロのヒント

長年これらの環境を管理してきた経験から、最後に3つの強化ステップを推奨します:

  1. セットアップの自動化: ユーザー作成には Ansible プレイブックや Bash スクリプトを使用しましょう。毎回手動でディレクトリを作成していると、いずれパーミッション設定をミスし、セキュリティホールを作ることになります。
  2. SSH鍵の強制: パスワードは簡単ですが、ブルートフォース攻撃に弱いです。本番環境では、 root 所有のホームディレクトリ内に .ssh/authorized_keys ファイルを配置します。パーミッションを 400 に設定し、ユーザーが自身の鍵を変更できないようにします。
  3. ディスククォータの設定: 隔離されたユーザーであっても、ディスクを使い果たすことでサーバーをクラッシュさせることは可能です。プロジェクトのニーズに応じて、 quotaedquota を使用し、 uploads フォルダを 5GB や 10GB などの適切なサイズに制限してください。

このセットアップは、コアシステムファイルを保護しながら、ユーザーに必要な権限だけを与える、プロフェッショナルで隔離された環境を提供します。これは、現代の Linux 管理における安全なファイル処理のゴールドスタンダードです。

Share: