source: main/kofacustom.nigeria/trunk/src/kofacustom/nigeria/remita/helpers.py @ 16277

Last change on this file since 16277 was 15917, checked in by Henrik Bettermann, 5 years ago

Be more verbose.

  • Property svn:keywords set to Id
File size: 6.0 KB
Line 
1## $Id: helpers.py 15917 2020-01-10 12:28:18Z henrik $
2##
3## Copyright (C) 2017 Uli Fouquet & Henrik Bettermann
4## This program is free software; you can redistribute it and/or modify
5## it under the terms of the GNU General Public License as published by
6## the Free Software Foundation; either version 2 of the License, or
7## (at your option) any later version.
8##
9## This program is distributed in the hope that it will be useful,
10## but WITHOUT ANY WARRANTY; without even the implied warranty of
11## MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12## GNU General Public License for more details.
13##
14## You should have received a copy of the GNU General Public License
15## along with this program; if not, write to the Free Software
16## Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
17##
18"""General helper functions for the remita module in custom packages.
19"""
20import grok
21from datetime import datetime
22import httplib
23import hashlib
24import json
25
26from zope.event import notify
27from kofacustom.nigeria.interfaces import MessageFactory as _
28
29def get_JSON_POST_response(
30        merchantId, serviceTypeId, api_key, orderId,
31        amount, responseurl, host, url, https, fullname, email, lineitems):
32    hashargs =  merchantId + serviceTypeId + orderId + str(amount) + responseurl + api_key
33    hashvalue = hashlib.sha512(hashargs).hexdigest()
34    headers={
35        'Content-Type':'application/json; charset=utf-8',
36    }
37    if https:
38        h = httplib.HTTPSConnection(host)
39    else:
40        h = httplib.HTTPConnection(host)
41    data = {
42            "merchantId":merchantId,
43            "serviceTypeId":serviceTypeId,
44            "totalAmount":amount,
45            "hash":hashvalue,
46            "payerName":fullname,
47            "payerEmail":email,
48            "orderId":orderId,
49            "responseurl":responseurl,
50            "lineItems":lineitems
51            }
52    try:
53        h.request("POST", url, body=json.dumps(data), headers=headers)
54        resp = h.getresponse()
55    except:
56        return {'error': 'Socket Error: Connection to Remita gateway refused.'}
57    if resp.status!=200:
58        return {'error': 'Connection Error (%s, %s)' % (resp.status, resp.reason)}
59    jsonout = resp.read()
60    try:
61        parsed_json = json.loads(jsonout[6:-1])
62    except ValueError:
63        return {'error': 'No JSON response'}
64    return parsed_json
65
66def get_payment_status_via_rrr(merchantId, api_key, RRR, host, https):
67    RRR = RRR.rstrip()
68    hashargs =  RRR + api_key + merchantId
69    hashvalue = hashlib.sha512(hashargs).hexdigest()
70    headers={
71        'Content-Type':'application/json; charset=utf-8',
72    }
73    url = '/remita/ecomm/%s/%s/%s/status.reg' % (merchantId, RRR, hashvalue)
74    if https:
75        h = httplib.HTTPSConnection(host)
76    else:
77        h = httplib.HTTPConnection(host)
78    try:
79        h.request("GET", url, headers=headers)
80        resp = h.getresponse()
81    except:
82        return {'error': 'Socket Error: Connection to Remita gateway refused.'}
83    if resp.status!=200:
84        return {'error': 'Connection error (%s, %s)' % (resp.status, resp.reason)}
85    jsonout = resp.read()
86    try:
87        parsed_json = json.loads(jsonout)
88    except ValueError:
89        return {'error': 'No JSON response'}
90    return parsed_json
91
92def query_remita(payment, merchantId, api_key, RRR, host, https, verify):
93
94    jr = get_payment_status_via_rrr(merchantId, api_key, RRR, host, https)
95    error = jr.get('error')
96    if error:
97        msg = log = error
98        return False, msg, log
99
100    # A typical JSON response
101    # {
102    # u'orderId': u'3456346346',
103    # u'status': u'021',
104    # u'amount': 1000.0,
105    # u'transactiontime': u'2017-07-31 11:17:24 AM',
106    # u'message': u'Transaction Pending',
107    # u'lineitems': [{u'status': u'021', u'lineItemsId': u'itemid1'},
108    #                {u'status': u'021', u'lineItemsId': u'itemid2'}
109    #               ],
110    # u'RRR': u'280007640804'}
111
112    payment.r_code = jr['status']
113    payment.r_desc = jr['message']
114    payment.r_amount_approved = jr['amount']
115    try:
116        payment.r_pay_reference = jr['RRR']
117    except KeyError:
118        msg = _('Error message from Remita: ${a}', mapping = {'a': payment.r_desc})
119        log = 'unsuccessful response for %s payment %s: %s' % (
120            payment.p_category, payment.p_id, str(jr))
121        payment.p_state = 'failed'
122        notify(grok.ObjectModifiedEvent(payment))
123        return False, msg, log
124    #payment.r_company = u'remita'
125    if payment.r_code not in ('00', '01'):
126        msg = _('Unsuccessful response: ${a}', mapping = {'a': payment.r_desc})
127        log = 'unsuccessful response for %s payment %s: %s' % (
128            payment.p_category, payment.p_id, payment.r_desc)
129        payment.p_state = 'failed'
130        notify(grok.ObjectModifiedEvent(payment))
131        return False, msg, log
132    if round(payment.r_amount_approved/10.0, 0) != round(
133        payment.amount_auth/10.0, 0):
134        msg = _('Response amount does not match.')
135        log = 'wrong response for %s payment %s: %s' % (
136            payment.p_category, payment.p_id, str(jr))
137        payment.p_state = 'failed'
138        notify(grok.ObjectModifiedEvent(payment))
139        return False, msg, log
140    orderId = jr.get('orderId', None)
141    if orderId and orderId != payment.p_id:
142        msg = _('Response order id does not match.')
143        log = 'wrong response for %s payment %s: %s' % (
144            payment.p_category, payment.p_id, str(jr))
145        payment.p_state = 'failed'
146        notify(grok.ObjectModifiedEvent(payment))
147        return False, msg, log
148    payment.p_state = 'paid'
149    if not verify:
150        payment.payment_date = datetime.utcnow()
151    msg = _('Successful response received')
152    log = 'valid response for %s payment %s: %s' % (
153        payment.p_category, payment.p_id, str(jr))
154    notify(grok.ObjectModifiedEvent(payment))
155    return True, msg, log
156
157def write_payments_log(id, payment):
158    payment.logger.info(
159        '%s,%s,%s,%s,%s,%s,%s,%s,,,' % (
160        id, payment.p_id, payment.p_category,
161        payment.amount_auth, payment.r_code,
162        payment.provider_amt, payment.gateway_amt,
163        payment.thirdparty_amt))
Note: See TracBrowser for help on using the repository browser.