AccountService.php 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300
  1. <?php
  2. namespace App\Services\Coach;
  3. use App\Enums\TechnicianAuthStatus;
  4. use Illuminate\Support\Facades\Cache;
  5. use Illuminate\Support\Facades\DB;
  6. use Illuminate\Support\Facades\Log;
  7. class AccountService
  8. {
  9. private const CACHE_KEY_PREFIX = 'coach_info_';
  10. private const CACHE_TTL = 300; // 5分钟
  11. /**
  12. * 提交技师基本信息
  13. */
  14. public function submitBaseInfo($user, array $data)
  15. {
  16. DB::beginTransaction();
  17. try {
  18. $this->setTransactionConfig();
  19. abort_if(! $user->coach, 404, '技师信息不存在');
  20. // 检查是否有待审核的记录
  21. $pendingRecord = $user->coach->infoRecords()
  22. ->where('state', TechnicianAuthStatus::AUDITING->value)
  23. ->exists();
  24. abort_if($pendingRecord, 422, '已有待审核的基本信息记录');
  25. // 创建技师信息
  26. $record = $user->coach->infoRecords()->create(array_merge($data, [
  27. 'state' => TechnicianAuthStatus::AUDITING->value,
  28. ]));
  29. // 清除技师信息缓存
  30. $this->clearCoachCache($user->coach->id);
  31. DB::commit();
  32. $this->logInfo('技师提交基本信息成功', $user, $data);
  33. return ['message' => '基本信息提交成功'];
  34. } catch (\Exception $e) {
  35. DB::rollBack();
  36. $this->logError('提交技师基本信息失败', $user, $data, $e);
  37. throw $e;
  38. }
  39. }
  40. /**
  41. * 提交技师资质信息
  42. */
  43. public function submitQualification($user, array $data)
  44. {
  45. DB::beginTransaction();
  46. try {
  47. $this->setTransactionConfig();
  48. abort_if(! $user->coach, 404, '技师信息不存在');
  49. // 检查是否有待审核的记录
  50. $pendingRecord = $user->coach->qualRecords()
  51. ->where('state', TechnicianAuthStatus::AUDITING->value)
  52. ->exists();
  53. abort_if($pendingRecord, 422, '已有待审核的资质信息记录');
  54. // 创建资质信息
  55. $record = $user->coach->qualRecords()->create(array_merge($data, [
  56. 'state' => TechnicianAuthStatus::AUDITING->value,
  57. ]));
  58. // 清除技师信息缓存
  59. $this->clearCoachCache($user->coach->id);
  60. DB::commit();
  61. $this->logInfo('技师提交资质信息成功', $user, $data);
  62. return ['message' => '资质信息提交成功'];
  63. } catch (\Exception $e) {
  64. DB::rollBack();
  65. $this->logError('提交技师资质信息失败', $user, $data, $e);
  66. throw $e;
  67. }
  68. }
  69. /**
  70. * 提交实名认证信息
  71. */
  72. public function submitRealName($user, array $data)
  73. {
  74. DB::beginTransaction();
  75. try {
  76. $this->setTransactionConfig();
  77. abort_if(! $user->coach, 404, '技师信息不存在');
  78. // 检查是否有待审核的记录
  79. $pendingRecord = $user->coach->realRecords()
  80. ->where('state', TechnicianAuthStatus::AUDITING->value)
  81. ->exists();
  82. abort_if($pendingRecord, 422, '已有待审核的实名认证信息');
  83. // 创建实名认证信息
  84. $record = $user->coach->realRecords()->create(array_merge($data, [
  85. 'state' => TechnicianAuthStatus::AUDITING->value,
  86. ]));
  87. // 清除技师信息缓存
  88. $this->clearCoachCache($user->coach->id);
  89. DB::commit();
  90. $this->logInfo('技师提交实名认证信息成功', $user, $this->maskSensitiveData($data));
  91. return ['message' => '实名认证信息提交成功'];
  92. } catch (\Exception $e) {
  93. DB::rollBack();
  94. $this->logError('提交实名认证信息失败', $user, $this->maskSensitiveData($data), $e);
  95. throw $e;
  96. }
  97. }
  98. /**
  99. * 获取技师信息
  100. */
  101. public function getCoachInfo($user)
  102. {
  103. try {
  104. abort_if(! $user, 404, '用户不存在');
  105. abort_if(! $user->coach, 404, '技师信息不存在');
  106. return Cache::remember(
  107. self::CACHE_KEY_PREFIX.$user->coach->id,
  108. self::CACHE_TTL,
  109. function () use ($user) {
  110. return $this->fetchCoachInfo($user->coach);
  111. }
  112. );
  113. } catch (\Exception $e) {
  114. $this->logError('获取技师信息失败', $user, [], $e);
  115. throw $e;
  116. }
  117. }
  118. /**
  119. * 设置事务配置
  120. */
  121. private function setTransactionConfig()
  122. {
  123. DB::statement('SET SESSION TRANSACTION ISOLATION LEVEL READ COMMITTED');
  124. DB::statement('SET SESSION innodb_lock_wait_timeout=10');
  125. }
  126. /**
  127. * 记录信息日志
  128. */
  129. private function logInfo(string $message, $user, array $data)
  130. {
  131. Log::info($message, [
  132. 'user_id' => $user->id,
  133. 'coach_id' => $user->coach->id,
  134. 'data' => $data,
  135. 'ip' => request()->ip(),
  136. 'timestamp' => now()->toDateTimeString(),
  137. ]);
  138. }
  139. /**
  140. * 记录错误日志
  141. */
  142. private function logError(string $message, $user, array $data, \Exception $e)
  143. {
  144. Log::error($message, [
  145. 'user_id' => $user->id,
  146. 'coach_id' => $user->coach->id ?? null,
  147. 'data' => $data,
  148. 'error' => $e->getMessage(),
  149. 'file' => $e->getFile(),
  150. 'line' => $e->getLine(),
  151. 'ip' => request()->ip(),
  152. 'timestamp' => now()->toDateTimeString(),
  153. ]);
  154. }
  155. /**
  156. * 获取技师详细信息
  157. */
  158. private function fetchCoachInfo($coach)
  159. {
  160. $baseInfo = $coach->infoRecords()->latest()->first();
  161. $qualification = $coach->qualRecords()->latest()->first();
  162. $realName = $coach->realRecords()->latest()->first();
  163. return [
  164. 'base_info' => $baseInfo ? $this->formatBaseInfo($baseInfo) : null,
  165. 'qualification' => $qualification ? $this->formatQualification($qualification) : null,
  166. 'real_name' => $realName ? $this->formatRealName($realName) : null,
  167. ];
  168. }
  169. /**
  170. * 格式化基本信息
  171. */
  172. private function formatBaseInfo($info)
  173. {
  174. return [
  175. 'nickname' => $info->nickname,
  176. 'avatar' => $info->avatar,
  177. 'gender' => $info->gender,
  178. 'mobile' => $this->maskMobile($info->mobile),
  179. 'birthday' => $info->birthday,
  180. 'work_years' => $info->work_years,
  181. 'intention_city' => $info->intention_city,
  182. 'introduction' => $info->introduction,
  183. 'state' => $info->state,
  184. 'state_text' => TechnicianAuthStatus::fromValue($info->state)->label(),
  185. 'audit_remark' => $info->audit_remark,
  186. ];
  187. }
  188. /**
  189. * 格式化资质信息
  190. */
  191. private function formatQualification($qual)
  192. {
  193. return [
  194. 'qual_type' => $qual->qual_type,
  195. 'qual_no' => $qual->qual_no,
  196. 'qual_photo' => $qual->qual_photo,
  197. 'valid_start' => $qual->valid_start,
  198. 'valid_end' => $qual->valid_end,
  199. 'state' => $qual->state,
  200. 'state_text' => TechnicianAuthStatus::fromValue($qual->state)->label(),
  201. 'audit_remark' => $qual->audit_remark,
  202. ];
  203. }
  204. /**
  205. * 格式化实名信息
  206. */
  207. private function formatRealName($real)
  208. {
  209. return [
  210. 'real_name' => $real->real_name,
  211. 'id_card' => $this->maskIdCard($real->id_card),
  212. 'id_card_front_photo' => $real->id_card_front_photo,
  213. 'id_card_back_photo' => $real->id_card_back_photo,
  214. 'id_card_hand_photo' => $real->id_card_hand_photo,
  215. 'state' => $real->state,
  216. 'state_text' => TechnicianAuthStatus::fromValue($real->state)->label(),
  217. 'audit_remark' => $real->audit_remark,
  218. ];
  219. }
  220. /**
  221. * 手机号脱敏
  222. */
  223. private function maskMobile($mobile)
  224. {
  225. return substr_replace($mobile, '****', 3, 4);
  226. }
  227. /**
  228. * 身份证号脱敏
  229. */
  230. private function maskIdCard($idCard)
  231. {
  232. return substr_replace($idCard, '****', 6, 8);
  233. }
  234. /**
  235. * 敏感数据脱敏
  236. */
  237. private function maskSensitiveData(array $data)
  238. {
  239. if (isset($data['id_card'])) {
  240. $data['id_card'] = $this->maskIdCard($data['id_card']);
  241. }
  242. if (isset($data['mobile'])) {
  243. $data['mobile'] = $this->maskMobile($data['mobile']);
  244. }
  245. return $data;
  246. }
  247. /**
  248. * 清除技师信息缓存
  249. */
  250. private function clearCoachCache($coachId)
  251. {
  252. Cache::forget(self::CACHE_KEY_PREFIX.$coachId);
  253. }
  254. }