MobileApi.php 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458
  1. <?php
  2. namespace app\admin\controller;
  3. use app\admin\model\Admin;
  4. use app\admin\model\AdminLog;
  5. use app\admin\model\Mobile as MO;
  6. use app\admin\model\MobileExportLog;
  7. use app\common\controller\Backend;
  8. use app\common\library\MobileConstant;
  9. use app\common\model\SysConfig;
  10. use app\common\service\MobileImport;
  11. use app\service\EsMobileService;
  12. use think\Db;
  13. use think\Loader;
  14. /**
  15. *
  16. *
  17. * @icon fa fa-mobile
  18. */
  19. class MobileApi extends Backend
  20. {
  21. const TYPE=MO::API;
  22. protected $noNeedRight=['mobile_rules','status','constant','exclude_nums'];
  23. /**
  24. * Mobile模型对象
  25. * @var \app\admin\model\Mobile
  26. */
  27. protected $model = null;
  28. protected $relationSearch=true;
  29. public function _initialize()
  30. {
  31. //ini_set('memory_limit',-1);
  32. parent::_initialize();
  33. $this->model = new \app\admin\model\Mobile;
  34. $this->assign('status',\app\common\model\Mobile::$status);
  35. $this->assign('network',MobileConstant::getNetworkString());
  36. $this->assign('network_color',MobileConstant::getNetworkColor());
  37. $this->assign('network_select',MobileConstant::getNetworkSelect());
  38. }
  39. /**
  40. * api号码管理
  41. */
  42. public function index()
  43. {
  44. //设置过滤方法
  45. $this->request->filter(['strip_tags', 'trim']);
  46. list($where, $sort, $order, $offset, $limit,$page) = $this->buildindexparams();
  47. if($this->request->get('export')==1 && $this->auth->check('mobile/_mobile_export')){
  48. if(MobileExportLog::exists()){
  49. die('有未完成导出,请稍后再试');
  50. }
  51. $export=MobileExportLog::make($this->admin('id'));
  52. ini_set('memory_limit',-1);
  53. ini_set('max_execution_time',0);
  54. $db=Db::name('mobile');
  55. $db->getConnection()->setConfig('resultset_type','array');
  56. $list= $db
  57. ->where($where)
  58. ->where('mobile.type',self::TYPE)
  59. ->orderRaw($this->getOrder())
  60. ->buildSql();
  61. //dd($list);
  62. $export['sql']=$list;
  63. $export->save();
  64. die('已添加到导出日志,请稍后查看');
  65. }
  66. if ($this->request->isAjax()) {
  67. //如果发送的来源是Selectpage,则转发到Selectpage
  68. if ($this->request->request('keyField')) {
  69. return $this->selectpage();
  70. }
  71. $deleteMode=input('_delete_condition');
  72. if($deleteMode==='true'){
  73. if(!$this->auth->check('mobile/del_condition')){
  74. $this->error('您无权进行此操作');
  75. }
  76. $num=$this->model
  77. ->alias(null)
  78. ->where('mobile.type',self::TYPE)
  79. ->where($where)
  80. ->delete();
  81. AdminLog::record('根据条件删除号码');
  82. \app\common\model\Mobile::deleteOtherTableInfo();
  83. $deleteConditionMsg=sprintf("成功删除%d个号码",$num);
  84. }
  85. if(!$this->admin('is_sub')){
  86. $list = $this->model
  87. ->with(['info','proxy'])
  88. ->where($where)
  89. ->where('mobile.type',self::TYPE)
  90. ->orderRaw($this->getOrder())
  91. ->page($page)
  92. ->limit($limit)
  93. ->select();
  94. }else{
  95. $list=$this->model
  96. ->with(['info'])
  97. ->field('mobile.*')
  98. ->where($where)
  99. ->where('mobile.type',self::TYPE)
  100. ->orderRaw($this->getOrder())
  101. ->page($page)
  102. ->limit($limit)
  103. ->select();
  104. }
  105. foreach ($list as $row) {
  106. $rules=[];
  107. foreach (MobileConstant::getFilters() as $rule=>$field){
  108. foreach (array_values($field) as $column){
  109. if($row[$column]==1){
  110. $rules[]=$rule;
  111. }
  112. }
  113. }
  114. $row['rules']=array_values(array_unique($rules));
  115. }
  116. $result = array("total" => EsMobileService::count(self::TYPE), "rows" => $list,'input'=>input(),'delete_condition_msg'=>$deleteConditionMsg??null,'url'=>$this->request->url().'&export=1');
  117. return json($result);
  118. }
  119. $this->assign('no_type',array_column(MobileConstant::getNoType(),'name','id'));
  120. $this->assign('filters',MobileConstant::getFilters());
  121. return $this->view->fetch();
  122. }
  123. protected function getOrder(){
  124. $filter=json_decode(input('filter'),true);
  125. $default='mobile.id desc';
  126. return $default;
  127. if(!$filter||empty($filter['no'])){
  128. return $default;
  129. }
  130. $value=str_replace('%','',$filter['no']);
  131. if(!$value){
  132. return $default;
  133. }
  134. $lastNum=substr($value,-1);
  135. if(strlen($value)==1){
  136. return "FIELD(`mobile.filter_no_pos_11`,$lastNum) DESC";
  137. }else{
  138. $arr=[];
  139. $numLen=strlen($value);
  140. $idx=1;
  141. for ($i=11;$i>=2;$i--){
  142. if($idx>$numLen){
  143. break;
  144. }
  145. $numPos=$value[$numLen-$idx];
  146. $arr[]="FIELD(mobile.filter_no_pos_{$i},$numPos) DESC";
  147. $idx++;
  148. }
  149. return implode(',',$arr);
  150. }
  151. }
  152. protected function buildindexparams($searchfields = null, $relationSearch = null)
  153. {
  154. $searchfields = is_null($searchfields) ? $this->searchFields : $searchfields;
  155. $relationSearch = is_null($relationSearch) ? $this->relationSearch : $relationSearch;
  156. $search = $this->request->get("search", '');
  157. $filter = $this->request->get("filter", '');
  158. $op = $this->request->get("op", '', 'trim');
  159. $sort = $this->request->get("sort", !empty($this->model) && $this->model->getPk() ? $this->model->getPk() : 'id');
  160. $order = $this->request->get("order", "DESC");
  161. $offset = $this->request->get("offset/d", 0);
  162. $limit = $this->request->get("limit/d", 999999);
  163. //新增自动计算页码
  164. $page = $limit ? intval($offset / $limit) + 1 : 1;
  165. if ($this->request->has("page")) {
  166. $page = $this->request->get("page/d", 1);
  167. }
  168. $this->request->get([config('paginate.var_page') => $page]);
  169. $filter = (array)json_decode($filter, true);
  170. $op = (array)json_decode($op, true);
  171. $filter = $filter ? $filter : [];
  172. $where = [];
  173. $alias = [];
  174. $bind = [];
  175. $name = '';
  176. $aliasName = '';
  177. if (!empty($this->model) && $this->relationSearch) {
  178. $name = $this->model->getTable();
  179. $alias[$name] = Loader::parseName(basename(str_replace('\\', '/', get_class($this->model))));
  180. $aliasName = $alias[$name] . '.';
  181. }
  182. $sortArr = explode(',', $sort);
  183. foreach ($sortArr as $index => & $item) {
  184. $item = stripos($item, ".") === false ? $aliasName . trim($item) : $item;
  185. }
  186. unset($item);
  187. $sort = implode(',', $sortArr);
  188. $adminIds = $this->getDataLimitAdminIds();
  189. if (is_array($adminIds)) {
  190. $where[] = [$aliasName . $this->dataLimitField, 'in', $adminIds];
  191. }
  192. if ($search) {
  193. $searcharr = is_array($searchfields) ? $searchfields : explode(',', $searchfields);
  194. foreach ($searcharr as $k => &$v) {
  195. $v = stripos($v, ".") === false ? $aliasName . $v : $v;
  196. }
  197. unset($v);
  198. $where[] = [implode("|", $searcharr), "LIKE", "%{$search}%"];
  199. }
  200. $index = 0;
  201. foreach ($filter as $k => $v) {
  202. if($k=='rules'){
  203. continue;
  204. }
  205. if (!preg_match('/^[a-zA-Z0-9_\-.]+$/', $k)) {
  206. continue;
  207. }
  208. $sym = isset($op[$k]) ? $op[$k] : '=';
  209. if (stripos($k, ".") === false) {
  210. $k = $aliasName . $k;
  211. }
  212. $v = !is_array($v) ? trim($v) : $v;
  213. $sym = strtoupper(isset($op[$k]) ? $op[$k] : $sym);
  214. //null和空字符串特殊处理
  215. if (!is_array($v)) {
  216. if (in_array(strtoupper($v), ['NULL', 'NOT NULL'])) {
  217. $sym = strtoupper($v);
  218. }
  219. if (in_array($v, ['""', "''"])) {
  220. $v = '';
  221. $sym = '=';
  222. }
  223. }
  224. switch ($sym) {
  225. case '=':
  226. case '<>':
  227. $where[] = [$k, $sym, (string)$v];
  228. break;
  229. case 'LIKE':
  230. case 'NOT LIKE':
  231. case 'LIKE %...%':
  232. case 'NOT LIKE %...%':
  233. $where[] = [$k, trim(str_replace('%...%', '', $sym)), "%{$v}%"];
  234. break;
  235. case '>':
  236. case '>=':
  237. case '<':
  238. case '<=':
  239. $where[] = [$k, $sym, intval($v)];
  240. break;
  241. case 'FINDIN':
  242. case 'FINDINSET':
  243. case 'FIND_IN_SET':
  244. $v = is_array($v) ? $v : explode(',', str_replace(' ', ',', $v));
  245. $findArr = array_values($v);
  246. foreach ($findArr as $idx => $item) {
  247. $bindName = "item_" . $index . "_" . $idx;
  248. $bind[$bindName] = $item;
  249. $where[] = "FIND_IN_SET(:{$bindName}, `" . str_replace('.', '`.`', $k) . "`)";
  250. }
  251. break;
  252. case 'IN':
  253. case 'IN(...)':
  254. case 'NOT IN':
  255. case 'NOT IN(...)':
  256. $where[] = [$k, str_replace('(...)', '', $sym), is_array($v) ? $v : explode(',', $v)];
  257. break;
  258. case 'BETWEEN':
  259. case 'NOT BETWEEN':
  260. $arr = array_slice(explode(',', $v), 0, 2);
  261. if (stripos($v, ',') === false || !array_filter($arr)) {
  262. continue 2;
  263. }
  264. //当出现一边为空时改变操作符
  265. if ($arr[0] === '') {
  266. $sym = $sym == 'BETWEEN' ? '<=' : '>';
  267. $arr = $arr[1];
  268. } elseif ($arr[1] === '') {
  269. $sym = $sym == 'BETWEEN' ? '>=' : '<';
  270. $arr = $arr[0];
  271. }
  272. $where[] = [$k, $sym, $arr];
  273. break;
  274. case 'RANGE':
  275. case 'NOT RANGE':
  276. $v = str_replace(' - ', ',', $v);
  277. $arr = array_slice(explode(',', $v), 0, 2);
  278. if (stripos($v, ',') === false || !array_filter($arr)) {
  279. continue 2;
  280. }
  281. //当出现一边为空时改变操作符
  282. if ($arr[0] === '') {
  283. $sym = $sym == 'RANGE' ? '<=' : '>';
  284. $arr = $arr[1];
  285. } elseif ($arr[1] === '') {
  286. $sym = $sym == 'RANGE' ? '>=' : '<';
  287. $arr = $arr[0];
  288. }
  289. $tableArr = explode('.', $k);
  290. if (count($tableArr) > 1 && $tableArr[0] != $name && !in_array($tableArr[0], $alias) && !empty($this->model)) {
  291. //修复关联模型下时间无法搜索的BUG
  292. $relation = Loader::parseName($tableArr[0], 1, false);
  293. $alias[$this->model->$relation()->getTable()] = $tableArr[0];
  294. }
  295. $where[] = [$k, str_replace('RANGE', 'BETWEEN', $sym) . ' TIME', $arr];
  296. break;
  297. case 'NULL':
  298. case 'IS NULL':
  299. case 'NOT NULL':
  300. case 'IS NOT NULL':
  301. $where[] = [$k, strtolower(str_replace('IS ', '', $sym))];
  302. break;
  303. default:
  304. break;
  305. }
  306. $index++;
  307. }
  308. if (!empty($this->model)) {
  309. $this->model->alias($alias);
  310. }
  311. if(isset($filter['rules']) && $filter['rules']){
  312. $temp=[];
  313. foreach (MobileConstant::getFilters()[$filter['rules']] as $pos=>$column){
  314. $temp[]=$column;
  315. }
  316. $flip=array_flip($temp);
  317. if(isset($filter['filter_first_last'])){
  318. if($filter['filter_first_last']==1){
  319. $where[] = [$temp[1],1];
  320. }elseif ($filter['filter_first_last']==2){
  321. $where[] = [$temp[0],1];
  322. }
  323. }else {
  324. $where[] = [implode("|", $temp), 1];
  325. }
  326. }
  327. $model = $this->model;
  328. $where = function ($query) use ($where, $alias, $bind, &$model,$filter) {
  329. if (!empty($model)) {
  330. $model->alias($alias);
  331. $model->bind($bind);
  332. }
  333. foreach ($where as $k => $v) {
  334. if (is_array($v)) {
  335. if(in_array($v[0],['mobile_sub.sub_top_time','mobile_sub.sub_rec_time'])){
  336. if($v[2]) {
  337. $query->whereIn($v[0], 1);
  338. }else{
  339. $query->where(function ($query)use ($v){
  340. $query->whereRaw("{$v[0]}=0 or {$v[0]} is null");
  341. });
  342. }
  343. continue;
  344. }elseif ($v[0]=='proxy.nickname'){
  345. $proxy_id=Admin::where('proxy',1)->where('nickname','like',"%{$v[2]}%")->column('id')?:[0];
  346. $query->whereIn('proxy_id',$proxy_id);
  347. continue;
  348. }elseif ($v[0]=='mobile.exclude_nums'){
  349. $arr=explode(',',$v[2]);
  350. foreach ($arr as $num){
  351. if(!is_numeric($num)){
  352. continue;
  353. }
  354. $query->where(sprintf("filter_num_%s",$num),0);
  355. }
  356. continue;
  357. }elseif ($v[0]=='mobile.filter_first_last'){
  358. continue;
  359. }elseif ($v[0]=='mobile.no'){
  360. $noSearch=$filter['no'];
  361. if(isset($filter['filter_first_last'])){
  362. if($filter['filter_first_last']==1) {
  363. $noPad = str_pad($noSearch, 11, 'a',STR_PAD_LEFT);
  364. for ($i = 11; $i >1; $i--) {
  365. if ($noPad[$i-1] !== 'a') {
  366. $query->where("filter_no_pos_{$i}", $noPad[$i-1]);
  367. }
  368. }
  369. }elseif ($filter['filter_first_last']==2){
  370. $query->where('filter_no_pos_11','<>',$noSearch[strlen($noSearch)-1])->whereRaw("LOCATE('{$noSearch}',no)");
  371. }
  372. }else{
  373. //$query->where('no','like',"%{$noSearch}%");
  374. $query->whereRaw("LOCATE('{$noSearch}',no)");
  375. }
  376. continue;
  377. }
  378. call_user_func_array([$query, 'where'], $v);
  379. } else {
  380. $query->where($v);
  381. }
  382. }
  383. $precise_search=input('precise_search/a');
  384. if(is_array($precise_search) && $precise_search){
  385. foreach ($precise_search as $idx=>$num){
  386. if(!is_numeric($num)){
  387. continue;
  388. }
  389. $query->where(sprintf('mobile.filter_no_pos_%d',$idx),(int)$num);
  390. }
  391. }
  392. };
  393. return [$where, $sort, $order, $offset, $limit, $page, $alias, $bind];
  394. }
  395. #规则定价
  396. public function price(){
  397. if($this->request->isGet()){
  398. $this->assign('rules',MobileConstant::getRuleKeys());
  399. $this->assign('row',SysConfig::look('api_rule_price',[]));
  400. return $this->fetch();
  401. }
  402. $rules=input('row/a');
  403. SysConfig::set('api_rule_price',$rules);
  404. $this->success();
  405. }
  406. public function del_all(){
  407. $data=['num'=>0];
  408. $data['num']= MO::where('type',self::TYPE)->delete();
  409. \app\common\model\Mobile::deleteOtherTableInfo();
  410. EsMobileService::clear(self::TYPE);
  411. AdminLog::record('清空号码');
  412. $this->success('',null,$data);
  413. }
  414. /**
  415. * 上架
  416. * @param $ids
  417. */
  418. public function up($ids){
  419. $mobiles=MO::where('type',self::TYPE)->whereIn('id',$ids)->select();
  420. foreach ($mobiles as $mobile){
  421. $mobile['type']=$mobile::BEAUTI;
  422. $mobile->save();
  423. }
  424. $this->success();
  425. }
  426. /**
  427. * 上架
  428. * @param $ids
  429. */
  430. public function all_up(){
  431. $ids=MO::where('type',self::TYPE)->column('id');
  432. // 改db
  433. (new MO)->where(['type'=>self::TYPE])->update(['type'=>MO::BEAUTI]);
  434. // 改es
  435. EsMobileService::updateById($ids,['type'=>MO::BEAUTI]);
  436. AdminLog::record('全部导入靓号');
  437. $this->success();
  438. }
  439. }