设为首页 收藏本站
查看: 596|回复: 0

[经验分享] 纯基于PHP的单点登陆

[复制链接]

尚未签到

发表于 2017-4-3 06:18:11 | 显示全部楼层 |阅读模式
纯基于PHP的单点登陆
  Discuz有一个通行证,类似于单点登陆。不过我觉得单点登陆最好应该是一个独立的程序,和CAS一样。由于所有的程序都是PHP的,所以就做了一个简单的单点登陆。借用了一下discuz的加密方法
  用户有以下几种情况会直接访问本系统
1 用户直接访问passport,希望登陆
2 用户从passport_app上点击登陆按钮转过来的
3 用户从passport_app1转到passport_app2时候,passport_app2转过来的
4 用户直接输入访问passport_app的URL
  2 3 4 或者有referer,或者有fromurl的参数
  ====
  1 使用通行证的应用程序上面的登陆按钮全部指向下面的链接
http://localhost/passport/login.php
  2 如果用户已经登陆了,转到5
  3 如果用户还没有登陆
转到一个登陆页面,要包含参数:
用户名,密码,fromurl(预处理过的refer)
提交到login.php?op=login

4 验证通过以后,设置自身的session或者cookies,
  5 根据用户http头里面的refer得到来源地址。
5.1 如果没有来源地址则显示本passport登陆成功页面,上面列出所有的passport应用
5.2 如果有refer,则跳回到refer的地址,即到6
  6 转到通行证应用的passport_login.php页面,传递的参数包括
$userinfo 一个数组,包含了用户名,角色,组等其它信息,一般不需要包含密码
$fromurl
$verify md5($auth.$fromurl) 保证用户信息没有被篡改
===============================
7 passport_login.php
首先检查参数有没有被改变
然后取出参数内的user_id

验证通过以后,

如果user_id在系统内存不做处理
如果不存在,则根据编码添加该用户

最后设置自己的session/cookies,然后跳转到fromurl

======

通行证的密匙,可以自行填写英文,可包含任何字母及数字,长度大于 10 字节
  -------
  代码如下
  session_start();
  $username = "";
$password = "";
$loginerror = "";
$fromurl = '';
  if(isset($_GET['fromurl']) && trim($_GET['fromurl'])!= '') {
$fromurl = $_GET['fromurl'];
}else if(isset($_SERVER['HTTP_REFERER']) && trim($_SERVER['HTTP_REFERER'])!= '') {
$fromurl = $_SERVER['HTTP_REFERER'];
}

//防止同一个服务器装了多个upassport互相干扰
$key = md5(DB_DATABASE.DB_USER.DB_PASSWORD);
if(!isset($_SESSION['passport_app'])) {
unset($_SESSION['u']);
$_SESSION['passport_app'] = $key;
include('themes/'.THEME.'/login.html');
exit;
}else if($_SESSION['passport_app'] != $key) {
unset($_SESSION['u']);
$_SESSION['passport_app'] = $key;
include('themes/'.THEME.'/login.html');
exit;
}
  
if (isset($_POST['op'])&&trim($_POST['op'])=='dologin') {
  $valid = 0;
$authnum = $_POST['authnum'];
$username = $_POST['username'];

$fromurl = $_POST['fromurl'];

  if ($authnum && trim($_SESSION['authnum'])==$authnum && $username) {

$password = $_POST['password'];

$user = $db->getRow("select * from $dbutils->user where username = '$username' and password = '$password'");
  if (!empty($user) && $user['user_id']!=0) {
$u['user_id'] = $user['user_id'];
$u['username'] = $user['username'];
$u['truename'] = $user['truename'];

$u['group'] = $db->getRow("select ug.group_id,g.group_name from $dbutils->user_group ug,$dbutils->group g where ug.group_id = g.group_id and ug.user_id = ".$user['user_id']);

$u['roles'] = $db->getAll("select ur.role_id,r.role_name,r.privileges from $dbutils->user_role ur,$dbutils->role r where ur.role_id = r.role_id and ur.user_id = ".$user['user_id']);

$privileges = Array();
foreach($u['roles'] as $role) {
$role_privis = explode(',',$role['privileges']);

foreach($role_privis as $p) {
if(!in_array($p,$privileges)) array_push($privileges,$p);

}

}

$u['privileges'] = $privileges ;

$_SESSION['u'] = $u;

$valid =1;
  }

}

if (!$valid) {
$loginerror = '<div class="error">'."非法登陆".'</div>';
$username = $_POST['username'];
unset($u);
}

  }
  //用户正常跳转或者访问
$u = isset($_SESSION['u']) ? $_SESSION['u'] : '';


if (empty($u)) { //用户不存在,跳到登陆界面

include('themes/'.THEME.'/login.html');
exit;

}else { //用户存在,表示已经登陆过了
  if(trim($fromurl)=='') { //没有referer,则显示默认主页,列出所有应用

$t->assign('u',$u);
$t->render('index.html', "欢迎登陆通行证",'wrap.html',true);

}else { //从别的应用转过来的,可能是上面 2 3 4,此时必有referer

  //根据referer的url得到当前的应用的key和完整的地址
  $passport_app = $db->GetRow("select * from $dbutils->app where INSTR('$fromurl', url)=1");

$userinfo= passport_encrypt(passport_encode($u), $passport_app['key']);
$verify= md5($userinfo.$fromurl.$passport_app['key']);

header("Location: ".$passport_app['login_url'].
"?userinfo=".rawurlencode($userinfo).
"&fromurl=".rawurlencode($fromurl).
"&verify=$verify");
  }


}
  客户应用检验单点登陆的方法如下
  $passport_key = '1234567890';
  
if($_GET['verify'] != md5($_GET['userinfo'].$_GET['fromurl'].$passport_key)) {
exit('Illegal request');
}
  
$u = array();
parse_str(passport_decrypt($_GET['userinfo'], $passport_key), $u);
  header("location: ".$_GET['fromurl']);

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-359303-1-1.html 上篇帖子: PHP文档生成工具 下篇帖子: PHP下解析swf文件头
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表