| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134 | import hashlibimport requestsimport base64from OpenSSL import cryptofrom Ansjer.config import BASE_DIR, SERVER_DOMAIN_SSLimport datetimefrom urllib.parse import parse_qsclass UnionPayObject:    def __init__(self):        self.cert_password = b'000000'        self.cert_path = '{}/Ansjer/file/unionpay/acp_test_sign.pfx'.format(BASE_DIR)        self.x509_filepath = '{}/Ansjer/file/unionpay/verify_sign_acp.cer'.format(BASE_DIR)        self.cert = self.get_cert()        self.pay_url = 'https://gateway.test.cup.com.cn/gateway/api/appTransReq.do'        self.query_url = 'https://gateway.test.95516.com/gateway/api/queryTrans.do'        self.refund_url = 'https://101.231.204.80:5000/gateway/api/backTransReq.do'        self.param = {            "version": '5.1.0',  # 版本            "encoding": "utf-8",  # 编码 可以使用UTF-8,GBK两种方式            "bizType": "000201",  # 产品类型  000201:B2C网关支付            "signMethod": "01",  # 签名方法  01:RSA签名            "accessType": "0",  # 接入类型  0:普通商户直连接入            "merId": '777290058205683',  # 商户代码            "txnTime": datetime.datetime.now().strftime("%Y%m%d%H%M%S"),  # 订单发送时间            "certId": self.cert['certid'],        }    def pay(self, order_id, price):        request_data = {            "txnType": '01',  # 交易类型  01:消费            "txnSubType": "01",  # 交易子类  01:自助消费            "backUrl": 'https://test.zositechc.cn/cloudstorage/doapplenotify',  # 后台通知地址 需外网            "currencyCode": "156",  # 交易币种  156:人民币            # "accType": "01",  # 账号类型 01:银行卡02:存折03:IC卡帐号类型(卡介质)            "txnAmt": price,  # 订单金额(单位: 分)            "orderId": order_id,            "channelType": "08",  # 渠道类型  08:移动        }        self.param.update(request_data)        self.get_sign(self.param)        response = requests.post(self.pay_url, self.param)        data = self.parse_arguments(response.content.decode('utf-8'))        if data['respCode'] != '00':            return False        # self.validate(data)        return data    def get_cert(self):        with open(self.cert_path, "rb") as f:            certs = crypto.load_pkcs12(f.read(), self.cert_password)            x509data = certs.get_certificate()            return {'certid': x509data.get_serial_number(), 'pkey': certs.get_privatekey()}    def get_sign(self, data):        sha256 = hashlib.sha256(self.build_sign_str(data).encode("utf-8")).hexdigest()        private = crypto.sign(self.cert["pkey"], sha256, "sha256")        data["signature"] = str(base64.b64encode(private), encoding="utf-8")    @staticmethod    def build_sign_str(data):        """        排序        :param data:        :return:        """        req = []        for key in sorted(data.keys()):            if data[key] != '':                req.append("%s=%s" % (key, data[key]))        return '&'.join(req)    @staticmethod    def parse_arguments(raw):        """        @raw: raw data to parse argument        """        data = {}        qs_params = parse_qs(str(raw))        for name in qs_params.keys():            data[name] = qs_params.get(name)[-1]        return data    def validate(self, data):        """        @data: a dict ready for validate, must contain "signature" key name        """        signature = base64.b64decode(data.pop('signature'))  # 获取签名        sign_pubkey_cert = data.get("signPubKeyCert", None)        digest = hashlib.sha256(self.build_sign_str(data).encode("utf-8")).hexdigest()        # sign_pubkey_cert = open(self.x509_filepath, 'rb').read()        try:            x509_ert = crypto.load_certificate(crypto.FILETYPE_PEM, sign_pubkey_cert.encode('utf-8'))            crypto.verify(x509_ert, signature, digest, 'sha256')            return True        except Exception as exc:            return False    def query_order(self, order_id):        request_data = {            "txnType": '00',  # 交易类型  00:查询交易            "txnSubType": "00",  # 交易子类  00:查询交易            "orderId": order_id,        }        self.param.update(request_data)        self.get_sign(self.param)        response = requests.post(self.query_url, self.param)        data = self.parse_arguments(response.content.decode('utf-8'))        if data['respCode'] != '00':            return False        return data    def refund(self, order_id, query_id, price):        request_data = {            "txnType": "04",  # 交易类型  00:退货            "txnSubType": "00",  # 交易子类  00:查询交易            "orderId": order_id,            "origQryId": query_id,            "backUrl": "https://test.zositechc.cn/cloudstorage/doapplenotify",            "currencyCode": "156",            "channelType": "08",            "txnAmt": price        }        self.param.update(request_data)        self.get_sign(self.param)        response = requests.post(self.refund_url, self.param)        data = self.parse_arguments(response.content.decode('utf-8'))        if data['respCode'] != '00':            return False        return data
 |