PHP Conference Japan 2024

ldap_set_rebind_proc

(PHP 4 >= 4.2.0, PHP 5, PHP 7, PHP 8)

ldap_set_rebind_proc设置回调函数以在转介跟踪中进行重新绑定

描述

ldap_set_rebind_proc(LDAP\Connection $ldap, ?callable $callback): bool
警告

此函数目前未记录;仅提供其参数列表。

变更日志

版本 描述
8.1.0 ldap 参数现在需要一个 LDAP\Connection 实例;以前,需要一个有效的 ldap link 资源
8.0.0 callback 现在可以为空。
添加注释

用户贡献的注释 7 条注释

mvanbeek at forgetaboutit dot net
11 年前
在回调函数中使用的 $referral 不是绑定 DN,而是正在访问的记录的 DN(或者更确切地说是它在主服务器上的位置,如果两者之间有区别的话),因此您需要使用现有的凭据重新绑定。连接 ($ldap) 似乎已建立到新服务器,因此它只是一个重新绑定过程,没有比这更复杂的了。底层库中一定有一个循环,它会重新提交导致转介的请求,直到返回成功或失败。

我想,如果您最初使用的绑定 DN 无法让您编辑主服务器上的记录,那么这是一个 LDAP 问题而不是 PHP 问题。但是,至少使用重新绑定过程,您可以先修改绑定 DN。

所以,现在我理解了它的工作原理,重新绑定过程实际上非常简单!我以为它会复杂得多。在最简单的形式中,这就是您需要的所有内容,假设您的绑定 $dn 和 $pass 是全局变量

<?php
function rebind($ldap, $referral) {
// 设置 LDAP 选项
ldap_set_option($ldap, LDAP_OPT_PROTOCOL_VERSION, 3);
ldap_set_option($ldap, LDAP_OPT_REFERRALS, True);
ldap_set_rebind_proc($ldap, 'rebind');
// 重新绑定
if (!ldap_bind($ldap, $dn, $pass)) {
echo
'无法绑定到转介服务器';
return
1; // 是的,1 表示失败。
}
return
0; // 是的,成功时返回 0。
}
?>
leon at leonux dot co dot za
13 年前
我终于可以使用 ldap_set_rebind_proc 函数使转介工作了。不要在您的回调函数中连接到转介服务器。这是为您完成的。您只需要绑定即可。如果绑定成功,回调函数必须返回 0,否则返回 1。

考虑一个主从 LDAP 设置,其中从服务器是只读的,并将写入操作转介到主服务器。对于从服务器上的 PHP,您需要类似以下内容

<?php

// 回调函数
function rebind($ldap, $referral) {
// LDAP 选项
ldap_set_option($ldap, LDAP_OPT_PROTOCOL_VERSION, 3);
ldap_set_option($ldap, LDAP_OPT_REFERRALS, True);
ldap_set_rebind_proc($ldap, 'rebind');
// 转介的形式为:
// ldaps://newhost/cn=user,ou=people,dc=example,dc=com
$refparts = explode('/', $referral);
if (
count($refparts) > 2) {
// 从转介中获取绑定 DN
$dn = $refparts[3];
// 绑定到新主机
if (!ldap_bind($ldap, $dn, $pass)) {
echo
'无法绑定到转介服务器';
return
1;
}
} else {
// 尝试匿名绑定
if (!ldap_bind($ldap)) {
echo
'无法匿名绑定到转介服务器';
return
1;
}
}
return
0;
}

// 初始 LDAP 连接到从服务器
$ldap_host = 'localhost'
$ds = ldap_connect($ldap_host);
// LDAP 选项
ldap_set_option($ds, LDAP_OPT_PROTOCOL_VERSION, 3)
ldap_set_option($ds, LDAP_OPT_REFERRALS, True)
// 设置回调函数
ldap_set_rebind_proc($ds, 'rebind'))
// 绑定
ldap_bind($ds, $dn, $pass)
// LDAP 写入
ldap_modify($ds, $dn, $attr);

?>

如果您使用类方法作为回调函数,则更容易从回调中访问密码和其他数据。回调将这样初始化

<?php

ldap_set_rebind_proc
($ldap, 'MyClass::rebind');

?>
mvanbeek at forgetaboutit dot net
14 年前
我很难找到关于跟踪转介的良好信息,因此我在这里添加了我的一点拙见。我的测试代码还没有完全运行,所以请查看页面下方的更新。

看起来这需要这样工作:使用此函数设置您自己的回调函数,以连接并绑定到推荐服务器。您需要设置此回调函数,以及强制使用 v3 ldap 并将推荐跟踪设置为开启,作为初始连接设置的一部分。因此,在连接之后但绑定之前,您需要类似以下的内容:

<?php
$ds
= ldap_connect($server);
ldap_set_option($ds, LDAP_OPT_PROTOCOL_VERSION, 3);
ldap_set_option($ds, LDAP_OPT_REFERRALS, 1);
ldap_set_rebind_proc($ds, "rebind");
ldap_bind($ds,$dn,$pass);
?>

