unionpei 1.0.0 → 1.3.1
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/.github/workflows/gem-push.yml +45 -0
- data/.gitignore +17 -0
- data/Gemfile +6 -0
- data/README.md +67 -0
- data/Rakefile +11 -0
- data/certs/acp_test_enc.cer +25 -0
- data/certs/acp_test_middle.cer +23 -0
- data/certs/acp_test_root.cer +22 -0
- data/certs/acp_test_sign.pfx +0 -0
- data/install.sh +2 -0
- data/lib/unionpei/acp_service.rb +37 -42
- data/lib/unionpei/cert_util.rb +61 -82
- data/lib/unionpei/configuration.rb +15 -0
- data/lib/unionpei/log_util.rb +24 -31
- data/lib/unionpei/payment.rb +78 -48
- data/lib/unionpei/sdk_config.rb +46 -48
- data/lib/unionpei/sdk_util.rb +164 -186
- data/lib/unionpei/version.rb +3 -7
- data/lib/unionpei.rb +17 -2
- data/unionpei.gemspec +31 -0
- metadata +87 -4
data/lib/unionpei/log_util.rb
CHANGED
@@ -1,4 +1,4 @@
|
|
1
|
-
#
|
1
|
+
# frozen_string_literal: true
|
2
2
|
|
3
3
|
require 'singleton'
|
4
4
|
require 'logger'
|
@@ -8,75 +8,68 @@ require_relative 'sdk_config'
|
|
8
8
|
|
9
9
|
module UnionPei
|
10
10
|
class LogUtil
|
11
|
-
|
12
11
|
@@logger = nil
|
13
12
|
|
14
13
|
private_class_method :new
|
15
14
|
|
16
|
-
|
17
|
-
|
18
|
-
|
19
|
-
|
20
|
-
|
21
|
-
|
22
|
-
|
23
|
-
else
|
24
|
-
@@logger = Logger.new(SDKConfig.instance.logFilePath)
|
25
|
-
end
|
15
|
+
def self.getLogger
|
16
|
+
unless @@logger
|
17
|
+
@@logger = if SDKConfig.instance.logFilePath.nil?
|
18
|
+
Logger.new($stdout)
|
19
|
+
else
|
20
|
+
Logger.new(SDKConfig.instance.logFilePath)
|
21
|
+
end
|
26
22
|
@@logger.datetime_format = '%Y-%m-%d %H:%M:%S'
|
27
23
|
@@logger.formatter = proc do |severity, datetime, progname, msg|
|
28
24
|
"#{datetime} [#{severity}] #{progname}: #{msg}\n"
|
29
25
|
end
|
30
26
|
@@logger.level = case SDKConfig.instance.logLevel.upcase
|
31
|
-
when 'INFO'
|
27
|
+
when 'INFO'
|
32
28
|
Logger::INFO
|
33
|
-
when 'DEBUG'
|
29
|
+
when 'DEBUG'
|
34
30
|
Logger::DEBUG
|
35
|
-
when 'WARN'
|
31
|
+
when 'WARN'
|
36
32
|
Logger::WARN
|
37
|
-
when 'ERROR'
|
33
|
+
when 'ERROR'
|
38
34
|
Logger::ERROR
|
39
|
-
when 'FATAL'
|
35
|
+
when 'FATAL'
|
40
36
|
Logger::FATAL
|
41
37
|
else
|
42
38
|
Logger::UNKNOWN
|
43
|
-
|
39
|
+
end
|
44
40
|
end
|
45
41
|
p = LogUtil.parse_caller(caller(0)[2])
|
46
|
-
@@logger.progname = p[0]
|
42
|
+
@@logger.progname = "#{p[0]}:#{p[1]}"
|
47
43
|
@@logger
|
48
44
|
end
|
49
45
|
|
50
|
-
def
|
46
|
+
def self.parse_caller(at)
|
51
47
|
if /^(.+?):(\d+)(?::in `(.*)')?/ =~ at
|
52
|
-
file =
|
53
|
-
line =
|
54
|
-
method =
|
48
|
+
file = Regexp.last_match(1)
|
49
|
+
line = Regexp.last_match(2).to_i
|
50
|
+
method = Regexp.last_match(3)
|
55
51
|
[file, line, method]
|
56
52
|
end
|
57
53
|
end
|
58
54
|
|
59
|
-
|
60
|
-
|
61
|
-
def LogUtil.info(msg)
|
55
|
+
def self.info(msg)
|
62
56
|
LogUtil.getLogger.info(msg)
|
63
57
|
end
|
64
58
|
|
65
|
-
def
|
59
|
+
def self.debug(msg)
|
66
60
|
LogUtil.getLogger.debug(msg)
|
67
61
|
end
|
68
62
|
|
69
|
-
def
|
63
|
+
def self.warn(msg)
|
70
64
|
LogUtil.getLogger.warn(msg)
|
71
65
|
end
|
72
66
|
|
73
|
-
def
|
67
|
+
def self.error(msg)
|
74
68
|
LogUtil.getLogger.error(msg)
|
75
69
|
end
|
76
70
|
|
77
|
-
def
|
71
|
+
def self.fatal(msg)
|
78
72
|
LogUtil.getLogger.fatal(msg)
|
79
73
|
end
|
80
|
-
|
81
74
|
end
|
82
75
|
end
|
data/lib/unionpei/payment.rb
CHANGED
@@ -1,67 +1,97 @@
|
|
1
|
-
#
|
1
|
+
# frozen_string_literal: true
|
2
2
|
|
3
|
-
require
|
3
|
+
require 'date'
|
4
4
|
require_relative 'sdk_config'
|
5
5
|
require_relative 'acp_service'
|
6
6
|
module UnionPei
|
7
7
|
class Payment
|
8
8
|
class << self
|
9
|
-
|
10
|
-
|
11
|
-
|
12
|
-
|
13
|
-
|
14
|
-
|
15
|
-
|
16
|
-
|
17
|
-
|
18
|
-
|
19
|
-
|
20
|
-
|
21
|
-
|
22
|
-
|
23
|
-
|
24
|
-
|
25
|
-
|
9
|
+
def default_b2c_req
|
10
|
+
{
|
11
|
+
'version' => UnionPei::SDKConfig.instance.version,
|
12
|
+
'encoding' => UnionPei::SDKConfig.instance.encoding,
|
13
|
+
'signMethod' => UnionPei::SDKConfig.instance.signMethod,
|
14
|
+
'frontUrl' => UnionPei::SDKConfig.instance.frontUrl,
|
15
|
+
'backUrl' => UnionPei::SDKConfig.instance.backUrl,
|
16
|
+
'txnType' => '01',
|
17
|
+
'txnSubType' => '01',
|
18
|
+
'bizType' => '000201', # 000201 是b2c / 000202 是 b2b
|
19
|
+
'channelType' => '07',
|
20
|
+
'currencyCode' => '156',
|
21
|
+
'txnAmt' => '881000',
|
22
|
+
'merId' => '777290058189920',
|
23
|
+
'orderId' => DateTime.parse(Time.now.to_s).strftime('%Y%m%d%H%M%S').to_s,
|
24
|
+
'txnTime' => DateTime.parse(Time.now.to_s).strftime('%Y%m%d%H%M%S').to_s,
|
25
|
+
'accessType' => '0'
|
26
|
+
}
|
27
|
+
end
|
26
28
|
|
27
29
|
# 【默认大于配置】哲学
|
28
|
-
def b2c(req
|
29
|
-
req =
|
30
|
-
UnionPei::AcpService.sign(req)
|
30
|
+
def b2c(req = {})
|
31
|
+
req = default_b2c_req.merge(req)
|
32
|
+
signature = UnionPei::AcpService.sign(req)
|
33
|
+
req['signature'] = signature
|
31
34
|
url = UnionPei::SDKConfig.instance.frontTransUrl
|
32
35
|
UnionPei::AcpService.createAutoFormHtml(req, url)
|
33
36
|
end
|
34
37
|
|
35
|
-
|
36
|
-
|
37
|
-
|
38
|
-
|
39
|
-
|
40
|
-
|
41
|
-
|
42
|
-
|
43
|
-
|
44
|
-
|
45
|
-
|
46
|
-
|
47
|
-
|
48
|
-
|
49
|
-
|
50
|
-
|
51
|
-
|
52
|
-
|
53
|
-
|
54
|
-
|
55
|
-
|
56
|
-
|
38
|
+
def default_b2b_req
|
39
|
+
{
|
40
|
+
'version' => UnionPei::SDKConfig.instance.version,
|
41
|
+
'encoding' => UnionPei::SDKConfig.instance.encoding,
|
42
|
+
'signMethod' => UnionPei::SDKConfig.instance.signMethod,
|
43
|
+
'frontUrl' => UnionPei::SDKConfig.instance.frontUrl,
|
44
|
+
'backUrl' => UnionPei::SDKConfig.instance.backUrl,
|
45
|
+
'txnType' => '01',
|
46
|
+
'txnSubType' => '01',
|
47
|
+
'bizType' => '000202', # 000201 是b2c / 000202 是 b2b,
|
48
|
+
'channelType' => '07',
|
49
|
+
'currencyCode' => '156',
|
50
|
+
'txnAmt' => '881000',
|
51
|
+
'merId' => '777290058189920',
|
52
|
+
'orderId' => DateTime.parse(Time.now.to_s).strftime('%Y%m%d%H%M%S').to_s,
|
53
|
+
'txnTime' => DateTime.parse(Time.now.to_s).strftime('%Y%m%d%H%M%S').to_s,
|
54
|
+
'accessType' => '0',
|
55
|
+
'payTimeout' => DateTime.parse((Time.now + 15 * 60 * 1000).to_s).strftime('%Y%m%d%H%M%S').to_s,
|
56
|
+
'bizScene' => '110001',
|
57
|
+
'payeeAcctNm' => 'xx商户',
|
58
|
+
'payeeAcctNo' => '12345678',
|
59
|
+
'payeeBankName' => 'xx行'
|
60
|
+
}
|
61
|
+
end
|
57
62
|
|
58
|
-
def b2b(req
|
59
|
-
req =
|
60
|
-
UnionPei::AcpService.sign(req)
|
63
|
+
def b2b(req = {})
|
64
|
+
req = default_b2b_req.merge(req)
|
65
|
+
signature = UnionPei::AcpService.sign(req)
|
66
|
+
req['signature'] = signature
|
61
67
|
url = UnionPei::SDKConfig.instance.frontTransUrl
|
62
68
|
UnionPei::AcpService.createAutoFormHtml(req, url)
|
63
69
|
end
|
70
|
+
|
71
|
+
def default_trans_req
|
72
|
+
{
|
73
|
+
'version' => UnionPei::SDKConfig.instance.version,
|
74
|
+
'encoding' => UnionPei::SDKConfig.instance.encoding,
|
75
|
+
'bizType' => '000000',
|
76
|
+
'txnTime' => DateTime.parse(Time.now.to_s).strftime('%Y%m%d%H%M%S').to_s,
|
77
|
+
'txnType' => '00',
|
78
|
+
'txnSubType' => '00',
|
79
|
+
'accessType' => '0',
|
80
|
+
'signMethod' => UnionPei::SDKConfig.instance.signMethod,
|
81
|
+
'merId' => '777290058189920',
|
82
|
+
'orderId' => DateTime.parse(Time.now.to_s).strftime('%Y%m%d%H%M%S').to_s
|
83
|
+
}
|
84
|
+
end
|
85
|
+
|
86
|
+
# query single transaction info
|
87
|
+
# doc https://open.unionpay.com/tjweb/acproduct/APIList?acpAPIId=757&apiservId=448&version=V2.2&bussType=0
|
88
|
+
def query_trans(req = {})
|
89
|
+
req = default_trans_req.merge(req)
|
90
|
+
signature = UnionPei::AcpService.sign(req)
|
91
|
+
req['signature'] = signature
|
92
|
+
url = UnionPei::SDKConfig.instance.singleQueryUrl
|
93
|
+
UnionPei::AcpService.post(req, url)
|
94
|
+
end
|
64
95
|
end
|
65
96
|
end
|
66
97
|
end
|
67
|
-
|
data/lib/unionpei/sdk_config.rb
CHANGED
@@ -1,4 +1,4 @@
|
|
1
|
-
#
|
1
|
+
# frozen_string_literal: true
|
2
2
|
|
3
3
|
require 'iniparse'
|
4
4
|
require 'singleton'
|
@@ -14,63 +14,61 @@ module UnionPei
|
|
14
14
|
:encoding, :secureKey, :logFilePath, :logLevel
|
15
15
|
|
16
16
|
def initialize
|
17
|
-
|
18
|
-
|
19
|
-
|
20
|
-
|
21
|
-
|
22
|
-
@
|
23
|
-
@
|
24
|
-
@
|
25
|
-
@
|
26
|
-
|
27
|
-
@
|
28
|
-
@
|
29
|
-
|
30
|
-
@
|
31
|
-
@
|
32
|
-
|
33
|
-
@
|
34
|
-
@
|
35
|
-
|
36
|
-
|
37
|
-
@
|
38
|
-
@
|
39
|
-
|
40
|
-
@
|
41
|
-
|
42
|
-
@
|
43
|
-
@version = "5.0.0" if @version.nil?
|
44
|
-
|
45
|
-
@ifValidateCNName = ini["acpsdk"]["acpsdk.ifValidateCNName"]
|
17
|
+
ini = parse_acpsdk
|
18
|
+
acpsdk = ini['acpsdk']
|
19
|
+
@frontTransUrl = acpsdk['acpsdk.frontTransUrl']
|
20
|
+
@singleQueryUrl = acpsdk['acpsdk.singleQueryUrl']
|
21
|
+
@backTransUrl = acpsdk['acpsdk.backTransUrl']
|
22
|
+
@batchTransUrl = acpsdk['acpsdk.batchTransUrl']
|
23
|
+
@fileTransUrl = acpsdk['acpsdk.fileTransUrl']
|
24
|
+
@appTransUrl = acpsdk['acpsdk.appTransUrl']
|
25
|
+
@cardTransUrl = acpsdk['acpsdk.cardTransUrl']
|
26
|
+
|
27
|
+
@jfFrontTransUrl = acpsdk['acpsdk.jfFrontTransUrl']
|
28
|
+
@jfSingleQueryUrl = acpsdk['acpsdk.jfSingleQueryUrl']
|
29
|
+
@jfBackTransUrl = acpsdk['acpsdk.jfBackTransUrl']
|
30
|
+
@jfCardTransUrl = acpsdk['acpsdk.jfCardTransUrl']
|
31
|
+
@jfAppTransUrl = acpsdk['acpsdk.jfAppTransUrl']
|
32
|
+
|
33
|
+
@qrcBackTransUrl = acpsdk['acpsdk.qrcBackTransUrl']
|
34
|
+
@qrcB2cIssBackTransUrl = acpsdk['acpsdk.qrcB2cIssBackTransUrl']
|
35
|
+
@qrcB2cMerBackTransUrl = acpsdk['acpsdk.qrcB2cMerBackTransUrl']
|
36
|
+
|
37
|
+
@signMethod = acpsdk['acpsdk.signMethod']
|
38
|
+
@signMethod = @signMethod.to_s unless @signMethod.nil?
|
39
|
+
@version = acpsdk['acpsdk.version']
|
40
|
+
@version = '5.0.0' if @version.nil?
|
41
|
+
|
42
|
+
@ifValidateCNName = acpsdk['acpsdk.ifValidateCNName']
|
46
43
|
@ifValidateCNName = true if @ifValidateCNName.nil?
|
47
|
-
@ifValidateRemoteCert =
|
44
|
+
@ifValidateRemoteCert = acpsdk['acpsdk.ifValidateRemoteCert']
|
48
45
|
@ifValidateRemoteCert = false if @ifValidateRemoteCert.nil?
|
49
46
|
|
50
|
-
@signCertPath =
|
51
|
-
@signCertPwd =
|
52
|
-
@signCertPwd = @signCertPwd.to_s
|
47
|
+
@signCertPath = acpsdk['acpsdk.signCert.path']
|
48
|
+
@signCertPwd = acpsdk['acpsdk.signCert.pwd']
|
49
|
+
@signCertPwd = @signCertPwd.to_s unless @signCertPwd.nil?
|
53
50
|
|
54
|
-
@validateCertDir =
|
55
|
-
@encryptCertPath =
|
56
|
-
@rootCertPath =
|
57
|
-
@middleCertPath =
|
51
|
+
@validateCertDir = acpsdk['acpsdk.validateCert.dir']
|
52
|
+
@encryptCertPath = acpsdk['acpsdk.encryptCert.path']
|
53
|
+
@rootCertPath = acpsdk['acpsdk.rootCert.path']
|
54
|
+
@middleCertPath = acpsdk['acpsdk.middleCert.path']
|
58
55
|
|
59
|
-
@frontUrl =
|
60
|
-
@backUrl =
|
56
|
+
@frontUrl = acpsdk['acpsdk.frontUrl']
|
57
|
+
@backUrl = acpsdk['acpsdk.backUrl']
|
61
58
|
|
62
|
-
@encoding =
|
63
|
-
@secureKey =
|
64
|
-
@secureKey = @secureKey.to_s
|
59
|
+
@encoding = acpsdk['acpsdk.encoding']
|
60
|
+
@secureKey = acpsdk['acpsdk.secureKey']
|
61
|
+
@secureKey = @secureKey.to_s unless @secureKey.nil?
|
65
62
|
|
66
|
-
@logFilePath =
|
67
|
-
@logLevel =
|
63
|
+
@logFilePath = acpsdk['acpsdk.log.file.path']
|
64
|
+
@logLevel = acpsdk['acpsdk.log.level']
|
68
65
|
|
69
66
|
@encoding = 'UTF-8'
|
67
|
+
end
|
70
68
|
|
69
|
+
def parse_acpsdk
|
70
|
+
acp_sdk_config_path = ::UnionPei.configuration.acp_sdk_config_path
|
71
|
+
IniParse.parse(File.read(acp_sdk_config_path).force_encoding('UTF-8'))
|
71
72
|
end
|
72
73
|
end
|
73
74
|
end
|
74
|
-
|
75
|
-
|
76
|
-
|