まったり技術ブログ

Technology is power.

LDAPインジェクションをしたかった話

f:id:motikan2010:20181005002942p:plain

今更ながらLDAPインジェクションの検証をやってみた。

 LDAPインジェクションは脆弱性としてそこそこ有名であり、名前だけは目にすることがあるが、イマイチ実際に検証を行う気になれない脆弱性でもあると思う。特にLDAPの環境構築は手間になりそうだし。
 このままだとLDAPインジェクションを体験しないまま死んでしまってもおかしくないので、DockerでささっとLDAPインジェクションを行える環境を作ってみて、検証してみるとする。

検証環境

ホスト

  • Docker 18.06.1

コンテナ

  • OpenLDAP 2.4.44
  • PHP 7.0 (php-ldap)

構築

OpenLDAP

 特に設定ファイルなどを書き換えることなく、下記のコマンドでLDAPサーバの起動までを行ってくれる。

$ docker run -p 389:389 --name openldap-container --detach osixia/openldap:1.2.2

github.com

LDAPクライアントアプリケーション

認証を行うクライアントアプリのソースは下記のリポジトリにあります。

github.com

1. アプリケーションの用意

ソースコード全体 LDAP-Injection-Vuln-App/index.php at 20181004 · motikan2010/LDAP-Injection-Vuln-App · GitHub

以下がソースコードの抜粋です。
LDAPに関係している部分のみを書き出しています。

/**
 * src/public/index.php
 */

<?php

//LDAPの接続情報
const LDAP_HOST = "openldap-container";
const LDAP_PORT = 389;
const LDAP_DC = "dc=example,dc=org";
const LDAP_DN = "cn=admin,dc=example,dc=org";
const LDAP_PASS = "admin";

// 省略

// LDAPに接続
$ldapConn = ldap_connect(LDAP_HOST, LDAP_PORT);
if (!$ldapConn) {
    exit('ldap_conn');
}

// バインド
ldap_set_option($ldapConn, LDAP_OPT_PROTOCOL_VERSION, 3); // バージョンをOpenLDAPの方に合わせる
$ldapBind = ldap_bind($ldapConn, LDAP_DN,LDAP_PASS);
if ($ldapBind) {

    // ログイン処理
    // 「$userId」と「$password」はユーザの入力値が格納されます。
    $filter = '(&(cn=' . $userId . ')(userPassword=' . $password . '))'; // IDとパスワードのAND条件でフィルタを作成
    $ldapSearch = ldap_search($ldapConn, LDAP_DC, $filter);
    $getEntries = ldap_get_entries($ldapConn, $ldapSearch);
    if ($getEntries['count'] > 0) {
        // 成功
    }
} else {
    // 失敗
}
?>

// 以下省略

 ユーザの入力値をそのままフィルタに指定しているのが脆弱性となっています。

$filter = '(&(cn=' . $userId . ')(userPassword=' . $password . '))';

2. コンテナの準備

# Dockerfile

FROM php:7.0-apache

