开源版禅道集成LDAP实现统一身份认证

一、版本说明

1、禅道版本为开源版,版本号为最新的15.2
docker安装请参考:https://hub.docker.com/r/idoop/zentao

docker pull idoop/zentao

docker run -d -p 8084:80 -p 3346:3306 -e USER="admin" -e PASSWD="123456" -e BIND_ADDRESS="false" -e SMTP_HOST="163.177.90.125 smtp.exmail.qq.com" -v /data/zbox/:/opt/zbox/ --name zentao-server idoop/zentao
  • 访问地址是:http://localhost:8084
  • 用户名和密码是admin/123456
    由于密码过于简单,禅道会引导你去修改密码。

2、LDAP插件,版本: 1.2 作者: TigerLau

二、下载并安装LDAP插件

管理员账号admin,进入后台--》插件--》本地安装


step1.png

我这里是安装好了,过程中按提示操作,可能会让你在服务器上新建或删除文件,保证你是管理员。

插件不要安装错了!!github上也有一个插件,不建议。亲自试过不行。为防止你少走弯路,我贴出它的地址https://github.com/iboxpay/ldap.

step2.png

安装成功后的,提示也很给力,直接引导你下一步去设置LDAP.

1.插件安装后,在后台页面会多出一个"LDAP"子页面,可在该页面配置LDAP服务器信息

2.在LDAP配置页面可以测试是否能够正常连接LDAP服务器 (这一步我是失败的,建议你忽略!!)

3.保存配置后,点击“手动同步”按钮,从LDAP服务器上同步用户信息 

4.同步用户信息以后,可以使用LDAP用户登录禅道 

5.本地用户,通过在账户名称前加“$”符号来登录禅道

三、登录说明

  • admin用户的登录用户名必须前加美元符,admin-->$admin,密码还是原先的密码。
  • LDAP用户一般使用手机号或邮箱作为账号登录。

四、LDAP的设置

LDAP设置.png

这里有个坑,“测试连接”,一直是报错说:“Can't contact LDAP server ”, 而我确保LDAP服务是正常的,且密码也输入正确。忽略不管该提示,直接保存并手动同步用户。

人员列表.png

一般地,你需要对人员设置权限。


image.png
image.png

admin管理员登录


admin管理员登录.png

LDAP用户登录


LDAP用户登录.png

五、修改Php源码

如果你想现在就能够LDAP登录成功,恐怕要让你失望了。还需要继续跟着修改以下几处代码。

  • 修改/opt/zbox/app/zentao/module/user/ext/model/identify.php 文件
# $account = $this->config->ldap->uid.'='.$account.','.$this->config->ldap->baseDN;
 # $pass = $ldap->identify($this->config->ldap->host, $account, $password);

#getUserDn的定义见下module/ldap/model.php
$dn = $ldap->getUserDn($this->config->ldap, $account);
$pass = $ldap->identify($this->config->ldap->host, $dn, $password);

完整的文件内容是:

<?php
public function identify($account, $password)
{
        if (0 == strcmp('$',substr($account, 0, 1))) {
                return parent::identify(ltrim($account, '$'), $password);
        } else {
                $user = false;
                $record = $this->dao->select('*')->from(TABLE_USER)
            ->where('account')->eq($account)
            ->andWhere('deleted')->eq(0)
            ->fetch();
        if ($record) {
                $ldap = $this->loadModel('ldap');
                #       $account = $this->config->ldap->uid.'='.$account.','.$this->config->ldap->baseDN;
                $dn = $ldap->getUserDn($this->config->ldap, $account);
                #       $pass = $ldap->identify($this->config->ldap->host, $account, $password);
                $pass = $ldap->identify($this->config->ldap->host, $dn, $password);

                if (0 == strcmp('Success', $pass)) {
                        $user = $record;
                        $ip   = $this->server->remote_addr;
                    $last = $this->server->request_time;
                    $this->dao->update(TABLE_USER)->set('visits = visits + 1')->set('ip')->eq($ip)->set('last')->eq($last)->where('account')->eq($account)->exec();
                    $user->last = date(DT_DATETIME1, $user->last);
                }
        }
                return $user;
        }
}
  • 修改/opt/zbox/app/zentao/module/ldap/model.php 文件
    增加函数
    public function getUserDn($config, $account) {
        $ret = null;
        $ds = ldap_connect($config->host);
        if ($ds) {
                ldap_set_option($ds,LDAP_OPT_PROTOCOL_VERSION,3);
                ldap_bind($ds, $config->bindDN, $config->bindPWD);
                $filter = "(cn=$account)";
                $rlt = ldap_search($ds, $config->baseDN, $filter);
                $count=ldap_count_entries($ds, $rlt);
                if($count > 0) {
                        $data = ldap_get_entries($ds, $rlt);
                        $ret = $data[0]['dn'];
                        $str = serialize($data);
                }
                ldap_unbind($ds);
                ldap_close($ds);
        }
        return $ret;
    }

