| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500 | <?phpnamespace App\Service;use App\Model\BasicType;use App\Model\Product;use App\Model\SalesOrder;use App\Model\SalesOrderInfo;use App\Model\SalesOrderProductInfo;use Illuminate\Support\Facades\DB;use Illuminate\Support\Facades\Log;use Illuminate\Support\Facades\Redis;class DouShopService extends Service{    public $shopId = '59866821';    public $appKey = '7385361854375216667';    public $appSecret = '1e8de180-8c02-49fc-960c-77062fdb9361';    public $host = 'https://openapi-fxg.jinritemai.com';    const RedisKey = 'DOUDIANACCESSTOKEN';    //获取token    public $url_create = '/token/create';    //订单列表    public $url_order_list = '/order/searchList';    //订单详情    public $url_order_detail = '/order/orderDetail';    //------------------获取token    public function getToken($data){        if(empty($data['shop_id'])) return [false, '请选择店铺不能为空'];        $key = $this->appKey . $data['shop_id'] . self::RedisKey;        $token = Redis::get($key);        if(! empty($token)) return [true, $token];        list($bool,$msg) = $this->curlForToken($key, $data['shop_id']);        if(! $bool) return [false, $msg];        return [true, $msg];    }    public function curlForToken($key, $shop_id){        $param['shop_id'] = $shop_id;        $param['grant_type'] = "authorization_self";        $param['code'] = "";        list($url, $paramJson) = $this->organization($param,$this->url_create);        //发送请求        list($bool,$return) = $this->post_helper($url,$paramJson);        if(! $bool) return [false, $return];        if($return['code'] != 10000) return [false, $return['sub_msg']];        file_put_contents('token_'. $shop_id . '.txt',json_encode($return));        $token = $return['data']['access_token'];        $expires_in = $return['data']['expires_in'] - 3600;        Redis::setex($key,$expires_in,$token);        return [true,$token];    }    //------------------获取token    //------------------获取订单列表    public function getOrderList($data,$user){        list($bool,$token) = $this->getToken($data);        if(! $bool) return [false, $token];        list($bool,$return) = $this->curlForOrderList($token,$data);        if(! $bool) return [false, $return];        return [true, $return];    }    public function curlForOrderList($token,$data){        if(empty($data['create_time'][0]) || empty($data['create_time'][0])) return [false, '下单时间不能为空'];        //下单时间:开始结束,秒级时间戳        $return = $this->changeDateToTimeStampAboutRange($data['create_time']);        $param['create_time_start'] = $return[0];        $param['create_time_end'] = $return[1];        if(! empty($data['product'])) {            //商品,number型代表商品ID,其它代表商品名称            $param['product'] = $data['product'];        }        if(isset($data['b_type'])){            //【下单端】 0、站外 1、火山 2、抖音 3、头条 4、西瓜 5、微信 6、值点app 7、头条lite 8、懂车帝 9、皮皮虾 11、抖音极速版 12、TikTok 13、musically 14、穿山甲 15、火山极速版 16、服务市场 26、番茄小说 27、UG教育营销电商平台 28、Jumanji 29、电商SDK            $param['b_type'] = $data['b_type'];        }        if(! empty($data['after_sale_status_desc'])){            //售后状态:all-全部,in_aftersale-售后中,refund-退款中,refund_success-退款成功,refund_fail-退款失败,exchange_success-换货成功 aftersale_close-售后关闭            $param['after_sale_status_desc'] = $data['after_sale_status_desc'];        }        if(! empty($data['tracking_no'])){            //物流单号            $param['tracking_no'] = $data['tracking_no'];        }        if(isset($data['presell_type'])){            //预售类型:0-普通订单;1-全款预售;2-定金预售;3-定金找货;            $param['presell_type'] = $data['presell_type'];        }        if(isset($data['order_type'])){            //订单类型 0、普通订单 2、虚拟商品订单 4、电子券(poi核销) 5、三方核销            $param['order_type'] = $data['order_type'];        }        if(! empty($data['abnormal_order'])) {            //异常订单,1-异常取消,2-风控审核中            $param['abnormal_order'] = $data['abnormal_order'];        }        if(isset($data['trade_type'])){            //交易类型:0-普通;1-拼团;2-定金预售;3-订金找货;4-拍卖;5-0元单;6-回收;7-寄卖;10-寄样;11-0元抽奖(超级福袋);12-达人买样;13-普通定制;16-大众竞拍;18-小时达;102-定金预售的赠品单;103-收款;            $param['trade_type'] = $data['trade_type'];        }        if(! empty($data['update_time'][0]) && ! empty($data['update_time'][0])){            //更新时间:开始结束,秒级时间戳            $return = $this->changeDateToTimeStampAboutRange($data['update_time']);            $param['update_time_start'] = $return[0];            $param['update_time_end'] = $return[1];        }        if(empty($data['size'])) {            //单页大小,限制100以内            $param['size'] = 20;        }else{            if($data['size'] > 100) return [false, '单页大小,限制100以内'];            $param['size'] = $data['size'];        }        if(! isset($data['page'])) {            $param['page'] = 0;        }else{            $param['page'] = $data['page'];        }        if(! empty($data['order_by'])){            //排序条件(create_time 订单创建时间;update_time 订单更新时间;默认create_time;)            $param['order_by'] = $data['order_by'];        }        //排序类型,小到大或大到小,默认大到小        if(! isset($data['order_asc'])){            $param['order_asc'] = false;        }else{            $param['order_asc'] = $data['order_asc'];        }        if(! empty($data['fulfil_status'])) {            //履约状态;如小时达未接单"no_accept"            $param['fulfil_status'] = $data['fulfil_status'];        }        //组织数据        list($url, $paramJson) = $this->organization($param,$this->url_order_list,$token);        //发送请求        list($bool,$return) = $this->post_helper($url,$paramJson);        if(! $bool) return [false, $return];        if($return['code'] != 10000) return [false, $return['sub_msg']];        return [true, $return['data'] ?? []];    }    //------------------获取订单列表    //------------------获取订单详情    public function getOrderDetail($data,$user){        list($bool,$token) = $this->getToken($data);        if(! $bool) return [false, $token];        list($bool,$return) = $this->curlForOrderDetail($token,$data);        if(! $bool) return [false, $return];        return [true, $return];    }    public function curlForOrderDetail($token,$data){        if(empty($data['shop_order_id'])) return [false, '店铺订单号不能为空'];        $param['shop_order_id'] = $data['shop_order_id'];        //组织数据        list($url, $paramJson) = $this->organization($param,$this->url_order_detail,$token);        //发送请求        list($bool,$return) = $this->post_helper($url,$paramJson);        if(! $bool) return [false, $return];        if($return['code'] != 10000) return [false, $return['sub_msg']];        return [true, $return['data'] ?? []];    }    //------------------获取订单详情    //------------------导入线上订单    public function insertDouOrder($data,$user){//        [//            'shop_id' => 111,//            'order_list'=>[//                [//                    'order_id' => '',//                    'order_amount' => '',//                    'product' => [//                        [//                            'product_name' => '',//                            'code' => '',//                            'item_num' => '',//                            'sum_amount' => '',//                        ]//                    ],//                ],//                [//                    'order_id' => '',//                    'order_amount' => '',//                    'product' => [//                        [//                            'product_name' => '',//                            'code' => '',//                            'item_num' => '',//                            'sum_amount' => '',//                        ]//                    ],//                ]//            ],//        ];        if(empty($data['order_list'])) return [false, '订单数据不能为空'];        if(empty($data['shop_id'])) return [false, '店铺ID不能为空'];        $basic = BasicType::where('del_time',0)            ->where('code', $data['shop_id'])            ->first();        if(empty($basic->id)) return [false, '未找到店铺绑定的平台,请联系管理员进行绑定'];        $plat_type = $basic->id;        $customer_short_name = 0;        $basic = BasicType::where('del_time',0)            ->whereIn('type',[29])            ->where('title','抖音平台')            ->where('top_depart_id',$user['head']['id'])            ->select('id')->first();        if(! empty($basic->id)) $customer_short_name = $basic->id;        $search = "" ;        $order_id = [];        foreach ($data['order_list'] as $value){            if(empty($value['order_id'])) return [false, '订单ID不能为空'];            $order_id[] = $value['order_id'];            if(! isset($value['order_amount']) || $value['order_amount'] < 0) return [false, '订单金额不能为空'];            //promotion_amount 优惠金额            if(empty($value['product'])) return [false,'订单下的商品不能为空'];            foreach ($value['product'] as $p){                if(empty($p['product_name'])) return [false,'商品名称不能为空'];                if(empty($p['code'])) return [false, '商品:' . $p['product_name'] . '的商家编码为空,请进入抖店商家后台补充完整'];                if(empty($p['item_num'])) return [false, '商品:' . $p['product_name'] . '的数量不能为空'];                if(empty($p['sum_amount'])) return [false, '商品:' . $p['sum_amount'] . '的金额不能为空'];                $search .= "(code = '".$p['code']."') or";            }        }        $search = rtrim($search,' or');        $model = Product::ProductClear($user,[]);        $product = $model->whereRaw($search)            ->where('del_time',0)            ->select('title','id','code','cost','retail_price')            ->get()->toArray();        $product_map = [];        foreach ($product as $value){            $product_map[$value['code']] = [                'id' => $value['id'],                'cost' => $value['cost'],                'retail_price' => $value['retail_price'],            ];        }        $sale = SalesOrder::where('del_time',0)            ->whereIn('plat_order',$order_id)            ->pluck('plat_order','id')            ->toArray();        $head = $user['head']['id'] ?? 0;        $prefix = SalesOrder::$prefix[salesOrder::Model_type_four];        $time = time();        $tmp = [            'sales_order_type' => SalesOrder::Order_type_one,            'model_type' => SalesOrder::Model_type_four,            'order_number' => '',            'customer_short_name' => $customer_short_name,            'plat_type' => $plat_type,            'plat_order' => '',            'sign_time' => $time,            'product_total' => 0,            'contract_fee' => 0,            'discount_fee' => 0,            'cdefine29' => '',//分社施工            'cdefine32' => '',//达人昵称            'cdefine30' => '',//业务员            'rate' => 100,            'depart_id' => $head,            'top_depart_id' => $head,            'crt_id' => $user['id'],            'crt_time' => $time,            'upd_time' => $time,        ];        $tmp_detail = [            'sales_order_id' => 0,            'product_id' => 0,            'cost' => 0,            'retail_price' => 0,            'basic_type_id' => 0,            'price' => 0,            'final_amount' => 0,            'number' => '',            'crt_time' => $time,        ];        $insert = $insert_detail = $insert_detail_man = [];        foreach ($data['order_list'] as $value){            if(isset($sale[$value['order_id']])) return [false, '平台单号:' . $value['order_id'] . '在系统中已录入!'];            $tmp['order_number'] = OrderNoService::createSalesOrderNumber($prefix);            $tmp['plat_order'] = $value['order_id'];            $tmp['discount_fee'] = 0;            $tmp['contract_fee'] = $value['order_amount'] / 100;            foreach ($value['product'] as $p){               if(! isset($product_map[$p['code']])) return [false, '产品编码:' . $p['code'] . '在系统中不存在或已被删除'];                $tmp['product_total'] += $p['sum_amount'] / 100;                //产品子表                $product_tmp = $product_map[$p['code']] ?? [];                $tmp_detail['product_id'] = $product_tmp['id'];                $tmp_detail['cost'] = $product_tmp['cost'];                $tmp_detail['retail_price'] = $product_tmp['retail_price'];                $tmp_detail['price'] = $product_tmp['retail_price'];                $tmp_detail['final_amount'] = $p['sum_amount'] / 100;                $tmp_detail['number'] = $p['item_num'];                $insert_detail[$value['order_id']][] = $tmp_detail;            }            //主表            $insert[$value['order_id']] = $tmp;            //人员子表            $insert_detail_man[$value['order_id']] = [                'type' => SalesOrderInfo::type_one,                'data_id' => $user['id'],                'crt_time' => $time,            ];        }        $insert = array_values($insert);        $insert_detail = array_values($insert_detail);        $insert_detail_man = array_values($insert_detail_man);        try{            DB::beginTransaction();            if(! empty($insert)) SalesOrder::insert($insert);            $last_insert_id = SalesOrder::where('crt_time',$time)                ->select('id')                ->get()->toArray();            $last_insert_id = array_column($last_insert_id,'id');            if(! empty($insert_detail)){                $insert2 = [];                foreach ($last_insert_id as $key => $value){                    if(isset($insert_detail[$key])) {                        foreach ($insert_detail[$key] as $val){                            $val['sales_order_id'] = $value;                            $insert2[] = $val;                        }                    }                }                SalesOrderProductInfo::insert($insert2);            }            if(! empty($insert_detail_man)){                $insert3 = [];                foreach ($last_insert_id as $key => $value){                    if(isset($insert_detail_man[$key])) {                        $insert_detail_man[$key]['sales_order_id'] = $value;                        $insert3[] = $insert_detail_man[$key];                    }                }                SalesOrderInfo::insert($insert3);            }            DB::commit();        }catch (\Exception $e){            DB::rollBack();            return [false, $e->getMessage() . $e->getLine() . $e->getCode()];        }        return [true, ''];    }    //------------------导入线上订单    //序列化排序    public function marshal($param){        if($param == null) return "{}";        $arr = $param;        $this->recKSort($arr); // 对关联数组中的kv,执行排序,需要递归        $json = json_encode($arr, JSON_UNESCAPED_SLASHES | JSON_UNESCAPED_UNICODE); // 重新序列化,确保所有key按字典序排序        return $json;    }    // 关联数组排序,递归    public function recKSort(&$arr)    {        $kstring = true;        foreach ($arr as $k => &$v) {            if (!is_string($k)) {                $kstring = false;            }            if (is_array($v)) {                $this->recKSort($v);            }        }        if ($kstring) {            ksort($arr);        }    }    //加密    public function sign($method, $timestamp, $paramJson){        $appKey = $this->appKey;        $appSecret = $this->appSecret;        $paramPattern = 'app_key' . $appKey . 'method' . $method . 'param_json' . $paramJson . 'timestamp' . $timestamp . 'v2';        $signPattern = $appSecret . $paramPattern . $appSecret;        return hash_hmac("sha256", $signPattern, $appSecret);    }    //获取方法    public function getMethod($urlPath) {        if (strlen($urlPath) == 0) {            return $urlPath;        }        $methodPath = "";        if (substr($urlPath, 0, 1) == "/") {            $methodPath = substr($urlPath, 1, strlen($urlPath));        } else {            $methodPath = $urlPath;        }        return str_replace("/", ".", $methodPath);    }    //组织请求参数    public function organization($param,$method,$token = ""){        $paramJson = $this->marshal($param);        //时间戳        $timestamp = time();        //方法        $method_2 = $this->getMethod($method);        //签名        $sign = $this->sign($method_2,$timestamp,$paramJson);        //地址        $url = $this->host . $method . '?app_key='. $this->appKey.'&method='. $method_2 .'¶m_json='. $paramJson .'×tamp='. $timestamp .'&v=2&sign=' . $sign . '&access_token='. $token . '&sign_method=hmac-sha256';        return [$url, $paramJson];    }    //发送请求    public function post_dou($url, $data, $timeout = 20){        $header = array("Content-type:application/json;charset='utf-8'");        Log::channel('apiLog')->info('抖店POST', ["api" => $url , "param" => $data ,"header" => $header]);        $ch = curl_init();        curl_setopt($ch, CURLOPT_URL, $url);        curl_setopt($ch,  CURLOPT_RETURNTRANSFER, true);        curl_setopt($ch, CURLOPT_ENCODING, '');        curl_setopt($ch, CURLOPT_POST, 1);        curl_setopt($ch, CURLOPT_CUSTOMREQUEST, 'POST');        curl_setopt($ch, CURLOPT_HTTPHEADER, $header);        curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);        curl_setopt($ch, CURLOPT_TIMEOUT, $timeout);        if(!is_null($data)) curl_setopt($ch, CURLOPT_POSTFIELDS, $data);        $r = curl_exec($ch);        if ($r === false) {            // 获取错误号            $errorNumber = curl_errno($ch);            // 获取错误信息            $errorMessage = curl_error($ch);            $message = "cURL Error #{$errorNumber}: {$errorMessage}";            Log::channel('apiLog')->info('抖店POST结果', ["message" => $message]);            return [false, "cURL Error #{$errorNumber}: {$errorMessage}"];        }        curl_close($ch);        Log::channel('apiLog')->info('抖店POST结果', ["message" => json_decode($r, true)]);        return [true, json_decode($r, true)];    }}
 |