接入钉钉,实现钉钉扫码登录
Signed-off-by: vilson <545522390@qq.com>
This commit is contained in:
parent
5448a88197
commit
a628429934
@ -5,7 +5,7 @@ use service\NodeService;
|
|||||||
use service\RandomService;
|
use service\RandomService;
|
||||||
use think\Db;
|
use think\Db;
|
||||||
use think\facade\Cache;
|
use think\facade\Cache;
|
||||||
use think\facade\Request;
|
use think\facade\Log;
|
||||||
|
|
||||||
|
|
||||||
function isDebug()
|
function isDebug()
|
||||||
@ -13,6 +13,54 @@ function isDebug()
|
|||||||
return config('app.app_debug');
|
return config('app.app_debug');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 日志记录
|
||||||
|
* @param string|array $content 内容
|
||||||
|
* @param string $type 日志类型
|
||||||
|
* @param string $path 日志地址
|
||||||
|
*/
|
||||||
|
function logRecord($content, $type = 'info', $path = 'default')
|
||||||
|
{
|
||||||
|
$path = 'log/' . $path;
|
||||||
|
Log::init(['path' => $path]);
|
||||||
|
if (is_array($content) || is_object($content)) {
|
||||||
|
$content = json_encode($content);
|
||||||
|
}
|
||||||
|
Log::write($content, $type);
|
||||||
|
Log::init();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
function getCurrentMember()
|
||||||
|
{
|
||||||
|
return session('member');
|
||||||
|
}
|
||||||
|
|
||||||
|
function setCurrentMember($data)
|
||||||
|
{
|
||||||
|
return session('member', $data);
|
||||||
|
}
|
||||||
|
|
||||||
|
function getCurrentOrganizationCode()
|
||||||
|
{
|
||||||
|
return session('currentOrganizationCode');
|
||||||
|
}
|
||||||
|
|
||||||
|
function setCurrentOrganizationCode($data)
|
||||||
|
{
|
||||||
|
return session('currentOrganizationCode', $data);
|
||||||
|
}
|
||||||
|
|
||||||
|
function getCurrentOrganization()
|
||||||
|
{
|
||||||
|
return session('organization');
|
||||||
|
}
|
||||||
|
|
||||||
|
function setCurrentOrganization($data)
|
||||||
|
{
|
||||||
|
return session('organization', $data);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 打印输出数据到文件
|
* 打印输出数据到文件
|
||||||
* @param mixed $data 输出的数据
|
* @param mixed $data 输出的数据
|
||||||
|
@ -2,6 +2,9 @@
|
|||||||
|
|
||||||
namespace app\common\Model;
|
namespace app\common\Model;
|
||||||
|
|
||||||
|
use service\JwtService;
|
||||||
|
use service\NodeService;
|
||||||
|
use service\RandomService;
|
||||||
use think\Db;
|
use think\Db;
|
||||||
use think\File;
|
use think\File;
|
||||||
|
|
||||||
@ -10,13 +13,38 @@ class Member extends CommonModel
|
|||||||
|
|
||||||
protected $append = [];
|
protected $append = [];
|
||||||
|
|
||||||
public function login($account)
|
public static function login($member)
|
||||||
{
|
{
|
||||||
if ($account == 'admin') {
|
// 更新登录信息
|
||||||
return [];
|
Db::name('Member')->where(['id' => $member['id']])->update([
|
||||||
|
'last_login_time' => Db::raw('now()'),
|
||||||
|
]);
|
||||||
|
$list = MemberAccount::where(['member_code' => $member['code']])->order('id asc')->select()->toArray();
|
||||||
|
$organizationList = [];
|
||||||
|
if ($list) {
|
||||||
|
foreach ($list as $item) {
|
||||||
|
$organization = Organization::where(['code' => $item['organization_code']])->find();
|
||||||
|
if ($organization) {
|
||||||
|
$organizationList[] = $organization;
|
||||||
}
|
}
|
||||||
$where[] = ['account', '=', $account];
|
}
|
||||||
return Db::name('member')->where($where)->find();
|
}
|
||||||
|
$member['account_id'] = $list[0]['id'];
|
||||||
|
$member['is_owner'] = $list[0]['is_owner'];
|
||||||
|
$member['authorize'] = $list[0]['authorize'];
|
||||||
|
$member['position'] = $list[0]['position'];
|
||||||
|
$member['department'] = $list[0]['department'];
|
||||||
|
|
||||||
|
setCurrentMember($member);
|
||||||
|
!empty($member['authorize']) && NodeService::applyProjectAuthNode();
|
||||||
|
$member = getCurrentMember();
|
||||||
|
$tokenList = JwtService::initToken($member);
|
||||||
|
$accessTokenExp = JwtService::decodeToken($tokenList['accessToken'])->exp;
|
||||||
|
$tokenList['accessTokenExp'] = $accessTokenExp;
|
||||||
|
$loginInfo = ['member' => $member, 'tokenList' => $tokenList, 'organizationList' => $organizationList];
|
||||||
|
session('loginInfo', $loginInfo);
|
||||||
|
logRecord($loginInfo, 'info', 'member/login');
|
||||||
|
return $loginInfo;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -30,6 +58,10 @@ class Member extends CommonModel
|
|||||||
{
|
{
|
||||||
//需要创建的信息。1、用户 2、用户所属组织 3、组织权限 4、所属组织账号
|
//需要创建的信息。1、用户 2、用户所属组织 3、组织权限 4、所属组织账号
|
||||||
$memberData['create_time'] = nowTime();
|
$memberData['create_time'] = nowTime();
|
||||||
|
(!isset($memberData['avatar']) || !$memberData['avatar']) && $memberData['avatar'] = 'https://static.vilson.xyz/cover.png';
|
||||||
|
!isset($memberData['status']) && $memberData['status'] = 1;
|
||||||
|
!isset($memberData['code']) && $memberData['code'] = createUniqueCode('member');
|
||||||
|
!isset($memberData['account']) && $memberData['account'] = RandomService::alnumLowercase();
|
||||||
$result = self::create($memberData);
|
$result = self::create($memberData);
|
||||||
|
|
||||||
|
|
||||||
@ -80,6 +112,25 @@ class Member extends CommonModel
|
|||||||
return $result;
|
return $result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static function dingtalkLogin($userInfo)
|
||||||
|
{
|
||||||
|
$unionid = $userInfo['unionid'];
|
||||||
|
$openid = $userInfo['openid'];
|
||||||
|
$member = self::where(['dingtalk_unionid' => $unionid])->find();
|
||||||
|
if (!$member) {
|
||||||
|
$memberData = [
|
||||||
|
'dingtalk_openid' => $openid,
|
||||||
|
'dingtalk_unionid' => $unionid,
|
||||||
|
'name' => $userInfo['nick'],
|
||||||
|
'avatar' => isset($userInfo['avatar']) ? $userInfo['avatar'] : '',
|
||||||
|
'mobile' => isset($userInfo['mobile']) ? $userInfo['mobile'] : '',
|
||||||
|
'email' => isset($userInfo['email']) ? $userInfo['email'] : '',
|
||||||
|
];
|
||||||
|
$member = self::createMember($memberData);
|
||||||
|
}
|
||||||
|
self::login($member);
|
||||||
|
return $member;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param File $file
|
* @param File $file
|
||||||
|
@ -65,7 +65,7 @@ class Organization extends CommonModel
|
|||||||
'create_time' => nowTime(),
|
'create_time' => nowTime(),
|
||||||
'avatar' => $memberData['avatar'],
|
'avatar' => $memberData['avatar'],
|
||||||
'name' => $memberData['name'],
|
'name' => $memberData['name'],
|
||||||
'email' => $memberData['email'],
|
'email' => isset($memberData['email']) ? $memberData['email'] : '',
|
||||||
];
|
];
|
||||||
MemberAccount::create($memberAccountData);
|
MemberAccount::create($memberAccountData);
|
||||||
return $organization;
|
return $organization;
|
||||||
|
85
application/index/controller/Oauth.php
Normal file
85
application/index/controller/Oauth.php
Normal file
@ -0,0 +1,85 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* Created by PhpStorm.
|
||||||
|
* User: Lenovo
|
||||||
|
* Date: 2019/1/2
|
||||||
|
* Time: 15:03
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace app\index\controller;
|
||||||
|
|
||||||
|
|
||||||
|
use app\common\Model\Client;
|
||||||
|
use app\common\Model\Member;
|
||||||
|
use controller\BasicApi;
|
||||||
|
use EasyDingTalk\Application;
|
||||||
|
use Symfony\Component\HttpFoundation\RedirectResponse;
|
||||||
|
use think\facade\Log;
|
||||||
|
use think\facade\Request;
|
||||||
|
use think\response\Redirect;
|
||||||
|
|
||||||
|
class Oauth extends BasicApi
|
||||||
|
{
|
||||||
|
|
||||||
|
public function index()
|
||||||
|
{
|
||||||
|
$app = new Application(config('dingtalk.'));
|
||||||
|
$userId = $app->user->getUseridByUnionid('3CnKFHEE7mX1hayPIHvpCwiEiE');
|
||||||
|
// echo json_encode($userId);die;
|
||||||
|
$userId = $userId['userid'];
|
||||||
|
$user = $app->user->get($userId, $lang = null);
|
||||||
|
echo json_encode($user);die;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function dingTalkOauth()
|
||||||
|
{
|
||||||
|
$currentMember = getCurrentMember();
|
||||||
|
if (!$currentMember) {
|
||||||
|
$app = new Application(config('dingtalk.'));
|
||||||
|
$response = $app->oauth->use('app-01')->withQrConnect()->redirect();
|
||||||
|
$redirect = $response->getTargetUrl();
|
||||||
|
if ($redirect) {
|
||||||
|
if ($redirect == 'undefined') {
|
||||||
|
$redirect = '/';
|
||||||
|
}
|
||||||
|
$_SESSION['target_url'] = $redirect;
|
||||||
|
}
|
||||||
|
return redirect($redirect);
|
||||||
|
} else {
|
||||||
|
//已登录跳转
|
||||||
|
return redirect(Request::domain() . '/index.html');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取钉钉授权
|
||||||
|
* @return RedirectResponse|Redirect
|
||||||
|
*/
|
||||||
|
public function dingTalkOauthCallback()
|
||||||
|
{
|
||||||
|
$app = new Application(config('dingtalk.'));
|
||||||
|
$data = Request::param();
|
||||||
|
logRecord($data);
|
||||||
|
$user = $app->oauth->use('app-01')->user();
|
||||||
|
$response = $app->oauth->use('app-01')->redirect();
|
||||||
|
$redirect = $response->getTargetUrl();
|
||||||
|
logRecord($redirect);
|
||||||
|
logRecord($user);
|
||||||
|
//用户注册/绑定微信
|
||||||
|
$targetUrl = '/index.html#/member/login?logged=1';
|
||||||
|
if (!$user['errcode']) {
|
||||||
|
$result = $app->user->getUseridByUnionid($user['user_info']['unionid']);
|
||||||
|
if (!$result['errcode']) {
|
||||||
|
$user['user_info']['avatar'] = $result['avatar'];
|
||||||
|
$user['user_info']['mobile'] = $result['mobile'];
|
||||||
|
$user['user_info']['email'] = $result['email'];
|
||||||
|
}
|
||||||
|
Member::dingtalkLogin($user['user_info']);
|
||||||
|
}else{
|
||||||
|
$targetUrl = '/index.html#/member/login?logged=0&message=钉钉登录失败,请重试';
|
||||||
|
}
|
||||||
|
// 登录成功跳转
|
||||||
|
return redirect(Request::domain() . $targetUrl);
|
||||||
|
}
|
||||||
|
}
|
@ -37,33 +37,3 @@ function _uploadFile(File $file, $path_name = '', $saveName = false)
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
function getCurrentMember()
|
|
||||||
{
|
|
||||||
return session('member');
|
|
||||||
}
|
|
||||||
|
|
||||||
function setCurrentMember($data)
|
|
||||||
{
|
|
||||||
return session('member', $data);
|
|
||||||
}
|
|
||||||
|
|
||||||
function getCurrentOrganizationCode()
|
|
||||||
{
|
|
||||||
return session('currentOrganizationCode');
|
|
||||||
}
|
|
||||||
|
|
||||||
function setCurrentOrganizationCode($data)
|
|
||||||
{
|
|
||||||
return session('currentOrganizationCode', $data);
|
|
||||||
}
|
|
||||||
|
|
||||||
function getCurrentOrganization()
|
|
||||||
{
|
|
||||||
return session('organization');
|
|
||||||
}
|
|
||||||
|
|
||||||
function setCurrentOrganization($data)
|
|
||||||
{
|
|
||||||
return session('organization', $data);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
@ -31,13 +31,13 @@ class Index extends BasicApi
|
|||||||
/**
|
/**
|
||||||
* @title 用户菜单
|
* @title 用户菜单
|
||||||
* @description 获取用户菜单列表
|
* @description 获取用户菜单列表
|
||||||
* @author PearProject
|
|
||||||
* @url /project/index
|
|
||||||
* @method GET
|
|
||||||
* @return void :名称
|
* @return void :名称
|
||||||
* @throws DataNotFoundException
|
* @throws DataNotFoundException
|
||||||
* @throws ModelNotFoundException
|
* @throws ModelNotFoundException
|
||||||
* @throws DbException
|
* @throws DbException
|
||||||
|
* @author PearProject
|
||||||
|
* @url /project/index
|
||||||
|
* @method GET
|
||||||
*/
|
*/
|
||||||
public function index()
|
public function index()
|
||||||
{
|
{
|
||||||
@ -81,7 +81,6 @@ class Index extends BasicApi
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 个人个信息
|
* 个人个信息
|
||||||
* @throws DataNotFoundException
|
* @throws DataNotFoundException
|
||||||
|
@ -42,15 +42,15 @@ class Login extends BasicApi
|
|||||||
/**
|
/**
|
||||||
* @title 用户登录
|
* @title 用户登录
|
||||||
* @description 用户登录
|
* @description 用户登录
|
||||||
* @author PearProject
|
|
||||||
* @url /project/login
|
|
||||||
* @method POST
|
|
||||||
* @return void :名称
|
* @return void :名称
|
||||||
* @throws DataNotFoundException
|
* @throws DataNotFoundException
|
||||||
* @throws DbException
|
* @throws DbException
|
||||||
* @throws ModelNotFoundException
|
* @throws ModelNotFoundException
|
||||||
* @throws \think\Exception
|
* @throws \think\Exception
|
||||||
* @throws PDOException
|
* @throws PDOException
|
||||||
|
* @author PearProject
|
||||||
|
* @url /project/login
|
||||||
|
* @method POST
|
||||||
*/
|
*/
|
||||||
public function index()
|
public function index()
|
||||||
{
|
{
|
||||||
@ -92,34 +92,8 @@ class Login extends BasicApi
|
|||||||
if (!$mobile) {
|
if (!$mobile) {
|
||||||
$member['password'] !== $data['password'] && $this->error('账号或密码错误', 201);
|
$member['password'] !== $data['password'] && $this->error('账号或密码错误', 201);
|
||||||
}
|
}
|
||||||
// 更新登录信息
|
$result = Member::login($member);
|
||||||
Db::name('Member')->where(['id' => $member['id']])->update([
|
$this->success('', $result);
|
||||||
'last_login_time' => Db::raw('now()'),
|
|
||||||
]);
|
|
||||||
$list = MemberAccount::where(['member_code' => $member['code']])->order('id asc')->select()->toArray();
|
|
||||||
$organizationList = [];
|
|
||||||
if ($list) {
|
|
||||||
foreach ($list as $item) {
|
|
||||||
$organization = Organization::where(['code' => $item['organization_code']])->find();
|
|
||||||
if ($organization) {
|
|
||||||
$organizationList[] = $organization;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
$member['account_id'] = $list[0]['id'];
|
|
||||||
$member['is_owner'] = $list[0]['is_owner'];
|
|
||||||
$member['authorize'] = $list[0]['authorize'];
|
|
||||||
$member['position'] = $list[0]['position'];
|
|
||||||
$member['department'] = $list[0]['department'];
|
|
||||||
|
|
||||||
setCurrentMember($member);
|
|
||||||
!empty($member['authorize']) && NodeService::applyProjectAuthNode();
|
|
||||||
$member = getCurrentMember();
|
|
||||||
Log::write(json_encode($member), "member-login");
|
|
||||||
$tokenList = JwtService::initToken($member);
|
|
||||||
$accessTokenExp = JwtService::decodeToken($tokenList['accessToken'])->exp;
|
|
||||||
$tokenList['accessTokenExp'] = $accessTokenExp;
|
|
||||||
$this->success('', ['member' => $member, 'tokenList' => $tokenList, 'organizationList' => $organizationList]);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -307,15 +281,23 @@ class Login extends BasicApi
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function _checkLogin()
|
||||||
|
{
|
||||||
|
$loginInfo = session('loginInfo');
|
||||||
|
$this->success('', $loginInfo);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 退出登录
|
* 退出登录
|
||||||
*/
|
*/
|
||||||
public function out()
|
public function _out()
|
||||||
{
|
{
|
||||||
session('user') && LogService::write('系统管理', '用户退出系统成功');
|
session('member') && LogService::write('系统管理', '用户退出系统成功');
|
||||||
!empty($_SESSION) && $_SESSION = [];
|
!empty($_SESSION) && $_SESSION = [];
|
||||||
[session_unset(), session_destroy()];
|
[session_unset(), session_destroy()];
|
||||||
$this->success('退出登录成功!');
|
session('loginInfo', null);
|
||||||
|
$this->success('');
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -27,7 +27,8 @@
|
|||||||
"overtrue/easy-sms": "^1.1",
|
"overtrue/easy-sms": "^1.1",
|
||||||
"phpoffice/phpspreadsheet": "^1.5",
|
"phpoffice/phpspreadsheet": "^1.5",
|
||||||
"firebase/php-jwt": "^5.0",
|
"firebase/php-jwt": "^5.0",
|
||||||
"phpmailer/phpmailer": "^6.0"
|
"phpmailer/phpmailer": "^6.0",
|
||||||
|
"mingyoung/dingtalk": "2.0"
|
||||||
},
|
},
|
||||||
"autoload": {
|
"autoload": {
|
||||||
"psr-4": {
|
"psr-4": {
|
||||||
|
78
config/dingtalk.php
Normal file
78
config/dingtalk.php
Normal file
@ -0,0 +1,78 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
return [
|
||||||
|
/*
|
||||||
|
|-----------------------------------------------------------
|
||||||
|
| 【必填】企业 corpId
|
||||||
|
|-----------------------------------------------------------
|
||||||
|
*/
|
||||||
|
'corp_id' => 'ccc',
|
||||||
|
|
||||||
|
/*
|
||||||
|
|-----------------------------------------------------------
|
||||||
|
| 【必填】应用 AppKey
|
||||||
|
|-----------------------------------------------------------
|
||||||
|
*/
|
||||||
|
'app_key' => 'bbb',
|
||||||
|
|
||||||
|
/*
|
||||||
|
|-----------------------------------------------------------
|
||||||
|
| 【必填】应用 AppSecret
|
||||||
|
|-----------------------------------------------------------
|
||||||
|
*/
|
||||||
|
'app_secret' => 'aaa',
|
||||||
|
|
||||||
|
/*
|
||||||
|
|-----------------------------------------------------------
|
||||||
|
| 【选填】加解密
|
||||||
|
|-----------------------------------------------------------
|
||||||
|
| 此处的 `token` 和 `aes_key` 用于事件通知的加解密
|
||||||
|
| 如果你用到事件回调功能,需要配置该两项
|
||||||
|
*/
|
||||||
|
'token' => 'uhl3CZbtsmf93bFPanmMenhWwrqbSwPc',
|
||||||
|
'aes_key' => 'qZEOmHU2qYYk6n6vqLfi3FAhcp9bGA2kgbfnsXDrGgN',
|
||||||
|
|
||||||
|
/*
|
||||||
|
|-----------------------------------------------------------
|
||||||
|
| 【选填】后台免登配置信息
|
||||||
|
|-----------------------------------------------------------
|
||||||
|
| 如果你用到应用管理后台免登功能,需要配置该项
|
||||||
|
*/
|
||||||
|
'sso_secret' => 'Fx9_i5dSW5tpGtjalksdf98JF8uj32xb4NJQR5G9-VSchasd98asfdMmLR',
|
||||||
|
|
||||||
|
/*
|
||||||
|
|-----------------------------------------------------------
|
||||||
|
| 【选填】第三方网站 OAuth 授权
|
||||||
|
|-----------------------------------------------------------
|
||||||
|
| 如果你用到扫码登录、钉钉内免登和密码登录第三方网站,需要配置该项
|
||||||
|
*/
|
||||||
|
'oauth' => [
|
||||||
|
/*
|
||||||
|
|-------------------------------------------
|
||||||
|
| `app-01` 为你自定义的名称,不要重复即可
|
||||||
|
|-------------------------------------------
|
||||||
|
| 数组内需要配置 `client_id`, `client_secret`, `scope` 和 `redirect` 四项
|
||||||
|
|
|
||||||
|
| `client_id` 为钉钉登录应用的 `appId`
|
||||||
|
| `client_secret` 为钉钉登录应用的 `appSecret`
|
||||||
|
| `scope`:
|
||||||
|
| - 扫码登录第三方网站和密码登录第三方网站填写 `snsapi_login`
|
||||||
|
| - 钉钉内免登第三方网站填写 `snsapi_auth`
|
||||||
|
| `redirect` 为回调地址
|
||||||
|
*/
|
||||||
|
'app-01' => [
|
||||||
|
'client_id' => 'aaa',
|
||||||
|
'client_secret' => 'bbb-Tu2CsUZN-ZlQXWQ',
|
||||||
|
'scope' => 'snsapi_login',
|
||||||
|
'redirect' => 'https://example.com',
|
||||||
|
],
|
||||||
|
/*
|
||||||
|
|-------------------------------------------
|
||||||
|
| 可配置多个 OAuth 应用,数组内内容同上
|
||||||
|
|-------------------------------------------
|
||||||
|
*/
|
||||||
|
'app-02' => [
|
||||||
|
// ...
|
||||||
|
]
|
||||||
|
]
|
||||||
|
];
|
BIN
static/image/default/logo.png
Normal file
BIN
static/image/default/logo.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 6.0 KiB |
Loading…
x
Reference in New Issue
Block a user