UserOpen.php 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426
  1. <?php
  2. namespace app\api\model\user;
  3. use app\api\model\plus\agent\Referee as RefereeModel;
  4. use app\api\model\plus\invitationgift\Partake;
  5. use app\common\model\user\User;
  6. use think\facade\Cache;
  7. use app\common\exception\BaseException;
  8. use app\common\model\user\User as UserModel;
  9. use app\common\model\user\Sms as SmsModel;
  10. use app\common\model\user\Grade as GradeModel;
  11. use think\facade\Db;
  12. /**
  13. * 公众号用户模型类
  14. */
  15. class UserOpen extends UserModel
  16. {
  17. private $token;
  18. /**
  19. * 隐藏字段
  20. */
  21. protected $hidden = [
  22. 'open_id',
  23. 'is_delete',
  24. 'app_id',
  25. 'create_time',
  26. 'update_time'
  27. ];
  28. /**
  29. * 用户登录
  30. */
  31. public function login($userInfo, $referee_id)
  32. {
  33. // 自动注册用户
  34. $user_id = $this->register($userInfo, $referee_id);
  35. // 生成token (session3rd)
  36. $this->token = $this->token($userInfo['openid']);
  37. // 记录缓存, 7天
  38. Cache::set($this->token, $user_id, 86400 * 7);
  39. return $user_id;
  40. }
  41. /**
  42. * 获取token
  43. */
  44. public function getToken()
  45. {
  46. return $this->token;
  47. }
  48. /**
  49. * 生成用户认证的token
  50. */
  51. private function token($openid)
  52. {
  53. return md5($openid . 'token_salt');
  54. }
  55. /**
  56. * 自动注册用户
  57. */
  58. private function register($userInfo, $referee_id = 0)
  59. {
  60. $data = [];
  61. //通过unionid查询用户是否存在
  62. $user = null;
  63. if (isset($userInfo['unionid']) && !empty($userInfo['unionid'])) {
  64. $data['union_id'] = $userInfo['unionid'];
  65. $user = self::detailByUnionid($userInfo['unionid']);
  66. }
  67. // 查询用户是否已存在
  68. if (!$user) {
  69. $user = self::detail(['appopen_id' => $userInfo['openid']]);
  70. }
  71. if ($user) {
  72. $model = $user;
  73. // 只修改union_id
  74. if (isset($data['union_id'])) {
  75. $data = [
  76. 'union_id' => $data['union_id'],
  77. ];
  78. }else{
  79. return $user['user_id'];
  80. }
  81. } else {
  82. $model = $this;
  83. $data['referee_id'] = $referee_id;
  84. $data['reg_source'] = 'app';
  85. //默认等级
  86. $data['grade_id'] = GradeModel::getDefaultGradeId();
  87. }
  88. $data['appopen_id'] = $userInfo['openid'];
  89. // 用户昵称
  90. if (!$user) {
  91. $data['nickName'] = preg_replace('/[\xf0-\xf7].{3}/', '', $userInfo['nickname']);
  92. }
  93. $data['avatarUrl'] = $userInfo['headimgurl'];
  94. $data['gender'] = $userInfo['sex'];
  95. $data['province'] = $userInfo['province'];
  96. $data['country'] = $userInfo['country'];
  97. $data['city'] = $userInfo['city'];
  98. $data['reg_source'] = 'app';
  99. try {
  100. $this->startTrans();
  101. // 保存/更新用户记录
  102. if (!$model->save(array_merge($data, [
  103. 'app_id' => self::$app_id
  104. ]))
  105. ) {
  106. throw new BaseException(['msg' => '用户注册失败']);
  107. }
  108. if (!$user && $referee_id > 0) {
  109. // 记录推荐人关系,
  110. RefereeModel::createRelation($model['user_id'], $referee_id);
  111. //更新用户邀请数量
  112. (new UserModel())->where('user_id', '=', $referee_id)->inc('total_invite')->update();
  113. }
  114. $this->commit();
  115. } catch (\Exception $e) {
  116. $this->rollback();
  117. throw new BaseException(['msg' => $e->getMessage()]);
  118. }
  119. return $model['user_id'];
  120. }
  121. /**
  122. * 手机号密码用户登录
  123. */
  124. public function phoneLogin($data)
  125. {
  126. $where = [
  127. ['user_no','=',$data['mobile']],
  128. ['is_delete','=',0],
  129. ];
  130. $user = $this->where($where)->order('user_id desc')->find();
  131. if (empty($user)) {
  132. $this->error = '登录账号错误';
  133. return false;
  134. }
  135. if ($user['password'] != md5($data['password'])) {
  136. $this->error = '登陆密码错误';
  137. return false;
  138. }
  139. if (!empty($data['openid'])) {
  140. $weixin = [
  141. // 'nickName' => authcode($data['nickName'], 'DECODE'),
  142. // 'avatarUrl' => authcode($data['avatarUrl'], 'DECODE'),
  143. 'mpopen_id' => authcode($data['openid'], 'DECODE'),
  144. ];
  145. // $mpopen_id = authcode($data['openid'], 'DECODE'),
  146. if ($weixin['mpopen_id'] != $user['mpopen_id']) {
  147. if ($this->where('user_id',$user['user_id'])->update(['mpopen_id'=>$weixin['mpopen_id']])) {
  148. $ceshi = [
  149. 'test' => json_encode($weixin,JSON_UNESCAPED_UNICODE),
  150. 'name' => '正确',
  151. 'time' => time(),
  152. ];
  153. Db::name('user_ceshi')->insertGetId($ceshi);
  154. // 生成token (session3rd)
  155. $this->token = $this->token($user['user_no']);
  156. // 记录缓存, 30天
  157. Cache::tag('cache')->set($this->token, $user['user_id'], 86400 * 30);
  158. return $user['user_id'];
  159. }else{
  160. $ceshi = [
  161. 'test' => json_encode($weixin,JSON_UNESCAPED_UNICODE),
  162. 'name' => '微信错误日志',
  163. 'time' => time(),
  164. ];
  165. Db::name('user_ceshi')->insertGetId($ceshi);
  166. $this->error = '会员绑定微信信息失败';
  167. return false;
  168. }
  169. }else{
  170. // 生成token (session3rd)
  171. $this->token = $this->token($user['user_no']);
  172. // 记录缓存, 30天
  173. Cache::tag('cache')->set($this->token, $user['user_id'], 86400 * 30);
  174. return $user['user_id'];
  175. }
  176. }else{
  177. // 生成token (session3rd)
  178. $this->token = $this->token($user['user_no']);
  179. // 记录缓存, 30天
  180. Cache::tag('cache')->set($this->token, $user['user_id'], 86400 * 30);
  181. return $user['user_id'];
  182. }
  183. }
  184. /**
  185. * 手机号验证码登录
  186. */
  187. public function smslogin($data)
  188. {
  189. $where = [
  190. [ 'user_no','=',$data['mobile']],
  191. ['is_delete','=',0],
  192. ];
  193. $user = $this->where($where)->order('user_id desc')->find();
  194. if (empty($user)) {
  195. $this->error = '登录账号错误';
  196. return false;
  197. }
  198. if (empty($user['mobile'])) {
  199. $this->error = '您的账号未绑定手机号';
  200. return false;
  201. }
  202. $vall = [
  203. 'mobile' => $user['mobile'],
  204. 'code' => $data['code'],
  205. ];
  206. if (!$this->check($vall)) {
  207. return false;
  208. }
  209. if (!empty($data['openid'])) {
  210. $weixin = [
  211. // 'nickName' => authcode($data['nickName'], 'DECODE'),
  212. // 'avatarUrl' => authcode($data['avatarUrl'], 'DECODE'),
  213. 'mpopen_id' => authcode($data['openid'], 'DECODE'),
  214. ];
  215. if ($weixin['mpopen_id'] != $user['mpopen_id']) {
  216. if ($this->where('user_id',$user['user_id'])->update(['mpopen_id'=> $weixin['mpopen_id']])) {
  217. $ceshi = [
  218. 'test' => json_encode($weixin,JSON_UNESCAPED_UNICODE),
  219. 'name' => '正确',
  220. 'time' => time(),
  221. ];
  222. Db::name('user_ceshi')->insertGetId($ceshi);
  223. // 生成token (session3rd)
  224. $this->token = $this->token($user['user_no']);
  225. // 记录缓存, 30天
  226. Cache::tag('cache')->set($this->token, $user['user_id'], 86400 * 30);
  227. return $user['user_id'];
  228. }else{
  229. $ceshi = [
  230. 'test' => json_encode($weixin,JSON_UNESCAPED_UNICODE),
  231. 'name' => '微信错误日志',
  232. 'time' => time(),
  233. ];
  234. Db::name('user_ceshi')->insertGetId($ceshi);
  235. $this->error = '会员绑定微信信息失败';
  236. return false;
  237. }
  238. }else{
  239. // 生成token (session3rd)
  240. $this->token = $this->token($user['user_no']);
  241. // 记录缓存, 30天
  242. Cache::tag('cache')->set($this->token, $user['user_id'], 86400 * 30);
  243. return $user['user_id'];
  244. }
  245. }else{
  246. // 生成token (session3rd)
  247. $this->token = $this->token($user['user_no']);
  248. // 记录缓存, 30天
  249. Cache::tag('cache')->set($this->token, $user['user_id'], 86400 * 30);
  250. return $user['user_id'];
  251. }
  252. }
  253. /*
  254. *重置密码
  255. */
  256. public function resetpassword($data)
  257. {
  258. $where = [[ 'user_no','=',$data['mobile']]];
  259. $user = $this->where($where)->order('user_id desc')->find();
  260. if (empty($user)) {
  261. $this->error = '登录账号错误';
  262. return false;
  263. }
  264. if ($user['is_delete'] == 1) {
  265. $this->error = '手机号被禁止或删除,请联系客服';
  266. return false;
  267. }
  268. if (empty($user['mobile'])) {
  269. $this->error = '您的账号未绑定手机号';
  270. return false;
  271. }
  272. $vall = [
  273. 'mobile' => $user['mobile'],
  274. 'code' => $data['code'],
  275. ];
  276. if (!$this->check($vall)) {
  277. return false;
  278. }
  279. if (md5($data['password']) == $user['password']) {
  280. $this->error = '登陆密码与原密码相同';
  281. return false;
  282. }
  283. return $this->where('user_id', $user['user_id'])->update([
  284. 'password' => md5($data['password'])
  285. ]);
  286. }
  287. /*
  288. *手机号注册
  289. */
  290. public function phoneRegister($data)
  291. {
  292. if (!$this->check($data)) {
  293. return false;
  294. }
  295. $user = $this->where('user_no',$data['user_no'])->find();
  296. if (empty($user)) {
  297. $vall = [
  298. 'mobile' => $data['mobile'],
  299. 'user_no' => $data['user_no']??'',
  300. 'reg_source' => isset($data['reg_source'])?$data['reg_source']:'app',
  301. 'grade_id' => GradeModel::getDefaultGradeId(),
  302. 'app_id' => self::$app_id,
  303. 'password' => md5($data['password'])
  304. ];
  305. if (empty($data['nickName'])) {
  306. $vall['nickName'] = $data['user_no']??'';
  307. }
  308. if (!empty($data['openid'])) {
  309. // $vall['avatarUrl'] = authcode($data['avatarUrl'], 'DECODE');
  310. // $vall['nickName'] = authcode($data['nickName'], 'DECODE');
  311. $vall['mpopen_id'] = authcode($data['openid'], 'DECODE');
  312. }
  313. $this->save($vall);
  314. if ($data['referee_id'] > 0) {
  315. // 记录推荐人关系
  316. RefereeModel::createRelation($this['user_id'], $data['referee_id']);
  317. //更新用户邀请数量
  318. (new UserModel())->setIncInvite($data['referee_id']);
  319. //邀请好友送好礼
  320. $data['invitation_id'] > 0 && (new Partake())->addPartake($data['invitation_id'], $data['referee_id'], $this['user_id']);
  321. }
  322. return true;
  323. } else {
  324. $this->error = '会员号已存在';
  325. return false;
  326. }
  327. }
  328. /**
  329. * 验证
  330. */
  331. private function check($data)
  332. {
  333. //判断验证码是否过期、是否正确
  334. $sms_model = new SmsModel();
  335. $sms_record_list = $sms_model
  336. ->where('mobile', '=', $data['mobile'])
  337. ->order(['create_time' => 'desc'])
  338. ->limit(1)->select();
  339. if (count($sms_record_list) == 0) {
  340. $this->error = '未查到短信发送记录';
  341. return false;
  342. }
  343. $sms_model = $sms_record_list[0];
  344. if ((time() - strtotime($sms_model['create_time'])) / 60 > 30) {
  345. $this->error = '短信验证码超时';
  346. return false;
  347. }
  348. if ($sms_model['code'] != $data['code']) {
  349. $this->error = '验证码不正确';
  350. return false;
  351. }
  352. return true;
  353. }
  354. /**
  355. * 手机号验证码登录
  356. */
  357. public function smsloginWx($data)
  358. {
  359. // 判断会员号如果是手机号,则查询是否有有此会员号
  360. $pattern = '/^1\d{10}$/';
  361. if (!preg_match($pattern, $data['mobile'])) {
  362. $this->error = '登录账号错误';
  363. return false;
  364. }
  365. $user = $this->where('user_no', $data['mobile'])->find();
  366. if (empty($user)) {
  367. $vall = [
  368. 'mobile' => $data['mobile'],
  369. 'user_no' => $data['mobile'],
  370. 'reg_source' => 'wx',
  371. 'grade_id' => GradeModel::getDefaultGradeId(),
  372. 'app_id' => self::$app_id,
  373. 'password' => md5('123456'),
  374. 'nickName' => $data['mobile'],
  375. 'openid' => $data['openid'], // authcode($data['openid'], 'DECODE'),
  376. 'avatarUrl' => 'http://cni-ekshop.cni.com.cn/image/agent/user_head.png',
  377. ];
  378. if (!$this->save($vall)) {
  379. $this->error = '登录失败';
  380. return false;
  381. }
  382. $user = $this->where('user_no', $data['mobile'])->find();
  383. }
  384. if (!$this->check($data)) {
  385. return false;
  386. }
  387. $info = [
  388. 'open_id' => $data['openid'],
  389. 'reg_source' => 'wx',
  390. ];
  391. $this->where('user_id', $user['user_id'])->update($info);
  392. // 生成token (session3rd)
  393. $this->token = $this->token($user['user_no']);
  394. // 记录缓存, 30天
  395. Cache::tag('cache')->set($this->token, $user['user_id'], 86400 * 30);
  396. return $user['user_id'];
  397. }
  398. }