RUN \
    apt-get update && \
    apt-get install libldap2-dev -y && \
    rm -rf /var/lib/apt/lists/* && \
    docker-php-ext-configure ldap --with-libdir=lib/x86_64-linux-gnu/ && \
    docker-php-ext-install ldap

ADD ./src/public /var/www/html/

3. コンテナの実行

$ docker build -t ldap-client-container .
$ docker run --link openldap-container -p 8888:80 ldap-client-container

動作確認

OpenLDAPの動作確認

-w オプションでパスワードを指定していますが、「admin」がパスワードとなっています。
末尾に記述されている「cn=admin」はフィルタであり、cn(Common Name)が「admin」のアカウントを表示しています。

$ ldapsearch -x -H ldap://localhost -b dc=example,dc=org -D "cn=admin,dc=example,dc=org" -w admin 'cn=admin'
# extended LDIF
#
# LDAPv3
# base <dc=example,dc=org> with scope subtree
# filter: cn=admin
# requesting: ALL
#

# admin, example.org
dn: cn=admin,dc=example,dc=org
objectClass: simpleSecurityObject
objectClass: organizationalRole
cn: admin
description: LDAP administrator
userPassword:: e1NTSEF9Z0RjWGl1QkR0d2xDcEZ5bVE4QWtoN09iRU1IZFVPN0s=

# search result
search: 2
result: 0 Success

# numResponses: 2
# numEntries: 1

予想外の事態発生

「cn(Common Name)」と「userPassword」で認証する予定でしたが、userPasswordはハッシュ化されているらしい。

$ echo "e1NTSEF9Z0RjWGl1QkR0d2xDcEZ5bVE4QWtoN09iRU1IZFVPN0s=" | base64 -D ; echo
{SSHA}gDcXiuBDtwlCpFymQ8Akh7ObEMHdUO7K

つまり、「(&(cn=admin)(userPassword=admin))」でフィルタした場合には、adminは表示されない。

$ ldapsearch -x -H ldap://localhost -b dc=example,dc=org -D "cn=admin,dc=example,dc=org" -w admin '(&(cn=admin)(userPassword=admin))'
# extended LDIF
#
# LDAPv3
# base <dc=example,dc=org> with scope subtree
# filter: (&(cn=admin)(userPassword=admin))
# requesting: ALL
#

# search result
search: 2
result: 0 Success

# numResponses: 1

だが、「(&(cn=admin)(userPassword={SSHA}gDcXiuBDtwlCpFymQ8Akh7ObEMHdUO7K))」でフィルタした場合には、adminを表示することができた。 LDAPクライアント側からもこの文字列を入力する必要がありそう。

脆弱性の検証

まずはアプリケーションとして正常系の動作確認を実施してみます。

1. 正しいパスワード入力

2. 認証できた

もちろんパスワードに別の文字列を入力した場合には、認証することはできない。というのがアプリケーションの正しい挙動なのだが、パスワードに「*」を入力してみる。

  1. 「*」をパスワードに入力

  2. 認証できてしまった・・・。

というのが、LDAPインジェクション。

OpenLDAP側のログを見てみると以下のようにフィルタが行われていた。
入力した通りパスワードにワイルドカード「*」が指定されており、認証が成功している。

5bb61de0 conn=1009 op=1 SRCH base="dc=example,dc=org" scope=2 deref=0 filter="(&(cn=admin)(userPassword=*))"

対策(ダメな奴)

ユーザから入力されたパスワード内の「*」を削除すればワイルドカードが指定されることがなくなる。
具体的にはフィルタの部分を以下の内容に修正する。

$filter = '(&(cn=' . $userId . ')(userPassword=' . str_replace('*', '', $password) . '))';

これで再度アプリを動かしてみると、パスワードに「*」では認証が成功することはなくなった。 だが、今度はログインIDとパスワードに以下の文字列を入力してみる。

  • ログインID:admin)(|(cn=admin
  • パスワード:hoge)

表示上少しおかしいが、デタラメなパスワードでログインすることができた。

LDAPのログでは下記のようになっていた。

5bb62412 conn=1020 op=1 SRCH base="dc=example,dc=org" scope=2 deref=0 filter="(&(cn=admin)(|(cn=admin)(userPassword=hoge)))"

対策

ldap_escape関数を利用することにより、どちらのパターンでも認証を突破することはできなくなった。

$filter = '(&(cn=' . ldap_escape($userId) . ')(userPassword=' . ldap_escape($password) . '))';

まとめ

 環境構築が面倒と思っていたLDAPですが、Dockerを利用したら1コマンドで構築できたというのが、1番の収穫。
 LDAPインジェクションを手元て検証してみて分かったのですが、LDAPにはパスワードがハッシュ値で格納されており、脆弱性のサンプルのようにフィルタで認証している実装というのはあまりなさそう。
 昔のOpenLDAP or 別のLDAPでは平文でパスワードが格納される設定になっていたんですかね。それか今回利用したDockerイメージがそのような設定がなされてたいのか。
 まだまだLDAPに関して分からないことだらけですので、また機会を見つけて学習しないと。。

参考

LDAP認証してみた。 - ばずなダイアリー

How to use LDAP Active Directory Authentication with PHP :: ExchangeCore

PayloadsAllTheThings/LDAP_FUZZ.txt at master · swisskyrepo/PayloadsAllTheThings · GitHub