此回调函数(在上例中称为 rebind)需要两个参数。这些参数是预设的,并在调用回调函数时提供。第一个是 ldap 链接标识符。我假设这是提供的,因为该函数可以被一系列连续的推荐依次使用。第二个是由初始服务器提供的 ldap 推荐 URL。我见过一些说明称此函数必须在由 ldap_set_rebind_proc 设置之前定义,但我目前无法确认这一点。

我的设置基于主从 ldap 服务器配置,PHP 应用程序驻留在从服务器上,它在那里执行本地主机查找。当您尝试写入从 ldap 服务器时,服务器会返回一个推荐 URL,然后内部 PHP 函数会调用回调函数。

尽管此页面上已经存在代码(似乎也用于测试 PHP 代码),但我认为它是错误的。我认为它只是重新连接到初始服务器。我认为回调函数应该做的是连接到新服务器,并绑定到它。我的测试代码目前如下所示:

<?php
function rebind($ldap, $referral) {
global
$dn;
global
$pass;
$server= preg_replace('!^(ldap://[^/]+)/.*$!', '\\1', $referral);
if (!(
$ldap = ldap_connect($server))){
echo
"reconnect failed - <br>";
return
1;
}
ldap_set_option($ldap, LDAP_OPT_PROTOCOL_VERSION, 3);
ldap_set_option($ldap, LDAP_OPT_REFERRALS, 1);
ldap_set_rebind_proc($ldap, "rebind");
if (!
ldap_bind($ldap,$dn,$pass)){
echo
"rebind failed - <br>";
return
1;
}
return
0;
}
?>

据我所知,返回值 0 表示成功,任何其他值表示失败。完全缺乏文档并没有帮助。

上面的代码一直工作到对新服务器进行身份验证,但目前在我尝试将记录写入新服务器之前,似乎收到了一个解绑请求,因此它失败了。

我还建议在绑定之前添加 ldap_start_tls。
pearcec at commnav dot com
21 年前
PHP 期望 ldap 函数 ldap_set_rebind_proc 具有三个参数。据我所知,这在 OpenLDAP 的 2.0 版本中不存在,但在 2.1 中实现了。Configure 会告诉你

检查 3 个参数的 ldap_set_rebind_proc... 不存在
night0wl at frost dot ath dot cx
21 年前
由于此函数没有示例代码,我遇到了很多麻烦才能使其正常工作。

因此,这是一个可行的示例

function rebind_on_ref ($ds, $ldap_url) {
global $binddn; // 用于绑定的 DN
global $bindpw; // 用于绑定的密码

// 大多数现代 LDAP 服务器都需要,使用 LDAPv3
ldap_set_option($a, LDAP_OPT_PROTOCOL_VERSION, 3);

if (!ldap_bind($a,$binddn,$bindpw)) {
print "无法绑定";
}
}
randy at kotmail dot com
22 年前
如果 slapd 中未编译 rebind_proc,则该函数将永远无法工作。查看 slapd 的新 alpha 版本并阅读手册。
匿名
13 年前
我现在已经花费了足够的时间来研究这个问题,可以断言 ldap 推荐,至少在尝试在从服务器上添加、修改或删除记录时(从服务器会正确地将推荐发送到主服务器),在 php 中不起作用。我的建议是关闭 ldap 推荐,并编写您自己的带内置推荐处理的添加、修改和删除函数。例如:

<?php
function ldap_referral_add($connection,$add_dn,$Add_entry,$bind_dn,$bind_pw)
{
$rconnection = $connection;
$loop = 10; # 最大推荐跳转次数。
# 关闭正常推荐
ldap_get_option($connection,LDAP_OPT_REFERRALS,$old_referral_setting)
do
{
$response = ldap_add($rconnection,$dn,$entry);
# 我们得到一条成功消息:
if ( $response )
{
ldap_unbind($rconnection);
$loop = 0; # 可能不需要
ldap_set_option($connection,LDAP_OPT_REFERRALS,$old_referral_setting);
return
true;
}
# 我们得到一条推荐消息:
elseif ( !$response && ldap_errno($rconnection) == 0x0a )
{
$new_server_url = $server= preg_replace('!^(ldap://[^/]+)/.*$!', '\\1', $ldap_error($rconnection)); # 可能需要在此处进行一些健全性检查
$rconnection = ldap_connect($new_server_url);
ldap_set_option($rconnection,LDAP_OPT_REFERRALS,0)
ldap_set_option($rconnection,LDAP_OPT_PROTOCOL_VERSION, 3)
ldap_bind($rconnection,$bind_dn,$bind_pw);
$loop = $loop - 1;
}
else
{
ldap_unbind($rconnection);
$loop = 0; # 可能不需要
ldap_set_option($connection,LDAP_OPT_REFERRALS,$old_referral_setting);
return
false;
}
} while (
$loop > 0);
}
?>
To Top