Alexey Pelykh
6 years ago
23 changed files with 1858 additions and 0 deletions
-
114account_bank_statement_import_online/README.rst
-
5account_bank_statement_import_online/__init__.py
-
30account_bank_statement_import_online/__manifest__.py
-
21account_bank_statement_import_online/data/account_bank_statement_import_online.xml
-
4account_bank_statement_import_online/models/__init__.py
-
96account_bank_statement_import_online/models/account_journal.py
-
388account_bank_statement_import_online/models/online_bank_statement_provider.py
-
23account_bank_statement_import_online/readme/CONFIGURE.rst
-
1account_bank_statement_import_online/readme/CONTRIBUTORS.rst
-
1account_bank_statement_import_online/readme/DESCRIPTION.rst
-
9account_bank_statement_import_online/readme/USAGE.rst
-
3account_bank_statement_import_online/security/ir.model.access.csv
-
16account_bank_statement_import_online/security/online_bank_statement_provider.xml
-
460account_bank_statement_import_online/static/description/index.html
-
3account_bank_statement_import_online/tests/__init__.py
-
55account_bank_statement_import_online/tests/online_bank_statement_provider_dummy.py
-
343account_bank_statement_import_online/tests/test_account_bank_statement_import_online.py
-
120account_bank_statement_import_online/views/account_journal.xml
-
96account_bank_statement_import_online/views/online_bank_statement_provider.xml
-
3account_bank_statement_import_online/wizards/__init__.py
-
34account_bank_statement_import_online/wizards/online_bank_statement_pull_wizard.py
-
32account_bank_statement_import_online/wizards/online_bank_statement_pull_wizard.xml
-
1oca_dependencies.txt
@ -0,0 +1,114 @@ |
|||||
|
====================== |
||||
|
Online Bank Statements |
||||
|
====================== |
||||
|
|
||||
|
.. !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! |
||||
|
!! This file is generated by oca-gen-addon-readme !! |
||||
|
!! changes will be overwritten. !! |
||||
|
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! |
||||
|
|
||||
|
.. |badge1| image:: https://img.shields.io/badge/maturity-Beta-yellow.png |
||||
|
:target: https://odoo-community.org/page/development-status |
||||
|
:alt: Beta |
||||
|
.. |badge2| image:: https://img.shields.io/badge/licence-AGPL--3-blue.png |
||||
|
:target: http://www.gnu.org/licenses/agpl-3.0-standalone.html |
||||
|
:alt: License: AGPL-3 |
||||
|
.. |badge3| image:: https://img.shields.io/badge/github-OCA%2Fbank--statement--import-lightgray.png?logo=github |
||||
|
:target: https://github.com/OCA/bank-statement-import/tree/12.0/account_bank_statement_import_online |
||||
|
:alt: OCA/bank-statement-import |
||||
|
.. |badge4| image:: https://img.shields.io/badge/weblate-Translate%20me-F47D42.png |
||||
|
:target: https://translation.odoo-community.org/projects/bank-statement-import-12-0/bank-statement-import-12-0-account_bank_statement_import_online |
||||
|
:alt: Translate me on Weblate |
||||
|
.. |badge5| image:: https://img.shields.io/badge/runbot-Try%20me-875A7B.png |
||||
|
:target: https://runbot.odoo-community.org/runbot/174/12.0 |
||||
|
:alt: Try me on Runbot |
||||
|
|
||||
|
|badge1| |badge2| |badge3| |badge4| |badge5| |
||||
|
|
||||
|
This module provides base for building online bank statements providers. |
||||
|
|
||||
|
**Table of contents** |
||||
|
|
||||
|
.. contents:: |
||||
|
:local: |
||||
|
|
||||
|
Configuration |
||||
|
============= |
||||
|
|
||||
|
To configure online bank statements provider: |
||||
|
|
||||
|
#. Go to *Invoicing > Configuration > Bank Accounts* |
||||
|
#. Open bank account to configure and edit it |
||||
|
#. Set *Bank Feeds* to *Online* |
||||
|
#. Select online bank statements provider in *Online Bank Statements (OCA)* |
||||
|
section |
||||
|
#. Save the bank account |
||||
|
#. Click on provider and configure provider-specific settings. |
||||
|
|
||||
|
or, alternatively: |
||||
|
|
||||
|
#. Go to *Invoicing > Overview* |
||||
|
#. Open settings of the corresponding journal account |
||||
|
#. Switch to *Bank Account* tab |
||||
|
#. Set *Bank Feeds* to *Online* |
||||
|
#. Select online bank statements provider in *Online Bank Statements (OCA)* |
||||
|
section |
||||
|
#. Save the bank account |
||||
|
#. Click on provider and configure provider-specific settings. |
||||
|
|
||||
|
**NOTE**: To access these features, user needs to belong to |
||||
|
*Show Full Accounting Features* group. |
||||
|
|
||||
|
Usage |
||||
|
===== |
||||
|
|
||||
|
To pull historical bank statements: |
||||
|
|
||||
|
#. Go to *Invoicing > Configuration > Bank Accounts* |
||||
|
#. Select specific bank accounts |
||||
|
#. Launch *Actions > Online Bank Statements Pull Wizard* |
||||
|
#. Configure date interval and click *Pull* |
||||
|
|
||||
|
**NOTE**: To access these features, user needs to belong to |
||||
|
*Show Full Accounting Features* group. |
||||
|
|
||||
|
Bug Tracker |
||||
|
=========== |
||||
|
|
||||
|
Bugs are tracked on `GitHub Issues <https://github.com/OCA/bank-statement-import/issues>`_. |
||||
|
In case of trouble, please check there if your issue has already been reported. |
||||
|
If you spotted it first, help us smashing it by providing a detailed and welcomed |
||||
|
`feedback <https://github.com/OCA/bank-statement-import/issues/new?body=module:%20account_bank_statement_import_online%0Aversion:%2012.0%0A%0A**Steps%20to%20reproduce**%0A-%20...%0A%0A**Current%20behavior**%0A%0A**Expected%20behavior**>`_. |
||||
|
|
||||
|
Do not contact contributors directly about support or help with technical issues. |
||||
|
|
||||
|
Credits |
||||
|
======= |
||||
|
|
||||
|
Authors |
||||
|
~~~~~~~ |
||||
|
|
||||
|
* Brainbean Apps |
||||
|
* Dataplug |
||||
|
|
||||
|
Contributors |
||||
|
~~~~~~~~~~~~ |
||||
|
|
||||
|
* Alexey Pelykh <alexey.pelykh@brainbeanapps.com> |
||||
|
|
||||
|
Maintainers |
||||
|
~~~~~~~~~~~ |
||||
|
|
||||
|
This module is maintained by the OCA. |
||||
|
|
||||
|
.. image:: https://odoo-community.org/logo.png |
||||
|
:alt: Odoo Community Association |
||||
|
:target: https://odoo-community.org |
||||
|
|
||||
|
OCA, or the Odoo Community Association, is a nonprofit organization whose |
||||
|
mission is to support the collaborative development of Odoo features and |
||||
|
promote its widespread use. |
||||
|
|
||||
|
This module is part of the `OCA/bank-statement-import <https://github.com/OCA/bank-statement-import/tree/12.0/account_bank_statement_import_online>`_ project on GitHub. |
||||
|
|
||||
|
You are welcome to contribute. To learn how please visit https://odoo-community.org/page/Contribute. |
@ -0,0 +1,5 @@ |
|||||
|
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). |
||||
|
|
||||
|
from . import models |
||||
|
from . import wizards |
||||
|
from .tests import online_bank_statement_provider_dummy |
@ -0,0 +1,30 @@ |
|||||
|
# Copyright 2019 Brainbean Apps (https://brainbeanapps.com) |
||||
|
# Copyright 2019 Dataplug (https://dataplug.io) |
||||
|
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). |
||||
|
|
||||
|
{ |
||||
|
'name': 'Online Bank Statements', |
||||
|
'version': '12.0.1.0.0', |
||||
|
'author': |
||||
|
'Brainbean Apps, ' |
||||
|
'Dataplug, ' |
||||
|
'Odoo Community Association (OCA)', |
||||
|
'website': 'https://github.com/OCA/bank-statement-import/', |
||||
|
'license': 'AGPL-3', |
||||
|
'category': 'Accounting', |
||||
|
'summary': 'Online bank statements update', |
||||
|
'depends': [ |
||||
|
'account', |
||||
|
'account_bank_statement_import', |
||||
|
'web_widget_dropdown_dynamic', |
||||
|
], |
||||
|
'data': [ |
||||
|
'data/account_bank_statement_import_online.xml', |
||||
|
'security/ir.model.access.csv', |
||||
|
'security/online_bank_statement_provider.xml', |
||||
|
'views/account_journal.xml', |
||||
|
'views/online_bank_statement_provider.xml', |
||||
|
'wizards/online_bank_statement_pull_wizard.xml', |
||||
|
], |
||||
|
'installable': True, |
||||
|
} |
@ -0,0 +1,21 @@ |
|||||
|
<?xml version="1.0" ?> |
||||
|
<!-- |
||||
|
Copyright 2019 Brainbean Apps (https://brainbeanapps.com) |
||||
|
Copyright 2019 Dataplug (https://dataplug.io) |
||||
|
License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). |
||||
|
--> |
||||
|
<odoo noupdate="1"> |
||||
|
|
||||
|
<record model="ir.cron" id="ir_cron_account_pull_online_bank_statements"> |
||||
|
<field name="name">Pull Online Bank Statements</field> |
||||
|
<field name="interval_number">1</field> |
||||
|
<field name="interval_type">hours</field> |
||||
|
<field name="numbercall">-1</field> |
||||
|
<field name="state">code</field> |
||||
|
<field name="nextcall">2019-01-01 00:10:00</field> |
||||
|
<field name="doall" eval="False"/> |
||||
|
<field name="model_id" ref="account_bank_statement_import_online.model_online_bank_statement_provider"/> |
||||
|
<field name="code">model._scheduled_pull()</field> |
||||
|
</record> |
||||
|
|
||||
|
</odoo> |
@ -0,0 +1,4 @@ |
|||||
|
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). |
||||
|
|
||||
|
from . import account_journal |
||||
|
from . import online_bank_statement_provider |
@ -0,0 +1,96 @@ |
|||||
|
# Copyright 2019 Brainbean Apps (https://brainbeanapps.com) |
||||
|
# Copyright 2019 Dataplug (https://dataplug.io) |
||||
|
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl.html). |
||||
|
|
||||
|
import logging |
||||
|
|
||||
|
from odoo import models, fields, api, _ |
||||
|
|
||||
|
_logger = logging.getLogger(__name__) |
||||
|
|
||||
|
|
||||
|
class AccountJournal(models.Model): |
||||
|
_inherit = 'account.journal' |
||||
|
|
||||
|
online_bank_statement_provider = fields.Selection( |
||||
|
selection=lambda self: self.env[ |
||||
|
'account.journal' |
||||
|
]._selection_online_bank_statement_provider(), |
||||
|
) |
||||
|
online_bank_statement_provider_id = fields.Many2one( |
||||
|
string='Statement Provider', |
||||
|
comodel_name='online.bank.statement.provider', |
||||
|
ondelete='restrict', |
||||
|
copy=False, |
||||
|
) |
||||
|
|
||||
|
def __get_bank_statements_available_sources(self): |
||||
|
result = super().__get_bank_statements_available_sources() |
||||
|
result.append(('online', _('Online (OCA)'))) |
||||
|
return result |
||||
|
|
||||
|
@api.model |
||||
|
def _selection_online_bank_statement_provider(self): |
||||
|
return self.env[ |
||||
|
'online.bank.statement.provider' |
||||
|
]._get_available_services() + [('dummy', 'Dummy')] |
||||
|
|
||||
|
@api.model |
||||
|
def values_online_bank_statement_provider(self): |
||||
|
return self.env[ |
||||
|
'online.bank.statement.provider' |
||||
|
]._get_available_services() |
||||
|
|
||||
|
@api.multi |
||||
|
def _update_online_bank_statement_provider_id(self): |
||||
|
OnlineBankStatementProvider = ( |
||||
|
self.env['online.bank.statement.provider'] |
||||
|
) |
||||
|
for journal in self.filtered('id'): |
||||
|
provider_id = journal.online_bank_statement_provider_id |
||||
|
if journal.bank_statements_source != 'online': |
||||
|
journal.online_bank_statement_provider_id = False |
||||
|
if provider_id: |
||||
|
provider_id.unlink() |
||||
|
continue |
||||
|
if provider_id.service == journal.online_bank_statement_provider: |
||||
|
continue |
||||
|
journal.online_bank_statement_provider_id = False |
||||
|
if provider_id: |
||||
|
provider_id.unlink() |
||||
|
journal.online_bank_statement_provider_id = ( |
||||
|
OnlineBankStatementProvider.create({ |
||||
|
'journal_id': journal.id, |
||||
|
'service': journal.online_bank_statement_provider, |
||||
|
}) |
||||
|
) |
||||
|
|
||||
|
@api.model |
||||
|
def create(self, vals): |
||||
|
rec = super().create(vals) |
||||
|
if 'bank_statements_source' in vals \ |
||||
|
or 'online_bank_statement_provider' in vals: |
||||
|
rec._update_online_bank_statement_provider_id() |
||||
|
return rec |
||||
|
|
||||
|
@api.multi |
||||
|
def write(self, vals): |
||||
|
res = super().write(vals) |
||||
|
if 'bank_statements_source' in vals \ |
||||
|
or 'online_bank_statement_provider' in vals: |
||||
|
self._update_online_bank_statement_provider_id() |
||||
|
return res |
||||
|
|
||||
|
@api.multi |
||||
|
def action_online_bank_statements_pull_wizard(self): |
||||
|
provider_ids = self.mapped('online_bank_statement_provider_id').ids |
||||
|
return { |
||||
|
'name': _('Online Bank Statement Pull Wizard'), |
||||
|
'type': 'ir.actions.act_window', |
||||
|
'res_model': 'online.bank.statement.pull.wizard', |
||||
|
'views': [[False, 'form']], |
||||
|
'target': 'new', |
||||
|
'context': { |
||||
|
'default_provider_ids': [(6, False, provider_ids)], |
||||
|
}, |
||||
|
} |
@ -0,0 +1,388 @@ |
|||||
|
# Copyright 2019 Brainbean Apps (https://brainbeanapps.com) |
||||
|
# Copyright 2019 Dataplug (https://dataplug.io) |
||||
|
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl.html). |
||||
|
|
||||
|
from dateutil.relativedelta import relativedelta, MO |
||||
|
from decimal import Decimal |
||||
|
import logging |
||||
|
|
||||
|
from odoo import models, fields, api, _ |
||||
|
from odoo.addons.base.models.res_bank import sanitize_account_number |
||||
|
|
||||
|
_logger = logging.getLogger(__name__) |
||||
|
|
||||
|
|
||||
|
class OnlineBankStatementProvider(models.Model): |
||||
|
_name = 'online.bank.statement.provider' |
||||
|
_inherit = ['mail.thread'] |
||||
|
_description = 'Online Bank Statement Provider' |
||||
|
|
||||
|
company_id = fields.Many2one( |
||||
|
related='journal_id.company_id', |
||||
|
store=True, |
||||
|
) |
||||
|
active = fields.Boolean() |
||||
|
name = fields.Char( |
||||
|
string='Name', |
||||
|
compute='_compute_name', |
||||
|
store=True, |
||||
|
) |
||||
|
journal_id = fields.Many2one( |
||||
|
comodel_name='account.journal', |
||||
|
required=True, |
||||
|
readonly=True, |
||||
|
ondelete='cascade', |
||||
|
domain=[ |
||||
|
('type', '=', 'bank'), |
||||
|
], |
||||
|
) |
||||
|
currency_id = fields.Many2one( |
||||
|
related='journal_id.currency_id', |
||||
|
) |
||||
|
account_number = fields.Char( |
||||
|
related='journal_id.bank_account_id.acc_number' |
||||
|
) |
||||
|
service = fields.Selection( |
||||
|
selection=lambda self: self._selection_service(), |
||||
|
required=True, |
||||
|
readonly=True, |
||||
|
) |
||||
|
interval_type = fields.Selection( |
||||
|
selection=[ |
||||
|
('minutes', 'Minute(s)'), |
||||
|
('hours', 'Hour(s)'), |
||||
|
('days', 'Day(s)'), |
||||
|
('weeks', 'Week(s)'), |
||||
|
], |
||||
|
default='hours', |
||||
|
required=True, |
||||
|
) |
||||
|
interval_number = fields.Integer( |
||||
|
string='Scheduled update interval', |
||||
|
default=1, |
||||
|
required=True, |
||||
|
) |
||||
|
update_schedule = fields.Char( |
||||
|
string='Update Schedule', |
||||
|
compute='_compute_update_schedule', |
||||
|
) |
||||
|
last_successful_run = fields.Datetime( |
||||
|
string='Last successful pull', |
||||
|
) |
||||
|
next_run = fields.Datetime( |
||||
|
string='Next scheduled pull', |
||||
|
default=fields.Datetime.now, |
||||
|
required=True, |
||||
|
) |
||||
|
statement_creation_mode = fields.Selection( |
||||
|
selection=[ |
||||
|
('daily', 'Daily statements'), |
||||
|
('weekly', 'Weekly statements'), |
||||
|
('monthly', 'Monthly statements'), |
||||
|
], |
||||
|
default='daily', |
||||
|
required=True, |
||||
|
) |
||||
|
api_base = fields.Char() |
||||
|
origin = fields.Char() |
||||
|
username = fields.Char() |
||||
|
password = fields.Char() |
||||
|
key = fields.Binary() |
||||
|
certificate = fields.Binary() |
||||
|
passphrase = fields.Char() |
||||
|
certificate_public_key = fields.Text() |
||||
|
certificate_private_key = fields.Text() |
||||
|
certificate_chain = fields.Text() |
||||
|
|
||||
|
_sql_constraints = [ |
||||
|
( |
||||
|
'journal_id_uniq', |
||||
|
'UNIQUE(journal_id)', |
||||
|
'Only one online banking statement provider per journal!' |
||||
|
), |
||||
|
( |
||||
|
'valid_interval_number', |
||||
|
'CHECK(interval_number > 0)', |
||||
|
'Scheduled update interval must be greater than zero!' |
||||
|
) |
||||
|
] |
||||
|
|
||||
|
@api.model |
||||
|
def _get_available_services(self): |
||||
|
"""Hook for extension""" |
||||
|
return [] |
||||
|
|
||||
|
@api.model |
||||
|
def _selection_service(self): |
||||
|
return self._get_available_services() + [('dummy', 'Dummy')] |
||||
|
|
||||
|
@api.model |
||||
|
def values_service(self): |
||||
|
return self._get_available_services() |
||||
|
|
||||
|
@api.multi |
||||
|
@api.depends('service') |
||||
|
def _compute_name(self): |
||||
|
for provider in self: |
||||
|
provider.name = list(filter( |
||||
|
lambda x: x[0] == provider.service, |
||||
|
self._selection_service() |
||||
|
))[0][1] |
||||
|
|
||||
|
@api.multi |
||||
|
@api.depends('active', 'interval_type', 'interval_number') |
||||
|
def _compute_update_schedule(self): |
||||
|
for provider in self: |
||||
|
if not provider.active: |
||||
|
provider.update_schedule = _('Inactive') |
||||
|
continue |
||||
|
|
||||
|
provider.update_schedule = _('%(number)s %(type)s') % { |
||||
|
'number': provider.interval_number, |
||||
|
'type': list(filter( |
||||
|
lambda x: x[0] == provider.interval_type, |
||||
|
self._fields['interval_type'].selection |
||||
|
))[0][1], |
||||
|
} |
||||
|
|
||||
|
@api.multi |
||||
|
def _pull(self, date_since, date_until): |
||||
|
AccountBankStatement = self.env['account.bank.statement'] |
||||
|
is_scheduled = self.env.context.get('scheduled') |
||||
|
if is_scheduled: |
||||
|
AccountBankStatement = AccountBankStatement.with_context( |
||||
|
tracking_disable=True, |
||||
|
) |
||||
|
AccountBankStatementLine = self.env['account.bank.statement.line'] |
||||
|
for provider in self: |
||||
|
statement_date_since = provider._get_statement_date_since( |
||||
|
date_since |
||||
|
) |
||||
|
while statement_date_since < date_until: |
||||
|
statement_date_until = ( |
||||
|
statement_date_since + provider._get_statement_date_step() |
||||
|
) |
||||
|
try: |
||||
|
data = provider._obtain_statement_data( |
||||
|
statement_date_since, |
||||
|
statement_date_until |
||||
|
) |
||||
|
except Exception as e: |
||||
|
if is_scheduled: |
||||
|
_logger.warning( |
||||
|
'Online Bank Statement Provider "%s" failed to' |
||||
|
' obtain statement data since %s until %s' % ( |
||||
|
provider.name, |
||||
|
statement_date_since, |
||||
|
statement_date_until, |
||||
|
), |
||||
|
exc_info=True, |
||||
|
) |
||||
|
provider.message_post( |
||||
|
body=_( |
||||
|
'Online Bank Statement Provider "%s" failed to' |
||||
|
' obtain statement data since %s until %s:\n%s' |
||||
|
) % ( |
||||
|
provider.name, |
||||
|
statement_date_since, |
||||
|
statement_date_until, |
||||
|
str(e), |
||||
|
), |
||||
|
subject=_( |
||||
|
'Online Bank Statement Provider failure' |
||||
|
), |
||||
|
) |
||||
|
break |
||||
|
raise |
||||
|
statement_date = provider._get_statement_date( |
||||
|
statement_date_since, |
||||
|
statement_date_until, |
||||
|
) |
||||
|
if not data: |
||||
|
statement_date_since = statement_date_until |
||||
|
continue |
||||
|
lines_data, statement_values = data |
||||
|
statement = AccountBankStatement.search([ |
||||
|
('journal_id', '=', provider.journal_id.id), |
||||
|
('state', '=', 'open'), |
||||
|
('date', '=', statement_date), |
||||
|
], limit=1) |
||||
|
if not statement: |
||||
|
statement_values.update({ |
||||
|
'name': provider.journal_id.sequence_id.with_context( |
||||
|
ir_sequence_date=statement_date, |
||||
|
).next_by_id(), |
||||
|
'journal_id': provider.journal_id.id, |
||||
|
'date': statement_date, |
||||
|
}) |
||||
|
statement = AccountBankStatement.create( |
||||
|
# NOTE: This is needed since create() alters values |
||||
|
statement_values.copy() |
||||
|
) |
||||
|
filtered_lines = [] |
||||
|
for line_values in lines_data: |
||||
|
date = fields.Datetime.from_string(line_values['date']) |
||||
|
if date < statement_date_since: |
||||
|
if 'balance_start' in statement_values: |
||||
|
statement_values['balance_start'] = ( |
||||
|
Decimal( |
||||
|
statement_values['balance_start'] |
||||
|
) + Decimal( |
||||
|
line_values['amount'] |
||||
|
) |
||||
|
) |
||||
|
continue |
||||
|
elif date >= statement_date_until: |
||||
|
if 'balance_end_real' in statement_values: |
||||
|
statement_values['balance_end_real'] = ( |
||||
|
Decimal( |
||||
|
statement_values['balance_end_real'] |
||||
|
) - Decimal( |
||||
|
line_values['amount'] |
||||
|
) |
||||
|
) |
||||
|
continue |
||||
|
elif date <= date_since or date > date_until: |
||||
|
continue |
||||
|
unique_import_id = line_values.get('unique_import_id') |
||||
|
if unique_import_id: |
||||
|
unique_import_id = provider._generate_unique_import_id( |
||||
|
unique_import_id |
||||
|
) |
||||
|
line_values.update({ |
||||
|
'unique_import_id': unique_import_id, |
||||
|
}) |
||||
|
if AccountBankStatementLine.sudo().search( |
||||
|
[('unique_import_id', '=', unique_import_id)], |
||||
|
limit=1): |
||||
|
continue |
||||
|
filtered_lines.append(line_values) |
||||
|
statement_values.update({ |
||||
|
'line_ids': [[0, False, line] for line in filtered_lines], |
||||
|
}) |
||||
|
if 'balance_start' in statement_values: |
||||
|
statement_values['balance_start'] = float( |
||||
|
statement_values['balance_start'] |
||||
|
) |
||||
|
if 'balance_start' in statement_values: |
||||
|
statement_values['balance_start'] = float( |
||||
|
statement_values['balance_start'] |
||||
|
) |
||||
|
statement.write(statement_values) |
||||
|
statement_date_since = statement_date_until |
||||
|
if is_scheduled: |
||||
|
provider._schedule_next_run() |
||||
|
|
||||
|
@api.multi |
||||
|
def _schedule_next_run(self): |
||||
|
self.ensure_one() |
||||
|
self.last_successful_run = self.next_run |
||||
|
self.next_run += self._get_next_run_period() |
||||
|
|
||||
|
@api.multi |
||||
|
def _get_statement_date_since(self, date): |
||||
|
self.ensure_one() |
||||
|
date = date.replace( |
||||
|
hour=0, |
||||
|
minute=0, |
||||
|
second=0, |
||||
|
microsecond=0, |
||||
|
) |
||||
|
if self.statement_creation_mode == 'daily': |
||||
|
return date |
||||
|
elif self.statement_creation_mode == 'weekly': |
||||
|
return date + relativedelta(weekday=MO(-1)) |
||||
|
elif self.statement_creation_mode == 'monthly': |
||||
|
return date.replace( |
||||
|
day=1, |
||||
|
) |
||||
|
|
||||
|
@api.multi |
||||
|
def _get_statement_date_step(self): |
||||
|
self.ensure_one() |
||||
|
if self.statement_creation_mode == 'daily': |
||||
|
return relativedelta( |
||||
|
days=1, |
||||
|
hour=0, |
||||
|
minute=0, |
||||
|
second=0, |
||||
|
microsecond=0, |
||||
|
) |
||||
|
elif self.statement_creation_mode == 'weekly': |
||||
|
return relativedelta( |
||||
|
weeks=1, |
||||
|
weekday=MO, |
||||
|
hour=0, |
||||
|
minute=0, |
||||
|
second=0, |
||||
|
microsecond=0, |
||||
|
) |
||||
|
elif self.statement_creation_mode == 'monthly': |
||||
|
return relativedelta( |
||||
|
months=1, |
||||
|
day=1, |
||||
|
hour=0, |
||||
|
minute=0, |
||||
|
second=0, |
||||
|
microsecond=0, |
||||
|
) |
||||
|
|
||||
|
@api.multi |
||||
|
def _get_statement_date(self, date_since, date_until): |
||||
|
self.ensure_one() |
||||
|
# NOTE: Statement date is treated by Odoo as start of period. Details |
||||
|
# - addons/account/models/account_journal_dashboard.py |
||||
|
# - def get_line_graph_datas() |
||||
|
return date_since.date() |
||||
|
|
||||
|
@api.multi |
||||
|
def _generate_unique_import_id(self, unique_import_id): |
||||
|
self.ensure_one() |
||||
|
sanitized_account_number = sanitize_account_number(self.account_number) |
||||
|
return ( |
||||
|
sanitized_account_number and sanitized_account_number + '-' or '' |
||||
|
) + str(self.journal_id.id) + '-' + unique_import_id |
||||
|
|
||||
|
@api.multi |
||||
|
def _get_next_run_period(self): |
||||
|
self.ensure_one() |
||||
|
if self.interval_type == 'minutes': |
||||
|
return relativedelta(minutes=self.interval_number) |
||||
|
elif self.interval_type == 'hours': |
||||
|
return relativedelta(hours=self.interval_number) |
||||
|
elif self.interval_type == 'days': |
||||
|
return relativedelta(days=self.interval_number) |
||||
|
elif self.interval_type == 'weeks': |
||||
|
return relativedelta(weeks=self.interval_number) |
||||
|
|
||||
|
@api.model |
||||
|
def _scheduled_pull(self): |
||||
|
_logger.info('Scheduled pull of online bank statements...') |
||||
|
|
||||
|
providers = self.search([ |
||||
|
('active', '=', True), |
||||
|
('next_run', '<=', fields.Datetime.now()), |
||||
|
]) |
||||
|
if providers: |
||||
|
_logger.info('Pulling online bank statements of: %s' % ', '.join( |
||||
|
providers.mapped('journal_id.name') |
||||
|
)) |
||||
|
for provider in providers.with_context({'scheduled': True}): |
||||
|
date_since = ( |
||||
|
provider.last_successful_run |
||||
|
) if provider.last_successful_run else ( |
||||
|
provider.next_run - provider._get_next_run_period() |
||||
|
) |
||||
|
date_until = provider.next_run |
||||
|
provider._pull(date_since, date_until) |
||||
|
|
||||
|
_logger.info('Scheduled pull of online bank statements complete.') |
||||
|
|
||||
|
@api.multi |
||||
|
def _obtain_statement_data( |
||||
|
self, date_since, date_until |
||||
|
): |
||||
|
"""Hook for extension""" |
||||
|
# Check tests/online_bank_statement_provider_dummy.py for reference |
||||
|
self.ensure_one() |
||||
|
return [] |
@ -0,0 +1,23 @@ |
|||||
|
To configure online bank statements provider: |
||||
|
|
||||
|
#. Go to *Invoicing > Configuration > Bank Accounts* |
||||
|
#. Open bank account to configure and edit it |
||||
|
#. Set *Bank Feeds* to *Online* |
||||
|
#. Select online bank statements provider in *Online Bank Statements (OCA)* |
||||
|
section |
||||
|
#. Save the bank account |
||||
|
#. Click on provider and configure provider-specific settings. |
||||
|
|
||||
|
or, alternatively: |
||||
|
|
||||
|
#. Go to *Invoicing > Overview* |
||||
|
#. Open settings of the corresponding journal account |
||||
|
#. Switch to *Bank Account* tab |
||||
|
#. Set *Bank Feeds* to *Online* |
||||
|
#. Select online bank statements provider in *Online Bank Statements (OCA)* |
||||
|
section |
||||
|
#. Save the bank account |
||||
|
#. Click on provider and configure provider-specific settings. |
||||
|
|
||||
|
**NOTE**: To access these features, user needs to belong to |
||||
|
*Show Full Accounting Features* group. |
@ -0,0 +1 @@ |
|||||
|
* Alexey Pelykh <alexey.pelykh@brainbeanapps.com> |
@ -0,0 +1 @@ |
|||||
|
This module provides base for building online bank statements providers. |
@ -0,0 +1,9 @@ |
|||||
|
To pull historical bank statements: |
||||
|
|
||||
|
#. Go to *Invoicing > Configuration > Bank Accounts* |
||||
|
#. Select specific bank accounts |
||||
|
#. Launch *Actions > Online Bank Statements Pull Wizard* |
||||
|
#. Configure date interval and click *Pull* |
||||
|
|
||||
|
**NOTE**: To access these features, user needs to belong to |
||||
|
*Show Full Accounting Features* group. |
@ -0,0 +1,3 @@ |
|||||
|
id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink |
||||
|
access_online_bank_statement_provider_admin,online.bank.statement.provider:base.group_system,model_online_bank_statement_provider,base.group_system,1,1,1,1 |
||||
|
access_online_bank_statement_provider_user,online.bank.statement.provider:account.group_account_user,model_online_bank_statement_provider,account.group_account_user,1,1,1,1 |
@ -0,0 +1,16 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||
|
<!-- |
||||
|
License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). |
||||
|
Copyright 2019 Brainbean Apps (https://brainbeanapps.com) |
||||
|
Copyright 2019 Dataplug (https://dataplug.io) |
||||
|
--> |
||||
|
<odoo noupdate="1"> |
||||
|
|
||||
|
<record id="online_bank_statement_provider_multicompany" model="ir.rule"> |
||||
|
<field name="name">Online Bank Statement Provider multi-company</field> |
||||
|
<field name="model_id" ref="model_online_bank_statement_provider"/> |
||||
|
<field eval="True" name="global"/> |
||||
|
<field name="domain_force">['|', ('company_id', '=', False), ('company_id', 'child_of', [user.company_id.id])]</field> |
||||
|
</record> |
||||
|
|
||||
|
</odoo> |
@ -0,0 +1,460 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8" ?> |
||||
|
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> |
||||
|
<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en"> |
||||
|
<head> |
||||
|
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> |
||||
|
<meta name="generator" content="Docutils 0.14: http://docutils.sourceforge.net/" /> |
||||
|
<title>Online Bank Statements</title> |
||||
|
<style type="text/css"> |
||||
|
|
||||
|
/* |
||||
|
:Author: David Goodger (goodger@python.org) |
||||
|
:Id: $Id: html4css1.css 7952 2016-07-26 18:15:59Z milde $ |
||||
|
:Copyright: This stylesheet has been placed in the public domain. |
||||
|
|
||||
|
Default cascading style sheet for the HTML output of Docutils. |
||||
|
|
||||
|
See http://docutils.sf.net/docs/howto/html-stylesheets.html for how to |
||||
|
customize this style sheet. |
||||
|
*/ |
||||
|
|
||||
|
/* used to remove borders from tables and images */ |
||||
|
.borderless, table.borderless td, table.borderless th { |
||||
|
border: 0 } |
||||
|
|
||||
|
table.borderless td, table.borderless th { |
||||
|
/* Override padding for "table.docutils td" with "! important". |
||||
|
The right padding separates the table cells. */ |
||||
|
padding: 0 0.5em 0 0 ! important } |
||||
|
|
||||
|
.first { |
||||
|
/* Override more specific margin styles with "! important". */ |
||||
|
margin-top: 0 ! important } |
||||
|
|
||||
|
.last, .with-subtitle { |
||||
|
margin-bottom: 0 ! important } |
||||
|
|
||||
|
.hidden { |
||||
|
display: none } |
||||
|
|
||||
|
.subscript { |
||||
|
vertical-align: sub; |
||||
|
font-size: smaller } |
||||
|
|
||||
|
.superscript { |
||||
|
vertical-align: super; |
||||
|
font-size: smaller } |
||||
|
|
||||
|
a.toc-backref { |
||||
|
text-decoration: none ; |
||||
|
color: black } |
||||
|
|
||||
|
blockquote.epigraph { |
||||
|
margin: 2em 5em ; } |
||||
|
|
||||
|
dl.docutils dd { |
||||
|
margin-bottom: 0.5em } |
||||
|
|
||||
|
object[type="image/svg+xml"], object[type="application/x-shockwave-flash"] { |
||||
|
overflow: hidden; |
||||
|
} |
||||
|
|
||||
|
/* Uncomment (and remove this text!) to get bold-faced definition list terms |
||||
|
dl.docutils dt { |
||||
|
font-weight: bold } |
||||
|
*/ |
||||
|
|
||||
|
div.abstract { |
||||
|
margin: 2em 5em } |
||||
|
|
||||
|
div.abstract p.topic-title { |
||||
|
font-weight: bold ; |
||||
|
text-align: center } |
||||
|
|
||||
|
div.admonition, div.attention, div.caution, div.danger, div.error, |
||||
|
div.hint, div.important, div.note, div.tip, div.warning { |
||||
|
margin: 2em ; |
||||
|
border: medium outset ; |
||||
|
padding: 1em } |
||||
|
|
||||
|
div.admonition p.admonition-title, div.hint p.admonition-title, |
||||
|
div.important p.admonition-title, div.note p.admonition-title, |
||||
|
div.tip p.admonition-title { |
||||
|
font-weight: bold ; |
||||
|
font-family: sans-serif } |
||||
|
|
||||
|
div.attention p.admonition-title, div.caution p.admonition-title, |
||||
|
div.danger p.admonition-title, div.error p.admonition-title, |
||||
|
div.warning p.admonition-title, .code .error { |
||||
|
color: red ; |
||||
|
font-weight: bold ; |
||||
|
font-family: sans-serif } |
||||
|
|
||||
|
/* Uncomment (and remove this text!) to get reduced vertical space in |
||||
|
compound paragraphs. |
||||
|
div.compound .compound-first, div.compound .compound-middle { |
||||
|
margin-bottom: 0.5em } |
||||
|
|
||||
|
div.compound .compound-last, div.compound .compound-middle { |
||||
|
margin-top: 0.5em } |
||||
|
*/ |
||||
|
|
||||
|
div.dedication { |
||||
|
margin: 2em 5em ; |
||||
|
text-align: center ; |
||||
|
font-style: italic } |
||||
|
|
||||
|
div.dedication p.topic-title { |
||||
|
font-weight: bold ; |
||||
|
font-style: normal } |
||||
|
|
||||
|
div.figure { |
||||
|
margin-left: 2em ; |
||||
|
margin-right: 2em } |
||||
|
|
||||
|
div.footer, div.header { |
||||
|
clear: both; |
||||
|
font-size: smaller } |
||||
|
|
||||
|
div.line-block { |
||||
|
display: block ; |
||||
|
margin-top: 1em ; |
||||
|
margin-bottom: 1em } |
||||
|
|
||||
|
div.line-block div.line-block { |
||||
|
margin-top: 0 ; |
||||
|
margin-bottom: 0 ; |
||||
|
margin-left: 1.5em } |
||||
|
|
||||
|
div.sidebar { |
||||
|
margin: 0 0 0.5em 1em ; |
||||
|
border: medium outset ; |
||||
|
padding: 1em ; |
||||
|
background-color: #ffffee ; |
||||
|
width: 40% ; |
||||
|
float: right ; |
||||
|
clear: right } |
||||
|
|
||||
|
div.sidebar p.rubric { |
||||
|
font-family: sans-serif ; |
||||
|
font-size: medium } |
||||
|
|
||||
|
div.system-messages { |
||||
|
margin: 5em } |
||||
|
|
||||
|
div.system-messages h1 { |
||||
|
color: red } |
||||
|
|
||||
|
div.system-message { |
||||
|
border: medium outset ; |
||||
|
padding: 1em } |
||||
|
|
||||
|
div.system-message p.system-message-title { |
||||
|
color: red ; |
||||
|
font-weight: bold } |
||||
|
|
||||
|
div.topic { |
||||
|
margin: 2em } |
||||
|
|
||||
|
h1.section-subtitle, h2.section-subtitle, h3.section-subtitle, |
||||
|
h4.section-subtitle, h5.section-subtitle, h6.section-subtitle { |
||||
|
margin-top: 0.4em } |
||||
|
|
||||
|
h1.title { |
||||
|
text-align: center } |
||||
|
|
||||
|
h2.subtitle { |
||||
|
text-align: center } |
||||
|
|
||||
|
hr.docutils { |
||||
|
width: 75% } |
||||
|
|
||||
|
img.align-left, .figure.align-left, object.align-left, table.align-left { |
||||
|
clear: left ; |
||||
|
float: left ; |
||||
|
margin-right: 1em } |
||||
|
|
||||
|
img.align-right, .figure.align-right, object.align-right, table.align-right { |
||||
|
clear: right ; |
||||
|
float: right ; |
||||
|
margin-left: 1em } |
||||
|
|
||||
|
img.align-center, .figure.align-center, object.align-center { |
||||
|
display: block; |
||||
|
margin-left: auto; |
||||
|
margin-right: auto; |
||||
|
} |
||||
|
|
||||
|
table.align-center { |
||||
|
margin-left: auto; |
||||
|
margin-right: auto; |
||||
|
} |
||||
|
|
||||
|
.align-left { |
||||
|
text-align: left } |
||||
|
|
||||
|
.align-center { |
||||
|
clear: both ; |
||||
|
text-align: center } |
||||
|
|
||||
|
.align-right { |
||||
|
text-align: right } |
||||
|
|
||||
|
/* reset inner alignment in figures */ |
||||
|
div.align-right { |
||||
|
text-align: inherit } |
||||
|
|
||||
|
/* div.align-center * { */ |
||||
|
/* text-align: left } */ |
||||
|
|
||||
|
.align-top { |
||||
|
vertical-align: top } |
||||
|
|
||||
|
.align-middle { |
||||
|
vertical-align: middle } |
||||
|
|
||||
|
.align-bottom { |
||||
|
vertical-align: bottom } |
||||
|
|
||||
|
ol.simple, ul.simple { |
||||
|
margin-bottom: 1em } |
||||
|
|
||||
|
ol.arabic { |
||||
|
list-style: decimal } |
||||
|
|
||||
|
ol.loweralpha { |
||||
|
list-style: lower-alpha } |
||||
|
|
||||
|
ol.upperalpha { |
||||
|
list-style: upper-alpha } |
||||
|
|
||||
|
ol.lowerroman { |
||||
|
list-style: lower-roman } |
||||
|
|
||||
|
ol.upperroman { |
||||
|
list-style: upper-roman } |
||||
|
|
||||
|
p.attribution { |
||||
|
text-align: right ; |
||||
|
margin-left: 50% } |
||||
|
|
||||
|
p.caption { |
||||
|
font-style: italic } |
||||
|
|
||||
|
p.credits { |
||||
|
font-style: italic ; |
||||
|
font-size: smaller } |
||||
|
|
||||
|
p.label { |
||||
|
white-space: nowrap } |
||||
|
|
||||
|
p.rubric { |
||||
|
font-weight: bold ; |
||||
|
font-size: larger ; |
||||
|
color: maroon ; |
||||
|
text-align: center } |
||||
|
|
||||
|
p.sidebar-title { |
||||
|
font-family: sans-serif ; |
||||
|
font-weight: bold ; |
||||
|
font-size: larger } |
||||
|
|
||||
|
p.sidebar-subtitle { |
||||
|
font-family: sans-serif ; |
||||
|
font-weight: bold } |
||||
|
|
||||
|
p.topic-title { |
||||
|
font-weight: bold } |
||||
|
|
||||
|
pre.address { |
||||
|
margin-bottom: 0 ; |
||||
|
margin-top: 0 ; |
||||
|
font: inherit } |
||||
|
|
||||
|
pre.literal-block, pre.doctest-block, pre.math, pre.code { |
||||
|
margin-left: 2em ; |
||||
|
margin-right: 2em } |
||||
|
|
||||
|
pre.code .ln { color: grey; } /* line numbers */ |
||||
|
pre.code, code { background-color: #eeeeee } |
||||
|
pre.code .comment, code .comment { color: #5C6576 } |
||||
|
pre.code .keyword, code .keyword { color: #3B0D06; font-weight: bold } |
||||
|
pre.code .literal.string, code .literal.string { color: #0C5404 } |
||||
|
pre.code .name.builtin, code .name.builtin { color: #352B84 } |
||||
|
pre.code .deleted, code .deleted { background-color: #DEB0A1} |
||||
|
pre.code .inserted, code .inserted { background-color: #A3D289} |
||||
|
|
||||
|
span.classifier { |
||||
|
font-family: sans-serif ; |
||||
|
font-style: oblique } |
||||
|
|
||||
|
span.classifier-delimiter { |
||||
|
font-family: sans-serif ; |
||||
|
font-weight: bold } |
||||
|
|
||||
|
span.interpreted { |
||||
|
font-family: sans-serif } |
||||
|
|
||||
|
span.option { |
||||
|
white-space: nowrap } |
||||
|
|
||||
|
span.pre { |
||||
|
white-space: pre } |
||||
|
|
||||
|
span.problematic { |
||||
|
color: red } |
||||
|
|
||||
|
span.section-subtitle { |
||||
|
/* font-size relative to parent (h1..h6 element) */ |
||||
|
font-size: 80% } |
||||
|
|
||||
|
table.citation { |
||||
|
border-left: solid 1px gray; |
||||
|
margin-left: 1px } |
||||
|
|
||||
|
table.docinfo { |
||||
|
margin: 2em 4em } |
||||
|
|
||||
|
table.docutils { |
||||
|
margin-top: 0.5em ; |
||||
|
margin-bottom: 0.5em } |
||||
|
|
||||
|
table.footnote { |
||||
|
border-left: solid 1px black; |
||||
|
margin-left: 1px } |
||||
|
|
||||
|
table.docutils td, table.docutils th, |
||||
|
table.docinfo td, table.docinfo th { |
||||
|
padding-left: 0.5em ; |
||||
|
padding-right: 0.5em ; |
||||
|
vertical-align: top } |
||||
|
|
||||
|
table.docutils th.field-name, table.docinfo th.docinfo-name { |
||||
|
font-weight: bold ; |
||||
|
text-align: left ; |
||||
|
white-space: nowrap ; |
||||
|
padding-left: 0 } |
||||
|
|
||||
|
/* "booktabs" style (no vertical lines) */ |
||||
|
table.docutils.booktabs { |
||||
|
border: 0px; |
||||
|
border-top: 2px solid; |
||||
|
border-bottom: 2px solid; |
||||
|
border-collapse: collapse; |
||||
|
} |
||||
|
table.docutils.booktabs * { |
||||
|
border: 0px; |
||||
|
} |
||||
|
table.docutils.booktabs th { |
||||
|
border-bottom: thin solid; |
||||
|
text-align: left; |
||||
|
} |
||||
|
|
||||
|
h1 tt.docutils, h2 tt.docutils, h3 tt.docutils, |
||||
|
h4 tt.docutils, h5 tt.docutils, h6 tt.docutils { |
||||
|
font-size: 100% } |
||||
|
|
||||
|
ul.auto-toc { |
||||
|
list-style-type: none } |
||||
|
|
||||
|
</style> |
||||
|
</head> |
||||
|
<body> |
||||
|
<div class="document" id="online-bank-statements"> |
||||
|
<h1 class="title">Online Bank Statements</h1> |
||||
|
|
||||
|
<!-- !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! |
||||
|
!! This file is generated by oca-gen-addon-readme !! |
||||
|
!! changes will be overwritten. !! |
||||
|
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! --> |
||||
|
<p><a class="reference external" href="https://odoo-community.org/page/development-status"><img alt="Beta" src="https://img.shields.io/badge/maturity-Beta-yellow.png" /></a> <a class="reference external" href="http://www.gnu.org/licenses/agpl-3.0-standalone.html"><img alt="License: AGPL-3" src="https://img.shields.io/badge/licence-AGPL--3-blue.png" /></a> <a class="reference external" href="https://github.com/OCA/bank-statement-import/tree/12.0/account_bank_statement_import_online"><img alt="OCA/bank-statement-import" src="https://img.shields.io/badge/github-OCA%2Fbank--statement--import-lightgray.png?logo=github" /></a> <a class="reference external" href="https://translation.odoo-community.org/projects/bank-statement-import-12-0/bank-statement-import-12-0-account_bank_statement_import_online"><img alt="Translate me on Weblate" src="https://img.shields.io/badge/weblate-Translate%20me-F47D42.png" /></a> <a class="reference external" href="https://runbot.odoo-community.org/runbot/174/12.0"><img alt="Try me on Runbot" src="https://img.shields.io/badge/runbot-Try%20me-875A7B.png" /></a></p> |
||||
|
<p>This module provides base for building online bank statements providers.</p> |
||||
|
<p><strong>Table of contents</strong></p> |
||||
|
<div class="contents local topic" id="contents"> |
||||
|
<ul class="simple"> |
||||
|
<li><a class="reference internal" href="#configuration" id="id1">Configuration</a></li> |
||||
|
<li><a class="reference internal" href="#usage" id="id2">Usage</a></li> |
||||
|
<li><a class="reference internal" href="#bug-tracker" id="id3">Bug Tracker</a></li> |
||||
|
<li><a class="reference internal" href="#credits" id="id4">Credits</a><ul> |
||||
|
<li><a class="reference internal" href="#authors" id="id5">Authors</a></li> |
||||
|
<li><a class="reference internal" href="#contributors" id="id6">Contributors</a></li> |
||||
|
<li><a class="reference internal" href="#maintainers" id="id7">Maintainers</a></li> |
||||
|
</ul> |
||||
|
</li> |
||||
|
</ul> |
||||
|
</div> |
||||
|
<div class="section" id="configuration"> |
||||
|
<h1><a class="toc-backref" href="#id1">Configuration</a></h1> |
||||
|
<p>To configure online bank statements provider:</p> |
||||
|
<ol class="arabic simple"> |
||||
|
<li>Go to <em>Invoicing > Configuration > Bank Accounts</em></li> |
||||
|
<li>Open bank account to configure and edit it</li> |
||||
|
<li>Set <em>Bank Feeds</em> to <em>Online</em></li> |
||||
|
<li>Select online bank statements provider in <em>Online Bank Statements (OCA)</em> |
||||
|
section</li> |
||||
|
<li>Save the bank account</li> |
||||
|
<li>Click on provider and configure provider-specific settings.</li> |
||||
|
</ol> |
||||
|
<p>or, alternatively:</p> |
||||
|
<ol class="arabic simple"> |
||||
|
<li>Go to <em>Invoicing > Overview</em></li> |
||||
|
<li>Open settings of the corresponding journal account</li> |
||||
|
<li>Switch to <em>Bank Account</em> tab</li> |
||||
|
<li>Set <em>Bank Feeds</em> to <em>Online</em></li> |
||||
|
<li>Select online bank statements provider in <em>Online Bank Statements (OCA)</em> |
||||
|
section</li> |
||||
|
<li>Save the bank account</li> |
||||
|
<li>Click on provider and configure provider-specific settings.</li> |
||||
|
</ol> |
||||
|
<p><strong>NOTE</strong>: To access these features, user needs to belong to |
||||
|
<em>Show Full Accounting Features</em> group.</p> |
||||
|
</div> |
||||
|
<div class="section" id="usage"> |
||||
|
<h1><a class="toc-backref" href="#id2">Usage</a></h1> |
||||
|
<p>To pull historical bank statements:</p> |
||||
|
<ol class="arabic simple"> |
||||
|
<li>Go to <em>Invoicing > Configuration > Bank Accounts</em></li> |
||||
|
<li>Select specific bank accounts</li> |
||||
|
<li>Launch <em>Actions > Online Bank Statements Pull Wizard</em></li> |
||||
|
<li>Configure date interval and click <em>Pull</em></li> |
||||
|
</ol> |
||||
|
<p><strong>NOTE</strong>: To access these features, user needs to belong to |
||||
|
<em>Show Full Accounting Features</em> group.</p> |
||||
|
</div> |
||||
|
<div class="section" id="bug-tracker"> |
||||
|
<h1><a class="toc-backref" href="#id3">Bug Tracker</a></h1> |
||||
|
<p>Bugs are tracked on <a class="reference external" href="https://github.com/OCA/bank-statement-import/issues">GitHub Issues</a>. |
||||
|
In case of trouble, please check there if your issue has already been reported. |
||||
|
If you spotted it first, help us smashing it by providing a detailed and welcomed |
||||
|
<a class="reference external" href="https://github.com/OCA/bank-statement-import/issues/new?body=module:%20account_bank_statement_import_online%0Aversion:%2012.0%0A%0A**Steps%20to%20reproduce**%0A-%20...%0A%0A**Current%20behavior**%0A%0A**Expected%20behavior**">feedback</a>.</p> |
||||
|
<p>Do not contact contributors directly about support or help with technical issues.</p> |
||||
|
</div> |
||||
|
<div class="section" id="credits"> |
||||
|
<h1><a class="toc-backref" href="#id4">Credits</a></h1> |
||||
|
<div class="section" id="authors"> |
||||
|
<h2><a class="toc-backref" href="#id5">Authors</a></h2> |
||||
|
<ul class="simple"> |
||||
|
<li>Brainbean Apps</li> |
||||
|
<li>Dataplug</li> |
||||
|
</ul> |
||||
|
</div> |
||||
|
<div class="section" id="contributors"> |
||||
|
<h2><a class="toc-backref" href="#id6">Contributors</a></h2> |
||||
|
<ul class="simple"> |
||||
|
<li>Alexey Pelykh <<a class="reference external" href="mailto:alexey.pelykh@brainbeanapps.com">alexey.pelykh@brainbeanapps.com</a>></li> |
||||
|
</ul> |
||||
|
</div> |
||||
|
<div class="section" id="maintainers"> |
||||
|
<h2><a class="toc-backref" href="#id7">Maintainers</a></h2> |
||||
|
<p>This module is maintained by the OCA.</p> |
||||
|
<a class="reference external image-reference" href="https://odoo-community.org"><img alt="Odoo Community Association" src="https://odoo-community.org/logo.png" /></a> |
||||
|
<p>OCA, or the Odoo Community Association, is a nonprofit organization whose |
||||
|
mission is to support the collaborative development of Odoo features and |
||||
|
promote its widespread use.</p> |
||||
|
<p>This module is part of the <a class="reference external" href="https://github.com/OCA/bank-statement-import/tree/12.0/account_bank_statement_import_online">OCA/bank-statement-import</a> project on GitHub.</p> |
||||
|
<p>You are welcome to contribute. To learn how please visit <a class="reference external" href="https://odoo-community.org/page/Contribute">https://odoo-community.org/page/Contribute</a>.</p> |
||||
|
</div> |
||||
|
</div> |
||||
|
</div> |
||||
|
</body> |
||||
|
</html> |
@ -0,0 +1,3 @@ |
|||||
|
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). |
||||
|
|
||||
|
from . import test_account_bank_statement_import_online |
@ -0,0 +1,55 @@ |
|||||
|
# Copyright 2019 Brainbean Apps (https://brainbeanapps.com) |
||||
|
# Copyright 2019 Dataplug (https://dataplug.io) |
||||
|
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl.html). |
||||
|
|
||||
|
from datetime import datetime, timedelta |
||||
|
from dateutil.relativedelta import relativedelta |
||||
|
from random import randrange |
||||
|
|
||||
|
from odoo import models, api |
||||
|
|
||||
|
|
||||
|
class OnlineBankStatementProviderDummy(models.Model): |
||||
|
_inherit = 'online.bank.statement.provider' |
||||
|
|
||||
|
@api.multi |
||||
|
def _obtain_statement_data(self, date_since, date_until): |
||||
|
self.ensure_one() |
||||
|
if self.service != 'dummy': |
||||
|
return super()._obtain_statement_data( |
||||
|
date_since, |
||||
|
date_until, |
||||
|
) # pragma: no cover |
||||
|
|
||||
|
if self.env.context.get('crash', False): |
||||
|
raise Exception('Expected') |
||||
|
|
||||
|
line_step_options = self.env.context.get('step', { |
||||
|
'minutes': 5, |
||||
|
}) |
||||
|
line_step = relativedelta(**line_step_options) |
||||
|
expand_by = self.env.context.get('expand_by', 0) |
||||
|
date_since -= expand_by * line_step |
||||
|
date_until += expand_by * line_step |
||||
|
|
||||
|
balance_start = randrange(-10000, 10000, 1) * 0.1 |
||||
|
balance = balance_start |
||||
|
lines = [] |
||||
|
date = date_since |
||||
|
while date < date_until: |
||||
|
amount = randrange(-100, 100, 1) * 0.1 |
||||
|
lines.append({ |
||||
|
'name': 'payment', |
||||
|
'amount': amount, |
||||
|
'date': date, |
||||
|
'unique_import_id': str(int( |
||||
|
(date - datetime(1970, 1, 1)) / timedelta(seconds=1) |
||||
|
)), |
||||
|
}) |
||||
|
balance += amount |
||||
|
date += line_step |
||||
|
balance_end = balance |
||||
|
return lines, { |
||||
|
'balance_start': balance_start, |
||||
|
'balance_end_real': balance_end, |
||||
|
} |
@ -0,0 +1,343 @@ |
|||||
|
# Copyright 2019 Brainbean Apps (https://brainbeanapps.com) |
||||
|
# Copyright 2019 Dataplug (https://dataplug.io) |
||||
|
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). |
||||
|
|
||||
|
from dateutil.relativedelta import relativedelta |
||||
|
|
||||
|
from psycopg2 import IntegrityError |
||||
|
|
||||
|
from odoo.tests import common |
||||
|
from odoo.tools import mute_logger |
||||
|
from odoo import fields |
||||
|
|
||||
|
|
||||
|
class TestAccountBankAccountStatementImportOnline(common.TransactionCase): |
||||
|
|
||||
|
def setUp(self): |
||||
|
super().setUp() |
||||
|
|
||||
|
self.now = fields.Datetime.now() |
||||
|
self.AccountJournal = self.env['account.journal'] |
||||
|
self.OnlineBankStatementProvider = self.env[ |
||||
|
'online.bank.statement.provider' |
||||
|
] |
||||
|
self.AccountBankStatement = self.env['account.bank.statement'] |
||||
|
self.AccountBankStatementLine = self.env['account.bank.statement.line'] |
||||
|
|
||||
|
def test_provider_unlink_restricted(self): |
||||
|
journal = self.AccountJournal.create({ |
||||
|
'name': 'Bank', |
||||
|
'type': 'bank', |
||||
|
'code': 'BANK', |
||||
|
}) |
||||
|
with common.Form(journal) as journal_form: |
||||
|
journal_form.bank_statements_source = 'online' |
||||
|
journal_form.online_bank_statement_provider = 'dummy' |
||||
|
journal_form.save() |
||||
|
|
||||
|
with self.assertRaises(IntegrityError), mute_logger('odoo.sql_db'): |
||||
|
journal.online_bank_statement_provider_id.unlink() |
||||
|
|
||||
|
def test_cascade_unlink(self): |
||||
|
journal = self.AccountJournal.create({ |
||||
|
'name': 'Bank', |
||||
|
'type': 'bank', |
||||
|
'code': 'BANK', |
||||
|
}) |
||||
|
with common.Form(journal) as journal_form: |
||||
|
journal_form.bank_statements_source = 'online' |
||||
|
journal_form.online_bank_statement_provider = 'dummy' |
||||
|
journal_form.save() |
||||
|
|
||||
|
self.assertTrue(journal.online_bank_statement_provider_id) |
||||
|
journal.unlink() |
||||
|
self.assertFalse(self.OnlineBankStatementProvider.search([])) |
||||
|
|
||||
|
def test_source_change_cleanup(self): |
||||
|
journal = self.AccountJournal.create({ |
||||
|
'name': 'Bank', |
||||
|
'type': 'bank', |
||||
|
'code': 'BANK', |
||||
|
}) |
||||
|
with common.Form(journal) as journal_form: |
||||
|
journal_form.bank_statements_source = 'online' |
||||
|
journal_form.online_bank_statement_provider = 'dummy' |
||||
|
journal_form.save() |
||||
|
|
||||
|
self.assertTrue(journal.online_bank_statement_provider_id) |
||||
|
|
||||
|
with common.Form(journal) as journal_form: |
||||
|
journal_form.bank_statements_source = 'undefined' |
||||
|
journal_form.save() |
||||
|
|
||||
|
self.assertFalse(journal.online_bank_statement_provider_id) |
||||
|
self.assertFalse(self.OnlineBankStatementProvider.search([])) |
||||
|
|
||||
|
def test_pull_boundary(self): |
||||
|
journal = self.AccountJournal.create({ |
||||
|
'name': 'Bank', |
||||
|
'type': 'bank', |
||||
|
'code': 'BANK', |
||||
|
'bank_statements_source': 'online', |
||||
|
'online_bank_statement_provider': 'dummy', |
||||
|
}) |
||||
|
|
||||
|
provider = journal.online_bank_statement_provider_id |
||||
|
provider.active = True |
||||
|
provider.with_context({ |
||||
|
'expand_by': 1, |
||||
|
})._pull( |
||||
|
self.now - relativedelta(hours=1), |
||||
|
self.now, |
||||
|
) |
||||
|
|
||||
|
statement = self.AccountBankStatement.search( |
||||
|
[('journal_id', '=', journal.id)], |
||||
|
) |
||||
|
self.assertEquals(len(statement), 1) |
||||
|
self.assertEquals(len(statement.line_ids), 12) |
||||
|
|
||||
|
def test_pull_mode_daily(self): |
||||
|
journal = self.AccountJournal.create({ |
||||
|
'name': 'Bank', |
||||
|
'type': 'bank', |
||||
|
'code': 'BANK', |
||||
|
'bank_statements_source': 'online', |
||||
|
'online_bank_statement_provider': 'dummy', |
||||
|
}) |
||||
|
|
||||
|
provider = journal.online_bank_statement_provider_id |
||||
|
provider.active = True |
||||
|
provider.statement_creation_mode = 'daily' |
||||
|
|
||||
|
provider.with_context(step={'hours': 2})._pull( |
||||
|
self.now - relativedelta(days=1), |
||||
|
self.now, |
||||
|
) |
||||
|
self.assertEquals( |
||||
|
len(self.AccountBankStatement.search( |
||||
|
[('journal_id', '=', journal.id)] |
||||
|
)), |
||||
|
2 |
||||
|
) |
||||
|
|
||||
|
def test_pull_mode_weekly(self): |
||||
|
journal = self.AccountJournal.create({ |
||||
|
'name': 'Bank', |
||||
|
'type': 'bank', |
||||
|
'code': 'BANK', |
||||
|
'bank_statements_source': 'online', |
||||
|
'online_bank_statement_provider': 'dummy', |
||||
|
}) |
||||
|
|
||||
|
provider = journal.online_bank_statement_provider_id |
||||
|
provider.active = True |
||||
|
provider.statement_creation_mode = 'weekly' |
||||
|
|
||||
|
provider.with_context(step={'hours': 8})._pull( |
||||
|
self.now - relativedelta(weeks=1), |
||||
|
self.now, |
||||
|
) |
||||
|
self.assertEquals( |
||||
|
len(self.AccountBankStatement.search( |
||||
|
[('journal_id', '=', journal.id)] |
||||
|
)), |
||||
|
2 |
||||
|
) |
||||
|
|
||||
|
def test_pull_mode_monthly(self): |
||||
|
journal = self.AccountJournal.create({ |
||||
|
'name': 'Bank', |
||||
|
'type': 'bank', |
||||
|
'code': 'BANK', |
||||
|
'bank_statements_source': 'online', |
||||
|
'online_bank_statement_provider': 'dummy', |
||||
|
}) |
||||
|
|
||||
|
provider = journal.online_bank_statement_provider_id |
||||
|
provider.active = True |
||||
|
provider.statement_creation_mode = 'monthly' |
||||
|
|
||||
|
provider.with_context(step={'hours': 8})._pull( |
||||
|
self.now - relativedelta(months=1), |
||||
|
self.now, |
||||
|
) |
||||
|
self.assertEquals( |
||||
|
len(self.AccountBankStatement.search( |
||||
|
[('journal_id', '=', journal.id)] |
||||
|
)), |
||||
|
2 |
||||
|
) |
||||
|
|
||||
|
def test_pull_scheduled(self): |
||||
|
journal = self.AccountJournal.create({ |
||||
|
'name': 'Bank', |
||||
|
'type': 'bank', |
||||
|
'code': 'BANK', |
||||
|
'bank_statements_source': 'online', |
||||
|
'online_bank_statement_provider': 'dummy', |
||||
|
}) |
||||
|
|
||||
|
provider = journal.online_bank_statement_provider_id |
||||
|
provider.active = True |
||||
|
provider.next_run = ( |
||||
|
self.now - relativedelta(days=15) |
||||
|
) |
||||
|
|
||||
|
self.assertFalse(self.AccountBankStatement.search( |
||||
|
[('journal_id', '=', journal.id)], |
||||
|
)) |
||||
|
|
||||
|
provider.with_context(step={'hours': 8})._scheduled_pull() |
||||
|
|
||||
|
statement = self.AccountBankStatement.search( |
||||
|
[('journal_id', '=', journal.id)], |
||||
|
) |
||||
|
self.assertEquals(len(statement), 1) |
||||
|
|
||||
|
def test_pull_skip_duplicates_by_unique_import_id(self): |
||||
|
journal = self.AccountJournal.create({ |
||||
|
'name': 'Bank', |
||||
|
'type': 'bank', |
||||
|
'code': 'BANK', |
||||
|
'bank_statements_source': 'online', |
||||
|
'online_bank_statement_provider': 'dummy', |
||||
|
}) |
||||
|
|
||||
|
provider = journal.online_bank_statement_provider_id |
||||
|
provider.active = True |
||||
|
provider.statement_creation_mode = 'weekly' |
||||
|
|
||||
|
provider.with_context(step={'hours': 8})._pull( |
||||
|
self.now - relativedelta(weeks=2), |
||||
|
self.now, |
||||
|
) |
||||
|
self.assertEquals( |
||||
|
len(self.AccountBankStatementLine.search( |
||||
|
[('journal_id', '=', journal.id)] |
||||
|
)), |
||||
|
14 * (24 / 8) |
||||
|
) |
||||
|
|
||||
|
provider.with_context(step={'hours': 8})._pull( |
||||
|
self.now - relativedelta(weeks=3), |
||||
|
self.now - relativedelta(weeks=1), |
||||
|
) |
||||
|
self.assertEquals( |
||||
|
len(self.AccountBankStatementLine.search( |
||||
|
[('journal_id', '=', journal.id)] |
||||
|
)), |
||||
|
21 * (24 / 8) |
||||
|
) |
||||
|
|
||||
|
provider.with_context(step={'hours': 8})._pull( |
||||
|
self.now - relativedelta(weeks=1), |
||||
|
self.now, |
||||
|
) |
||||
|
self.assertEquals( |
||||
|
len(self.AccountBankStatementLine.search( |
||||
|
[('journal_id', '=', journal.id)] |
||||
|
)), |
||||
|
21 * (24 / 8) |
||||
|
) |
||||
|
|
||||
|
def test_interval_type_minutes(self): |
||||
|
journal = self.AccountJournal.create({ |
||||
|
'name': 'Bank', |
||||
|
'type': 'bank', |
||||
|
'code': 'BANK', |
||||
|
'bank_statements_source': 'online', |
||||
|
'online_bank_statement_provider': 'dummy', |
||||
|
}) |
||||
|
|
||||
|
provider = journal.online_bank_statement_provider_id |
||||
|
provider.active = True |
||||
|
provider.interval_type = 'minutes' |
||||
|
provider._compute_update_schedule() |
||||
|
|
||||
|
def test_interval_type_hours(self): |
||||
|
journal = self.AccountJournal.create({ |
||||
|
'name': 'Bank', |
||||
|
'type': 'bank', |
||||
|
'code': 'BANK', |
||||
|
'bank_statements_source': 'online', |
||||
|
'online_bank_statement_provider': 'dummy', |
||||
|
}) |
||||
|
|
||||
|
provider = journal.online_bank_statement_provider_id |
||||
|
provider.active = True |
||||
|
provider.interval_type = 'hours' |
||||
|
provider._compute_update_schedule() |
||||
|
|
||||
|
def test_interval_type_days(self): |
||||
|
journal = self.AccountJournal.create({ |
||||
|
'name': 'Bank', |
||||
|
'type': 'bank', |
||||
|
'code': 'BANK', |
||||
|
'bank_statements_source': 'online', |
||||
|
'online_bank_statement_provider': 'dummy', |
||||
|
}) |
||||
|
|
||||
|
provider = journal.online_bank_statement_provider_id |
||||
|
provider.active = True |
||||
|
provider.interval_type = 'days' |
||||
|
provider._compute_update_schedule() |
||||
|
|
||||
|
def test_interval_type_weeks(self): |
||||
|
journal = self.AccountJournal.create({ |
||||
|
'name': 'Bank', |
||||
|
'type': 'bank', |
||||
|
'code': 'BANK', |
||||
|
'bank_statements_source': 'online', |
||||
|
'online_bank_statement_provider': 'dummy', |
||||
|
}) |
||||
|
|
||||
|
provider = journal.online_bank_statement_provider_id |
||||
|
provider.active = True |
||||
|
provider.interval_type = 'weeks' |
||||
|
provider._compute_update_schedule() |
||||
|
|
||||
|
def test_pull_no_crash(self): |
||||
|
journal = self.AccountJournal.create({ |
||||
|
'name': 'Bank', |
||||
|
'type': 'bank', |
||||
|
'code': 'BANK', |
||||
|
'bank_statements_source': 'online', |
||||
|
'online_bank_statement_provider': 'dummy', |
||||
|
}) |
||||
|
|
||||
|
provider = journal.online_bank_statement_provider_id |
||||
|
provider.active = True |
||||
|
provider.statement_creation_mode = 'weekly' |
||||
|
|
||||
|
provider.with_context( |
||||
|
crash=True, |
||||
|
scheduled=True, |
||||
|
)._pull( |
||||
|
self.now - relativedelta(hours=1), |
||||
|
self.now, |
||||
|
) |
||||
|
self.assertFalse(self.AccountBankStatement.search( |
||||
|
[('journal_id', '=', journal.id)], |
||||
|
)) |
||||
|
|
||||
|
def test_pull_crash(self): |
||||
|
journal = self.AccountJournal.create({ |
||||
|
'name': 'Bank', |
||||
|
'type': 'bank', |
||||
|
'code': 'BANK', |
||||
|
'bank_statements_source': 'online', |
||||
|
'online_bank_statement_provider': 'dummy', |
||||
|
}) |
||||
|
|
||||
|
provider = journal.online_bank_statement_provider_id |
||||
|
provider.active = True |
||||
|
provider.statement_creation_mode = 'weekly' |
||||
|
|
||||
|
with self.assertRaises(Exception): |
||||
|
provider.with_context( |
||||
|
crash=True, |
||||
|
)._pull( |
||||
|
self.now - relativedelta(hours=1), |
||||
|
self.now, |
||||
|
) |
@ -0,0 +1,120 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||
|
<!-- |
||||
|
Copyright 2019 Brainbean Apps (https://brainbeanapps.com) |
||||
|
Copyright 2019 Dataplug (https://dataplug.io) |
||||
|
License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). |
||||
|
--> |
||||
|
<odoo> |
||||
|
|
||||
|
<record model="ir.ui.view" id="view_account_journal_form"> |
||||
|
<field name="name">account.journal.form</field> |
||||
|
<field name="model">account.journal</field> |
||||
|
<field name="inherit_id" ref="account.view_account_journal_form"/> |
||||
|
<field name="arch" type="xml"> |
||||
|
<page name="bank_account"> |
||||
|
<group |
||||
|
name="online_bank_statements" |
||||
|
string="Online Bank Statements (OCA)" |
||||
|
groups="account.group_account_user" |
||||
|
attrs="{'invisible': [('bank_statements_source', '!=', 'online')]}" |
||||
|
> |
||||
|
<label |
||||
|
for="online_bank_statement_provider" |
||||
|
string="Provider" |
||||
|
attrs="{'required': [('bank_statements_source', '=', 'online')]}" |
||||
|
class="oe_edit_only" |
||||
|
groups="account.group_account_user" |
||||
|
/> |
||||
|
<field |
||||
|
name="online_bank_statement_provider" |
||||
|
nolabel="1" |
||||
|
attrs="{'required': [('bank_statements_source', '=', 'online')]}" |
||||
|
class="oe_edit_only" |
||||
|
groups="account.group_account_user" |
||||
|
/> |
||||
|
<label |
||||
|
for="online_bank_statement_provider_id" |
||||
|
string="Provider" |
||||
|
attrs="{'invisible': [('online_bank_statement_provider_id', '=', False)]}" |
||||
|
class="oe_read_only" |
||||
|
/> |
||||
|
<field |
||||
|
name="online_bank_statement_provider_id" |
||||
|
nolabel="1" |
||||
|
attrs="{'invisible': [('online_bank_statement_provider_id', '=', False)]}" |
||||
|
class="oe_read_only" |
||||
|
/> |
||||
|
</group> |
||||
|
</page> |
||||
|
</field> |
||||
|
</record> |
||||
|
|
||||
|
<record model="ir.ui.view" id="view_account_bank_journal_form"> |
||||
|
<field name="name">account.journal.form</field> |
||||
|
<field name="model">account.journal</field> |
||||
|
<field name="inherit_id" ref="account.view_account_bank_journal_form"/> |
||||
|
<field name="arch" type="xml"> |
||||
|
<group name="bank_statement" position="after"> |
||||
|
<group |
||||
|
name="online_bank_statements" |
||||
|
string="Online Bank Statements (OCA)" |
||||
|
groups="account.group_account_user" |
||||
|
attrs="{'invisible': [('bank_statements_source', '!=', 'online')]}" |
||||
|
> |
||||
|
<label |
||||
|
for="online_bank_statement_provider" |
||||
|
string="Provider" |
||||
|
attrs="{'required': [('bank_statements_source', '=', 'online')]}" |
||||
|
class="oe_edit_only" |
||||
|
groups="account.group_account_user" |
||||
|
/> |
||||
|
<field |
||||
|
name="online_bank_statement_provider" |
||||
|
nolabel="1" |
||||
|
attrs="{'required': [('bank_statements_source', '=', 'online')]}" |
||||
|
class="oe_edit_only" |
||||
|
groups="account.group_account_user" |
||||
|
widget="dynamic_dropdown" |
||||
|
values="values_online_bank_statement_provider" |
||||
|
/> |
||||
|
<label |
||||
|
for="online_bank_statement_provider_id" |
||||
|
string="Provider" |
||||
|
attrs="{'invisible': [('online_bank_statement_provider_id', '=', False)]}" |
||||
|
class="oe_read_only" |
||||
|
/> |
||||
|
<field |
||||
|
name="online_bank_statement_provider_id" |
||||
|
nolabel="1" |
||||
|
attrs="{'invisible': [('online_bank_statement_provider_id', '=', False)]}" |
||||
|
class="oe_read_only" |
||||
|
/> |
||||
|
</group> |
||||
|
</group> |
||||
|
</field> |
||||
|
</record> |
||||
|
|
||||
|
<record id="account_journal_dashboard_kanban_view" model="ir.ui.view"> |
||||
|
<field name="name">account.journal.dashboard.kanban</field> |
||||
|
<field name="model">account.journal</field> |
||||
|
<field name="inherit_id" ref="account.account_journal_dashboard_kanban_view"/> |
||||
|
<field name="arch" type="xml"> |
||||
|
<div name="bank_statement_create_button" position="attributes"> |
||||
|
<attribute name="t-if">dashboard.bank_statements_source != 'online_sync' and dashboard.bank_statements_source != 'online'</attribute> |
||||
|
</div> |
||||
|
</field> |
||||
|
</record> |
||||
|
|
||||
|
<record id="action_online_bank_statements_pull_wizard" model="ir.actions.server"> |
||||
|
<field name="name">Online Bank Statements Pull Wizard</field> |
||||
|
<field name="type">ir.actions.server</field> |
||||
|
<field name="model_id" ref="account.model_account_journal"/> |
||||
|
<field name="binding_model_id" ref="account.model_account_journal"/> |
||||
|
<field name="state">code</field> |
||||
|
<field name="code"> |
||||
|
if records: |
||||
|
action = records.action_online_bank_statements_pull_wizard() |
||||
|
</field> |
||||
|
</record> |
||||
|
|
||||
|
</odoo> |
@ -0,0 +1,96 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||
|
<!-- |
||||
|
Copyright 2019 Brainbean Apps (https://brainbeanapps.com) |
||||
|
Copyright 2019 Dataplug (https://dataplug.io) |
||||
|
License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). |
||||
|
--> |
||||
|
<odoo> |
||||
|
|
||||
|
<record model="ir.ui.view" id="online_bank_statement_provider_filter"> |
||||
|
<field name="name">online.bank.statement.provider.filter</field> |
||||
|
<field name="model">online.bank.statement.provider</field> |
||||
|
<field name="arch" type="xml"> |
||||
|
<search string="Online Bank Statement Providers"> |
||||
|
<filter name="active" string="Inactive" domain="[('active', '=', False)]"/> |
||||
|
</search> |
||||
|
</field> |
||||
|
</record> |
||||
|
|
||||
|
<record model="ir.ui.view" id="online_bank_statement_provider_tree"> |
||||
|
<field name="name">online.bank.statement.provider.tree</field> |
||||
|
<field name="model">online.bank.statement.provider</field> |
||||
|
<field name="arch" type="xml"> |
||||
|
<tree> |
||||
|
<field name="journal_id"/> |
||||
|
<field name="company_id" groups="base.group_multi_company"/> |
||||
|
<field name="service"/> |
||||
|
<field name="currency_id"/> |
||||
|
<field name="update_schedule"/> |
||||
|
<field name="next_run"/> |
||||
|
</tree> |
||||
|
</field> |
||||
|
</record> |
||||
|
|
||||
|
<record model="ir.ui.view" id="online_bank_statement_provider_form"> |
||||
|
<field name="name">online.bank.statement.provider.form</field> |
||||
|
<field name="model">online.bank.statement.provider</field> |
||||
|
<field name="arch" type="xml"> |
||||
|
<form string="Online Bank Statement Provider"> |
||||
|
<sheet> |
||||
|
<div class="oe_button_box" name="button_box"> |
||||
|
<button class="oe_stat_button" type="object" name="toggle_active" icon="fa-archive"> |
||||
|
<field name="active" widget="boolean_button" options='{"terminology": "active"}'/> |
||||
|
</button> |
||||
|
</div> |
||||
|
<notebook> |
||||
|
<page name="details" string="Details"> |
||||
|
<group> |
||||
|
<group groups="base.group_multi_company"> |
||||
|
<field name="company_id"/> |
||||
|
</group> |
||||
|
<group> |
||||
|
<field name="journal_id"/> |
||||
|
<field name="currency_id"/> |
||||
|
<field name="account_number"/> |
||||
|
</group> |
||||
|
<group> |
||||
|
<field |
||||
|
name="service" |
||||
|
widget="dynamic_dropdown" |
||||
|
values="values_service" |
||||
|
/> |
||||
|
</group> |
||||
|
</group> |
||||
|
</page> |
||||
|
<page name="pull" string="Scheduled Pull"> |
||||
|
<group> |
||||
|
<group> |
||||
|
<label for="interval_number"/> |
||||
|
<div class="o_row"> |
||||
|
<field name="interval_number" class="ml8"/> |
||||
|
<field name="interval_type"/> |
||||
|
</div> |
||||
|
</group> |
||||
|
<group> |
||||
|
<field name="next_run"/> |
||||
|
</group> |
||||
|
</group> |
||||
|
</page> |
||||
|
<page name="configuration" string="Configuration"> |
||||
|
<group> |
||||
|
<group> |
||||
|
<field name="statement_creation_mode"/> |
||||
|
</group> |
||||
|
</group> |
||||
|
</page> |
||||
|
</notebook> |
||||
|
</sheet> |
||||
|
<div class="oe_chatter"> |
||||
|
<field name="message_follower_ids" widget="mail_followers"/> |
||||
|
<field name="message_ids" widget="mail_thread"/> |
||||
|
</div> |
||||
|
</form> |
||||
|
</field> |
||||
|
</record> |
||||
|
|
||||
|
</odoo> |
@ -0,0 +1,3 @@ |
|||||
|
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). |
||||
|
|
||||
|
from . import online_bank_statement_pull_wizard |
@ -0,0 +1,34 @@ |
|||||
|
# Copyright 2019 Brainbean Apps (https://brainbeanapps.com) |
||||
|
# Copyright 2019 Dataplug (https://dataplug.io) |
||||
|
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). |
||||
|
|
||||
|
from odoo import fields, models, api |
||||
|
|
||||
|
|
||||
|
class OnlineBankStatementPullWizard(models.TransientModel): |
||||
|
_name = 'online.bank.statement.pull.wizard' |
||||
|
_description = 'Online Bank Statement Pull Wizard' |
||||
|
|
||||
|
date_since = fields.Datetime( |
||||
|
string='Since', |
||||
|
required=True, |
||||
|
default=fields.Datetime.now, |
||||
|
) |
||||
|
date_until = fields.Datetime( |
||||
|
string='Until', |
||||
|
required=True, |
||||
|
default=fields.Datetime.now, |
||||
|
) |
||||
|
provider_ids = fields.Many2many( |
||||
|
string='Providers', |
||||
|
comodel_name='online.bank.statement.provider', |
||||
|
column1='wizard_id', |
||||
|
column2='provider_id', |
||||
|
relation='online_bank_statement_provider_pull_wizard_rel' |
||||
|
) |
||||
|
|
||||
|
@api.multi |
||||
|
def action_pull(self): |
||||
|
self.ensure_one() |
||||
|
self.provider_ids._pull(self.date_since, self.date_until) |
||||
|
return {'type': 'ir.actions.act_window_close'} |
@ -0,0 +1,32 @@ |
|||||
|
<?xml version="1.0" encoding="UTF-8" ?> |
||||
|
<!-- |
||||
|
Copyright 2019 Brainbean Apps (https://brainbeanapps.com) |
||||
|
Copyright 2019 Dataplug (https://dataplug.io) |
||||
|
License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). |
||||
|
--> |
||||
|
<odoo> |
||||
|
|
||||
|
<record id="online_bank_statement_pull_wizard_form" model="ir.ui.view"> |
||||
|
<field name="name">online.bank.statement.pull.wizard.form</field> |
||||
|
<field name="model">online.bank.statement.pull.wizard</field> |
||||
|
<field name="arch" type="xml"> |
||||
|
<form> |
||||
|
<group name="filter"> |
||||
|
<group name="date_range" colspan="2"> |
||||
|
<group> |
||||
|
<field name="date_since"/> |
||||
|
</group> |
||||
|
<group> |
||||
|
<field name="date_until"/> |
||||
|
</group> |
||||
|
</group> |
||||
|
</group> |
||||
|
<footer> |
||||
|
<button name="action_pull" string="Pull" type="object" default_focus="1" class="oe_highlight"/> |
||||
|
<button string="Cancel" class="oe_link" special="cancel" /> |
||||
|
</footer> |
||||
|
</form> |
||||
|
</field> |
||||
|
</record> |
||||
|
|
||||
|
</odoo> |
@ -0,0 +1 @@ |
|||||
|
web |
Write
Preview
Loading…
Cancel
Save
Reference in new issue