source: main/kofacustom.dspg/trunk/src/kofacustom/dspg/interswitch/browser.py @ 15227

Last change on this file since 15227 was 15227, checked in by Henrik Bettermann, 6 years ago

Change SPAT account.

  • Property svn:keywords set to Id
File size: 20.4 KB
Line 
1## $Id: browser.py 15227 2018-11-09 18:37:38Z henrik $
2##
3## Copyright (C) 2012 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##
18import httplib
19import hashlib
20import grok
21from kofacustom.nigeria.interswitch.browser import (
22    InterswitchPaymentRequestWebservicePageApplicant,
23    InterswitchPaymentRequestWebservicePageStudent,
24    InterswitchPaymentVerifyWebservicePageApplicant,
25    InterswitchPaymentVerifyWebservicePageStudent,
26    InterswitchPageStudent, InterswitchPageApplicant,
27    )
28from kofacustom.dspg.students.interfaces import ICustomStudentOnlinePayment
29from kofacustom.dspg.applicants.interfaces import ICustomApplicantOnlinePayment
30from kofacustom.dspg.interfaces import MessageFactory as _
31
32PRODUCT_ID = '7269' # must be provided by Interswitch
33SITE_NAME = 'dspg.waeup.org'
34PROVIDER_ACCT = '2028964403'
35PROVIDER_BANK_ID = '8'
36PROVIDER_ITEM_NAME = 'BT Education'
37INSTITUTION_NAME = 'Delta State Polytechnic Ogwashi-Uku'
38CURRENCY = '566'
39GATEWAY_AMT = 250.0
40MAC = 'D43ED93B7A307B152C7111D04E1F384428057CD1E75428D3A4C5631C28ACB9C8C822A88CA76DEDC5F9A520CF4BFDB2824FF8B207AAFC667DF3CBB13685B66627'
41
42POST_ACTION = 'https://webpay.interswitchng.com/paydirect/pay'
43#POST_ACTION = 'https://sandbox.interswitchng.com/webpay/pay'
44HOST = 'webpay.interswitchng.com'
45#HOST = 'sandbox.interswitchng.com'
46URL = '/paydirect/api/v1/gettransaction.json'
47#URL = '/webpay/api/v1/gettransaction.json'
48
49httplib.HTTPSConnection.debuglevel = 0
50HTTPS = True
51
52SPECIAL_PAYMENT_PARAMS = {
53
54    'carryover1': ('101', 500.0, '1012808851', '117'),
55    'carryover4': ('101', 500.0, '1012808851', '117'),
56    'carryover5': ('101', 500.0, '1012808851', '117'),
57    'carryover6': ('101', 500.0, '1012808851', '117'),
58    'carryover7': ('101', 500.0, '1012808851', '117'),
59    'carryover8': ('101', 500.0, '1012808851', '117'),
60    'carryover9': ('101', 500.0, '1012808851', '117'),
61    'carryover10': ('101', 500.0, '1012808851', '117'),
62    'carryover11': ('101', 500.0, '1012808851', '117'),
63    'carryover12': ('101', 500.0, '1012808851', '117'),
64    'balance': ('101', 500.0, '1012808851', '117'),
65
66    'certificate': ('111', 500.0, '1012808851', '117'),
67    'state_result': ('112', 500.0, '1012808851', '117'),
68    'transcript_local': ('113', 500.0, '1012808851', '117'),
69    'transcript_foreign': ('114', 500.0, '1012808851', '117'),
70    'ver_result': ('115', 500.0, '1012808851', '117'),
71    'change_course': ('116', 500.0, '1012808851', '117'),
72    'change_inst': ('117', 500.0, '1012808851', '117'),
73    'jamb_reject': ('118', 500.0, '1012808851', '117'),
74    'cert_of_cert': ('119', 500.0, '1012808851', '117'),
75    'ref_let': ('120', 500.0, '1012808851', '117'),
76    'proc_cert': ('121', 500.0, '1012808851', '117'),
77    'loss_idcard': ('122', 9.0, '1012808851', '117'),
78    'loss_examcard': ('123', 4.5, '1012808851', '117'),
79    'loss_result': ('124', 500.0, '1012808851', '117'),
80    'loss_receipt': ('125', 500.0, '1012808851', '117'),
81    'loss_clearance': ('126', 500.0, '1012808851', '117'),
82    'conv_brochure': ('127', 500.0, '1012808851', '117'),
83    'hnd_certificate': ('128', 500.0, '1012808851', '117'),
84    'hnd_state_result': ('129', 500.0, '1012808851', '117'),
85    'hnd_transcript_local': ('130', 500.0, '1012808851', '117'),
86    'hnd_transcript_foreign': ('131', 500.0, '1012808851', '117'),
87    'staff_loss_idcard': ('132', 9.0, '1012808851', '117'),
88    'hnd_conv_brochure': ('133', 500.0, '1012808851', '117'),
89    'pgd_state_result': ('134', 500.0, '1012808851', '117'),
90    'pgd_conv_brochure': ('135', 500.0, '1012808851', '117'),
91    'pgd_certificate': ('136', 500.0, '1012808851', '117'),
92    'log_book': ('137', 4.5, '1012808851', '117'),
93    'pgd_transcript_foreign': ('138', 500.0, '1012808851', '117'),
94    'pgd_transcript_local': ('139', 500.0, '1012808851', '117'),
95    'jamb_regularization': ('140', 500.0, '1012808851', '117'),
96    'utme_registration': ('142', 9.0, '1012808851', '117'),
97    'utme_cbt': ('143', 9.0, '1012808851', '117'),
98    'nysc_id_card': ('144', 9.0, '1012808851', '117'),
99    'ijmb_result': ('147', 500.0, '1012808851', '117'),
100    }
101
102
103# Not yet readily configured. Bank account numbers are
104# contradictory. pay_item_ids are not yet assigned.
105
106class CustomInterswitchPageStudent(InterswitchPageStudent):
107    """ View which sends a POST request to the Interswitch
108    CollegePAY payment gateway.
109    """
110    grok.context(ICustomStudentOnlinePayment)
111    action = POST_ACTION
112    site_name = SITE_NAME
113    currency = CURRENCY
114    product_id = PRODUCT_ID
115    mac = MAC
116    pay_item_id = '000'
117
118    def update(self):
119        error = self.init_update()
120        if error:
121            self.flash(error, type='danger')
122            self.redirect(self.url(self.context, '@@index'))
123            return
124        student = self.student
125        xmldict = self.xmldict
126        # Provider data
127        xmldict['detail_ref'] = self.context.p_id
128        xmldict['provider_acct'] = PROVIDER_ACCT
129        xmldict['provider_bank_id'] = PROVIDER_BANK_ID
130        xmldict['provider_item_name'] = PROVIDER_ITEM_NAME
131        # Institution data
132        xmldict['institution_acct'] = '00000000'
133        xmldict['institution_bank_id'] = '00'
134        xmldict['institution_amt'] = '0.0'
135        provider_amt = 0.0
136        tech_fee = 0.0
137        if self.context.p_category.startswith('schoolfee'):
138            self.pay_item_id = '102'
139            provider_amt = 2800.0
140            tech_fee = 1200.0
141            if student.faccode == 'SPAT':
142                xmldict['institution_acct'] = '1015220292'
143                xmldict['institution_bank_id'] = '117'
144            elif student.current_mode in ('nd_ft', 'hnd_ft'):
145                xmldict['institution_acct'] = '1012808851'
146                xmldict['institution_bank_id'] = '117'
147        elif self.context.p_category == 'clearance':
148            self.pay_item_id = '103'
149            provider_amt = 500.0
150            if student.faccode == 'SPAT':
151                xmldict['institution_acct'] = '1015220292'
152                xmldict['institution_bank_id'] = '117'
153            elif student.current_mode in ('nd_ft', 'hnd_ft'):
154                xmldict['institution_acct'] = '5210039869'
155                xmldict['institution_bank_id'] = '51'
156        elif self.context.p_category == 'gown':
157            self.pay_item_id = '102' # We use school fee item id.
158            xmldict['institution_acct'] = '2006612782'
159            xmldict['institution_bank_id'] = '8'
160        elif self.context.p_category == 'sports':
161            self.pay_item_id = '110'
162            xmldict['institution_acct'] = '6060300688'
163            xmldict['institution_bank_id'] = '51'
164        elif self.context.p_category == 'siwes':
165            self.pay_item_id = '111'
166            xmldict['institution_acct'] = '1001174679'
167            xmldict['institution_bank_id'] = '123'
168        elif self.context.p_category == 'eed_1':
169            self.pay_item_id = '112'
170            xmldict['institution_acct'] = '0039782431'
171            xmldict['institution_bank_id'] = '11'
172        elif self.context.p_category == 'eed_2':
173            self.pay_item_id = '116'
174            xmldict['institution_acct'] = '0039782431'
175            xmldict['institution_bank_id'] = '11'
176        elif self.context.p_category == 'olevel_results_1':
177            self.pay_item_id = '113'
178            xmldict['institution_acct'] = '2015327204'
179            xmldict['institution_bank_id'] = '8'
180        elif self.context.p_category == 'olevel_results_2':
181            self.pay_item_id = '115'
182            xmldict['institution_acct'] = '2015327204'
183            xmldict['institution_bank_id'] = '8'
184        elif self.context.p_category == 'hostel_maintenance':
185            provider_amt = 500.0
186            self.pay_item_id = '117'
187            xmldict['institution_acct'] = '2011810353'
188            xmldict['institution_bank_id'] = '8'
189            xmldict['provider_amt'] = 100 * provider_amt
190            xmldict['institution_amt'] = 100 * (
191                gateway_net_amt(self.context.amount_auth) - provider_amt)
192        elif self.context.p_category in SPECIAL_PAYMENT_PARAMS.keys():
193            self.pay_item_id = SPECIAL_PAYMENT_PARAMS[self.context.p_category][0]
194            provider_amt = SPECIAL_PAYMENT_PARAMS[self.context.p_category][1]
195            xmldict['institution_acct'] = SPECIAL_PAYMENT_PARAMS[self.context.p_category][2]
196            xmldict['institution_bank_id'] = SPECIAL_PAYMENT_PARAMS[self.context.p_category][3]
197        xmldict['provider_amt'] = 100 * provider_amt
198        xmldict['tech_fee'] = 100 * tech_fee
199        xmldict['institution_item_name'] = self.context.category
200        xmldict['institution_name'] = INSTITUTION_NAME
201        xmldict['institution_amt'] = institution_amt = 100 * (
202            self.context.amount_auth - provider_amt - tech_fee - GATEWAY_AMT)
203
204        if provider_amt == 0:
205            xmltext = """<payment_item_detail>
206<item_details detail_ref="%(detail_ref)s" college="%(institution_name)s" department="%(department)s" faculty="%(faculty)s">
207<item_detail item_id="1" item_name="%(institution_item_name)s" item_amt="%(institution_amt)d" bank_id="%(institution_bank_id)s" acct_num="%(institution_acct)s" />
208</item_details>
209</payment_item_detail>""" % xmldict
210        elif tech_fee == 0:
211            xmltext = """<payment_item_detail>
212<item_details detail_ref="%(detail_ref)s" college="%(institution_name)s" department="%(department)s" faculty="%(faculty)s">
213<item_detail item_id="1" item_name="%(institution_item_name)s" item_amt="%(institution_amt)d" bank_id="%(institution_bank_id)s" acct_num="%(institution_acct)s" />
214<item_detail item_id="2" item_name="%(provider_item_name)s" item_amt="%(provider_amt)d" bank_id="%(provider_bank_id)s" acct_num="%(provider_acct)s" />
215</item_details>
216</payment_item_detail>""" % xmldict
217        else:
218            xmltext = """<payment_item_detail>
219<item_details detail_ref="%(detail_ref)s" college="%(institution_name)s" department="%(department)s" faculty="%(faculty)s">
220<item_detail item_id="1" item_name="%(institution_item_name)s" item_amt="%(institution_amt)d" bank_id="%(institution_bank_id)s" acct_num="%(institution_acct)s" />
221<item_detail item_id="2" item_name="%(provider_item_name)s" item_amt="%(provider_amt)d" bank_id="%(provider_bank_id)s" acct_num="%(provider_acct)s" />
222<item_detail item_id="3" item_name="Technology Fee" item_amt="%(tech_fee)d" bank_id="307" acct_num="5100189030" />
223</item_details>
224</payment_item_detail>""" % xmldict
225
226        # isouaba: All certificates in SPAT are part time thus any payments
227        # coming from those certificates (Application, Acceptance, school fees
228        # etc) 40% should be split to keystone account.
229        if student.faccode == 'SPAT':
230            xmldict['institution_amt_2'] = 0.4 * institution_amt
231            xmldict['institution_amt_1'] = 0.6 * institution_amt
232            if provider_amt == 0:
233                xmltext = """<payment_item_detail>
234<item_details detail_ref="%(detail_ref)s" college="%(institution_name)s" department="%(department)s" faculty="%(faculty)s">
235<item_detail item_id="1" item_name="%(institution_item_name)s 1" item_amt="%(institution_amt_1)d" bank_id="%(institution_bank_id)s" acct_num="%(institution_acct)s" />
236<item_detail item_id="2" item_name="%(institution_item_name)s 2" item_amt="%(institution_amt_2)d" bank_id="123" acct_num="1002883141" />
237</item_details>
238</payment_item_detail>""" % xmldict
239            elif tech_fee == 0:
240                xmltext = """<payment_item_detail>
241<item_details detail_ref="%(detail_ref)s" college="%(institution_name)s" department="%(department)s" faculty="%(faculty)s">
242<item_detail item_id="1" item_name="%(institution_item_name)s 1" item_amt="%(institution_amt_1)d" bank_id="%(institution_bank_id)s" acct_num="%(institution_acct)s" />
243<item_detail item_id="2" item_name="%(institution_item_name)s 2" item_amt="%(institution_amt_2)d" bank_id="123" acct_num="1002883141" />
244<item_detail item_id="3" item_name="%(provider_item_name)s" item_amt="%(provider_amt)d" bank_id="%(provider_bank_id)s" acct_num="%(provider_acct)s" />
245</item_details>
246</payment_item_detail>""" % xmldict
247            else:
248                xmltext = """<payment_item_detail>
249<item_details detail_ref="%(detail_ref)s" college="%(institution_name)s" department="%(department)s" faculty="%(faculty)s">
250<item_detail item_id="1" item_name="%(institution_item_name)s 1" item_amt="%(institution_amt_1)d" bank_id="%(institution_bank_id)s" acct_num="%(institution_acct)s" />
251<item_detail item_id="2" item_name="%(institution_item_name)s 2" item_amt="%(institution_amt_2)d" bank_id="123" acct_num="1002883141" />
252<item_detail item_id="3" item_name="%(provider_item_name)s" item_amt="%(provider_amt)d" bank_id="%(provider_bank_id)s" acct_num="%(provider_acct)s" />
253<item_detail item_id="4" item_name="Technology Fee" item_amt="%(tech_fee)d" bank_id="307" acct_num="5100189030" />
254</item_details>
255</payment_item_detail>""" % xmldict
256        # dep_sug payments are treated completely separately. The following lines
257        # override the configuration above.
258        if self.context.p_category == 'dep_sug':
259            self.pay_item_id = '102' # We use school fee item id
260            # amount_auth is being ignored. However, the sum of all split payments
261            # must coincide with the amount_auth configured in students/utils.py
262            xmldict['amt1'] = 100 * 1000
263            xmldict['amt2'] = 100 * 500
264            xmldict['amt3'] = 100 * 900
265            xmldict['amt4'] = 100 * 500
266
267            xmltext = """<payment_item_detail>
268<item_details detail_ref="%(detail_ref)s" college="%(institution_name)s" department="%(department)s" faculty="%(faculty)s">
269<item_detail item_id="1" item_name="SUG" item_amt="%(amt1)d" bank_id="11" acct_num="0038079930" />
270<item_detail item_id="2" item_name="Students Welfare" item_amt="%(amt2)d" bank_id="11" acct_num="0037892949" />
271<item_detail item_id="3" item_name="Anti-Cult Book" item_amt="%(amt3)d" bank_id="11" acct_num="0037892949" />
272<item_detail item_id="4" item_name="NADESSTU" item_amt="%(amt4)d" bank_id="11" acct_num="0036375968" />
273</item_details>
274</payment_item_detail>""" % xmldict
275
276        self.xml_data = """<input type="hidden" name="xml_data" value='%s'  />""" % xmltext
277        self.context.provider_amt = provider_amt
278        self.context.gateway_amt = GATEWAY_AMT
279        hashargs = (
280            self.context.p_id +
281            PRODUCT_ID +
282            self.pay_item_id +
283            str(int(self.amount_auth)) +
284            self.site_redirect_url +
285            self.mac)
286        self.hashvalue = hashlib.sha512(hashargs).hexdigest()
287        return
288
289class CustomInterswitchPageApplicant(InterswitchPageApplicant):
290    """ View which sends a POST request to the Interswitch
291    CollegePAY payment gateway.
292    """
293    grok.context(ICustomApplicantOnlinePayment)
294    action = POST_ACTION
295    site_name = SITE_NAME
296    currency = CURRENCY
297    pay_item_id = '101' # must be provided by Interswitch
298    product_id = PRODUCT_ID
299    mac = MAC
300
301    def update(self):
302        error = self.init_update()
303        if error:
304            self.flash(error, type='danger')
305            self.redirect(self.url(self.context, '@@index'))
306            return
307        xmldict = {}
308        provider_amt = 500.0
309        gateway_amt = GATEWAY_AMT
310        xmldict['institution_acct'] = '00000000'
311        xmldict['institution_bank_id'] = '00'
312        if self.applicant.__parent__.application_category in ('ndft', 'hndft'):
313            xmldict['institution_acct'] = '5210039869'
314            xmldict['institution_bank_id'] = '51'
315        elif self.applicant.__parent__.application_category in (
316            'ndpt', 'hndpt', 'ndwe', 'hndwe'):
317            xmldict['institution_acct'] = '1015220292'
318            xmldict['institution_bank_id'] = '117'
319        elif self.applicant.applicant_id.startswith('special'):
320            if self.context.p_category in SPECIAL_PAYMENT_PARAMS.keys():
321                self.pay_item_id = SPECIAL_PAYMENT_PARAMS[self.context.p_category][0]
322                provider_amt = SPECIAL_PAYMENT_PARAMS[self.context.p_category][1]
323                xmldict['institution_acct'] = SPECIAL_PAYMENT_PARAMS[self.context.p_category][2]
324                xmldict['institution_bank_id'] = SPECIAL_PAYMENT_PARAMS[self.context.p_category][3]
325        if self.pay_item_id in ('122', '123', '132', '137', '141',
326                                '142', '143', '144', '146', '147',):
327                gateway_amt = round(0.015 * self.context.amount_auth, 2)
328        xmldict['detail_ref'] = self.context.p_id
329        xmldict['provider_amt'] = 100 * provider_amt
330        xmldict['provider_acct'] = PROVIDER_ACCT
331        xmldict['provider_bank_id'] = PROVIDER_BANK_ID
332        xmldict['provider_item_name'] = PROVIDER_ITEM_NAME
333        xmldict['institution_amt'] = institution_amt = 100 * (
334            self.context.amount_auth - provider_amt - gateway_amt)
335        xmldict['institution_item_name'] = self.context.category
336        xmldict['institution_name'] = INSTITUTION_NAME
337        xmltext = """<payment_item_detail>
338<item_details detail_ref="%(detail_ref)s" college="%(institution_name)s">
339<item_detail item_id="1" item_name="%(institution_item_name)s" item_amt="%(institution_amt)d" bank_id="%(institution_bank_id)s" acct_num="%(institution_acct)s" />
340<item_detail item_id="2" item_name="%(provider_item_name)s" item_amt="%(provider_amt)d" bank_id="%(provider_bank_id)s" acct_num="%(provider_acct)s" />
341</item_details>
342</payment_item_detail>""" % xmldict
343
344        # isouaba: All certificates in SPAT are part time thus any payments
345        # coming from those certificates (Application, Acceptance, school fees
346        # etc) 40% should be split to keystone account.
347        if self.applicant.__parent__.application_category in (
348            'ndpt', 'hndpt', 'ndwe', 'hndwe'):
349            xmldict['institution_amt_2'] = 0.4 * institution_amt
350            xmldict['institution_amt_1'] = 0.6 * institution_amt
351            xmltext = """<payment_item_detail>
352<item_details detail_ref="%(detail_ref)s" college="%(institution_name)s">
353<item_detail item_id="1" item_name="%(institution_item_name)s 1" item_amt="%(institution_amt_1)d" bank_id="%(institution_bank_id)s" acct_num="%(institution_acct)s" />
354<item_detail item_id="2" item_name="%(institution_item_name)s 2" item_amt="%(institution_amt_2)d" bank_id="123" acct_num="1002883141" />
355<item_detail item_id="3" item_name="%(provider_item_name)s" item_amt="%(provider_amt)d" bank_id="%(provider_bank_id)s" acct_num="%(provider_acct)s" />
356</item_details>
357</payment_item_detail>""" % xmldict
358
359        self.xml_data = """<input type="hidden" name="xml_data" value='%s'  />""" % xmltext
360        self.context.provider_amt = provider_amt
361        self.context.gateway_amt = GATEWAY_AMT
362
363        hashargs = (
364            self.context.p_id +
365            PRODUCT_ID +
366            self.pay_item_id +
367            str(int(self.amount_auth)) +
368            self.site_redirect_url +
369            self.mac)
370        self.hashvalue = hashlib.sha512(hashargs).hexdigest()
371        return
372
373class CustomInterswitchPaymentRequestWebservicePageStudent(
374    InterswitchPaymentRequestWebservicePageStudent):
375    """Request webservice view for the CollegePAY gateway
376    """
377    grok.context(ICustomStudentOnlinePayment)
378    product_id = PRODUCT_ID
379    gateway_host = HOST
380    gateway_url = URL
381    mac = MAC
382
383class CustomInterswitchPaymentVerifyWebservicePageStudent(
384    InterswitchPaymentVerifyWebservicePageStudent):
385    """Payment verify view for the CollegePAY gateway
386    """
387    grok.context(ICustomStudentOnlinePayment)
388    product_id = PRODUCT_ID
389    gateway_host = HOST
390    gateway_url = URL
391    mac = MAC
392
393class CustomInterswitchPaymentRequestWebservicePageApplicant(
394    InterswitchPaymentRequestWebservicePageApplicant):
395    """Request webservice view for the CollegePAY gateway
396    """
397    grok.context(ICustomApplicantOnlinePayment)
398    product_id = PRODUCT_ID
399    gateway_host = HOST
400    gateway_url = URL
401    mac = MAC
402
403class CustomInterswitchPaymentVerifyWebservicePageApplicant(
404    InterswitchPaymentVerifyWebservicePageApplicant):
405    """Payment verify view for the CollegePAY gateway
406    """
407    grok.context(ICustomApplicantOnlinePayment)
408    product_id = PRODUCT_ID
409    gateway_host = HOST
410    gateway_url = URL
411    mac = MAC
Note: See TracBrowser for help on using the repository browser.