Ei kuvausta
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

Admin.php 68KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547
  1. <?php
  2. /**
  3. * 易优CMS
  4. * ============================================================================
  5. * 版权所有 2016-2028 海口快推科技有限公司,并保留所有权利。
  6. * 网站地址: http://www.eyoucms.com
  7. * ----------------------------------------------------------------------------
  8. * 如果商业用途务必到官方购买正版授权, 以免引起不必要的法律纠纷.
  9. * ============================================================================
  10. * Author: 小虎哥 <1105415366@qq.com>
  11. * Date: 2018-4-3
  12. */
  13. namespace app\admin\controller;
  14. use think\Page;
  15. use think\Verify;
  16. use think\Db;
  17. use think\db\Query;
  18. use think\Session;
  19. use app\admin\model\AuthRole;
  20. use app\admin\logic\AjaxLogic;
  21. class Admin extends Base {
  22. private $admin_info = [];
  23. public function _initialize() {
  24. parent::_initialize();
  25. $this->admin_info = session('?admin_info') ? session('admin_info') : [];
  26. $this->assign('admin_info', $this->admin_info);
  27. }
  28. public function index()
  29. {
  30. $list = array();
  31. $keywords = input('keywords/s');
  32. $keywords = addslashes(trim($keywords));
  33. $condition = array();
  34. if (!empty($keywords)) {
  35. $condition['a.user_name|a.true_name'] = array('LIKE', "%{$keywords}%");
  36. }
  37. /*权限控制 by 小虎哥*/
  38. if (0 < intval($this->admin_info['role_id'])) {
  39. $condition['a.admin_id|a.parent_id'] = $this->admin_info['admin_id'];
  40. } else {
  41. if (!empty($this->admin_info['parent_id'])) {
  42. $condition['a.admin_id|a.parent_id'] = $this->admin_info['admin_id'];
  43. }
  44. }
  45. /*--end*/
  46. /**
  47. * 数据查询
  48. */
  49. $count = DB::name('admin')->alias('a')->where($condition)->count();// 查询满足要求的总记录数
  50. $Page = new Page($count, config('paginate.list_rows'));// 实例化分页类 传入总记录数和每页显示的记录数
  51. $list = DB::name('admin')->field('a.*, b.name AS role_name')
  52. ->alias('a')
  53. ->join('__AUTH_ROLE__ b', 'a.role_id = b.id', 'LEFT')
  54. ->where($condition)
  55. ->order('a.admin_id asc')
  56. ->limit($Page->firstRow.','.$Page->listRows)
  57. ->select();
  58. $locklist = tpSetting('adminlogin');
  59. foreach ($list as $key => $val) {
  60. if (0 >= intval($val['role_id'])) {
  61. $val['role_name'] = !empty($val['parent_id']) ? '超级管理员' : '创始人';
  62. }
  63. // 是否被锁定
  64. $login_lock_key = 'adminlogin_'.md5('login_lock_'.$val['user_name'].clientIP()); // 是否被锁定
  65. $val['is_locklogin'] = !empty($locklist[$login_lock_key]) ? 1 : 0;
  66. $list[$key] = $val;
  67. }
  68. $show = $Page->show();// 分页显示输出
  69. $this->assign('page',$show);// 赋值分页输出
  70. $this->assign('list',$list);// 赋值数据集
  71. $this->assign('pager',$Page);// 赋值分页集
  72. // 第三方扫码绑定与解绑
  73. $wxlist = [];
  74. $thirdata = login_third_type();
  75. if ('EyouGzhLogin' == $thirdata['type']) {
  76. $wxlist = Db::name('admin_wxlogin')->where(['type'=>1])->getAllWithIndex('admin_id');
  77. } else if ('WechatLogin' == $thirdata['type']) {
  78. $wxlist = Db::name('admin_wxlogin')->where(['type'=>2])->getAllWithIndex('admin_id');
  79. }
  80. $this->assign('wxlist', $wxlist);
  81. $this->assign('thirdata', $thirdata);
  82. /*第一次同步CMS用户的栏目ID到权限组里*/
  83. $this->syn_built_auth_role();
  84. /*--end*/
  85. return $this->fetch();
  86. }
  87. /*
  88. * 管理员登陆
  89. */
  90. public function login()
  91. {
  92. if (session('?admin_id') && session('admin_id') > 0) {
  93. $web_adminbasefile = tpCache('global.web_adminbasefile');
  94. $web_adminbasefile = !empty($web_adminbasefile) ? $web_adminbasefile : $this->root_dir.'/login.php';
  95. if (stristr($web_adminbasefile, 'index.php')) {
  96. $baseFile = explode('/', request()->baseFile());
  97. $web_adminbasefile = end($baseFile);
  98. $web_adminbasefile = $this->root_dir.'/'.$web_adminbasefile;
  99. }
  100. $this->success("您已登录", $web_adminbasefile);
  101. }
  102. $is_vertify = 1; // 默认开启验证码
  103. $admin_login_captcha = config('captcha.admin_login');
  104. if (!function_exists('imagettftext') || empty($admin_login_captcha['is_on'])) {
  105. $is_vertify = 0; // 函数不存在,不符合开启的条件
  106. } else if (is_file('./data/conf/admin_vertify.txt')) {
  107. $is_exist = @file_get_contents('./data/conf/admin_vertify.txt');
  108. if ($is_exist !== false && empty($is_exist)) {
  109. $is_vertify = 0;
  110. }
  111. }
  112. $this->assign('is_vertify', $is_vertify);
  113. /*----------------微信扫码登录 start---------------*/
  114. $login_type = 1; //仅账号密码登录 2-账号密码登录&微信扫码登录 3-仅微信扫码登录
  115. $thirdata = login_third_type();
  116. $third_login = !empty($thirdata['type']) ? $thirdata['type'] : '';
  117. if ('EyouGzhLogin' == $third_login) {
  118. if (empty($thirdata['data']['force'])){
  119. $login_type = 2; //2-账号密码登录&微信扫码登录
  120. } else {
  121. $login_type = 3; //仅微信扫码登录
  122. }
  123. } else if ('WechatLogin' == $third_login) {
  124. if (empty($thirdata['data']['security_wechat_forcelogin'])) {
  125. $login_type = 2; //2-账号密码登录&微信扫码登录
  126. } else {
  127. $login_type = 3; //仅微信扫码登录
  128. }
  129. }
  130. $this->assign('login_type', $login_type);
  131. $this->assign('third_login', $third_login);
  132. /*----------------微信扫码登录 end---------------*/
  133. if (IS_POST) {
  134. if (!in_array($login_type, [1,2])) {
  135. $this->error('强制扫码,不支持普通登录!');
  136. }
  137. $post = input('post.');
  138. if (!function_exists('session_start')) {
  139. $this->error('请联系空间商,开启php的session扩展!');
  140. }
  141. if (!testWriteAble(ROOT_PATH.config('session.path').'/')) {
  142. $this->error('请仔细检查以下问题:<br/>1、磁盘空间大小是否100%;<br/>2、站点目录权限是否为755;<br/>3、站点所有目录的权限,禁止用root:root ;<br/>4、如还没解决,请点击:<a href="http://www.eyoucms.com/wenda/6958.html" target="_blank">查看教程</a>');
  143. }
  144. if (1 == $is_vertify) {
  145. $verify = new Verify();
  146. if (!$verify->check(input('post.vertify'), "admin_login")) {
  147. $this->error('验证码错误');
  148. }
  149. }
  150. $is_clicap = 0; // 默认关闭文字验证码
  151. if (is_dir('./weapp/Clicap/')) {
  152. $ClicapRow = model('Weapp')->getWeappList('Clicap');
  153. if (!empty($ClicapRow['status']) && 1 == $ClicapRow['status']) {
  154. if (!empty($ClicapRow['data']) && $ClicapRow['data']['captcha']['admin_login']['is_on'] == 1) {
  155. $clicaptcha_info = input('post.clicaptcha-submit-info');
  156. $clicaptcha = new \weapp\Clicap\vendor\Clicaptcha;
  157. if (empty($clicaptcha_info) || !$clicaptcha->check($clicaptcha_info, false)) {
  158. $this->error('文字点击验证错误!');
  159. }
  160. }
  161. }
  162. }
  163. $admin_count = 0;
  164. $user_name = input('post.user_name/s');
  165. $password = input('post.password/s');
  166. /*---------登录错误次数的限制 start----------*/
  167. $globalConfing = tpCache('global');
  168. $web_login_lockopen = 0; // 是否开启登录失败锁定
  169. if (!isset($globalConfing['web_login_lockopen']) || !empty($globalConfing['web_login_lockopen'])) {
  170. $web_login_lockopen = 1;
  171. $admin_count = Db::name('admin')->where(['user_name'=>$user_name])->count();
  172. if (!empty($admin_count)) {
  173. $loginErrtotal = !empty($globalConfing['web_login_errtotal']) ? intval($globalConfing['web_login_errtotal']) : config('login_errtotal'); // 登录错误最大次数
  174. $loginErrexpire = !empty($globalConfing['web_login_errexpire']) ? intval($globalConfing['web_login_errexpire']) : config('login_errexpire'); // 登录错误最大限制时间
  175. $clientIP = clientIP();
  176. $login_errnum_key = 'adminlogin_'.md5('login_errnum_'.$user_name.$clientIP);
  177. $login_errtime_key = 'adminlogin_'.md5('login_errtime_'.$user_name.$clientIP);
  178. $login_lock_key = 'adminlogin_'.md5('login_lock_'.$user_name.$clientIP); // 是否被锁定
  179. $loginErrnum = (int)tpSetting('adminlogin.'.$login_errnum_key); // 登录错误次数
  180. $loginErrtime = tpSetting('adminlogin.'.$login_errtime_key); // 最后一次登录错误时间
  181. if ($loginErrnum >= $loginErrtotal) {
  182. if (getTime() < $loginErrtime + $loginErrexpire) {
  183. adminLog("登录失败(已被锁定,登录错误超限{$loginErrtotal}次)");
  184. $surplus_time = ($loginErrtime + $loginErrexpire) - getTime();
  185. if ($surplus_time <= 0) {
  186. $surplus_time = 1;
  187. }
  188. $this->error("多次登录失败,距离解锁还有".ceil($surplus_time/60)."分钟!");
  189. } else {
  190. // 重置登录错误次数
  191. $loginErrnum = $loginErrtime = $login_lock = 0;
  192. tpSetting('adminlogin', [$login_errnum_key => $loginErrnum]);
  193. tpSetting('adminlogin', [$login_errtime_key => $loginErrtime]);
  194. tpSetting('adminlogin', [$login_lock_key => $login_lock]);
  195. }
  196. }
  197. }
  198. }
  199. /*---------登录错误次数的限制 end----------*/
  200. if (!empty($user_name) && !empty($password)) {
  201. $condition['user_name'] = $user_name;
  202. $admin_info = Db::name('admin')->where($condition)->find();
  203. if (!empty($admin_info)) {
  204. /*等保密码复杂度验证 start*/
  205. if (is_dir('./weapp/Equal/')) {
  206. $equal_privkey = input('post.equal_privkey/s');
  207. $equalLogic = new \weapp\Equal\logic\EqualLogic;
  208. $equalLogic->loginLogic($password, $equal_privkey);
  209. }
  210. /*等保密码复杂度验证 end*/
  211. $entry = pwd_encry_type($admin_info['password']);
  212. $encry_password = func_encrypt($password, true, $entry);
  213. if ($admin_info['password'] == $encry_password) {
  214. if ($admin_info['status'] == 0) {
  215. adminLog('登录失败(用户名被禁用)');
  216. $this->error('用户名被禁用!');
  217. }
  218. $admin_info = adminLoginAfter($admin_info['admin_id'], $this->session_id);
  219. // 检查密码复杂度
  220. session('admin_login_pwdlevel', checkPasswordLevel($password));
  221. adminLog('后台登录');
  222. $url = session('from_url') ? session('from_url') : $this->request->baseFile();
  223. session('isset_author', null); // 内置勿动
  224. // 同步追加一个后台管理员到会员用户表
  225. $isFounder = !empty($admin_info['parent_id']) ? 0 : 1;
  226. $this->syn_users_login($admin_info, $isFounder);
  227. $this->success('登录成功', $url);
  228. }
  229. }
  230. }
  231. /*----------记录登录错误次数 start-----------*/
  232. if (!empty($admin_count) && !empty($web_login_lockopen)) {
  233. $login_errnum = $loginErrnum + 1;
  234. $login_num = $loginErrtotal - $login_errnum;
  235. tpSetting('adminlogin', [$login_errnum_key=>$login_errnum]);
  236. tpSetting('adminlogin', [$login_errtime_key=>getTime()]);
  237. if ($login_num > 0) {
  238. $this->error("用户名或密码错误,您还可以尝试[{$login_num}]次!");
  239. } else {
  240. tpSetting('adminlogin', [$login_lock_key => 1]);
  241. $this->error("登录错误超限{$loginErrtotal}次,账号将被锁定".ceil($loginErrexpire/60)."分钟!");
  242. }
  243. }
  244. /*----------记录登录错误次数 end-----------*/
  245. adminLog("登录失败({$user_name})");
  246. $this->error("用户名或密码错误!");
  247. }
  248. $ajaxLogic = new AjaxLogic;
  249. $ajaxLogic->login_handle();
  250. // 仅微信扫码登录
  251. // if ('WechatLogin' == $third_login && 3 == $login_type) {
  252. // $this->wechatLogin();
  253. // }
  254. $this->global = tpCache('global');
  255. $this->assign('global', $this->global);
  256. $this->assign('time', getTime());
  257. /*等保密码复杂度验证 start*/
  258. $pwdJsCode = '';
  259. if (is_dir('./weapp/Equal/')) {
  260. $equalLogic = new \weapp\Equal\logic\EqualLogic;
  261. $pwdJsCode = $equalLogic->pwdJsCode();
  262. }
  263. if ('close' == $pwdJsCode) {
  264. $pwdJsCode = '';
  265. }
  266. $this->assign('pwdJsCode', $pwdJsCode);
  267. /*等保密码复杂度验证 end*/
  268. session('admin_info', null);
  269. $viewfile = 'admin/login';
  270. if (2 <= $this->php_servicemeal) {
  271. $viewfile = 'admin/login_zy';
  272. }
  273. $web_theme_login_tplname = empty($this->globalConfig['web_theme_login_tplname']) ? '' : $this->globalConfig['web_theme_login_tplname'];
  274. if (!empty($web_theme_login_tplname) && file_exists("application/admin/template/theme/{$web_theme_login_tplname}")) {
  275. $login_tplname = str_ireplace('.htm', '', $web_theme_login_tplname);
  276. $viewfile = "theme/{$login_tplname}";
  277. }
  278. if (is_dir('./weapp/Mbackend/') && isMobile()) {
  279. $viewfile = 'weapp/Mbackend/template/admin/login_m.htm';
  280. if (2 <= $this->php_servicemeal) {
  281. $viewfile = 'weapp/Mbackend/template/admin/login_zy_m.htm';
  282. }
  283. // 是否配置微信公众号登录信息
  284. $wechat = tpSetting("OpenMinicode.conf_wechat") ? json_decode(tpSetting("OpenMinicode.conf_wechat"), true) : [];
  285. $this->assign('wechat', $wechat);
  286. return $this->fetch("{$viewfile}");
  287. } else {
  288. return $this->fetch(":{$viewfile}");
  289. }
  290. }
  291. // 后台管理插件(手机版)--微信公众号登录
  292. public function ajax_admin_wechat_login()
  293. {
  294. if (is_dir('./weapp/Mbackend/') && isMobile()) {
  295. // 调用逻辑层
  296. $mbackendLogic = new \weapp\Mbackend\logic\MbackendLogic;
  297. $url = $mbackendLogic->ajaxAdminWechatLogin();
  298. $this->success('授权成功', $url);
  299. } else {
  300. $this->error('请先安装后台管理插件(手机版)');
  301. }
  302. }
  303. // 后台管理插件(手机版)--获取微信登录用户信息
  304. public function get_admin_wechat_users()
  305. {
  306. if (is_dir('./weapp/Mbackend/') && isMobile()) {
  307. // 调用逻辑层
  308. $mbackendLogic = new \weapp\Mbackend\logic\MbackendLogic;
  309. $we_user = $mbackendLogic->getAdminWechatUsers();
  310. $admin_info = adminLoginAfter($we_user['admin_id'], session_id());
  311. if (!empty($admin_info)) {
  312. adminLog('微信授权登录成功');
  313. $this->success('登录成功', weapp_url('Mbackend/Mbackend/index'));
  314. } else {
  315. $this->success('404:您没有操作权限,请联系超级管理员分配权限');
  316. }
  317. } else {
  318. $this->error('请先安装后台管理插件(手机版)');
  319. }
  320. }
  321. private function wechatLogin()
  322. {
  323. $url = url('Admin/wechat_login', [], true, true);
  324. $url = preg_replace('/^http(s?)/i', $this->request->scheme(), $url);
  325. $this->redirect($url);
  326. exit;
  327. }
  328. /**
  329. * 解除锁定登录
  330. * @return [type] [description]
  331. */
  332. public function ajax_unlock_login()
  333. {
  334. $admin_id = input('param.id/d');
  335. if (!empty($admin_id) && IS_POST) {
  336. if (!empty($this->admin_info['parent_id']) || -1 != $this->admin_info['role_id']) {
  337. $this->error('该功能仅限于创始人操作!');
  338. }
  339. $clientIP = clientIP();
  340. $user_name = Db::name('admin')->where(['admin_id'=>$admin_id])->value('user_name');
  341. $login_errnum_key = 'adminlogin_'.md5('login_errnum_'.$user_name.$clientIP);
  342. $login_errtime_key = 'adminlogin_'.md5('login_errtime_'.$user_name.$clientIP);
  343. $login_lock_key = 'adminlogin_'.md5('login_lock_'.$user_name.$clientIP); // 是否被锁定
  344. tpSetting('adminlogin', [$login_errnum_key => 0]);
  345. tpSetting('adminlogin', [$login_errtime_key => 0]);
  346. tpSetting('adminlogin', [$login_lock_key => 0]);
  347. adminLog('解除锁定:'.$user_name);
  348. $this->success('操作成功');
  349. }
  350. $this->error('操作失败');
  351. }
  352. /**
  353. * 验证码获取
  354. */
  355. public function vertify()
  356. {
  357. /*验证码插件开关*/
  358. $admin_login_captcha = config('captcha.admin_login');
  359. $config = (!empty($admin_login_captcha['is_on']) && !empty($admin_login_captcha['config'])) ? $admin_login_captcha['config'] : config('captcha.default');
  360. /*--end*/
  361. ob_clean(); // 清空缓存,才能显示验证码
  362. $Verify = new Verify($config);
  363. $Verify->entry('admin_login');
  364. exit();
  365. }
  366. /**
  367. * 退出登陆
  368. */
  369. public function logout()
  370. {
  371. adminLog('安全退出');
  372. session_unset();
  373. // session_destroy();
  374. session::clear();
  375. cookie('admin-treeClicked', null); // 清除并恢复栏目列表的展开方式
  376. cookie('admin-treeClicked-1649642233', null); // 清除并恢复内容管理的展开方式
  377. $this->success("安全退出", request()->baseFile());
  378. }
  379. /**
  380. * 新增管理员时,检测用户名是否与前台用户名相同
  381. */
  382. public function ajax_add_user_name()
  383. {
  384. if (IS_AJAX_POST) {
  385. $user_name = input('post.user_name/s');
  386. if (Db::name('admin')->where("user_name", $user_name)->count()) {
  387. $this->error("此用户名已被注册,请更换!");
  388. }
  389. $row = Db::name('users')->field('users_id')->where([
  390. 'username' => $user_name,
  391. 'lang' => $this->admin_lang,
  392. ])->find();
  393. if (!empty($row)) {
  394. $this->error('已有相同会员名,将其转为系统账号?');
  395. } else {
  396. $this->success('会员名不存在,无需提示!');
  397. }
  398. }
  399. }
  400. /**
  401. * 新增管理员
  402. */
  403. public function admin_add()
  404. {
  405. $this->language_access(); // 多语言功能操作权限
  406. if (IS_POST) {
  407. $data = input('post.');
  408. if (0 < $this->admin_info['role_id']) {
  409. $this->error("超级管理员才能操作!");
  410. }
  411. if (empty($data['password']) || !trim($data['password'])) {
  412. $this->error("用户密码不能为空!", null, ['input_name'=>'password']);
  413. } else {
  414. /*等保密码复杂度验证 start*/
  415. if (is_dir('./weapp/Equal/')) {
  416. $equalLogic = new \weapp\Equal\logic\EqualLogic;
  417. $eqData = $equalLogic->pwdValidate($data['password']);
  418. if (isset($eqData['code']) && empty($eqData['code'])) {
  419. $this->error($eqData['msg']);
  420. }
  421. }
  422. /*等保密码复杂度验证 end*/
  423. }
  424. $data['user_name'] = trim($data['user_name']);
  425. $data['password'] = func_encrypt($data['password'], true, pwd_encry_type('bcrypt'));
  426. $data['role_id'] = intval($data['role_id']);
  427. $data['parent_id'] = $this->admin_info['admin_id'];
  428. $data['add_time'] = getTime();
  429. if (empty($data['pen_name'])) {
  430. $data['pen_name'] = '小编';
  431. }
  432. // 处理数据验证
  433. $error = handleEyouDataValidate('user_name', '__token_admin_add__', $data, '用户名不能为空!');
  434. if (!empty($error)) $this->error($error);
  435. if (Db::name('admin')->where(['user_name'=>$data['user_name']])->count()) {
  436. $this->error("此用户名已被注册,请更换", url('Admin/admin_add'), ['input_name'=>'user_name']);
  437. } else {
  438. $admin_id = Db::name('admin')->insertGetId($data);
  439. if ($admin_id !== false) {
  440. adminLog('新增管理员:'.$data['user_name']);
  441. /*同步追加一个后台管理员到会员用户表*/
  442. try {
  443. $usersInfo = Db::name('users')->field('users_id')->where([
  444. 'username' => $data['user_name'],
  445. 'lang' => $this->admin_lang,
  446. ])->find();
  447. if (!empty($usersInfo)) {
  448. $r = Db::name('users')->where(['users_id'=>$usersInfo['users_id']])->update([
  449. 'nickname' => $data['user_name'],
  450. 'admin_id' => $admin_id,
  451. 'is_activation' => 1,
  452. 'is_lock' => 0,
  453. 'is_del' => 0,
  454. 'update_time' => getTime(),
  455. ]);
  456. if ($r !== false) {
  457. $users_id = $usersInfo['users_id'];
  458. }
  459. } else {
  460. // 获取要添加的用户名
  461. $username = $this->GetUserName($data['user_name']);
  462. $password = getTime();
  463. $password = func_encrypt($password, false, pwd_encry_type('bcrypt'));
  464. $AddData = [
  465. 'username' => $username,
  466. 'nickname' => $username,
  467. 'password' => $password,
  468. 'level' => 1,
  469. 'lang' => $this->admin_lang,
  470. 'reg_time' => getTime(),
  471. 'head_pic' => ROOT_DIR . '/public/static/common/images/dfboy.png',
  472. 'register_place' => 1,
  473. 'admin_id' => $admin_id,
  474. ];
  475. $users_id = Db::name('users')->insertGetId($AddData);
  476. }
  477. if ($users_id !== false) {
  478. Db::name('admin')->where(['admin_id'=>$admin_id])->update([
  479. 'syn_users_id' => intval($users_id),
  480. 'update_time' => getTime(),
  481. ]);
  482. }
  483. } catch (\Exception $e) {}
  484. /* END */
  485. $this->success("操作成功", url('Admin/index'));
  486. } else {
  487. $this->error("操作失败");
  488. }
  489. }
  490. }
  491. // 权限组
  492. $admin_role_list = model('AuthRole')->getRoleAll();
  493. $this->assign('admin_role_list', $admin_role_list);
  494. // 模块组
  495. $modules = getAllMenu();
  496. $this->assign('modules', $modules);
  497. // 权限集
  498. $auth_rules = get_auth_rule(['is_modules'=>1]);
  499. $auth_rule_list = group_same_key($auth_rules, 'menu_id');
  500. foreach ($auth_rule_list as $key => $val) {
  501. if (is_array($val)) {
  502. $sort_order = [];
  503. foreach ($val as $_k => $_v) {
  504. $sort_order[$_k] = $_v['sort_order'];
  505. }
  506. array_multisort($sort_order, SORT_ASC, $val);
  507. $auth_rule_list[$key] = $val;
  508. }
  509. }
  510. $this->assign('auth_rule_list', $auth_rule_list);
  511. // 栏目
  512. $arctype_list = Db::name('arctype')->where([
  513. 'is_del' => 0,
  514. ])->order("grade desc")->select();
  515. $arctype_p_html = $arctype_child_html = "";
  516. $arctype_all = list_to_tree($arctype_list);
  517. foreach ($arctype_all as $key => $arctype) {
  518. if (!empty($arctype['children'])) {
  519. if ($key > 0) {
  520. $arctype_p_html .= '<em class="arctype_bg expandable"></em>';
  521. } else {
  522. $arctype_p_html .= '<em class="arctype_bg collapsable"></em>';
  523. }
  524. $arctype_child_html .= '<div class="arctype_child" id="arctype_child_' . $arctype['id'] . '"';
  525. if ($arctype_all[0]['id'] == $arctype['id']) {
  526. $arctype_child_html .= ' style="display: block;" ';
  527. }
  528. $arctype_child_html .= '>';
  529. $arctype_child_html .= $this->get_arctype_child_html($arctype);
  530. $arctype_child_html .= '</div>';
  531. }
  532. $arctype_p_html .= '<label>';
  533. $arctype_p_html .= '<img class="cboximg" src="'.ROOT_DIR.'/public/static/admin/images/ok.png" />';
  534. $arctype_p_html .= '<input type="checkbox" class="arctype_cbox arctype_id_' . $arctype['id'] . ' none" name="permission[arctype][]" value="' . $arctype['id'] . '"';
  535. $arctype_p_html .= ' checked="checked" ';
  536. $arctype_p_html .= ' />' . $arctype['typename'] . '</label>&nbsp;';
  537. }
  538. $this->assign('arctype_p_html', $arctype_p_html);
  539. $this->assign('arctype_child_html', $arctype_child_html);
  540. // 插件
  541. $plugins = model('Weapp')->getList(['status'=>1]);
  542. $this->assign('plugins', $plugins);
  543. /*等保密码复杂度验证 start*/
  544. $pwdJsCode = '';
  545. if (is_dir('./weapp/Equal/')) {
  546. $equalLogic = new \weapp\Equal\logic\EqualLogic;
  547. $pwdJsCode = $equalLogic->pwdJsCode();
  548. }
  549. if ('close' == $pwdJsCode) {
  550. $pwdJsCode =<<<EOF
  551. if (password.length < 5) {
  552. showErrorMsg('用户密码至少5位或以上!');
  553. $('input[name=password]').focus();
  554. return false;
  555. }
  556. EOF;
  557. }
  558. $this->assign('pwdJsCode', $pwdJsCode);
  559. /*等保密码复杂度验证 end*/
  560. return $this->fetch();
  561. }
  562. /**
  563. * 编辑管理员
  564. */
  565. public function admin_edit()
  566. {
  567. if (IS_POST) {
  568. $data = input('post.');
  569. $id = $data['admin_id'] = intval($data['admin_id']);
  570. $user_name = $data['user_name'] = isset($data['user_name']) ? trim($data['user_name']) : '';
  571. empty($data['pen_name']) && $data['pen_name'] = '小编';
  572. if ($id == $this->admin_info['admin_id']) {
  573. unset($data['role_id']); // 不能修改自己的权限组
  574. } else if (0 < $this->admin_info['role_id'] && $this->admin_info['admin_id'] != $id) {
  575. $this->error('禁止更改别人的信息!');
  576. }
  577. if (empty($this->admin_info['parent_id'])) { // 创始人才可以修改所有管理员的用户名
  578. if (empty($user_name)) {
  579. $this->error('用户名不能为空!', null, ['input_name'=>'user_name']);
  580. } else {
  581. if ($user_name == $this->admin_info['user_name']) {
  582. unset($data['user_name']);
  583. } else {
  584. $count = Db::name('admin')->where(['user_name'=>$user_name, 'admin_id'=>['NEQ', $id]])->count();
  585. if (!empty($count)) {
  586. $this->error("此用户名已被注册,请更换", null, ['input_name'=>'user_name']);
  587. }
  588. }
  589. }
  590. } else {
  591. unset($data['user_name']);
  592. }
  593. $password = $data['password'];
  594. if (empty($password) || !trim($password)) {
  595. unset($data['password']);
  596. }else{
  597. /*等保密码复杂度验证 start*/
  598. if (is_dir('./weapp/Equal/')) {
  599. $equalLogic = new \weapp\Equal\logic\EqualLogic;
  600. $eqData = $equalLogic->pwdValidate($password);
  601. if (isset($eqData['code']) && empty($eqData['code'])) {
  602. $this->error($eqData['msg']);
  603. }
  604. }
  605. /*等保密码复杂度验证 end*/
  606. if (!empty($this->globalConfig['security_verifyfunc']) && in_array('edit_pwd', $this->globalConfig['security_verifyfunc'])) {
  607. if (true !== security_answer_verify()) {
  608. $this->error("请先密保答案验证");
  609. }
  610. }
  611. $entry = pwd_encry_type('bcrypt');
  612. $data['password'] = func_encrypt($password, true, $entry);
  613. }
  614. // 处理数据验证
  615. $error = handleEyouDataValidate('admin_id', '__token_admin_edit__', $data, '用户名不能为空!');
  616. if (!empty($error)) $this->error($error);
  617. /*不允许修改自己的权限组*/
  618. if (isset($data['role_id'])) {
  619. if (0 < $this->admin_info['role_id'] && intval($data['role_id']) != $this->admin_info['role_id']) {
  620. $data['role_id'] = $this->admin_info['role_id'];
  621. }
  622. }
  623. /*--end*/
  624. $data['update_time'] = getTime();
  625. $r = Db::name('admin')->where('admin_id', $id)->save($data);
  626. if ($r !== false) {
  627. if ($id == $this->admin_info['admin_id']) {
  628. // 检查密码复杂度
  629. session('admin_login_pwdlevel', checkPasswordLevel($password));
  630. // 过滤存储在session文件的敏感信息
  631. $this->admin_info = array_merge($this->admin_info, $data);
  632. foreach (['user_name','true_name','password'] as $key => $val) {
  633. unset($this->admin_info[$val]);
  634. }
  635. session('admin_info', $this->admin_info);
  636. }
  637. /*同步相同数据到会员表对应的会员*/
  638. $syn_users_id = Db::name('admin')->where(['admin_id'=>$data['admin_id']])->getField('syn_users_id');
  639. if (!empty($syn_users_id)) {
  640. $updateData = [
  641. 'nickname' => $data['pen_name'],
  642. 'head_pic' => $data['head_pic'],
  643. 'update_time' => getTime(),
  644. ];
  645. Db::name('users')->where(['users_id'=>$syn_users_id])->update($updateData);
  646. }
  647. /*end*/
  648. adminLog('编辑管理员:'.$user_name);
  649. $this->success("操作成功", url('Admin/index'));
  650. } else {
  651. $this->error("操作失败");
  652. }
  653. }
  654. $id = input('param.id/d', 0);
  655. if (empty($id)) {
  656. $this->error('数据不存在,退出尝试登录!');
  657. exit;
  658. }
  659. $info = Db::name('admin')->field('password', true)->find($id);
  660. if (empty($info)) {
  661. $this->error('数据不存在,请联系管理员!');
  662. exit;
  663. }
  664. $this->assign('info',$info);
  665. $iframe = input('param.iframe/d', 0);
  666. $this->assign('iframe',$iframe);
  667. // 有权限查看的管理员列表
  668. $condition = array();
  669. if (0 < intval($this->admin_info['role_id'])) {
  670. $condition['a.admin_id|a.parent_id'] = $this->admin_info['admin_id'];
  671. } else {
  672. if (!empty($this->admin_info['parent_id'])) {
  673. $condition['a.admin_id|a.parent_id'] = $this->admin_info['admin_id'];
  674. }
  675. }
  676. $admin_list = Db::name('admin')->field('a.*')
  677. ->alias('a')
  678. ->where($condition)
  679. ->order('a.admin_id asc')
  680. ->getAllWithIndex('admin_id');
  681. if (empty($admin_list[$info['admin_id']])) {
  682. $this->error('您没有操作权限,请联系超级管理员分配权限');
  683. exit;
  684. }
  685. // 当前角色信息
  686. $admin_role_model = model('AuthRole');
  687. $role_info = $admin_role_model->getRole(array('id' => $info['role_id']));
  688. $this->assign('role_info', $role_info);
  689. // 权限组
  690. $admin_role_list = $admin_role_model->getRoleAll();
  691. $this->assign('admin_role_list', $admin_role_list);
  692. // 模块组
  693. $modules = getAllMenu();
  694. $this->assign('modules', $modules);
  695. // 权限集
  696. $auth_rules = get_auth_rule(['is_modules'=>1]);
  697. $auth_rule_list = group_same_key($auth_rules, 'menu_id');
  698. foreach ($auth_rule_list as $key => $val) {
  699. if (is_array($val)) {
  700. $sort_order = [];
  701. foreach ($val as $_k => $_v) {
  702. $sort_order[$_k] = $_v['sort_order'];
  703. }
  704. array_multisort($sort_order, SORT_ASC, $val);
  705. $auth_rule_list[$key] = $val;
  706. }
  707. }
  708. $this->assign('auth_rule_list', $auth_rule_list);
  709. // 栏目
  710. $arctype_list = Db::name('arctype')->where([
  711. 'is_del' => 0,
  712. ])->order("grade desc")->select();
  713. $arctype_p_html = $arctype_child_html = "";
  714. $arctype_all = list_to_tree($arctype_list);
  715. foreach ($arctype_all as $key => $arctype) {
  716. if (!empty($arctype['children'])) {
  717. if ($key > 0) {
  718. $arctype_p_html .= '<em class="arctype_bg expandable"></em>';
  719. } else {
  720. $arctype_p_html .= '<em class="arctype_bg collapsable"></em>';
  721. }
  722. $arctype_child_html .= '<div class="arctype_child" id="arctype_child_' . $arctype['id'] . '"';
  723. if ($arctype_all[0]['id'] == $arctype['id']) {
  724. $arctype_child_html .= ' style="display: block;" ';
  725. }
  726. $arctype_child_html .= '>';
  727. $arctype_child_html .= $this->get_arctype_child_html($arctype,$role_info);
  728. $arctype_child_html .= '</div>';
  729. }
  730. $arctype_p_html .= '<label>';
  731. if (!empty($role_info['permission']['arctype']) && in_array($arctype['id'], $role_info['permission']['arctype'])) {
  732. $arctype_p_html .= '<img class="cboximg" src="'.ROOT_DIR.'/public/static/admin/images/ok.png" />';
  733. }else{
  734. $arctype_p_html .= '<img class="cboximg" src="'.ROOT_DIR.'/public/static/admin/images/del.png" />';
  735. }
  736. $arctype_p_html .= '<input type="checkbox" class="arctype_cbox arctype_id_' . $arctype['id'] . ' none" name="permission[arctype][]" value="' . $arctype['id'] . '"';
  737. if (!empty($role_info['permission']['arctype']) && in_array($arctype['id'], $role_info['permission']['arctype'])) {
  738. $arctype_p_html .= ' checked="checked" ';
  739. }
  740. $arctype_p_html .= ' />' . $arctype['typename'] . '</label>&nbsp;';
  741. }
  742. $this->assign('arctype_p_html', $arctype_p_html);
  743. $this->assign('arctype_child_html', $arctype_child_html);
  744. // 插件
  745. $plugins = model('Weapp')->getList(['status'=>1]);
  746. $this->assign('plugins', $plugins);
  747. // 是否使用第三方扫码登录
  748. $wechatInfo = [];
  749. $thirdata = login_third_type();
  750. if ('WechatLogin' == $thirdata['type']) { // 扫码微信应用
  751. if (!empty($thirdata['data']['security_wechat_open'])) {
  752. $wechatInfo = Db::name('admin_wxlogin')->where(['admin_id'=>$id, 'type'=>2])->find();
  753. }
  754. }
  755. else if ('EyouGzhLogin' == $thirdata['type']) { // 扫码官方公众号
  756. if (!empty($thirdata['data']['switch'])) {
  757. $wechatInfo = Db::name('admin_wxlogin')->where(['admin_id'=>$id, 'type'=>1])->find();
  758. }
  759. }
  760. $this->assign('thirdata', $thirdata);
  761. $this->assign('wechatInfo', $wechatInfo);
  762. /*等保密码复杂度验证 start*/
  763. $pwdJsCode = '';
  764. if (is_dir('./weapp/Equal/')) {
  765. $equalLogic = new \weapp\Equal\logic\EqualLogic;
  766. $pwdJsCode = $equalLogic->pwdJsCode();
  767. }
  768. if ('close' == $pwdJsCode) {
  769. $pwdJsCode =<<<EOF
  770. if (password.length < 5) {
  771. showErrorMsg('用户密码至少5位或以上!');
  772. $('input[name=password]').focus();
  773. return false;
  774. }
  775. EOF;
  776. }
  777. $this->assign('pwdJsCode', $pwdJsCode);
  778. /*等保密码复杂度验证 end*/
  779. return $this->fetch();
  780. }
  781. /*
  782. * 递归生成$arctype_child_html
  783. * $vo 栏目tree
  784. * $info 权限集合(用于edit是否已经选中)
  785. * return 完整html
  786. */
  787. private function get_arctype_child_html($vo,$info = []){
  788. $arctype_child_html = "";
  789. if (!empty($vo['children'])) {
  790. $arctype_child_html .= '<div class="arctype_child1" id="arctype_child_' . $vo['id'] . '">';
  791. //判断当前下级是否还存在下级,true为竖着,false为横着
  792. $has_chldren = true;
  793. if ($vo['grade'] != 0 && !empty($vo['has_chldren']) && $vo['has_chldren'] == count($vo['children'])){
  794. $has_chldren = false;
  795. }
  796. if ($has_chldren){
  797. foreach ($vo['children'] as $vo1) {
  798. $arctype_child_html .= '<div class="arctype_child1">';
  799. $arctype_child_html .= ' <span class="button level1 switch center_docu"></span><label>';
  800. if (!empty($info['permission']['arctype']) && in_array($vo1['id'], $info['permission']['arctype'])) {
  801. $arctype_child_html .= '<img class="cboximg" src="'.ROOT_DIR.'/public/static/admin/images/ok.png" />';
  802. }else{
  803. $arctype_child_html .= '<img class="cboximg" src="'.ROOT_DIR.'/public/static/admin/images/del.png" />';
  804. }
  805. $arctype_child_html .= '<input type="checkbox" class="arctype_cbox arctype_id_' . $vo1['id'] . ' none" name="permission[arctype][]" value="' . $vo1['id'] . '" data-pid="' . $vo1['parent_id'] . '"';
  806. if (!empty($info['permission']['arctype']) && in_array($vo1['id'], $info['permission']['arctype'])) {
  807. $arctype_child_html .= ' checked="checked" ';
  808. }
  809. $arctype_child_html .= '/>' . $vo1['typename'] . '</label></div>';
  810. $arctype_child_html .= $this->get_arctype_child_html($vo1,$info);
  811. }
  812. }else{
  813. $arctype_child_html .= '<div class="arctype_child2"> <span class="button level1 switch center_docu"></span>';
  814. foreach ($vo['children'] as $vo1) {
  815. $arctype_child_html .= ' <label>';
  816. if (!empty($info['permission']['arctype']) && in_array($vo1['id'], $info['permission']['arctype'])) {
  817. $arctype_child_html .= '<img class="cboximg" src="'.ROOT_DIR.'/public/static/admin/images/ok.png" />';
  818. }else{
  819. $arctype_child_html .= '<img class="cboximg" src="'.ROOT_DIR.'/public/static/admin/images/del.png" />';
  820. }
  821. $arctype_child_html .= '<input type="checkbox" class="arctype_cbox arctype_id_' . $vo1['id'] . ' none" name="permission[arctype][]" value="' . $vo1['id'] . '" data-pid="' . $vo1['parent_id'] . '"';
  822. if (!empty($info['permission']['arctype']) && in_array($vo1['id'], $info['permission']['arctype'])) {
  823. $arctype_child_html .= ' checked="checked" ';
  824. }
  825. $arctype_child_html .= '/>' . $vo1['typename'] . '</label>';
  826. $arctype_child_html .= $this->get_arctype_child_html($vo1,$info);
  827. }
  828. $arctype_child_html .= '</div>';
  829. }
  830. $arctype_child_html .= '</div>';
  831. }
  832. return $arctype_child_html;
  833. }
  834. /**
  835. * 删除管理员
  836. */
  837. public function admin_del()
  838. {
  839. $this->language_access(); // 多语言功能操作权限
  840. if (IS_POST) {
  841. $id_arr = input('del_id/a');
  842. $id_arr = eyIntval($id_arr);
  843. if (in_array(session('admin_id'), $id_arr)) {
  844. $this->error('禁止删除自己');
  845. }
  846. if (!empty($id_arr)) {
  847. if (0 < $this->admin_info['role_id'] || !empty($this->admin_info['parent_id']) ) {
  848. $count = Db::name('admin')->where("admin_id in (".implode(',', $id_arr).") AND role_id = -1")
  849. ->count();
  850. if (!empty($count)) {
  851. $this->error('禁止删除超级管理员');
  852. }
  853. }
  854. $result = Db::name('admin')->field('user_name')->where("admin_id",'IN',$id_arr)->select();
  855. $user_names = get_arr_column($result, 'user_name');
  856. $r = Db::name('admin')->where("admin_id",'IN',$id_arr)->delete();
  857. if($r){
  858. adminLog('删除管理员:'.implode(',', $user_names));
  859. /*同步删除管理员关联的前台会员*/
  860. Db::name('users')->where(['admin_id'=>['IN', $id_arr],'lang'=>$this->admin_lang])->delete();
  861. /*end*/
  862. $this->success('删除成功');
  863. }else{
  864. $this->error('删除失败');
  865. }
  866. }else{
  867. $this->error('参数有误');
  868. }
  869. }
  870. $this->error('非法操作');
  871. }
  872. /*
  873. * 第一次同步CMS用户的栏目ID到权限组里
  874. * 默认赋予内置权限所有的内容栏目权限
  875. */
  876. private function syn_built_auth_role()
  877. {
  878. $authRole = new AuthRole;
  879. $roleRow = $authRole->getRoleAll(['built_in'=>1,'update_time'=>['elt',0]]);
  880. if (!empty($roleRow)) {
  881. $saveData = [];
  882. foreach ($roleRow as $key => $val) {
  883. $permission = $val['permission'];
  884. $arctype = Db::name('arctype')->where('status',1)->column('id');
  885. if (!empty($arctype)) {
  886. $permission['arctype'] = $arctype;
  887. } else {
  888. unset($permission['arctype']);
  889. }
  890. $saveData[] = array(
  891. 'id' => $val['id'],
  892. 'permission' => $permission,
  893. 'update_time' => getTime(),
  894. );
  895. }
  896. $authRole->saveAll($saveData);
  897. }
  898. }
  899. /*
  900. * 设置admin表数据
  901. */
  902. public function ajax_setfield()
  903. {
  904. $field = input('field'); // 修改哪个字段
  905. $field = preg_replace('/([^\w\-])/i', '', $field);
  906. $field = str_replace(['password'], '', $field);
  907. if (IS_POST && !empty($field)) {
  908. $value = input('value', '', null); // 修改字段值
  909. if (!empty($this->admin_info['admin_id'])) {
  910. $r = Db::name('admin')->where('admin_id', $this->admin_info['admin_id'])->save([
  911. $field=>$value,
  912. 'update_time'=>getTime(),
  913. ]); // 根据条件保存修改的数据
  914. if ($r !== false) {
  915. /*更新存储在session里的信息*/
  916. $this->admin_info[$field] = $value;
  917. session('admin_info', $this->admin_info);
  918. /*--end*/
  919. $this->success('操作成功');
  920. }
  921. }
  922. }
  923. $this->error('操作失败');
  924. }
  925. /*
  926. * 检测密码的复杂程度
  927. */
  928. public function ajax_checkPasswordLevel()
  929. {
  930. $password = input('post.password/s');
  931. if (IS_AJAX_POST && !empty($password)) {
  932. $pwdLevel = checkPasswordLevel($password);
  933. if (3 >= $pwdLevel) {
  934. $this->success("<font color='red'>当前密码复杂度为 {$pwdLevel} ,建议复杂度在 4~7 范围内,避免容易被暴力破解!</font>", null, ['pwdLevel'=>$pwdLevel]);
  935. } else {
  936. $this->success("<font color='green'>当前密码复杂度为 {$pwdLevel} ,在系统设定 4~7 安全范围内!</font>", null, ['pwdLevel'=>$pwdLevel]);
  937. }
  938. }
  939. $this->error('操作失败');
  940. }
  941. // 确保用户名唯一
  942. private function GetUserName($username = null)
  943. {
  944. $count = Db::name('users')->where('username',$username)->count();
  945. if (!empty($count)) {
  946. $username_new = $username.rand(1000,9999);
  947. $username = $this->GetUserName($username_new);
  948. }
  949. return $username;
  950. }
  951. /**
  952. * 同步追加一个后台管理员到会员用户表,并同步前台登录
  953. */
  954. private function syn_users_login($admin_info = [], $isFounder = 0)
  955. {
  956. $where_new = [
  957. 'admin_id' => $admin_info['admin_id'],
  958. 'lang' => $this->admin_lang,
  959. ];
  960. $users_id = Db::name('users')->where($where_new)->getField('users_id');
  961. try {
  962. if (empty($users_id) && empty($admin_info['syn_users_id'])) {
  963. $usersInfo = [];
  964. if (1 == $isFounder) {
  965. // 如果是创始人,强制将与会员名相同的改为管理员前台用户名
  966. $usersInfo = Db::name('users')->field('users_id')->where([
  967. 'username' => $admin_info['user_name'],
  968. 'lang' => $this->admin_lang,
  969. ])->find();
  970. }
  971. if (!empty($usersInfo)) {
  972. $r = Db::name('users')->where(['users_id'=>$usersInfo['users_id']])->update([
  973. 'nickname' => $admin_info['user_name'],
  974. 'admin_id' => $admin_info['admin_id'],
  975. 'is_activation' => 1,
  976. 'is_lock' => 0,
  977. 'is_del' => 0,
  978. 'update_time' => getTime(),
  979. 'last_login' => getTime(),
  980. ]);
  981. !empty($r) && $users_id = $usersInfo['users_id'];
  982. } else {
  983. // 获取要添加的用户名
  984. $username = $this->GetUserName($admin_info['user_name']);
  985. $password = getTime();
  986. $password = func_encrypt($password, false, pwd_encry_type('bcrypt'));
  987. $AddData = [
  988. 'username' => $username,
  989. 'nickname' => $username,
  990. 'password' => $password,
  991. 'level' => 1,
  992. 'lang' => $this->admin_lang,
  993. 'reg_time' => getTime(),
  994. 'head_pic' => ROOT_DIR . '/public/static/common/images/dfboy.png',
  995. 'add_time' => getTime(),
  996. 'last_login' => getTime(),
  997. 'register_place' => 1,
  998. 'admin_id' => $admin_info['admin_id'],
  999. ];
  1000. $users_id = Db::name('users')->insertGetId($AddData);
  1001. }
  1002. if (!empty($users_id)) {
  1003. Db::name('admin')->where(['admin_id'=>$admin_info['admin_id']])->update([
  1004. 'syn_users_id' => $users_id,
  1005. 'update_time' => getTime(),
  1006. ]);
  1007. $admin_info['syn_users_id'] = $users_id;
  1008. session('admin_info', $admin_info);
  1009. }
  1010. } else if (!empty($users_id) && empty($admin_info['syn_users_id'])) {
  1011. Db::name('admin')->where(['admin_id'=>$admin_info['admin_id']])->update([
  1012. 'syn_users_id' => $users_id,
  1013. 'update_time' => getTime(),
  1014. ]);
  1015. $admin_info['syn_users_id'] = $users_id;
  1016. session('admin_info', $admin_info);
  1017. }
  1018. } catch (\Exception $e) {}
  1019. // 加载前台session
  1020. if (!empty($users_id)) {
  1021. $users = Db::name('users')->field('a.*,b.level_name,b.level_value,b.discount as level_discount')
  1022. ->alias('a')
  1023. ->join('__USERS_LEVEL__ b', 'a.level = b.level_id', 'LEFT')
  1024. ->where([
  1025. 'a.users_id' => $users_id,
  1026. 'a.lang' => $this->admin_lang,
  1027. 'a.is_activation' => 1,
  1028. ])->find();
  1029. if (!empty($users)) {
  1030. Db::name('users')->where(['users_id'=>$users_id])->update([
  1031. 'update_time' => getTime(),
  1032. 'last_login' => getTime(),
  1033. ]);
  1034. GetUsersLatestData($users_id);
  1035. }
  1036. }
  1037. }
  1038. /*-----------------------------------扫码微信应用 start--------------------------*/
  1039. /**
  1040. * 微信应用登录
  1041. * @return [type] [description]
  1042. */
  1043. public function wechat_login()
  1044. {
  1045. $redirect_uri = url('Admin/wechat_callback', [], true, true);
  1046. $redirect_uri = urlencode($redirect_uri);//该回调需要url编码
  1047. $security = tpSetting('security');
  1048. $scope = "snsapi_login";//写死,微信暂时只支持这个值
  1049. //准备向微信发请求
  1050. $url = "https://open.weixin.qq.com/connect/qrconnect?appid=" . $security['security_wechat_appid'] . "&redirect_uri=" . $redirect_uri
  1051. . "&response_type=code&scope=" . $scope . "&state=STATE#wechat_redirect";
  1052. $this->redirect($url);
  1053. exit;
  1054. }
  1055. /**
  1056. * 立即绑定微信应用
  1057. * @return [type] [description]
  1058. */
  1059. public function wechat_bind()
  1060. {
  1061. $origin = input('param.origin/s');
  1062. $admin_id = input('param.admin_id/d');
  1063. $gourl = input('param.gourl/s');
  1064. $gourl = htmlspecialchars_decode($gourl);
  1065. $redirect_uri = url('Admin/wechat_callback', ['bind'=>1,'admin_id'=>$admin_id,'origin'=>$origin,'gourl'=>$gourl], true, true);
  1066. $redirect_uri = urlencode($redirect_uri);//该回调需要url编码
  1067. $security = tpSetting('security');
  1068. $scope = "snsapi_login";//写死,微信暂时只支持这个值
  1069. //准备向微信发请求
  1070. $url = "https://open.weixin.qq.com/connect/qrconnect?appid=" . $security['security_wechat_appid'] . "&redirect_uri=" . $redirect_uri
  1071. . "&response_type=code&scope=" . $scope . "&state=STATE&self_redirect=true#wechat_redirect";
  1072. $this->redirect($url);
  1073. exit;
  1074. }
  1075. /**
  1076. * 微信应用扫描回调
  1077. * @return [type] [description]
  1078. */
  1079. public function wechat_callback()
  1080. {
  1081. $code = input('param.code/s');
  1082. $bind = input('param.bind/d');
  1083. $isframe = 0; // 是否在弹窗内跳转
  1084. if (!empty($bind)) {
  1085. $isframe = 1;
  1086. }
  1087. if (empty($code)) {
  1088. if (empty($isframe)) {
  1089. $this->error('微信回调参数错误');
  1090. } else {
  1091. $html = <<<EOF
  1092. <script type="application/javascript" src="{$this->root_dir}/public/static/common/js/jquery.min.js?v={$this->version}"></script>
  1093. <script type="application/javascript" src="{$this->root_dir}/public/plugins/layer-v3.1.0/layer.js"></script>
  1094. <script type="text/javascript">
  1095. var _parent = parent;
  1096. _parent.layer.closeAll();
  1097. _parent.layer.alert("微信回调参数错误", {icon: 5, title: false});
  1098. </script>
  1099. EOF;
  1100. echo $html;
  1101. exit;
  1102. }
  1103. }
  1104. $security = tpSetting('security');
  1105. $appid = $security['security_wechat_appid'];
  1106. $secret = $security['security_wechat_secret'];
  1107. //通过code获得 access_token + openid
  1108. $url = "https://api.weixin.qq.com/sns/oauth2/access_token?appid=" . $appid
  1109. . "&secret=" . $secret . "&code=" . $code . "&grant_type=authorization_code";
  1110. $jsonResult = httpRequest($url);
  1111. $resultArray = json_decode($jsonResult, true);
  1112. if (!empty($resultArray['errcode'])) {
  1113. if (empty($isframe)) {
  1114. $this->error($resultArray['errmsg']);
  1115. } else {
  1116. $html = <<<EOF
  1117. <script type="application/javascript" src="{$this->root_dir}/public/static/common/js/jquery.min.js?v={$this->version}"></script>
  1118. <script type="application/javascript" src="{$this->root_dir}/public/plugins/layer-v3.1.0/layer.js"></script>
  1119. <script type="text/javascript">
  1120. var _parent = parent;
  1121. _parent.layer.closeAll();
  1122. _parent.layer.alert("{$resultArray['errmsg']}", {icon: 5, title: false});
  1123. </script>
  1124. EOF;
  1125. echo $html;
  1126. exit;
  1127. }
  1128. }
  1129. $access_token = $resultArray["access_token"];
  1130. $openid = $resultArray["openid"];
  1131. $unionid = !empty($resultArray["unionid"]) ? $resultArray["unionid"] : '';
  1132. //通过access_token + openid 获得用户所有信息,结果全部存储在$infoArray里
  1133. $infoUrl = "https://api.weixin.qq.com/sns/userinfo?access_token=" . $access_token . "&openid=" . $openid;
  1134. $infoResult = httpRequest($infoUrl);
  1135. $infoArray = json_decode($infoResult, true);
  1136. empty($infoArray['nickname']) && $infoArray['nickname'] = '';
  1137. $nickname = $infoArray['nickname'] = filterNickname($infoArray['nickname']);
  1138. if (!empty($infoArray['errcode'])) {
  1139. if (empty($isframe)) {
  1140. $this->error($infoArray['errmsg']);
  1141. } else {
  1142. $html = <<<EOF
  1143. <script type="application/javascript" src="{$this->root_dir}/public/static/common/js/jquery.min.js?v={$this->version}"></script>
  1144. <script type="application/javascript" src="{$this->root_dir}/public/plugins/layer-v3.1.0/layer.js"></script>
  1145. <script type="text/javascript">
  1146. var _parent = parent;
  1147. _parent.layer.closeAll();
  1148. _parent.layer.alert("{$infoArray['errmsg']}", {icon: 5, title: false});
  1149. </script>
  1150. EOF;
  1151. echo $html;
  1152. exit;
  1153. }
  1154. }
  1155. if (!empty($bind)) { // 绑定
  1156. $admin_id = input('param.admin_id/d');
  1157. $origin = input('param.origin/s');
  1158. $origin = preg_replace('/([^\w\-]+)/i', '', $origin);
  1159. $this->wechat_bind_handle($openid, $unionid, $infoArray, $admin_id, $origin);
  1160. return true;
  1161. }
  1162. else { // 登录
  1163. $this->wechat_login_handle($openid);
  1164. return true;
  1165. }
  1166. }
  1167. /**
  1168. * 微信应用扫码登录处理
  1169. * @param string $openid [description]
  1170. * @return [type] [description]
  1171. */
  1172. private function wechat_login_handle($openid = '')
  1173. {
  1174. $web_adminbasefile = tpCache('global.web_adminbasefile');
  1175. $web_adminbasefile = !empty($web_adminbasefile) ? $web_adminbasefile : $this->root_dir.'/login.php';
  1176. $we_user = Db::name('admin_wxlogin')->field('a.openid, b.admin_id, b.user_name')
  1177. ->alias('a')
  1178. ->join('admin b', 'a.admin_id=b.admin_id', 'LEFT')
  1179. ->where(['a.openid'=>$openid, 'a.type'=>2])
  1180. ->find();
  1181. if (empty($we_user['user_name'])) {
  1182. adminLog('登录失败(微信用户不存在)');
  1183. $this->error('微信用户不存在!', $web_adminbasefile);
  1184. } else {
  1185. $admin_info = adminLoginAfter($we_user['admin_id'], session_id(), 'WechatLogin');
  1186. if (!empty($admin_info)) {
  1187. adminLog('扫码登录成功');
  1188. $this->success('登录成功', $web_adminbasefile);
  1189. }
  1190. adminLog('扫码登录失败');
  1191. $this->error('登录失败', $web_adminbasefile);
  1192. }
  1193. }
  1194. /**
  1195. * 微信应用绑定处理
  1196. * @param string $openid [description]
  1197. * @param string $unionid [description]
  1198. * @param array $wx_info [description]
  1199. * @return [type] [description]
  1200. */
  1201. private function wechat_bind_handle($openid = '', $unionid = '', $wx_info = [], $admin_id = 0, $origin = '')
  1202. {
  1203. if (empty($this->admin_info['parent_id']) && -1 == $this->admin_info['role_id']) { // 创始人
  1204. $is_founder = 1;
  1205. empty($admin_id) && $admin_id = $this->admin_info['admin_id'];
  1206. $admin_info = Db::name('admin')->where(['admin_id'=>$admin_id])->find();
  1207. } else {
  1208. $is_founder = 0;
  1209. $admin_info = $this->admin_info;
  1210. $admin_id = $this->admin_info['admin_id'];
  1211. }
  1212. if (empty($admin_info)) {
  1213. $html = <<<EOF
  1214. <script type="application/javascript" src="{$this->root_dir}/public/static/common/js/jquery.min.js?v={$this->version}"></script>
  1215. <script type="application/javascript" src="{$this->root_dir}/public/plugins/layer-v3.1.0/layer.js"></script>
  1216. <script type="text/javascript">
  1217. var _parent = parent;
  1218. _parent.layer.closeAll();
  1219. _parent.layer.alert("查不到管理员信息", {icon: 5, title: false});
  1220. </script>
  1221. EOF;
  1222. echo $html;
  1223. exit;
  1224. }
  1225. $row = Db::name('admin_wxlogin')->where(['openid'=>$openid, 'type'=>2])->find();
  1226. if(!empty($row))
  1227. {
  1228. if (!empty($row['admin_id'])) {
  1229. $count = Db::name('admin')->where(['admin_id'=>$row['admin_id']])->count();
  1230. if (!empty($count)) {
  1231. $html = <<<EOF
  1232. <script type="application/javascript" src="{$this->root_dir}/public/static/common/js/jquery.min.js?v={$this->version}"></script>
  1233. <script type="application/javascript" src="{$this->root_dir}/public/plugins/layer-v3.1.0/layer.js"></script>
  1234. <script type="text/javascript">
  1235. var _parent = parent;
  1236. _parent.layer.closeAll();
  1237. _parent.layer.alert("当前微信已被绑定", {icon: 5, title: false});
  1238. </script>
  1239. EOF;
  1240. echo $html;
  1241. exit;
  1242. }
  1243. }
  1244. $saveData = [
  1245. 'admin_id' => $admin_id,
  1246. 'nickname' => $wx_info['nickname'],
  1247. 'headimgurl' => $wx_info['headimgurl'],
  1248. 'update_time'=> getTime(),
  1249. ];
  1250. $r = Db::name('admin_wxlogin')->where([
  1251. 'wx_id' => $row['wx_id'],
  1252. ])->update($saveData);
  1253. } else {
  1254. $saveData = [
  1255. 'admin_id' => $admin_id,
  1256. 'nickname' => $wx_info['nickname'],
  1257. 'headimgurl' => $wx_info['headimgurl'],
  1258. 'type' => 2,
  1259. 'openid' => $openid,
  1260. 'unionid' => $unionid,
  1261. 'add_time'=> getTime(),
  1262. 'update_time'=> getTime(),
  1263. ];
  1264. $r = Db::name('admin_wxlogin')->insert($saveData);
  1265. }
  1266. if ($r !== false) {
  1267. \think\Cache::clear("admin_wxlogin");
  1268. // 同步昵称、头像
  1269. $updateData = [
  1270. 'update_time'=> getTime(),
  1271. ];
  1272. if (empty($admin_info['head_pic']) && !empty($wx_info['headimgurl'])) {
  1273. $updateData['head_pic'] = $wx_info['headimgurl'];
  1274. if ($admin_id == $this->admin_info['admin_id']) {
  1275. $this->admin_info['head_pic'] = $wx_info['headimgurl'];
  1276. }
  1277. }
  1278. Db::name('admin')->where(['admin_id'=>$admin_id])->update($updateData);
  1279. if ($admin_id == $this->admin_info['admin_id']) {
  1280. $this->admin_info['openid'] = $openid;
  1281. session('admin_info', $this->admin_info);
  1282. } else {
  1283. if (1 == $is_founder) {
  1284. $openid = Db::name('admin_wxlogin')->where(['admin_id'=>$this->admin_info['admin_id'], 'type'=>2])->value('openid');
  1285. if (!empty($openid)) {
  1286. $this->admin_info['openid'] = $openid;
  1287. session('admin_info', $this->admin_info);
  1288. }
  1289. }
  1290. }
  1291. $gourl = input('param.gourl/s');
  1292. $gourl = htmlspecialchars_decode($gourl);
  1293. $html = <<<EOF
  1294. <script type="application/javascript" src="{$this->root_dir}/public/static/common/js/jquery.min.js?v={$this->version}"></script>
  1295. <script type="application/javascript" src="{$this->root_dir}/public/plugins/layer-v3.1.0/layer.js"></script>
  1296. <script type="text/javascript">
  1297. var origin = "{$origin}";
  1298. var _parent = parent;
  1299. if ('list' != origin) {
  1300. var documentOjb = window.parent.document.htm;
  1301. $('#span_wechat_nickname', documentOjb).html("{$wx_info['nickname']}");
  1302. $('#wechat_bind', documentOjb).hide();
  1303. $('#wechat_unbind', documentOjb).show();
  1304. }
  1305. _parent.layer.closeAll();
  1306. _parent.layer.msg("绑定成功", {time: 1000}, function(){
  1307. if ('list' == origin) {
  1308. _parent.window.location.reload();
  1309. }
  1310. });
  1311. </script>
  1312. EOF;
  1313. echo $html;
  1314. exit;
  1315. }
  1316. $html = <<<EOF
  1317. <script type="application/javascript" src="{$this->root_dir}/public/static/common/js/jquery.min.js?v={$this->version}"></script>
  1318. <script type="application/javascript" src="{$this->root_dir}/public/plugins/layer-v3.1.0/layer.js"></script>
  1319. <script type="text/javascript">
  1320. var _parent = parent;
  1321. _parent.layer.closeAll();
  1322. _parent.layer.alert("绑定失败", {icon: 5, title: false});
  1323. </script>
  1324. EOF;
  1325. echo $html;
  1326. exit;
  1327. }
  1328. /**
  1329. * 解除绑定微信应用
  1330. * @return [type] [description]
  1331. */
  1332. public function wechat_unbind_handle()
  1333. {
  1334. if (empty($this->admin_info['parent_id']) && -1 == $this->admin_info['role_id']) {
  1335. $admin_id = input('param.admin_id/d', $this->admin_info['admin_id']);
  1336. } else {
  1337. $admin_id = intval($this->admin_info['admin_id']);
  1338. }
  1339. if (IS_POST && !empty($admin_id)) {
  1340. $security_wechat_forcelogin = tpSetting('security.security_wechat_forcelogin');
  1341. if (!empty($security_wechat_forcelogin)) {
  1342. $this->error('检测已开启强制扫码登录,禁止解绑');
  1343. }
  1344. $r = Db::name('admin_wxlogin')->where(['admin_id'=>$admin_id, 'type'=>2])->delete();
  1345. if ($r !== false) {
  1346. \think\Cache::clear("admin_wxlogin");
  1347. if ($admin_id == $this->admin_info['admin_id'] && isset($this->admin_info['openid'])) {
  1348. unset($this->admin_info['openid']);
  1349. }
  1350. session('admin_info', $this->admin_info);
  1351. $this->success("操作成功");
  1352. }
  1353. }
  1354. $this->error("操作失败");
  1355. }
  1356. /*-----------------------------------扫码微信应用 end--------------------------*/
  1357. /*--------------------------------扫码微信公众号 start--------------------------*/
  1358. //获取官方微信公众号二维码
  1359. public function mp_getqrcode()
  1360. {
  1361. $eyouGzhLoginLogic = new \weapp\EyouGzhLogin\logic\EyouGzhLoginLogic;
  1362. $eyouGzhLoginLogic->mp_getqrcode();
  1363. }
  1364. //绑定官方微信公众号openid
  1365. public function mp_bingwxgzhopenid()
  1366. {
  1367. $eyouGzhLoginLogic = new \weapp\EyouGzhLogin\logic\EyouGzhLoginLogic;
  1368. $eyouGzhLoginLogic->mp_bingwxgzhopenid();
  1369. }
  1370. //解绑官方微信公众号
  1371. public function mp_unbindwx()
  1372. {
  1373. $eyouGzhLoginLogic = new \weapp\EyouGzhLogin\logic\EyouGzhLoginLogic;
  1374. $eyouGzhLoginLogic->mp_unbindwx();
  1375. }
  1376. /*--------------------------------扫码微信公众号 end--------------------------*/
  1377. /*--------------------------------微信公众号扫码关注 start--------------------------*/
  1378. // 微信公众号扫码关注
  1379. public function wechat_followed()
  1380. {
  1381. $admin_id = input('post.admin_id/d', 0);
  1382. // 默认授权页面链接
  1383. $defaultAuthorize = request()->domain() . ROOT_DIR . '/index.php?m=api&c=Ajax&a=defaultAuthorize&admin_id=' . $admin_id;
  1384. // 二维码图片完整目录
  1385. $defaultAuthorizePic = UPLOAD_PATH . 'system/wechat_followed/' . $admin_id . '/';
  1386. // 创建文件夹
  1387. @mkdir($defaultAuthorizePic, 0777, true);
  1388. // 二维码图片完整链接
  1389. $defaultAuthorizePic = $defaultAuthorizePic . md5($admin_id) . '.png';
  1390. // 生成二维码
  1391. vendor('wechatpay.phpqrcode.phpqrcode');
  1392. $qrcode = new \QRcode;
  1393. $qrcode->png($defaultAuthorize, $defaultAuthorizePic);
  1394. $defaultAuthorizePic = handle_subdir_pic(get_default_pic('/' . $defaultAuthorizePic, true));
  1395. $this->success("获取成功", $defaultAuthorizePic);
  1396. }
  1397. public function polling_wechat_followed()
  1398. {
  1399. $admin_id = input('post.admin_id/d', 0);
  1400. if (!empty($admin_id)) {
  1401. $admin = Db::name('admin')->where('admin_id', $admin_id)->find();
  1402. if (!empty($admin['wechat_open_id'])) {
  1403. // 查询用户是否关注了公众号
  1404. $tokenData = get_wechat_access_token();
  1405. if (!empty($tokenData)) {
  1406. $userInfo = 'https://api.weixin.qq.com/cgi-bin/user/info?access_token=' . $tokenData['access_token'] . '&openid=' . $admin['wechat_open_id'] . '&lang=zh_CN';
  1407. $userInfo = json_decode(httpRequest($userInfo), true);
  1408. // 关注则执行关联关注
  1409. if (!empty($userInfo['subscribe']) && $userInfo['openid'] == $admin['wechat_open_id']) {
  1410. $update = [
  1411. 'wechat_followed' => 1,
  1412. 'update_time' => getTime(),
  1413. ];
  1414. Db::name('admin')->where('admin_id', $admin_id)->update($update);
  1415. // 显示成功信息
  1416. $this->success("关注公众号成功!", null, ['code' => 1]);
  1417. }
  1418. }
  1419. }
  1420. }
  1421. $this->success("尚未关注", null, ['code' => 2]);
  1422. }
  1423. }