You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

558 lines
21 KiB

  1. # Copyright 2019-2020 Brainbean Apps (https://brainbeanapps.com)
  2. # License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl.html).
  3. from base64 import b64encode
  4. from datetime import datetime
  5. from dateutil.relativedelta import relativedelta
  6. import dateutil.parser
  7. from decimal import Decimal
  8. import itertools
  9. import json
  10. import pytz
  11. from urllib.error import HTTPError
  12. from urllib.parse import urlencode
  13. import urllib.request
  14. from odoo import models, api, _
  15. from odoo.exceptions import UserError
  16. PAYPAL_API_BASE = 'https://api.paypal.com'
  17. TRANSACTIONS_SCOPE = 'https://uri.paypal.com/services/reporting/search/read'
  18. EVENT_DESCRIPTIONS = {
  19. 'T0000': _('General PayPal-to-PayPal payment'),
  20. 'T0001': _('MassPay payment'),
  21. 'T0002': _('Subscription payment'),
  22. 'T0003': _('Pre-approved payment (BillUser API)'),
  23. 'T0004': _('eBay auction payment'),
  24. 'T0005': _('Direct payment API'),
  25. 'T0006': _('PayPal Checkout APIs'),
  26. 'T0007': _('Website payments standard payment'),
  27. 'T0008': _('Postage payment to carrier'),
  28. 'T0009': _('Gift certificate payment, purchase of gift certificate'),
  29. 'T0010': _('Third-party auction payment'),
  30. 'T0011': _('Mobile payment, made through a mobile phone'),
  31. 'T0012': _('Virtual terminal payment'),
  32. 'T0013': _('Donation payment'),
  33. 'T0014': _('Rebate payments'),
  34. 'T0015': _('Third-party payout'),
  35. 'T0016': _('Third-party recoupment'),
  36. 'T0017': _('Store-to-store transfers'),
  37. 'T0018': _('PayPal Here payment'),
  38. 'T0019': _('Generic instrument-funded payment'),
  39. 'T0100': _('General non-payment fee'),
  40. 'T0101': _('Website payments. Pro account monthly fee'),
  41. 'T0102': _('Foreign bank withdrawal fee'),
  42. 'T0103': _('WorldLink check withdrawal fee'),
  43. 'T0104': _('Mass payment batch fee'),
  44. 'T0105': _('Check withdrawal'),
  45. 'T0106': _('Chargeback processing fee'),
  46. 'T0107': _('Payment fee'),
  47. 'T0108': _('ATM withdrawal'),
  48. 'T0109': _('Auto-sweep from account'),
  49. 'T0110': _('International credit card withdrawal'),
  50. 'T0111': _('Warranty fee for warranty purchase'),
  51. 'T0112': _('Gift certificate expiration fee'),
  52. 'T0113': _('Partner fee'),
  53. 'T0200': _('General currency conversion'),
  54. 'T0201': _('User-initiated currency conversion'),
  55. 'T0202': _('Currency conversion required to cover negative balance'),
  56. 'T0300': _('General funding of PayPal account'),
  57. 'T0301': _('PayPal balance manager funding of PayPal account'),
  58. 'T0302': _('ACH funding for funds recovery from account balance'),
  59. 'T0303': _('Electronic funds transfer (EFT)'),
  60. 'T0400': _('General withdrawal from PayPal account'),
  61. 'T0401': _('AutoSweep'),
  62. 'T0500': _('General PayPal debit card transaction'),
  63. 'T0501': _('Virtual PayPal debit card transaction'),
  64. 'T0502': _('PayPal debit card withdrawal to ATM'),
  65. 'T0503': _('Hidden virtual PayPal debit card transaction'),
  66. 'T0504': _('PayPal debit card cash advance'),
  67. 'T0505': _('PayPal debit authorization'),
  68. 'T0600': _('General credit card withdrawal'),
  69. 'T0700': _('General credit card deposit'),
  70. 'T0701': _('Credit card deposit for negative PayPal account balance'),
  71. 'T0800': _('General bonus'),
  72. 'T0801': _('Debit card cash back bonus'),
  73. 'T0802': _('Merchant referral account bonus'),
  74. 'T0803': _('Balance manager account bonus'),
  75. 'T0804': _('PayPal buyer warranty bonus'),
  76. 'T0805': _(
  77. 'PayPal protection bonus, payout for PayPal buyer protection, payout '
  78. 'for full protection with PayPal buyer credit.'
  79. ),
  80. 'T0806': _('Bonus for first ACH use'),
  81. 'T0807': _('Credit card security charge refund'),
  82. 'T0808': _('Credit card cash back bonus'),
  83. 'T0900': _('General incentive or certificate redemption'),
  84. 'T0901': _('Gift certificate redemption'),
  85. 'T0902': _('Points incentive redemption'),
  86. 'T0903': _('Coupon redemption'),
  87. 'T0904': _('eBay loyalty incentive'),
  88. 'T0905': _('Offers used as funding source'),
  89. 'T1000': _('Bill pay transaction'),
  90. 'T1100': _('General reversal'),
  91. 'T1101': _('Reversal of ACH withdrawal transaction'),
  92. 'T1102': _('Reversal of debit card transaction'),
  93. 'T1103': _('Reversal of points usage'),
  94. 'T1104': _('Reversal of ACH deposit'),
  95. 'T1105': _('Reversal of general account hold'),
  96. 'T1106': _('Payment reversal, initiated by PayPal'),
  97. 'T1107': _('Payment refund, initiated by merchant'),
  98. 'T1108': _('Fee reversal'),
  99. 'T1109': _('Fee refund'),
  100. 'T1110': _('Hold for dispute investigation'),
  101. 'T1111': _('Cancellation of hold for dispute resolution'),
  102. 'T1112': _('MAM reversal'),
  103. 'T1113': _('Non-reference credit payment'),
  104. 'T1114': _('MassPay reversal transaction'),
  105. 'T1115': _('MassPay refund transaction'),
  106. 'T1116': _('Instant payment review (IPR) reversal'),
  107. 'T1117': _('Rebate or cash back reversal'),
  108. 'T1118': _('Generic instrument/Open Wallet reversals (seller side)'),
  109. 'T1119': _('Generic instrument/Open Wallet reversals (buyer side)'),
  110. 'T1200': _('General account adjustment'),
  111. 'T1201': _('Chargeback'),
  112. 'T1202': _('Chargeback reversal'),
  113. 'T1203': _('Charge-off adjustment'),
  114. 'T1204': _('Incentive adjustment'),
  115. 'T1205': _('Reimbursement of chargeback'),
  116. 'T1207': _('Chargeback re-presentment rejection'),
  117. 'T1208': _('Chargeback cancellation'),
  118. 'T1300': _('General authorization'),
  119. 'T1301': _('Reauthorization'),
  120. 'T1302': _('Void of authorization'),
  121. 'T1400': _('General dividend'),
  122. 'T1500': _('General temporary hold'),
  123. 'T1501': _('Account hold for open authorization'),
  124. 'T1502': _('Account hold for ACH deposit'),
  125. 'T1503': _('Temporary hold on available balance'),
  126. 'T1600': _('PayPal buyer credit payment funding'),
  127. 'T1601': _('BML credit, transfer from BML'),
  128. 'T1602': _('Buyer credit payment'),
  129. 'T1603': _('Buyer credit payment withdrawal, transfer to BML'),
  130. 'T1700': _('General withdrawal to non-bank institution'),
  131. 'T1701': _('WorldLink withdrawal'),
  132. 'T1800': _('General buyer credit payment'),
  133. 'T1801': _('BML withdrawal, transfer to BML'),
  134. 'T1900': _('General adjustment without business-related event'),
  135. 'T2000': _('General intra-account transfer'),
  136. 'T2001': _('Settlement consolidation'),
  137. 'T2002': _('Transfer of funds from payable'),
  138. 'T2003': _('Transfer to external GL entity'),
  139. 'T2101': _('General hold'),
  140. 'T2102': _('General hold release'),
  141. 'T2103': _('Reserve hold'),
  142. 'T2104': _('Reserve release'),
  143. 'T2105': _('Payment review hold'),
  144. 'T2106': _('Payment review release'),
  145. 'T2107': _('Payment hold'),
  146. 'T2108': _('Payment hold release'),
  147. 'T2109': _('Gift certificate purchase'),
  148. 'T2110': _('Gift certificate redemption'),
  149. 'T2111': _('Funds not yet available'),
  150. 'T2112': _('Funds available'),
  151. 'T2113': _('Blocked payments'),
  152. 'T2201': _('Transfer to and from a credit-card-funded restricted balance'),
  153. 'T3000': _('Generic instrument/Open Wallet transaction'),
  154. 'T5000': _('Deferred disbursement, funds collected for disbursement'),
  155. 'T5001': _('Delayed disbursement, funds disbursed'),
  156. 'T9700': _('Account receivable for shipping'),
  157. 'T9701': _('Funds payable: PayPal-provided funds that must be paid back'),
  158. 'T9702': _(
  159. 'Funds receivable: PayPal-provided funds that are being paid back'
  160. ),
  161. 'T9800': _('Display only transaction'),
  162. 'T9900': _('Other'),
  163. }
  164. NO_DATA_FOR_DATE_AVAIL_MSG = 'Data for the given start date is not available.'
  165. class OnlineBankStatementProviderPayPal(models.Model):
  166. _inherit = 'online.bank.statement.provider'
  167. @api.model
  168. def _get_available_services(self):
  169. return super()._get_available_services() + [
  170. ('paypal', 'PayPal.com'),
  171. ]
  172. @api.multi
  173. def _obtain_statement_data(self, date_since, date_until):
  174. self.ensure_one()
  175. if self.service != 'paypal':
  176. return super()._obtain_statement_data(
  177. date_since,
  178. date_until,
  179. ) # pragma: no cover
  180. currency = (
  181. self.currency_id or self.company_id.currency_id
  182. ).name
  183. if date_since.tzinfo:
  184. date_since = date_since.astimezone(pytz.utc).replace(tzinfo=None)
  185. if date_until.tzinfo:
  186. date_until = date_until.astimezone(pytz.utc).replace(tzinfo=None)
  187. if date_since < datetime.utcnow() - relativedelta(years=3):
  188. raise UserError(_(
  189. 'PayPal allows retrieving transactions only up to 3 years in '
  190. 'the past. Please import older transactions manually. See '
  191. 'https://www.paypal.com/us/smarthelp/article/why-can\'t-i'
  192. '-access-transaction-history-greater-than-3-years-ts2241'
  193. ))
  194. token = self._paypal_get_token()
  195. transactions = self._paypal_get_transactions(
  196. token,
  197. currency,
  198. date_since,
  199. date_until
  200. )
  201. if not transactions:
  202. balance = self._paypal_get_balance(
  203. token,
  204. currency,
  205. date_since
  206. )
  207. return [], {
  208. 'balance_start': balance,
  209. 'balance_end_real': balance,
  210. }
  211. # Normalize transactions, sort by date, and get lines
  212. transactions = list(sorted(
  213. transactions,
  214. key=lambda transaction: self._paypal_get_transaction_date(
  215. transaction
  216. )
  217. ))
  218. lines = list(itertools.chain.from_iterable(map(
  219. lambda x: self._paypal_transaction_to_lines(x),
  220. transactions
  221. )))
  222. first_transaction = transactions[0]
  223. first_transaction_id = \
  224. first_transaction['transaction_info']['transaction_id']
  225. first_transaction_date = self._paypal_get_transaction_date(
  226. first_transaction
  227. )
  228. first_transaction = self._paypal_get_transaction(
  229. token,
  230. first_transaction_id,
  231. first_transaction_date
  232. )
  233. if not first_transaction:
  234. raise UserError(_('Failed to resolve transaction %s (%s)') % (
  235. first_transaction_id,
  236. first_transaction_date
  237. ))
  238. balance_start = self._paypal_get_transaction_ending_balance(
  239. first_transaction
  240. )
  241. balance_start -= self._paypal_get_transaction_total_amount(
  242. first_transaction
  243. )
  244. balance_start -= self._paypal_get_transaction_fee_amount(
  245. first_transaction
  246. )
  247. last_transaction = transactions[-1]
  248. last_transaction_id = \
  249. last_transaction['transaction_info']['transaction_id']
  250. last_transaction_date = self._paypal_get_transaction_date(
  251. last_transaction
  252. )
  253. last_transaction = self._paypal_get_transaction(
  254. token,
  255. last_transaction_id,
  256. last_transaction_date
  257. )
  258. if not last_transaction:
  259. raise UserError(_('Failed to resolve transaction %s (%s)') % (
  260. last_transaction_id,
  261. last_transaction_date
  262. ))
  263. balance_end = self._paypal_get_transaction_ending_balance(
  264. last_transaction
  265. )
  266. return lines, {
  267. 'balance_start': balance_start,
  268. 'balance_end_real': balance_end,
  269. }
  270. @api.model
  271. def _paypal_preparse_transaction(self, transaction):
  272. date = dateutil.parser.parse(
  273. self._paypal_get_transaction_date(transaction)
  274. ).astimezone(pytz.utc).replace(tzinfo=None)
  275. transaction['transaction_info']['transaction_updated_date'] = date
  276. return transaction
  277. @api.model
  278. def _paypal_transaction_to_lines(self, data):
  279. transaction = data['transaction_info']
  280. payer = data['payer_info']
  281. transaction_id = transaction['transaction_id']
  282. event_code = transaction['transaction_event_code']
  283. date = self._paypal_get_transaction_date(data)
  284. total_amount = self._paypal_get_transaction_total_amount(data)
  285. fee_amount = self._paypal_get_transaction_fee_amount(data)
  286. transaction_subject = transaction.get('transaction_subject')
  287. transaction_note = transaction.get('transaction_note')
  288. invoice = transaction.get('invoice_id')
  289. payer_name = payer.get('payer_name', {})
  290. payer_email = payer_name.get('email_address')
  291. if invoice:
  292. invoice = _('Invoice %s') % invoice
  293. note = transaction_id
  294. if transaction_subject or transaction_note:
  295. note = '%s: %s' % (
  296. note,
  297. transaction_subject or transaction_note
  298. )
  299. if payer_email:
  300. note += ' (%s)' % payer_email
  301. unique_import_id = '%s-%s' % (
  302. transaction_id,
  303. int(date.timestamp()),
  304. )
  305. name = invoice \
  306. or transaction_subject \
  307. or transaction_note \
  308. or EVENT_DESCRIPTIONS.get(event_code) \
  309. or ''
  310. line = {
  311. 'name': name,
  312. 'amount': str(total_amount),
  313. 'date': date,
  314. 'note': note,
  315. 'unique_import_id': unique_import_id,
  316. }
  317. payer_full_name = payer_name.get('full_name') or \
  318. payer_name.get('alternate_full_name')
  319. if payer_full_name:
  320. line.update({
  321. 'partner_name': payer_full_name,
  322. })
  323. lines = [line]
  324. if fee_amount:
  325. lines += [{
  326. 'name': _('Fee for %s') % (name or transaction_id),
  327. 'amount': str(fee_amount),
  328. 'date': date,
  329. 'partner_name': 'PayPal',
  330. 'unique_import_id': '%s-FEE' % unique_import_id,
  331. 'note': _('Transaction fee for %s') % note,
  332. }]
  333. return lines
  334. @api.multi
  335. def _paypal_get_token(self):
  336. self.ensure_one()
  337. data = self._paypal_retrieve(
  338. (self.api_base or PAYPAL_API_BASE) + '/v1/oauth2/token',
  339. (self.username, self.password),
  340. data=urlencode({
  341. 'grant_type': 'client_credentials',
  342. }).encode('utf-8')
  343. )
  344. if 'scope' not in data or TRANSACTIONS_SCOPE not in data['scope']:
  345. raise UserError(_(
  346. 'PayPal App features are configured incorrectly!'
  347. ))
  348. if 'token_type' not in data or data['token_type'] != 'Bearer':
  349. raise UserError(_('Invalid token type!'))
  350. if 'access_token' not in data:
  351. raise UserError(_(
  352. 'Failed to acquire token using Client ID and Secret!'
  353. ))
  354. return data['access_token']
  355. @api.multi
  356. def _paypal_get_balance(self, token, currency, as_of_timestamp):
  357. self.ensure_one()
  358. url = (self.api_base or PAYPAL_API_BASE) \
  359. + '/v1/reporting/balances?currency_code=%s&as_of_time=%s' % (
  360. currency,
  361. as_of_timestamp.isoformat() + 'Z',
  362. )
  363. data = self._paypal_retrieve(url, token)
  364. available_balance = data['balances'][0].get('available_balance')
  365. if not available_balance:
  366. return Decimal()
  367. return Decimal(available_balance['value'])
  368. @api.multi
  369. def _paypal_get_transaction(self, token, transaction_id, timestamp):
  370. self.ensure_one()
  371. transaction_date = timestamp.isoformat() + 'Z'
  372. url = (self.api_base or PAYPAL_API_BASE) \
  373. + '/v1/reporting/transactions' \
  374. + (
  375. '?start_date=%s'
  376. '&end_date=%s'
  377. '&fields=all'
  378. ) % (
  379. transaction_date,
  380. transaction_date,
  381. )
  382. data = self._paypal_retrieve(url, token)
  383. transactions = data['transaction_details']
  384. for transaction in transactions:
  385. if transaction['transaction_info']['transaction_id'] != \
  386. transaction_id:
  387. continue
  388. return transaction
  389. return None
  390. @api.multi
  391. def _paypal_get_transactions(self, token, currency, since, until):
  392. self.ensure_one()
  393. # NOTE: Not more than 31 days in a row
  394. # NOTE: start_date <= date <= end_date, thus check every transaction
  395. interval_step = relativedelta(days=31)
  396. interval_start = since
  397. transactions = []
  398. while interval_start < until:
  399. interval_end = min(interval_start + interval_step, until)
  400. page = 1
  401. total_pages = None
  402. while total_pages is None or page <= total_pages:
  403. url = (self.api_base or PAYPAL_API_BASE) \
  404. + '/v1/reporting/transactions' \
  405. + (
  406. '?transaction_currency=%s'
  407. '&start_date=%s'
  408. '&end_date=%s'
  409. '&fields=all'
  410. '&balance_affecting_records_only=Y'
  411. '&page_size=500'
  412. '&page=%d'
  413. % (
  414. currency,
  415. interval_start.isoformat() + 'Z',
  416. interval_end.isoformat() + 'Z',
  417. page,
  418. ))
  419. # NOTE: Workaround for INVALID_REQUEST (see ROADMAP.rst)
  420. invalid_data_workaround = self.env.context.get(
  421. 'test_account_bank_statement_import_online_paypal_monday',
  422. interval_start.weekday() == 0 and (
  423. datetime.utcnow() - interval_start
  424. ).total_seconds() < 28800
  425. )
  426. data = self.with_context(
  427. invalid_data_workaround=invalid_data_workaround,
  428. )._paypal_retrieve(url, token)
  429. interval_transactions = map(
  430. lambda transaction: self._paypal_preparse_transaction(
  431. transaction
  432. ),
  433. data['transaction_details']
  434. )
  435. transactions += list(filter(
  436. lambda transaction:
  437. interval_start <= self._paypal_get_transaction_date(
  438. transaction
  439. ) < interval_end,
  440. interval_transactions
  441. ))
  442. total_pages = data['total_pages']
  443. page += 1
  444. interval_start += interval_step
  445. return transactions
  446. @api.model
  447. def _paypal_get_transaction_date(self, transaction):
  448. # NOTE: CSV reports from PayPal use this date, search as well
  449. return transaction['transaction_info']['transaction_updated_date']
  450. @api.model
  451. def _paypal_get_transaction_total_amount(self, transaction):
  452. transaction_amount = \
  453. transaction['transaction_info'].get('transaction_amount')
  454. if not transaction_amount:
  455. return Decimal()
  456. return Decimal(transaction_amount['value'])
  457. @api.model
  458. def _paypal_get_transaction_fee_amount(self, transaction):
  459. fee_amount = transaction['transaction_info'].get('fee_amount')
  460. if not fee_amount:
  461. return Decimal()
  462. return Decimal(fee_amount['value'])
  463. @api.model
  464. def _paypal_get_transaction_ending_balance(self, transaction):
  465. # NOTE: 'available_balance' instead of 'ending_balance' as per CSV file
  466. transaction_amount = \
  467. transaction['transaction_info'].get('available_balance')
  468. if not transaction_amount:
  469. return Decimal()
  470. return Decimal(transaction_amount['value'])
  471. @api.model
  472. def _paypal_decode_error(self, content):
  473. if 'name' in content:
  474. return UserError('%s: %s' % (
  475. content['name'],
  476. content.get('message', _('Unknown error')),
  477. ))
  478. if 'error' in content:
  479. return UserError('%s: %s' % (
  480. content['error'],
  481. content.get('error_description', _('Unknown error')),
  482. ))
  483. return None
  484. @api.model
  485. def _paypal_retrieve(self, url, auth, data=None):
  486. try:
  487. with self._paypal_urlopen(url, auth, data) as response:
  488. content = response.read().decode('utf-8')
  489. except HTTPError as e:
  490. content = json.loads(e.read().decode('utf-8'))
  491. # NOTE: Workaround for INVALID_REQUEST (see ROADMAP.rst)
  492. if self.env.context.get('invalid_data_workaround') \
  493. and content.get('name') == 'INVALID_REQUEST' \
  494. and content.get('message') == NO_DATA_FOR_DATE_AVAIL_MSG:
  495. return {
  496. 'transaction_details': [],
  497. 'page': 1,
  498. 'total_items': 0,
  499. 'total_pages': 0,
  500. }
  501. raise self._paypal_decode_error(content) or e
  502. return json.loads(content)
  503. @api.model
  504. def _paypal_urlopen(self, url, auth, data=None):
  505. if not auth:
  506. raise UserError(_('No authentication specified!'))
  507. request = urllib.request.Request(url, data=data)
  508. if isinstance(auth, tuple):
  509. request.add_header(
  510. 'Authorization',
  511. 'Basic %s' % str(
  512. b64encode(('%s:%s' % (auth[0], auth[1])).encode('utf-8')),
  513. 'utf-8'
  514. )
  515. )
  516. elif isinstance(auth, str):
  517. request.add_header(
  518. 'Authorization',
  519. 'Bearer %s' % auth
  520. )
  521. else:
  522. raise UserError(_('Unknown authentication specified!'))
  523. return urllib.request.urlopen(request)