quaility.php 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435
  1. <?php
  2. namespace refill;
  3. use Log;
  4. use mtopcard;
  5. use scope_trace;
  6. class Quality
  7. {
  8. const LowestQuality = 1;
  9. const Normal = 1;
  10. const Quick = 2;
  11. const CardKey = 3;
  12. const ThirdShop = 4;
  13. const SlowTwentyFour = 5;
  14. const SlowSix = 6;
  15. const SlowTwo = 7;
  16. const SlowFortyEight = 8;
  17. const SlowSeventyTwo = 9;
  18. const Fastest = 10;
  19. const HighestQuality = 10;
  20. const SlowNormal = 11; // 7 -> 1
  21. const ThirdNormal = 12; // 4 -> 1
  22. const DefSuccess = 13; // 4 -> 3 -> 1 -> 2
  23. const NormalQuick = 14;
  24. const SlowSixNormal = 15;
  25. const OilWithoutSN = 1;
  26. const OilQuick = 2;
  27. const OilCardKey = 3;
  28. const OilWithSN = 5;
  29. const OilSN_NONE_HAS = 20;
  30. const OilSN_HAS_NONE = 21;
  31. const OIL_SN_CARDKEY = 22;
  32. const OIL_SNNONE_CARDKEY = 23;
  33. const OIL_SN_SNNONE_CARDKEY = 24;
  34. const OIL_SNNONE_SN_CARDKEY = 25;
  35. const OIL_SNNONE_SN_CARDKEY_QUICK = 26;
  36. protected $mMchPhonectl;
  37. protected $mMchoilctl;
  38. protected $mSpeeds;
  39. protected $mQualities;
  40. protected $mTryAdjuster;
  41. protected $mRatioCtl;
  42. public function __construct()
  43. {
  44. $this->mMchPhonectl = new mchctl();
  45. $this->mMchoilctl = new mchoilctl();
  46. $this->mTryAdjuster = new try_judge();
  47. $this->mRatioCtl = null;
  48. }
  49. public function load()
  50. {
  51. $this->mMchPhonectl->load();
  52. $this->mMchoilctl->load();
  53. $this->mTryAdjuster->load();
  54. }
  55. public function setRatioCtl($ratio_ctl) {
  56. $this->mRatioCtl = $ratio_ctl;
  57. }
  58. public function qualities($quality)
  59. {
  60. if (array_key_exists($quality, $this->mQualities)) {
  61. return $this->mQualities[$quality];
  62. } else {
  63. return [];
  64. }
  65. }
  66. public function mechants_quality() {
  67. return $this->mMchPhonectl->mechants_quality();
  68. }
  69. public function find_quality($mchid, $card_type, $spec, $org_quality, $cur_quality, $times, $used_time, $caller): array
  70. {
  71. $trace = new scope_trace(__METHOD__);
  72. if($card_type == mtopcard\ChinaMobileCard || $card_type == mtopcard\ChinaUnicomCard || $card_type == mtopcard\ChinaTelecomCard) {
  73. return $this->mobile_quality($mchid, $card_type, $spec, $org_quality, $cur_quality, $times, $used_time);
  74. }
  75. elseif($card_type == mtopcard\PetroChinaCard || $card_type == mtopcard\SinopecCard) {
  76. return $this->oil_quality($mchid, $org_quality, $times,$used_time,$caller);
  77. }
  78. else {
  79. return [0, []];
  80. }
  81. }
  82. private function oil_quality($mchid, $quality, $times,$used_time,$caller): array
  83. {
  84. Log::record("oil_quality mchid={$mchid},quality={$quality},times={$times}",Log::DEBUG);
  85. if($quality == 0)
  86. {
  87. [$success,$setting_quality,$time_out] = $this->mMchoilctl->getCtls($mchid);
  88. if($success)
  89. {
  90. if(array_key_exists($setting_quality,$this->mQualities)) {
  91. $org = $setting_quality;
  92. $qualities = $this->mQualities[$setting_quality];
  93. }
  94. else {
  95. $org = $setting_quality;
  96. $qualities = [$setting_quality];
  97. }
  98. }
  99. else {
  100. $org = self::Normal;
  101. $qualities = $this->mQualities[$org];
  102. }
  103. }
  104. elseif(array_key_exists($quality,$this->mQualities)) {
  105. $org = $quality;
  106. $qualities = $this->mQualities[$quality];
  107. }
  108. else {
  109. Log::record("find_quality: cannot find any quality",Log::DEBUG);
  110. return [0,[]];
  111. }
  112. Log::record("oil_quality find qualities = " . implode(',',$qualities),Log::DEBUG);
  113. $qualities = $this->calc_oil_quality($qualities,$times,$used_time,$caller);
  114. return [$org,$qualities];
  115. }
  116. //通过每种类型通道耗时,倒推当前可用通道,并优先走推荐通道.
  117. private function calc_oil_quality($qualities, $times, $used_time, $caller)
  118. {
  119. $result = [];
  120. if($used_time > 900) {
  121. return $result;
  122. }
  123. $total_times = 0;
  124. foreach ($qualities as $quality)
  125. {
  126. $cur_times = $caller->calc_times($quality);
  127. if($cur_times <= 0) continue;
  128. $total_times += $cur_times;
  129. if($total_times > $times) {
  130. $result[] = $quality;
  131. }
  132. }
  133. Log::record("calc_oil_quality result = " . implode(',',$result),Log::DEBUG);
  134. return $result;
  135. }
  136. private function mobile_quality($mchid, $card_type, $spec, $org_quality, $cur_quality, $times, $used_time): array
  137. {
  138. if($org_quality == 0)
  139. {
  140. [$success,$setting_quality,$time_out] = $this->mMchPhonectl->getCtls($mchid);
  141. if($success)
  142. {
  143. if(array_key_exists($setting_quality,$this->mQualities)) {
  144. $org_quality = $setting_quality;
  145. }
  146. else {
  147. $org_quality = $setting_quality;
  148. }
  149. }
  150. else {
  151. $org_quality = self::Normal; //如果没设置质量,默认为普通
  152. $time_out = $this->mSpeeds[$org_quality]['retry_timeout'];
  153. }
  154. }
  155. elseif(array_key_exists($org_quality,$this->mQualities)) {
  156. [$success, $setting_quality, $time_out] = $this->mMchPhonectl->getCtls($mchid);
  157. }
  158. else {
  159. Log::record("find_quality: cannot find any quality",Log::DEBUG);
  160. return [0,[]];
  161. }
  162. if($time_out <= 0) {
  163. $time_out = $this->mSpeeds[$org_quality]['retry_timeout'];
  164. }
  165. $qualities = $this->qualities($org_quality);
  166. $max_times = $this->max_times($mchid,$org_quality,$qualities);
  167. if ($this->mRatioCtl->exist($mchid)) {
  168. Log::record("ratio_phone_quality exist=true",Log::DEBUG);
  169. $qualities = $this->ratio_phone_quality($org_quality, $qualities, $cur_quality, $times, $used_time, $mchid, $card_type, $spec);
  170. return [$org_quality, $qualities];
  171. } else {
  172. $fMixed = PolicyUtil::mixed_quality($org_quality);
  173. $qualities = $this->calc_phone_quality($qualities, $fMixed, $cur_quality, $times, $used_time, $time_out, $max_times, $mchid);
  174. return [$org_quality, $qualities];
  175. }
  176. }
  177. private function ratio_phone_quality($org_quality,$qualities, $cur_quality, $times, $used_time, $mchid, $card_type, $spec)
  178. {
  179. [$succ,$max_times,$time_out] = $this->mRatioCtl->total($mchid,$qualities);
  180. $left_time = $time_out - $used_time;
  181. if ($left_time <= 0 || $max_times <= $times) {
  182. return [];
  183. }
  184. $match_ratio = $this->mRatioCtl->ratio_match($mchid, $org_quality, $cur_quality, $card_type, $spec, $qualities);
  185. Log::record("ratio_phone_quality begin qualities= " . implode(',',$qualities),Log::DEBUG);
  186. $times_checker = function ($qualities, $times) use ($match_ratio, $cur_quality, $max_times, $mchid)
  187. {
  188. $result = [];
  189. $pre_times = 0;
  190. $compare = $cur_quality > 0 ? false : true;
  191. foreach ($qualities as $quality)
  192. {
  193. if($compare == false)
  194. {
  195. if($quality == $cur_quality) {
  196. $compare = true;
  197. }
  198. }
  199. if($compare == false) continue;
  200. if ($match_ratio) {
  201. $result[] = $quality;
  202. }
  203. else
  204. {
  205. $qtimes = $this->mRatioCtl->times($mchid, $quality);
  206. if($qtimes == false) continue;
  207. $pre_times += $qtimes;
  208. if ($times < $pre_times) {
  209. $result[] = $quality;
  210. }
  211. }
  212. }
  213. return $result;
  214. };
  215. $timeout_checker = function ($qualities, $left_time) use($cur_quality,$match_ratio,$mchid)
  216. {
  217. if($match_ratio)
  218. {
  219. $result = [];
  220. $compare = $cur_quality > 0 ? false : true;
  221. foreach ($qualities as $quality)
  222. {
  223. if($compare == false)
  224. {
  225. if($quality == $cur_quality) {
  226. $compare = true;
  227. }
  228. }
  229. if($compare == false) continue;
  230. $per_secs = $this->mSpeeds[$quality]['per_secs'];
  231. $left_time -= $per_secs;
  232. if($left_time > 0) {
  233. $result[] = $quality;
  234. } else {
  235. break;
  236. }
  237. }
  238. return $result;
  239. }
  240. else
  241. {
  242. $result = [];
  243. $qualities = array_reverse($qualities);
  244. foreach ($qualities as $quality)
  245. {
  246. $per_secs = $this->mSpeeds[$quality]['per_secs'];
  247. $qsecs = $this->mRatioCtl->seconds($mchid, $quality);
  248. Log::record("left_time={$left_time} quality={$quality} secs={$qsecs}",Log::DEBUG);
  249. if($qsecs == false) continue;
  250. if($left_time - $qsecs > 0) {
  251. $left_time -= $qsecs;
  252. $result[] = $quality;
  253. }
  254. elseif($left_time - $per_secs > 0) {
  255. $left_time -= $per_secs;
  256. $result[] = $quality;
  257. }
  258. else {
  259. break;
  260. }
  261. if($quality == $cur_quality) {
  262. break;
  263. }
  264. }
  265. $result = array_reverse($result);
  266. return $result;
  267. }
  268. };
  269. $pTryAdjuster = $this->mTryAdjuster;
  270. $timeing_checker = function ($qualities) use($pTryAdjuster,$mchid)
  271. {
  272. foreach ($qualities as $quality)
  273. {
  274. $ret = $pTryAdjuster->can_try($mchid,$quality);
  275. if($ret == false) {
  276. return false;
  277. }
  278. }
  279. return true;
  280. };
  281. if($times > 0)
  282. {
  283. $qualities = $times_checker($qualities,$times);
  284. Log::record("calc_quality times_checker result = " . implode(',', $qualities), Log::DEBUG);
  285. $qualities = $timeout_checker($qualities,$left_time);
  286. Log::record("calc_quality timeout_checker result = " . implode(',', $qualities), Log::DEBUG);
  287. if(!$timeing_checker($qualities)) {
  288. $qualities = [];
  289. Log::record("calc_quality timeing_checker result is empty", Log::DEBUG);
  290. }
  291. }
  292. return $qualities;
  293. }
  294. private function max_times($mchid,$quality,$qualities)
  295. {
  296. [$succ, $times,$secs] = $this->mRatioCtl->total($mchid,$qualities);
  297. if ($succ) {
  298. return $times;
  299. } else {
  300. return $this->mSpeeds[$quality]['retry_times'];
  301. }
  302. }
  303. //通过每种类型通道耗时,倒推当前可用通道,并优先走推荐通道.
  304. private function calc_phone_quality($qualities,$fMixed, $cur_quality, $times, $used_time, $time_out, $max_times, $mchid)
  305. {
  306. $left_time = $time_out - $used_time;
  307. if(!$fMixed)
  308. {
  309. if($left_time <= 0 || $max_times <= $times) {
  310. return [];
  311. }
  312. }
  313. Log::record("calc_phone_quality begin qualities= " . implode(',',$qualities),Log::DEBUG);
  314. $times_checker = function($qualities, $times)
  315. {
  316. $result = [];
  317. $pre_times = 0;
  318. foreach ($qualities as $quality)
  319. {
  320. $pre_times += $this->mSpeeds[$quality]['retry_times'];
  321. if($times < $pre_times) {
  322. $result[] = $quality;
  323. }
  324. }
  325. return $result;
  326. };
  327. $timeout_checker = function ($qualities, $left_time) use($fMixed,$cur_quality)
  328. {
  329. $qualities = array_reverse($qualities);
  330. if ($fMixed) {
  331. $fLasted = $cur_quality != $qualities[0];
  332. } else {
  333. $fLasted = false;
  334. }
  335. $result = [];
  336. foreach ($qualities as $quality)
  337. {
  338. $per_secs = $this->mSpeeds[$quality]['per_secs'];
  339. $left_time -= $per_secs;
  340. if($left_time > 0) {
  341. $result[] = $quality;
  342. } else {
  343. break;
  344. }
  345. }
  346. if(empty($result) && $fLasted) {
  347. $result[] = $qualities[0];
  348. }
  349. $result = array_reverse($result);
  350. return $result;
  351. };
  352. $pTryAdjuster = $this->mTryAdjuster;
  353. $timeing_checker = function ($qualities) use($pTryAdjuster,$mchid)
  354. {
  355. foreach ($qualities as $quality)
  356. {
  357. $ret = $pTryAdjuster->can_try($mchid,$quality);
  358. if($ret == false) {
  359. return false;
  360. }
  361. }
  362. return true;
  363. };
  364. if($times > 0)
  365. {
  366. $qualities = $times_checker($qualities,$times);
  367. Log::record("calc_quality times_checker result = " . implode(',', $qualities), Log::DEBUG);
  368. $qualities = $timeout_checker($qualities,$left_time);
  369. Log::record("calc_quality timeout_checker result = " . implode(',', $qualities), Log::DEBUG);
  370. if(!$timeing_checker($qualities)) {
  371. $qualities = [];
  372. Log::record("calc_quality timeing_checker result is empty", Log::DEBUG);
  373. }
  374. }
  375. return $qualities;
  376. }
  377. }