完整的文件内容是:

<?php
/**
 * The model file of ldap module of ZenTaoPMS.
 *
 * @license     ZPL (http://zpl.pub/page/zplv11.html)
 * @author      TigerLau
 * @package     ldap
 * @link        http://www.zentao.net
 */
?>
<?php
class ldapModel extends model
{
    public function identify($host, $dn, $pwd)
    {
        $ret = '';
        $ds = ldap_connect($host);
        if ($ds) {
                ldap_set_option($ds,LDAP_OPT_PROTOCOL_VERSION,3);
                ldap_bind($ds, $dn, $pwd);

            $ret = ldap_error($ds);
                ldap_close($ds);
        }  else {
            $ret = ldap_error($ds);
        }

        return $ret;
    }

    public function getUsers($config)
    {
        $ds = ldap_connect($config->host);
        if ($ds) {
            ldap_set_option($ds,LDAP_OPT_PROTOCOL_VERSION,3);
            ldap_bind($ds, $config->bindDN, $config->bindPWD);

            $attrs = [$config->uid, $config->mail, $config->name];

            $rlt = ldap_search($ds, $config->baseDN, $config->searchFilter, $attrs);
            $data = ldap_get_entries($ds, $rlt);
            return $data;
        }

        return null;
    }

    public function sync2db($config)
    {
        $ldapUsers = $this->getUsers($config);
        $user = new stdclass();
        $account = '';
        $i=0;
        for (; $i < $ldapUsers['count']; $i++) {         
            $user->account = $ldapUsers[$i][$config->uid][0];
            $user->email = $ldapUsers[$i][$config->mail][0];
            $user->realname = $ldapUsers[$i][$config->name][0];

            $account = $this->dao->select('*')->from(TABLE_USER)->where('account')->eq($user->account)->fetch('account');
            if ($account == $user->account) {
                $this->dao->update(TABLE_USER)->data($user)->where('account')->eq($user->account)->autoCheck()->exec();
            } else {
                $this->dao->insert(TABLE_USER)->data($user)->autoCheck()->exec();
            }

            if(dao::isError()) 
            {
                echo js::error(dao::getError());
                die(js::reload('parent'));
            }
        }

        return $i;
    }


    public function getUserDn($config, $account){
            $ret = null;
            $ds = ldap_connect($config->host);
            if ($ds) {
                ldap_set_option($ds,LDAP_OPT_PROTOCOL_VERSION,3);
                ldap_bind($ds, $config->bindDN, $config->bindPWD);
                $filter = "(uid=$account)";
                $rlt = ldap_search($ds, $config->baseDN, $filter);
                $count=ldap_count_entries($ds, $rlt);

                if($count > 0){
                    $data = ldap_get_entries($ds, $rlt);
                    $ret = $data[0]['dn'];
                    $str = serialize($data);
                }
                ldap_unbind($ds);
                ldap_close($ds);
         }
         return $ret;
    }

}
  • 修改 /opt/zbox/app/zentao/config/my.php 文件
    增加
$config->notMd5Pwd = true;

以关闭md5加密,否则认证不能通过

修改后的文件全文是:

<?php^M
$config->installed       = true;^M
$config->debug           = false;^M
$config->requestType     = 'PATH_INFO';^M
$config->db->host        = '127.0.0.1';^M
$config->db->port        = '3306';^M
$config->db->user        = 'root';^M
$config->db->prefix      = 'zt_';^M
$config->webRoot         = getWebRoot();^M

$config->db->name      = 'zentao';
$config->db->password  = '123456';
$config->default->lang = 'zh-cn';

#新增
$config->notMd5Pwd = true;

六、修改用户名

公司原先创建的账号是拼音字母的格式,后面采用LDAP,需要统一为手机号格式。所以老用户会有两个用户,在不影响之前账户的使用情况下(主要是需要保留数据),我们采用修改旧账号的登录名为手机号。

但是禅道不允许登录名重复,所以我们需要先把新账户的登录名为一个不存在的值,再删除它。否则按姓名检索,会出现两个账号。
第二步,修改旧账号的登录名为手机号。


根据姓名检索出两个用户.png
image.png

执行删除,因为禅道是逻辑删除,实际上该用户名是已经不能重复了。


image.png

最后就剩下一个老用户。


image.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,126评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,254评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,445评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,185评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,178评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,970评论 1 284
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,276评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,927评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,400评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,883评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,997评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,646评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,213评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,204评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,423评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,423评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,722评论 2 345

推荐阅读更多精彩内容