policy.php 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211
  1. <?php
  2. namespace refill;
  3. use Log;
  4. use mtopcard;
  5. class policy extends ProviderManager implements IPolicy
  6. {
  7. protected $mChannelControl;
  8. protected $mQuality;
  9. protected $mPrices;
  10. protected $mGroupCtl;
  11. public function __construct()
  12. {
  13. parent::__construct();
  14. $this->mChannelControl = new chctl();
  15. $this->mQuality = new quality_ploy();
  16. $this->mPrices = new merchant_price();
  17. $this->mGroupCtl = new rgroup_ctl();
  18. }
  19. public function load()
  20. {
  21. parent::load();
  22. $this->mChannelControl->load();
  23. $this->mQuality->load();
  24. $this->mPrices->load();
  25. $this->mGroupCtl->load();
  26. }
  27. public function find_providers(int $mchid, int $spec, int $card_type, int $quality, $regin_no): array
  28. {
  29. $providers = parent::find_providers($mchid,$spec,$card_type,$quality,$regin_no);
  30. if(empty($providers)) {
  31. return [$providers,false];
  32. }
  33. $names = [];
  34. foreach ($providers as $provider) {
  35. $names[] = $provider->name();
  36. }
  37. Log::record("mGroupCtl find_providers result=" . implode(',',$names),Log::DEBUG);
  38. [$hasGroup,$can_others,$channels] = $this->mGroupCtl->find_providers($mchid, $spec, $card_type, $quality);
  39. if($hasGroup)
  40. {
  41. if(empty($channels))
  42. {
  43. if(!$can_others) {
  44. return [[],false];
  45. }
  46. }
  47. else
  48. {
  49. $ret = array_intersect($names, $channels);
  50. if (empty($ret)) {
  51. return [[],false];
  52. }
  53. else {
  54. $names = $ret;
  55. }
  56. }
  57. }
  58. Log::record("after mGroupCtl find_providers result=" . implode(',',$names),Log::DEBUG);
  59. $name_overloads = $this->mChannelControl->match($names,$spec,$card_type,$quality);
  60. Log::record("policy::find_providers match result=" . implode(',',$names),Log::DEBUG);
  61. $result = [];
  62. foreach ($name_overloads as $name => $overload)
  63. {
  64. Log::record("channel {$name} has overloaded = {$overload}",Log::DEBUG);
  65. if(!isset($first)) {
  66. $first = $overload;
  67. }
  68. if($overload) continue;
  69. foreach ($providers as $provider)
  70. {
  71. if($name == $provider->name()) {
  72. $result[] = $provider;
  73. }
  74. }
  75. }
  76. if(!isset($first)) {
  77. $first = false;
  78. }
  79. return [$result,$first];
  80. }
  81. public function price($mchid,$spec,$card_type,$quality,$pcode)
  82. {
  83. return $this->mPrices->price($mchid,$card_type,$spec,$quality,$pcode);
  84. }
  85. public function channeles(int $mchid, int $spec, int $card_type, int $quality, $regin_no)
  86. {
  87. $providers = parent::find_providers($mchid, $spec, $card_type, $quality, $regin_no);
  88. return count($providers);
  89. }
  90. public function find_quality($mchid,$spec,$card_type,$org_quality,$times,$used_time,$pcode): array
  91. {
  92. if($card_type == mtopcard\SinopecCard || $card_type == mtopcard\PetroChinaCard) {
  93. $caller = new times_caller($mchid,$spec,$card_type,-1,$this);
  94. } else {
  95. $caller = null;
  96. }
  97. [$org_quality,$qualities] = $this->mQuality->find_quality($mchid,$card_type,$org_quality,$times,$used_time,$caller);
  98. if(empty($qualities)) {
  99. return [$org_quality,0];
  100. }
  101. foreach ($qualities as $quality)
  102. {
  103. $price = $this->mPrices->price($mchid,$card_type,$spec,$quality,$pcode);
  104. if($price === false) {
  105. Log::record("{$mchid} 没有协商 quality = {$quality} 价格",Log::DEBUG);
  106. continue;
  107. }
  108. [$providers,$overload] = $this->find_providers($mchid,$spec,$card_type,$quality,-1);
  109. if (!empty($providers))
  110. {
  111. if (!$overload) {
  112. Log::record("Policy::find_quality:{$quality}-{$spec}-{$card_type} is ok", Log::DEBUG);
  113. }
  114. else {
  115. Log::record("Policy::find_quality:{$quality}-{$spec}-{$card_type} is overload", Log::DEBUG);
  116. }
  117. return [$org_quality, $quality];
  118. } else {
  119. Log::record("Policy::find_quality:{$quality}-{$spec}-{$card_type} is fail", Log::DEBUG);
  120. }
  121. }
  122. return [$org_quality,0];
  123. }
  124. public function allow($mchid, $card_type, $amount, $quality): bool
  125. {
  126. return true;
  127. }
  128. public function notify($order_info, $refill_info) : bool
  129. {
  130. $order_state = $order_info['order_state'];
  131. if ($order_state == ORDER_STATE_CANCEL) {
  132. $state = 2;
  133. } else {
  134. $state = 1;
  135. }
  136. $mchid = $refill_info['mchid'];
  137. $mch_info = Model('merchant')->getMerchantInfo(['mchid' => $mchid]);
  138. [$params, $sign] = $this->body($state, $refill_info, $mch_info);
  139. $params['sgn'] = $sign;
  140. $notify_url = $refill_info['notify_url'];
  141. $resp = http_request($notify_url, $params, 'GET');
  142. return $resp == "ok";
  143. }
  144. private function body($state, $refill_info, $mch_info)
  145. {
  146. $params = [
  147. "usr" => $refill_info['mchid'],
  148. "ord" => $refill_info['mch_order'],
  149. 'bz' => $refill_info['official_sn'] ?? "",
  150. "state" => $state];
  151. $secure_key = $mch_info['secure_key'];
  152. $card_type = $refill_info['card_type'];
  153. if($card_type == mtopcard\ThirdRefillCard)
  154. {
  155. $mod_third = Model('thrid_refill');
  156. $thrid_info = $mod_third->getThird($refill_info['order_id']);
  157. if (!empty($thrid_info))
  158. {
  159. $card_info = $thrid_info['card_info'];
  160. if (!empty($card_info)) {
  161. $encrypt = openssl_encrypt($card_info,'AES-128-CBC',$secure_key);
  162. if($encrypt != false) {
  163. $params['card_info'] = $encrypt;
  164. }
  165. }
  166. }
  167. }
  168. $sign = $this->sign($params, $secure_key);
  169. return [$params, $sign];
  170. }
  171. private function sign($params, $key)
  172. {
  173. $body = "{$params['ord']}{$params['state']}{$key}";
  174. Log::record("notify body={$body}",Log::DEBUG);
  175. return strtoupper(md5($body));
  176. }
  177. }