Commit 7a475ed376a50b872bb44c73fbb3700e846c7cd3
1 parent
9b569c12be
Exists in
master
typo
Showing 1 changed file with 1 additions and 1 deletions Inline Diff
partner-kospinjasa.js
1 | var winston = require('winston'); | 1 | var winston = require('winston'); |
2 | var soap = require('soap'); | 2 | var soap = require('soap'); |
3 | var crypto = require('crypto'); | 3 | var crypto = require('crypto'); |
4 | var strftime = require('strftime'); | 4 | var strftime = require('strftime'); |
5 | var fs = require("fs"); | 5 | var fs = require("fs"); |
6 | var whiskers = require("whiskers"); | 6 | var whiskers = require("whiskers"); |
7 | var http = require("http"); | 7 | var http = require("http"); |
8 | var url = require("url"); | 8 | var url = require("url"); |
9 | 9 | ||
10 | process.chdir(__dirname); | 10 | process.chdir(__dirname); |
11 | var soapTemplate = fs.readFileSync("message.xml"); | 11 | var soapTemplate = fs.readFileSync("message.xml"); |
12 | 12 | ||
13 | var max_retry = 10; | 13 | var max_retry = 10; |
14 | var sleep_before_retry = 5000; | 14 | var sleep_before_retry = 5000; |
15 | 15 | ||
16 | var config; | 16 | var config; |
17 | var callbackReport; | 17 | var callbackReport; |
18 | var aaa; | 18 | var aaa; |
19 | var logger; | 19 | var logger; |
20 | var options; | 20 | var options; |
21 | 21 | ||
22 | function start(_config, _callbackReport, options) { | 22 | function start(_config, _callbackReport, options) { |
23 | config = _config; | 23 | config = _config; |
24 | callbackReport = _callbackReport | 24 | callbackReport = _callbackReport |
25 | 25 | ||
26 | if (options && options.aaa) { | 26 | if (options && options.aaa) { |
27 | aaa = options.aaa; | 27 | aaa = options.aaa; |
28 | } | 28 | } |
29 | 29 | ||
30 | if (options && options.logger) { | 30 | if (options && options.logger) { |
31 | logger = options.logger; | 31 | logger = options.logger; |
32 | } else { | 32 | } else { |
33 | logger = new winston.Logger({ | 33 | logger = new winston.Logger({ |
34 | transports: [ | 34 | transports: [ |
35 | new (winston.transports.Console)() | 35 | new (winston.transports.Console)() |
36 | ] | 36 | ] |
37 | }); | 37 | }); |
38 | } | 38 | } |
39 | } | 39 | } |
40 | 40 | ||
41 | function topupRequest(task, retry) { | 41 | function topupRequest(task, retry) { |
42 | saldoCheck(billpayment, task); | 42 | saldoCheck(billpayment, task); |
43 | } | 43 | } |
44 | 44 | ||
45 | function saldoCheck(callback, task) { | 45 | function saldoCheck(callback, task) { |
46 | 46 | ||
47 | var params = { | 47 | var params = { |
48 | userName: config.h2h_out.userid, | 48 | userName: config.h2h_out.userid, |
49 | productCode: '00000' , | 49 | productCode: '00000' , |
50 | terminal: 'H2HIPN10', | 50 | terminal: 'H2HIPN10', |
51 | transactionType: '61', | 51 | transactionType: '61', |
52 | reff: Math.ceil( Math.random() * 99999999 ), | 52 | reff: Math.ceil( Math.random() * 99999999 ), |
53 | timeStamp: strftime('%Y-%m-%d %H:%M:%S', new Date()) | 53 | timeStamp: strftime('%Y-%m-%d %H:%M:%S', new Date()) |
54 | } | 54 | } |
55 | 55 | ||
56 | params.signature = createSignatureForSaldoCheck(params, config.h2h_out.password); | 56 | params.signature = createSignatureForSaldoCheck(params, config.h2h_out.password); |
57 | 57 | ||
58 | soap.createClient(config.h2h_out.partner, function(err, soapClient) { | 58 | soap.createClient(config.h2h_out.partner, function(err, soapClient) { |
59 | 59 | ||
60 | if (err) { | 60 | if (err) { |
61 | logger.warn('Error creating soap client for saldoCheck', {err: err}); | 61 | logger.warn('Error creating soap client for saldoCheck', {err: err}); |
62 | callbackReport(task.requestId, '40', 'Error creating soap client for saldoCheck: ' + err); | 62 | callbackReport(task.requestId, '40', 'Error creating soap client for saldoCheck: ' + err); |
63 | return; | 63 | return; |
64 | } | 64 | } |
65 | 65 | ||
66 | logger.info('Requesting to service', {url: config.h2h_out.partner, params: params}); | 66 | logger.info('Requesting to service', {url: config.h2h_out.partner, params: params}); |
67 | soapClient.apih2h.apih2hPort.saldoCheck({ inputSaldo: params }, function(err, result) { | 67 | soapClient.apih2h.apih2hPort.saldoCheck({ inputSaldo: params }, function(err, result) { |
68 | 68 | ||
69 | logger.verbose( | 69 | logger.verbose( |
70 | 'Got saldoCheck response', | 70 | 'Got saldoCheck response', |
71 | { | 71 | { |
72 | lastEndpoint: soapClient.lastEndpoint, | 72 | lastEndpoint: soapClient.lastEndpoint, |
73 | lastRequest: soapClient.lastRequest, | 73 | lastRequest: soapClient.lastRequest, |
74 | lastMessage: soapClient.lastMessage, | 74 | lastMessage: soapClient.lastMessage, |
75 | lastResponse: soapClient.lastResponse, | 75 | lastResponse: soapClient.lastResponse, |
76 | } | 76 | } |
77 | ); | 77 | ); |
78 | 78 | ||
79 | if (err) { | 79 | if (err) { |
80 | logger.warn('Error requesting saldoCheck', {err: err}); | 80 | logger.warn('Error requesting saldoCheck', {err: err}); |
81 | } | 81 | } |
82 | 82 | ||
83 | var balance; | 83 | var balance; |
84 | logger.verbose('saldoCheck result', {result: result}); | 84 | logger.verbose('saldoCheck result', {result: result}); |
85 | 85 | ||
86 | 86 | ||
87 | if (task) { | 87 | if (task) { |
88 | callback(task, balance); | 88 | callback(task, balance); |
89 | } | 89 | } |
90 | }); | 90 | }); |
91 | } | 91 | }); |
92 | 92 | ||
93 | } | 93 | } |
94 | 94 | ||
95 | function billpayment(task, balance) { | 95 | function billpayment(task, balance) { |
96 | 96 | ||
97 | var remoteProduct = task.remoteProduct.split(','); | 97 | var remoteProduct = task.remoteProduct.split(','); |
98 | var params = { | 98 | var params = { |
99 | userName: config.h2h_out.userid, | 99 | userName: config.h2h_out.userid, |
100 | productCode: remoteProduct[0] , | 100 | productCode: remoteProduct[0] , |
101 | terminal: 'H2HIPN10', | 101 | terminal: 'H2HIPN10', |
102 | transactionType: '50', | 102 | transactionType: '50', |
103 | billNumber: createBillNumber(task.destination), | 103 | billNumber: createBillNumber(task.destination), |
104 | amount: remoteProduct[1], | 104 | amount: remoteProduct[1], |
105 | bit61: createBillNumber(task.destination), | 105 | bit61: createBillNumber(task.destination), |
106 | reff: task.requestId, | 106 | reff: task.requestId, |
107 | timeStamp: strftime('%Y-%m-%d %H:%M:%S', new Date()) | 107 | timeStamp: strftime('%Y-%m-%d %H:%M:%S', new Date()) |
108 | } | 108 | } |
109 | 109 | ||
110 | var signature = createSignature(params, config.h2h_out.password); | 110 | var signature = createSignature(params, config.h2h_out.password); |
111 | params.signature = signature; | 111 | params.signature = signature; |
112 | 112 | ||
113 | soap.createClient(config.h2h_out.partner, function(err, soapClient) { | 113 | soap.createClient(config.h2h_out.partner, function(err, soapClient) { |
114 | 114 | ||
115 | var _params = { | 115 | var _params = { |
116 | userName: params.userName, | 116 | userName: params.userName, |
117 | signature: params.signature, | 117 | signature: params.signature, |
118 | productCode: params.productCode, | 118 | productCode: params.productCode, |
119 | terminal: params.terminal, | 119 | terminal: params.terminal, |
120 | transactionType: params.transactionType, | 120 | transactionType: params.transactionType, |
121 | billNumber: params.billNumber, | 121 | billNumber: params.billNumber, |
122 | amount: params.amount, | 122 | amount: params.amount, |
123 | bit61: params.bit61, | 123 | bit61: params.bit61, |
124 | reff: params.reff, | 124 | reff: params.reff, |
125 | timeStamp: params.timeStamp | 125 | timeStamp: params.timeStamp |
126 | } | 126 | } |
127 | 127 | ||
128 | logger.info('Requesting to service', {url: config.h2h_out.partner, params: _params}); | 128 | logger.info('Requesting to service', {url: config.h2h_out.partner, params: _params}); |
129 | 129 | ||
130 | soapClient.apih2h.apih2hPort.billpayment({ inputCheck: _params }, function(err, result) { | 130 | soapClient.apih2h.apih2hPort.billpayment({ inputCheck: _params }, function(err, result) { |
131 | logger.verbose( | 131 | logger.verbose( |
132 | 'Got response', | 132 | 'Got response', |
133 | { | 133 | { |
134 | lastEndpoint: soapClient.lastEndpoint, | 134 | lastEndpoint: soapClient.lastEndpoint, |
135 | lastRequest: soapClient.lastRequest, | 135 | lastRequest: soapClient.lastRequest, |
136 | lastMessage: soapClient.lastMessage, | 136 | lastMessage: soapClient.lastMessage, |
137 | lastResponse: soapClient.lastResponse, | 137 | lastResponse: soapClient.lastResponse, |
138 | } | 138 | } |
139 | ); | 139 | ); |
140 | 140 | ||
141 | if (err) { | 141 | if (err) { |
142 | logger.warn('Error requesting service', {err: err}); | 142 | logger.warn('Error requesting service', {err: err}); |
143 | callbackReport(task.requestId, '68', 'something wrong'); | 143 | callbackReport(task.requestId, '68', 'something wrong'); |
144 | return; | 144 | return; |
145 | } | 145 | } |
146 | 146 | ||
147 | topupResponseHandler(task, result); | 147 | topupResponseHandler(task, result); |
148 | }); | 148 | }); |
149 | }); | 149 | }); |
150 | } | 150 | } |
151 | 151 | ||
152 | function topupResponseHandler(task, response) { | 152 | function topupResponseHandler(task, response) { |
153 | var st24rc = '68'; | 153 | var st24rc = '68'; |
154 | var st24message = response.outputParameter.resultDesc.$value; | 154 | var st24message = response.outputParameter.resultDesc.$value; |
155 | 155 | ||
156 | var resultCode = parseInt(response.outputParameter.resultCode.$value); | 156 | var resultCode = parseInt(response.outputParameter.resultCode.$value); |
157 | var bit39 = parseInt(response.outputParameter.bit39.$value); | 157 | var bit39 = parseInt(response.outputParameter.bit39.$value); |
158 | 158 | ||
159 | if ( resultCode == 1 ) { | 159 | if ( resultCode == 1 ) { |
160 | // product disabled | 160 | // product disabled |
161 | st24rc = '13'; | 161 | st24rc = '13'; |
162 | } | 162 | } |
163 | else if ( resultCode == 2 ) { | 163 | else if ( resultCode == 2 ) { |
164 | // prodcode disable | 164 | // prodcode disable |
165 | st24rc = '13'; | 165 | st24rc = '13'; |
166 | } | 166 | } |
167 | else if ( resultCode == 3 ) { | 167 | else if ( resultCode == 3 ) { |
168 | // duplicate reff | 168 | // duplicate reff |
169 | st24rc = '55'; | 169 | st24rc = '55'; |
170 | } | 170 | } |
171 | else if ( resultCode == 4 ) { | 171 | else if ( resultCode == 4 ) { |
172 | // not enough balance | 172 | // not enough balance |
173 | st24rc = '40'; | 173 | st24rc = '40'; |
174 | } | 174 | } |
175 | else if ( resultCode == 5 ) { | 175 | else if ( resultCode == 5 ) { |
176 | // username blocked | 176 | // username blocked |
177 | st24rc = '40'; | 177 | st24rc = '40'; |
178 | } | 178 | } |
179 | else if ( resultCode == 6 ) { | 179 | else if ( resultCode == 6 ) { |
180 | // not exists username | 180 | // not exists username |
181 | st24rc = '40'; | 181 | st24rc = '40'; |
182 | } | 182 | } |
183 | else if ( resultCode == 11 ) { | 183 | else if ( resultCode == 11 ) { |
184 | // invalid request | 184 | // invalid request |
185 | st24rc = '40' | 185 | st24rc = '40' |
186 | } | 186 | } |
187 | else if ( resultCode == 12 ) { | 187 | else if ( resultCode == 12 ) { |
188 | // no route to host | 188 | // no route to host |
189 | st24rc = '40' | 189 | st24rc = '40' |
190 | } | 190 | } |
191 | else if ( resultCode == 13 ) { | 191 | else if ( resultCode == 13 ) { |
192 | // invalid signature | 192 | // invalid signature |
193 | st24rc = '40' | 193 | st24rc = '40' |
194 | } | 194 | } |
195 | else if ( bit39 == 6 ) { | 195 | else if ( bit39 == 6 ) { |
196 | st24rc = '40'; | 196 | st24rc = '40'; |
197 | st24message = 'Error Transaksi ditolak karena terjadi error di H2H dengan response code diluar daftar ini. Silahkan hubungi H2H'; | 197 | st24message = 'Error Transaksi ditolak karena terjadi error di H2H dengan response code diluar daftar ini. Silahkan hubungi H2H'; |
198 | } | 198 | } |
199 | else if ( bit39 == 12 ) { | 199 | else if ( bit39 == 12 ) { |
200 | st24rc = '40'; | 200 | st24rc = '40'; |
201 | st24message = 'Invalid Transaction Transaksi ditolak karena flow transaksi tidak valid'; | 201 | st24message = 'Invalid Transaction Transaksi ditolak karena flow transaksi tidak valid'; |
202 | } | 202 | } |
203 | else if ( bit39 == 13 ) { | 203 | else if ( bit39 == 13 ) { |
204 | st24rc = '13'; | 204 | st24rc = '13'; |
205 | st24message = 'Invalid voucher nominal'; | 205 | st24message = 'Invalid voucher nominal'; |
206 | } | 206 | } |
207 | else if ( bit39 == 14 ) { | 207 | else if ( bit39 == 14 ) { |
208 | st24rc = '14'; | 208 | st24rc = '14'; |
209 | st24message = 'MSISDN tidak ditemukan'; | 209 | st24message = 'MSISDN tidak ditemukan'; |
210 | } | 210 | } |
211 | else if ( bit39 == 30 ) { | 211 | else if ( bit39 == 30 ) { |
212 | st24rc = '40'; | 212 | st24rc = '40'; |
213 | st24message = 'Format Error'; | 213 | st24message = 'Format Error'; |
214 | } | 214 | } |
215 | else if ( bit39 == 31 ) { | 215 | else if ( bit39 == 31 ) { |
216 | st24rc = '40'; | 216 | st24rc = '40'; |
217 | st24message = 'Kode bank tidak terdaftar'; | 217 | st24message = 'Kode bank tidak terdaftar'; |
218 | } | 218 | } |
219 | else if ( bit39 == 63 ) { | 219 | else if ( bit39 == 63 ) { |
220 | st24rc = '40'; | 220 | st24rc = '40'; |
221 | st24message = 'Reversal denied'; | 221 | st24message = 'Reversal denied'; |
222 | } | 222 | } |
223 | else if ( bit39 == 68 ) { | 223 | else if ( bit39 == 68 ) { |
224 | st24rc = '68'; | 224 | st24rc = '68'; |
225 | st24message = 'Transaksi sedang dalam proses'; | 225 | st24message = 'Transaksi sedang dalam proses'; |
226 | } | 226 | } |
227 | else if ( bit39 == 69 ) { | 227 | else if ( bit39 == 69 ) { |
228 | st24rc = '68'; | 228 | st24rc = '68'; |
229 | st24message = 'Respon Ok lebih dari 24 detik'; | 229 | st24message = 'Respon Ok lebih dari 24 detik'; |
230 | } | 230 | } |
231 | else if ( bit39 == 70 ) { | 231 | else if ( bit39 == 70 ) { |
232 | st24rc = '13'; | 232 | st24rc = '13'; |
233 | st24message = 'Voucher out of stock'; | 233 | st24message = 'Voucher out of stock'; |
234 | } | 234 | } |
235 | else if ( bit39 == 79 ) { | 235 | else if ( bit39 == 79 ) { |
236 | st24rc = '14'; | 236 | st24rc = '14'; |
237 | st24message = 'Phone number is blocked by Telkomsel'; | 237 | st24message = 'Phone number is blocked by Telkomsel'; |
238 | } | 238 | } |
239 | else if ( bit39 == 81 ) { | 239 | else if ( bit39 == 81 ) { |
240 | st24rc = '14'; | 240 | st24rc = '14'; |
241 | st24message = 'Phone number is expired'; | 241 | st24message = 'Phone number is expired'; |
242 | } | 242 | } |
243 | else if ( bit39 == 89 ) { | 243 | else if ( bit39 == 89 ) { |
244 | st24rc = '40'; | 244 | st24rc = '40'; |
245 | st24message = 'Link to billing provider is down'; | 245 | st24message = 'Link to billing provider is down'; |
246 | } | 246 | } |
247 | else if ( bit39 == 91 ) { | 247 | else if ( bit39 == 91 ) { |
248 | st24rc = '40'; | 248 | st24rc = '40'; |
249 | st24message = 'Database problem'; | 249 | st24message = 'Database problem'; |
250 | } | 250 | } |
251 | else if ( bit39 == 92 ) { | 251 | else if ( bit39 == 92 ) { |
252 | st24rc = '40'; | 252 | st24rc = '40'; |
253 | st24message = 'Unable to route transaction'; | 253 | st24message = 'Unable to route transaction'; |
254 | } | 254 | } |
255 | else if ( bit39 == 94 ) { | 255 | else if ( bit39 == 94 ) { |
256 | st24rc = '40'; | 256 | st24rc = '40'; |
257 | st24message = 'Duplicate reversal request'; | 257 | st24message = 'Duplicate reversal request'; |
258 | } | 258 | } |
259 | else if ( resultCode == 0 && bit39 == 0) { | 259 | else if ( resultCode == 0 && bit39 == 0) { |
260 | 260 | ||
261 | var sn = ''; | 261 | var sn = ''; |
262 | 262 | ||
263 | try { | 263 | try { |
264 | sn = response.outputParameter.bit61.$value.substring(43); | 264 | sn = response.outputParameter.bit61.$value.substring(43); |
265 | } | 265 | } |
266 | catch(e) { | 266 | catch(e) { |
267 | sn = ''; | 267 | sn = ''; |
268 | } | 268 | } |
269 | 269 | ||
270 | st24message = 'SN=' + sn + '; ' + st24message; | 270 | st24message = 'SN=' + sn + '; ' + st24message; |
271 | st24rc = '00'; | 271 | st24rc = '00'; |
272 | } | 272 | } |
273 | 273 | ||
274 | var message = | 274 | var message = |
275 | response.outputParameter.resultCode.$value | 275 | response.outputParameter.resultCode.$value |
276 | + " " + response.outputParameter.resultDesc.$value | 276 | + " " + response.outputParameter.resultDesc.$value |
277 | + "; BIT39: " + response.outputParameter.bit39.$value | 277 | + "; BIT39: " + response.outputParameter.bit39.$value |
278 | ; | 278 | ; |
279 | 279 | ||
280 | if (response.outputParameter.resultDesc.$value != st24message) { | 280 | if (response.outputParameter.resultDesc.$value != st24message) { |
281 | var message = message + " " + st24message; | 281 | var message = message + " " + st24message; |
282 | } | 282 | } |
283 | 283 | ||
284 | logger.info('Got result: ' + message, {response: response}); | 284 | logger.info('Got result: ' + message, {response: response}); |
285 | callbackReport(task.requestId, st24rc, st24message); | 285 | callbackReport(task.requestId, st24rc, st24message); |
286 | } | 286 | } |
287 | 287 | ||
288 | function createSignature(params, password) { | 288 | function createSignature(params, password) { |
289 | var passwordHash = crypto.createHash('sha256').update(password).digest().toString('hex'); | 289 | var passwordHash = crypto.createHash('sha256').update(password).digest().toString('hex'); |
290 | var plain = | 290 | var plain = |
291 | params.userName | 291 | params.userName |
292 | + passwordHash | 292 | + passwordHash |
293 | + params.productCode | 293 | + params.productCode |
294 | + params.terminal | 294 | + params.terminal |
295 | + params.transactionType | 295 | + params.transactionType |
296 | + params.billNumber | 296 | + params.billNumber |
297 | + params.amount | 297 | + params.amount |
298 | + params.reff | 298 | + params.reff |
299 | + params.timeStamp; | 299 | + params.timeStamp; |
300 | 300 | ||
301 | return crypto.createHash('sha1').update(plain).digest().toString('hex'); | 301 | return crypto.createHash('sha1').update(plain).digest().toString('hex'); |
302 | } | 302 | } |
303 | 303 | ||
304 | function createSignatureForSaldoCheck(params, password) { | 304 | function createSignatureForSaldoCheck(params, password) { |
305 | var passwordHash = crypto.createHash('sha256').update(password).digest().toString('hex'); | 305 | var passwordHash = crypto.createHash('sha256').update(password).digest().toString('hex'); |
306 | var plain = | 306 | var plain = |
307 | params.userName | 307 | params.userName |
308 | + passwordHash | 308 | + passwordHash |
309 | + params.productCode | 309 | + params.productCode |
310 | + params.terminal | 310 | + params.terminal |
311 | + params.transactionType | 311 | + params.transactionType |
312 | + params.reff | 312 | + params.reff |
313 | + params.timeStamp; | 313 | + params.timeStamp; |
314 | 314 | ||
315 | return crypto.createHash('sha1').update(plain).digest().toString('hex'); | 315 | return crypto.createHash('sha1').update(plain).digest().toString('hex'); |
316 | } | 316 | } |
317 | 317 | ||
318 | function createBillNumber(destination) { | 318 | function createBillNumber(destination) { |
319 | return ("0000000000000" + destination).slice(-13); | 319 | return ("0000000000000" + destination).slice(-13); |
320 | } | 320 | } |
321 | 321 | ||
322 | exports.start = start; | 322 | exports.start = start; |
323 | exports.topupRequest = topupRequest; | 323 | exports.topupRequest = topupRequest; |
324 | exports.createSignature = createSignature; | 324 | exports.createSignature = createSignature; |
325 | exports.createBillNumber = createBillNumber; | 325 | exports.createBillNumber = createBillNumber; |
326 | 326 |