Browse Source
[IMP] base_external_dbsource: Refactor & Split by source
[IMP] base_external_dbsource: Refactor & Split by source
* Heavily refactor code for reusability * Split all sources into independent modules * Add more test coverage * Add CRUD methods * Add iterator execute return to roadmappull/659/head
Dave Lasley
8 years ago
No known key found for this signature in database
GPG Key ID: 7DDBA4BA81B934CF
56 changed files with 1827 additions and 217 deletions
-
2.travis.yml
-
33base_external_dbsource/README.rst
-
1base_external_dbsource/__init__.py
-
6base_external_dbsource/__manifest__.py
-
13base_external_dbsource/exceptions.py
-
1base_external_dbsource/models/__init__.py
-
425base_external_dbsource/models/base_external_dbsource.py
-
2base_external_dbsource/tests/__init__.py
-
247base_external_dbsource/tests/test_base_external_dbsource.py
-
66base_external_dbsource/tests/test_create_dbsource.py
-
77base_external_dbsource_firebird/README.rst
-
2base_external_dbsource_firebird/__init__.py
-
28base_external_dbsource_firebird/__manifest__.py
-
9base_external_dbsource_firebird/demo/base_external_dbsource.xml
-
3base_external_dbsource_firebird/models/__init__.py
-
56base_external_dbsource_firebird/models/base_external_dbsource.py
-
3base_external_dbsource_firebird/tests/__init__.py
-
41base_external_dbsource_firebird/tests/test_base_external_dbsource.py
-
80base_external_dbsource_mssql/README.rst
-
2base_external_dbsource_mssql/__init__.py
-
29base_external_dbsource_mssql/__manifest__.py
-
9base_external_dbsource_mssql/demo/base_external_dbsource.xml
-
3base_external_dbsource_mssql/models/__init__.py
-
45base_external_dbsource_mssql/models/base_external_dbsource.py
-
3base_external_dbsource_mssql/tests/__init__.py
-
42base_external_dbsource_mssql/tests/test_base_external_dbsource.py
-
80base_external_dbsource_mysql/README.rst
-
2base_external_dbsource_mysql/__init__.py
-
29base_external_dbsource_mysql/__manifest__.py
-
9base_external_dbsource_mysql/demo/base_external_dbsource.xml
-
3base_external_dbsource_mysql/models/__init__.py
-
43base_external_dbsource_mysql/models/base_external_dbsource.py
-
3base_external_dbsource_mysql/tests/__init__.py
-
42base_external_dbsource_mysql/tests/test_base_external_dbsource.py
-
79base_external_dbsource_odbc/README.rst
-
2base_external_dbsource_odbc/__init__.py
-
29base_external_dbsource_odbc/__manifest__.py
-
9base_external_dbsource_odbc/demo/base_external_dbsource.xml
-
3base_external_dbsource_odbc/models/__init__.py
-
42base_external_dbsource_odbc/models/base_external_dbsource.py
-
3base_external_dbsource_odbc/tests/__init__.py
-
46base_external_dbsource_odbc/tests/test_base_external_dbsource.py
-
79base_external_dbsource_oracle/README.rst
-
2base_external_dbsource_oracle/__init__.py
-
25base_external_dbsource_oracle/__manifest__.py
-
3base_external_dbsource_oracle/models/__init__.py
-
47base_external_dbsource_oracle/models/base_external_dbsource.py
-
77base_external_dbsource_sqlite/README.rst
-
2base_external_dbsource_sqlite/__init__.py
-
28base_external_dbsource_sqlite/__manifest__.py
-
9base_external_dbsource_sqlite/demo/base_external_dbsource.xml
-
3base_external_dbsource_sqlite/models/__init__.py
-
59base_external_dbsource_sqlite/models/base_external_dbsource.py
-
3base_external_dbsource_sqlite/tests/__init__.py
-
42base_external_dbsource_sqlite/tests/test_base_external_dbsource.py
-
5requirements.txt
@ -1,2 +1,3 @@ |
|||
# -*- coding: utf-8 -*- |
|||
|
|||
from . import models |
@ -0,0 +1,13 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# Copyright 2016 LasLabs Inc. |
|||
# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl). |
|||
|
|||
from odoo.exceptions import UserError |
|||
|
|||
|
|||
class ConnectionFailedError(UserError): |
|||
pass |
|||
|
|||
|
|||
class ConnectionSuccessError(UserError): |
|||
pass |
@ -1,2 +1,3 @@ |
|||
# -*- coding: utf-8 -*- |
|||
|
|||
from . import base_external_dbsource |
@ -1,3 +1,3 @@ |
|||
# -*- encoding: utf-8 -*- |
|||
|
|||
from . import test_create_dbsource |
|||
from . import test_base_external_dbsource |
@ -0,0 +1,247 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# Copyright 2016 LasLabs Inc. |
|||
|
|||
import mock |
|||
|
|||
from odoo.tests import common |
|||
|
|||
from ..exceptions import ConnectionFailedError, ConnectionSuccessError |
|||
|
|||
|
|||
class TestBaseExternalDbsource(common.TransactionCase): |
|||
|
|||
def setUp(self): |
|||
super(TestBaseExternalDbsource, self).setUp() |
|||
self.dbsource = self.env.ref('base_external_dbsource.demo_postgre') |
|||
|
|||
def _test_adapter_method( |
|||
self, method_name, side_effect=None, return_value=None, |
|||
create=False, args=None, kwargs=None, |
|||
): |
|||
if args is None: |
|||
args = [] |
|||
if kwargs is None: |
|||
kwargs = {} |
|||
adapter = '%s_postgresql' % method_name |
|||
with mock.patch.object(self.dbsource, |
|||
adapter, create=create) as adapter: |
|||
if side_effect is not None: |
|||
adapter.side_effect = side_effect |
|||
elif return_value is not None: |
|||
adapter.return_value = return_value |
|||
res = getattr(self.dbsource, method_name)(*args, **kwargs) |
|||
return res, adapter |
|||
|
|||
def test_conn_string_full(self): |
|||
""" It should add password if string interpolation not detected """ |
|||
self.dbsource.conn_string = 'User=Derp;' |
|||
self.dbsource.password = 'password' |
|||
expect = self.dbsource.conn_string + 'PWD=%s;' % self.dbsource.password |
|||
self.assertEqual( |
|||
self.dbsource.conn_string_full, expect, |
|||
) |
|||
|
|||
# Interface |
|||
|
|||
def test_connection_success(self): |
|||
""" It should raise for successful connection """ |
|||
with self.assertRaises(ConnectionSuccessError): |
|||
self.dbsource.connection_test() |
|||
|
|||
def test_connection_fail(self): |
|||
""" It should raise for failed/invalid connection """ |
|||
with mock.patch.object(self.dbsource, 'connection_open') as conn: |
|||
conn.side_effect = Exception |
|||
with self.assertRaises(ConnectionFailedError): |
|||
self.dbsource.connection_test() |
|||
|
|||
def test_connection_open_calls_close(self): |
|||
""" It should close connection after context ends """ |
|||
with mock.patch.object( |
|||
self.dbsource, 'connection_close', |
|||
) as close: |
|||
with self.dbsource.connection_open(): |
|||
pass |
|||
close.assert_called_once() |
|||
|
|||
def test_connection_close(self): |
|||
""" It should call adapter's close method """ |
|||
args = [mock.MagicMock()] |
|||
res, adapter = self._test_adapter_method( |
|||
'connection_close', args=args, |
|||
) |
|||
adapter.assert_called_once_with(args[0]) |
|||
|
|||
def test_execute_asserts_query_arg(self): |
|||
""" It should raise a TypeError if query and sqlquery not in args """ |
|||
with self.assertRaises(TypeError): |
|||
self.dbsource.execute() |
|||
|
|||
def test_execute_calls_adapter(self): |
|||
""" It should call the adapter methods with proper args """ |
|||
expect = ('query', 'execute', 'metadata') |
|||
return_value = 'rows', 'cols' |
|||
res, adapter = self._test_adapter_method( |
|||
'execute', args=expect, return_value=return_value, |
|||
) |
|||
adapter.assert_called_once_with(*expect) |
|||
|
|||
def test_execute_return(self): |
|||
""" It should return rows if not metadata """ |
|||
expect = (True, True, False) |
|||
return_value = 'rows', 'cols' |
|||
res, adapter = self._test_adapter_method( |
|||
'execute', args=expect, return_value=return_value, |
|||
) |
|||
self.assertEqual(res, return_value[0]) |
|||
|
|||
def test_execute_return_metadata(self): |
|||
""" It should return rows and cols if metadata """ |
|||
expect = (True, True, True) |
|||
return_value = 'rows', 'cols' |
|||
res, adapter = self._test_adapter_method( |
|||
'execute', args=expect, return_value=return_value, |
|||
) |
|||
self.assertEqual( |
|||
res, |
|||
{'rows': return_value[0], |
|||
'cols': return_value[1]}, |
|||
) |
|||
|
|||
def test_remote_browse(self): |
|||
""" It should call the adapter method with proper args """ |
|||
args = [1], 'args' |
|||
kwargs = {'kwargs': True} |
|||
self.dbsource.current_table = 'table' |
|||
res, adapter = self._test_adapter_method( |
|||
'remote_browse', create=True, args=args, kwargs=kwargs, |
|||
) |
|||
adapter.assert_called_once_with(*args, **kwargs) |
|||
self.assertEqual(res, adapter()) |
|||
|
|||
def test_remote_browse_asserts_current_table(self): |
|||
""" It should raise AssertionError if a table not selected """ |
|||
args = [1], 'args' |
|||
kwargs = {'kwargs': True} |
|||
with self.assertRaises(AssertionError): |
|||
res, adapter = self._test_adapter_method( |
|||
'remote_browse', create=True, args=args, kwargs=kwargs, |
|||
) |
|||
|
|||
def test_remote_create(self): |
|||
""" It should call the adapter method with proper args """ |
|||
args = {'val': 'Value'}, 'args' |
|||
kwargs = {'kwargs': True} |
|||
self.dbsource.current_table = 'table' |
|||
res, adapter = self._test_adapter_method( |
|||
'remote_create', create=True, args=args, kwargs=kwargs, |
|||
) |
|||
adapter.assert_called_once_with(*args, **kwargs) |
|||
self.assertEqual(res, adapter()) |
|||
|
|||
def test_remote_create_asserts_current_table(self): |
|||
""" It should raise AssertionError if a table not selected """ |
|||
args = [1], 'args' |
|||
kwargs = {'kwargs': True} |
|||
with self.assertRaises(AssertionError): |
|||
res, adapter = self._test_adapter_method( |
|||
'remote_create', create=True, args=args, kwargs=kwargs, |
|||
) |
|||
|
|||
def test_remote_delete(self): |
|||
""" It should call the adapter method with proper args """ |
|||
args = [1], 'args' |
|||
kwargs = {'kwargs': True} |
|||
self.dbsource.current_table = 'table' |
|||
res, adapter = self._test_adapter_method( |
|||
'remote_delete', create=True, args=args, kwargs=kwargs, |
|||
) |
|||
adapter.assert_called_once_with(*args, **kwargs) |
|||
self.assertEqual(res, adapter()) |
|||
|
|||
def test_remote_delete_asserts_current_table(self): |
|||
""" It should raise AssertionError if a table not selected """ |
|||
args = [1], 'args' |
|||
kwargs = {'kwargs': True} |
|||
with self.assertRaises(AssertionError): |
|||
res, adapter = self._test_adapter_method( |
|||
'remote_delete', create=True, args=args, kwargs=kwargs, |
|||
) |
|||
|
|||
def test_remote_search(self): |
|||
""" It should call the adapter method with proper args """ |
|||
args = {'search': 'query'}, 'args' |
|||
kwargs = {'kwargs': True} |
|||
self.dbsource.current_table = 'table' |
|||
res, adapter = self._test_adapter_method( |
|||
'remote_search', create=True, args=args, kwargs=kwargs, |
|||
) |
|||
adapter.assert_called_once_with(*args, **kwargs) |
|||
self.assertEqual(res, adapter()) |
|||
|
|||
def test_remote_search_asserts_current_table(self): |
|||
""" It should raise AssertionError if a table not selected """ |
|||
args = [1], 'args' |
|||
kwargs = {'kwargs': True} |
|||
with self.assertRaises(AssertionError): |
|||
res, adapter = self._test_adapter_method( |
|||
'remote_search', create=True, args=args, kwargs=kwargs, |
|||
) |
|||
|
|||
def test_remote_update(self): |
|||
""" It should call the adapter method with proper args """ |
|||
args = [1], {'vals': 'Value'}, 'args' |
|||
kwargs = {'kwargs': True} |
|||
self.dbsource.current_table = 'table' |
|||
res, adapter = self._test_adapter_method( |
|||
'remote_update', create=True, args=args, kwargs=kwargs, |
|||
) |
|||
adapter.assert_called_once_with(*args, **kwargs) |
|||
self.assertEqual(res, adapter()) |
|||
|
|||
def test_remote_update_asserts_current_table(self): |
|||
""" It should raise AssertionError if a table not selected """ |
|||
args = [1], 'args' |
|||
kwargs = {'kwargs': True} |
|||
with self.assertRaises(AssertionError): |
|||
res, adapter = self._test_adapter_method( |
|||
'remote_update', create=True, args=args, kwargs=kwargs, |
|||
) |
|||
|
|||
# Postgres |
|||
|
|||
def test_execute_postgresql(self): |
|||
""" It should call generic executor with proper args """ |
|||
expect = ('query', 'execute', 'metadata') |
|||
with mock.patch.object( |
|||
self.dbsource, '_execute_generic', autospec=True, |
|||
) as execute: |
|||
execute.return_value = 'rows', 'cols' |
|||
self.dbsource.execute(*expect) |
|||
execute.assert_called_once_with(*expect) |
|||
|
|||
# Old API Compat |
|||
|
|||
def test_execute_calls_adapter_old_api(self): |
|||
""" It should call the adapter correctly if old kwargs provided """ |
|||
expect = [None, None, 'metadata'] |
|||
with mock.patch.object( |
|||
self.dbsource, 'execute_postgresql', autospec=True, |
|||
) as psql: |
|||
psql.return_value = 'rows', 'cols' |
|||
self.dbsource.execute( |
|||
*expect, sqlparams='params', sqlquery='query' |
|||
) |
|||
expect[0], expect[1] = 'query', 'params' |
|||
psql.assert_called_once_with(*expect) |
|||
|
|||
def test_conn_open(self): |
|||
""" It should return open connection for use """ |
|||
with mock.patch.object( |
|||
self.dbsource, 'connection_open', autospec=True, |
|||
) as connection: |
|||
res = self.dbsource.conn_open() |
|||
self.assertEqual( |
|||
res, |
|||
connection().__enter__(), |
|||
) |
@ -1,66 +0,0 @@ |
|||
# -*- coding: utf-8 -*- |
|||
|
|||
from odoo.exceptions import Warning as UserError |
|||
from odoo.tests import common |
|||
import logging |
|||
|
|||
|
|||
class TestCreateDbsource(common.TransactionCase): |
|||
"""Test class for base_external_dbsource.""" |
|||
|
|||
def test_create_dbsource(self): |
|||
"""source creation should succeed.""" |
|||
dbsource = self.env.ref('base_external_dbsource.demo_postgre') |
|||
try: |
|||
dbsource.connection_test() |
|||
except UserError as e: |
|||
logging.warning("Log = " + str(e)) |
|||
self.assertTrue(u'Everything seems properly set up!' in str(e)) |
|||
|
|||
def test_create_dbsource_failed(self): |
|||
"""source creation without connection string should failed.""" |
|||
dbsource = self.env.ref('base_external_dbsource.demo_postgre') |
|||
|
|||
# Connection without connection_string |
|||
dbsource.conn_string = "" |
|||
try: |
|||
dbsource.connection_test() |
|||
except UserError as e: |
|||
logging.warning("Log = " + str(e)) |
|||
self.assertTrue(u'Here is what we got instead:' in str(e)) |
|||
|
|||
def test_create_dbsource_without_connector_failed(self): |
|||
"""source creation with other connector should failed.""" |
|||
dbsource = self.env.ref('base_external_dbsource.demo_postgre') |
|||
|
|||
# Connection to mysql |
|||
try: |
|||
dbsource.connector = "mysql" |
|||
dbsource.connection_test() |
|||
except ValueError as e: |
|||
logging.warning("Log = " + str(e)) |
|||
self.assertTrue(u'Wrong value for' in str(e)) |
|||
|
|||
# Connection to mysql |
|||
try: |
|||
dbsource.connector = "pyodbc" |
|||
dbsource.connection_test() |
|||
except ValueError as e: |
|||
logging.warning("Log = " + str(e)) |
|||
self.assertTrue(u'Wrong value for' in str(e)) |
|||
|
|||
# Connection to oracle |
|||
try: |
|||
dbsource.connector = "cx_Oracle" |
|||
dbsource.connection_test() |
|||
except ValueError as e: |
|||
logging.warning("Log = " + str(e)) |
|||
self.assertTrue(u'Wrong value for' in str(e)) |
|||
|
|||
# Connection to firebird |
|||
try: |
|||
dbsource.connector = "fdb" |
|||
dbsource.connection_test() |
|||
except Exception as e: |
|||
logging.warning("Log = " + str(e)) |
|||
self.assertTrue(u'Wrong value for' in str(e)) |
@ -0,0 +1,77 @@ |
|||
.. image:: https://img.shields.io/badge/licence-LGPL--3-blue.svg |
|||
:target: http://www.gnu.org/licenses/lgpl-3.0-standalone.html |
|||
:alt: License: LGPL-3 |
|||
|
|||
=================================== |
|||
External Database Source - Firebird |
|||
=================================== |
|||
|
|||
This module extends ``base_external_dbsource``, allowing you to connect to |
|||
foreign Firebird databases. |
|||
|
|||
Installation |
|||
============ |
|||
|
|||
* Install ``fdb`` python library |
|||
|
|||
Configuration |
|||
============= |
|||
|
|||
Database sources can be configured in Settings > Configuration -> Data sources. |
|||
|
|||
|
|||
Usage |
|||
===== |
|||
|
|||
To use this module: |
|||
|
|||
* Go to Settings > Database Structure > Database Sources |
|||
* Click on Create to enter the following information: |
|||
|
|||
* Datasource name |
|||
* Pasword |
|||
* Connector: Choose the database to which you want to connect |
|||
* Connection string : Specify how to connect to database |
|||
|
|||
.. image:: https://odoo-community.org/website/image/ir.attachment/5784_f2813bd/datas |
|||
:alt: Try me on Runbot |
|||
:target: https://runbot.odoo-community.org/runbot/149/10.0 for server-tools |
|||
|
|||
Known issues / Roadmap |
|||
====================== |
|||
|
|||
* Setting ``metadata`` to ``True`` in ``execute_fdb`` will do nothing. |
|||
* ``execute`` is susceptible to SQL injection. |
|||
|
|||
Bug Tracker |
|||
=========== |
|||
|
|||
Bugs are tracked on `GitHub Issues <https://github.com/OCA/server-tools/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. |
|||
|
|||
Credits |
|||
======= |
|||
|
|||
Contributors |
|||
------------ |
|||
|
|||
* Daniel Reis <dreis.pt@hotmail.com> |
|||
* Maxime Chambreuil <maxime.chambreuil@savoirfairelinux.com> |
|||
* Gervais Naoussi <gervaisnaoussi@gmail.com> |
|||
* Dave Lasley <dave@laslabs.com> |
|||
|
|||
Maintainer |
|||
---------- |
|||
|
|||
.. image:: https://odoo-community.org/logo.png |
|||
:alt: Odoo Community Association |
|||
:target: https://odoo-community.org |
|||
|
|||
This module is maintained by the OCA. |
|||
|
|||
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. |
|||
|
|||
To contribute to this module, please visit http://odoo-community.org. |
@ -0,0 +1,2 @@ |
|||
# -*- coding: utf-8 -*- |
|||
from . import models |
@ -0,0 +1,28 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# Copyright <2011> <Daniel Reis, Maxime Chambreuil, Savoir-faire Linux> |
|||
# Copyright 2016 LasLabs Inc. |
|||
# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl). |
|||
{ |
|||
'name': 'External Database Source - Firebird', |
|||
'version': '10.0.1.0.0', |
|||
'category': 'Tools', |
|||
'author': "Daniel Reis, " |
|||
"LasLabs, " |
|||
"Odoo Community Association (OCA)", |
|||
'website': 'https://github.com/OCA/server-tools', |
|||
'license': 'LGPL-3', |
|||
'depends': [ |
|||
'base_external_dbsource', |
|||
], |
|||
# Uncomment this for v11 |
|||
# 'external_dependencies': { |
|||
# 'python': [ |
|||
# 'fdb', |
|||
# ] |
|||
# }, |
|||
'demo': [ |
|||
'demo/base_external_dbsource.xml', |
|||
], |
|||
'installable': True, |
|||
'auto_install': True, # Remove this key for v11 |
|||
} |
@ -0,0 +1,9 @@ |
|||
<?xml version="1.0"?> |
|||
<odoo> |
|||
<record model="base.external.dbsource" id="demo_firebird"> |
|||
<field name="name">Firebird Demo</field> |
|||
<field name="conn_string">User=firebird;Database=SampleDatabase.fdb;DataSource=localhost;Port=3050;Dialect=3;Charset=NONE;Role=;Connection lifetime=15;Pooling=true;MinPoolSize=0;MaxPoolSize=50;Packet Size=8192;ServerType=0;</field> |
|||
<field name="password">password</field> |
|||
<field name="connector">fdb</field> |
|||
</record> |
|||
</odoo> |
@ -0,0 +1,3 @@ |
|||
# -*- coding: utf-8 -*- |
|||
|
|||
from . import base_external_dbsource |
@ -0,0 +1,56 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# Copyright 2011 Daniel Reis |
|||
# Copyright 2016 LasLabs Inc. |
|||
# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl.html). |
|||
|
|||
import logging |
|||
|
|||
from odoo import api |
|||
from odoo import models |
|||
|
|||
_logger = logging.getLogger(__name__) |
|||
|
|||
try: |
|||
from odoo.addons.base_external_dbsource.models import ( |
|||
base_external_dbsource, |
|||
) |
|||
CONNECTORS = base_external_dbsource.BaseExternalDbsource.CONNECTORS |
|||
try: |
|||
import fdb |
|||
CONNECTORS.append(('fdb', 'Firebird')) |
|||
except: |
|||
_logger.info('Firebird library not available. Please install "fdb" ' |
|||
'python package.') |
|||
except ImportError: |
|||
_logger.info('base_external_dbsource Odoo module not found.') |
|||
|
|||
|
|||
class BaseExternalDbsource(models.Model): |
|||
""" It provides logic for connection to an Firebird data source. """ |
|||
|
|||
_inherit = "base.external.dbsource" |
|||
|
|||
PWD_STRING_FDB = 'Password=%s;' |
|||
|
|||
@api.multi |
|||
def connection_close_fdb(self, connection): |
|||
return connection.close() |
|||
|
|||
@api.multi |
|||
def connection_open_fdb(self): |
|||
kwargs = {} |
|||
for option in self.conn_string_full.split(';'): |
|||
try: |
|||
key, value = option.split('=') |
|||
except ValueError: |
|||
continue |
|||
kwargs[key.lower()] = value |
|||
return fdb.connect(**kwargs) |
|||
|
|||
@api.multi |
|||
def execute_fdb(self, sqlquery, sqlparams, metadata): |
|||
with self.connection_open_fdb() as conn: |
|||
cur = conn.cursor() |
|||
cur.execute(sqlquery % sqlparams) |
|||
rows = cur.fetchall() |
|||
return rows, [] |
@ -0,0 +1,3 @@ |
|||
# -*- encoding: utf-8 -*- |
|||
|
|||
from . import test_base_external_dbsource |
@ -0,0 +1,41 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# Copyright 2016 LasLabs Inc. |
|||
|
|||
import mock |
|||
|
|||
from odoo.tests import common |
|||
|
|||
|
|||
ADAPTER = ('odoo.addons.base_external_dbsource_firebird.models' |
|||
'.base_external_dbsource.fdb') |
|||
|
|||
|
|||
class TestBaseExternalDbsource(common.TransactionCase): |
|||
|
|||
def setUp(self): |
|||
super(TestBaseExternalDbsource, self).setUp() |
|||
self.dbsource = self.env.ref( |
|||
'base_external_dbsource_firebird.demo_firebird', |
|||
) |
|||
|
|||
def test_connection_close_fdb(self): |
|||
""" It should close the connection """ |
|||
connection = mock.MagicMock() |
|||
res = self.dbsource.connection_close_fdb(connection) |
|||
self.assertEqual(res, connection.close()) |
|||
|
|||
@mock.patch(ADAPTER) |
|||
def test_connection_open_fdb(self, fdb): |
|||
""" It should open the connection with the split conn string """ |
|||
self.dbsource.conn_string = 'User=User;' |
|||
self.dbsource.connection_open_fdb() |
|||
fdb.connect.assert_called_once_with(**{ |
|||
'user': 'User', |
|||
'password': 'password', |
|||
}) |
|||
|
|||
@mock.patch(ADAPTER) |
|||
def test_connection_open_fdb_return(self, fdb): |
|||
""" It should return the newly opened connection """ |
|||
res = self.dbsource.connection_open_fdb() |
|||
self.assertEqual(res, fdb.connect()) |
@ -0,0 +1,80 @@ |
|||
.. image:: https://img.shields.io/badge/licence-LGPL--3-blue.svg |
|||
:target: http://www.gnu.org/licenses/lgpl-3.0-standalone.html |
|||
:alt: License: LGPL-3 |
|||
|
|||
================================ |
|||
External Database Source - MSSQL |
|||
================================ |
|||
|
|||
This module extends ``base_external_dbsource``, allowing you to connect to |
|||
foreign MSSQL databases using SQLAlchemy. |
|||
|
|||
|
|||
|
|||
Installation |
|||
============ |
|||
|
|||
* Install & configure FreeTDS driver (tdsodbc package) |
|||
* Install ``sqlalchemy`` and ``pymssql`` python libraries |
|||
* Install ``base_external_dbsource_sqlite`` Odoo module |
|||
|
|||
Configuration |
|||
============= |
|||
|
|||
Database sources can be configured in Settings > Configuration -> Data sources. |
|||
|
|||
|
|||
Usage |
|||
===== |
|||
|
|||
To use this module: |
|||
|
|||
* Go to Settings > Database Structure > Database Sources |
|||
* Click on Create to enter the following information: |
|||
|
|||
* Datasource name |
|||
* Pasword |
|||
* Connector: Choose the database to which you want to connect |
|||
* Connection string: Specify how to connect to database |
|||
|
|||
.. image:: https://odoo-community.org/website/image/ir.attachment/5784_f2813bd/datas |
|||
:alt: Try me on Runbot |
|||
:target: https://runbot.odoo-community.org/runbot/149/10.0 for server-tools |
|||
|
|||
Known issues / Roadmap |
|||
====================== |
|||
|
|||
* Add X.509 authentication |
|||
|
|||
Bug Tracker |
|||
=========== |
|||
|
|||
Bugs are tracked on `GitHub Issues <https://github.com/OCA/server-tools/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. |
|||
|
|||
Credits |
|||
======= |
|||
|
|||
Contributors |
|||
------------ |
|||
|
|||
* Daniel Reis <dreis.pt@hotmail.com> |
|||
* Maxime Chambreuil <maxime.chambreuil@savoirfairelinux.com> |
|||
* Gervais Naoussi <gervaisnaoussi@gmail.com> |
|||
* Dave Lasley <dave@laslabs.com> |
|||
|
|||
Maintainer |
|||
---------- |
|||
|
|||
.. image:: https://odoo-community.org/logo.png |
|||
:alt: Odoo Community Association |
|||
:target: https://odoo-community.org |
|||
|
|||
This module is maintained by the OCA. |
|||
|
|||
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. |
|||
|
|||
To contribute to this module, please visit http://odoo-community.org. |
@ -0,0 +1,2 @@ |
|||
# -*- coding: utf-8 -*- |
|||
from . import models |
@ -0,0 +1,29 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# Copyright <2011> <Daniel Reis, Maxime Chambreuil, Savoir-faire Linux> |
|||
# Copyright 2016 LasLabs Inc. |
|||
# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl). |
|||
{ |
|||
'name': 'External Database Source - MSSQL', |
|||
'version': '10.0.1.0.0', |
|||
'category': 'Tools', |
|||
'author': "Daniel Reis, " |
|||
"LasLabs, " |
|||
"Odoo Community Association (OCA)", |
|||
'website': 'https://github.com/OCA/server-tools', |
|||
'license': 'LGPL-3', |
|||
'depends': [ |
|||
'base_external_dbsource_sqlite', |
|||
], |
|||
# Uncomment this for v11 |
|||
# 'external_dependencies': [ |
|||
# 'python': [ |
|||
# 'sqlalchemy', |
|||
# 'pymssql', |
|||
# ] |
|||
# ], |
|||
'demo': [ |
|||
'demo/base_external_dbsource.xml', |
|||
], |
|||
'installable': True, |
|||
'auto_install': True, # Remove this key for v11 |
|||
} |
@ -0,0 +1,9 @@ |
|||
<?xml version="1.0"?> |
|||
<odoo> |
|||
<record model="base.external.dbsource" id="demo_mssql"> |
|||
<field name="name">MSSQL Demo</field> |
|||
<field name="conn_string">Server=myServerAddress;Database=myDataBase;User Id=myUsername;</field> |
|||
<field name="password">password</field> |
|||
<field name="connector">mssql</field> |
|||
</record> |
|||
</odoo> |
@ -0,0 +1,3 @@ |
|||
# -*- coding: utf-8 -*- |
|||
|
|||
from . import base_external_dbsource |
@ -0,0 +1,45 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# Copyright 2011 Daniel Reis |
|||
# Copyright 2016 LasLabs Inc. |
|||
# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl.html). |
|||
|
|||
import logging |
|||
|
|||
from odoo import api, models |
|||
|
|||
_logger = logging.getLogger(__name__) |
|||
|
|||
try: |
|||
from odoo.addons.base_external_dbsource.models import ( |
|||
base_external_dbsource, |
|||
) |
|||
CONNECTORS = base_external_dbsource.BaseExternalDbsource.CONNECTORS |
|||
try: |
|||
import pymssql |
|||
CONNECTORS.append(('mssql', 'Microsoft SQL Server')) |
|||
assert pymssql |
|||
except (ImportError, AssertionError): |
|||
_logger.info('MS SQL Server not available. Please install "pymssql" ' |
|||
'python package.') |
|||
except ImportError: |
|||
_logger.info('base_external_dbsource Odoo module not found.') |
|||
|
|||
|
|||
class BaseExternalDbsource(models.Model): |
|||
""" It provides logic for connection to a MSSQL data source. """ |
|||
|
|||
_inherit = "base.external.dbsource" |
|||
|
|||
PWD_STRING_MSSQL = 'Password=%s;' |
|||
|
|||
@api.multi |
|||
def connection_close_mssql(self, connection): |
|||
return connection.close() |
|||
|
|||
@api.multi |
|||
def connection_open_mssql(self): |
|||
return self._connection_open_sqlalchemy() |
|||
|
|||
@api.multi |
|||
def execute_mssql(self, sqlquery, sqlparams, metadata): |
|||
return self._execute_sqlalchemy(sqlquery, sqlparams, metadata) |
@ -0,0 +1,3 @@ |
|||
# -*- encoding: utf-8 -*- |
|||
|
|||
from . import test_base_external_dbsource |
@ -0,0 +1,42 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# Copyright 2016 LasLabs Inc. |
|||
|
|||
import mock |
|||
|
|||
from odoo.tests import common |
|||
|
|||
|
|||
ADAPTER = ('odoo.addons.base_external_dbsource_mssql.models' |
|||
'.base_external_dbsource.pymssql') |
|||
|
|||
|
|||
class TestBaseExternalDbsource(common.TransactionCase): |
|||
|
|||
def setUp(self): |
|||
super(TestBaseExternalDbsource, self).setUp() |
|||
self.dbsource = self.env.ref( |
|||
'base_external_dbsource_mssql.demo_mssql', |
|||
) |
|||
|
|||
def test_connection_close_mssql(self): |
|||
""" It should close the connection """ |
|||
connection = mock.MagicMock() |
|||
res = self.dbsource.connection_close_mssql(connection) |
|||
self.assertEqual(res, connection.close()) |
|||
|
|||
def test_connection_open_mssql(self): |
|||
""" It should call SQLAlchemy open """ |
|||
with mock.patch.object( |
|||
self.dbsource, '_connection_open_sqlalchemy' |
|||
) as parent_method: |
|||
self.dbsource.connection_open_mssql() |
|||
parent_method.assert_called_once_with() |
|||
|
|||
def test_excecute_mssql(self): |
|||
""" It should pass args to SQLAlchemy execute """ |
|||
expect = 'sqlquery', 'sqlparams', 'metadata' |
|||
with mock.patch.object( |
|||
self.dbsource, '_execute_sqlalchemy' |
|||
) as parent_method: |
|||
self.dbsource.execute_mssql(*expect) |
|||
parent_method.assert_called_once_with(*expect) |
@ -0,0 +1,80 @@ |
|||
.. image:: https://img.shields.io/badge/licence-LGPL--3-blue.svg |
|||
:target: http://www.gnu.org/licenses/lgpl-3.0-standalone.html |
|||
:alt: License: LGPL-3 |
|||
|
|||
================================ |
|||
External Database Source - MySQL |
|||
================================ |
|||
|
|||
This module extends ``base_external_dbsource``, allowing you to connect to |
|||
foreign MySQL databases using SQLAlchemy. |
|||
|
|||
|
|||
|
|||
Installation |
|||
============ |
|||
|
|||
* Install ``sqlalchemy`` and ``MySQLdb`` python libraries |
|||
* Install ``base_external_dbsource_sqlite`` Odoo module |
|||
|
|||
Configuration |
|||
============= |
|||
|
|||
Database sources can be configured in Settings > Configuration -> Data sources. |
|||
|
|||
|
|||
Usage |
|||
===== |
|||
|
|||
To use this module: |
|||
|
|||
* Go to Settings > Database Structure > Database Sources |
|||
* Click on Create to enter the following information: |
|||
|
|||
* Datasource name |
|||
* Pasword |
|||
* Connector: Choose the database to which you want to connect |
|||
* Connection string: Specify how to connect to database |
|||
|
|||
.. image:: https://odoo-community.org/website/image/ir.attachment/5784_f2813bd/datas |
|||
:alt: Try me on Runbot |
|||
:target: https://runbot.odoo-community.org/runbot/149/10.0 for server-tools |
|||
|
|||
Known issues / Roadmap |
|||
====================== |
|||
|
|||
|
|||
* Add X.509 authentication |
|||
|
|||
Bug Tracker |
|||
=========== |
|||
|
|||
Bugs are tracked on `GitHub Issues <https://github.com/OCA/server-tools/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. |
|||
|
|||
Credits |
|||
======= |
|||
|
|||
Contributors |
|||
------------ |
|||
|
|||
* Daniel Reis <dreis.pt@hotmail.com> |
|||
* Maxime Chambreuil <maxime.chambreuil@savoirfairelinux.com> |
|||
* Gervais Naoussi <gervaisnaoussi@gmail.com> |
|||
* Dave Lasley <dave@laslabs.com> |
|||
|
|||
Maintainer |
|||
---------- |
|||
|
|||
.. image:: https://odoo-community.org/logo.png |
|||
:alt: Odoo Community Association |
|||
:target: https://odoo-community.org |
|||
|
|||
This module is maintained by the OCA. |
|||
|
|||
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. |
|||
|
|||
To contribute to this module, please visit http://odoo-community.org. |
@ -0,0 +1,2 @@ |
|||
# -*- coding: utf-8 -*- |
|||
from . import models |
@ -0,0 +1,29 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# Copyright <2011> <Daniel Reis, Maxime Chambreuil, Savoir-faire Linux> |
|||
# Copyright 2016 LasLabs Inc. |
|||
# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl). |
|||
{ |
|||
'name': 'External Database Source - MySQL', |
|||
'version': '10.0.1.0.0', |
|||
'category': 'Tools', |
|||
'author': "Daniel Reis, " |
|||
"LasLabs, " |
|||
"Odoo Community Association (OCA)", |
|||
'website': 'https://github.com/OCA/server-tools', |
|||
'license': 'LGPL-3', |
|||
'depends': [ |
|||
'base_external_dbsource_sqlite', |
|||
], |
|||
# Uncomment this for v11 |
|||
# 'external_dependencies': [ |
|||
# 'python': [ |
|||
# 'sqlalchemy', |
|||
# 'MySQLdb', |
|||
# ] |
|||
# ], |
|||
'demo': [ |
|||
'demo/base_external_dbsource.xml', |
|||
], |
|||
'installable': True, |
|||
'auto_install': True, # Remove this key for v11 |
|||
} |
@ -0,0 +1,9 @@ |
|||
<?xml version="1.0"?> |
|||
<odoo> |
|||
<record model="base.external.dbsource" id="demo_mysql"> |
|||
<field name="name">MySQL Demo</field> |
|||
<field name="conn_string">Server=myServerAddress;Database=myDataBase;Uid=myUsername;</field> |
|||
<field name="password">password</field> |
|||
<field name="connector">mysql</field> |
|||
</record> |
|||
</odoo> |
@ -0,0 +1,3 @@ |
|||
# -*- coding: utf-8 -*- |
|||
|
|||
from . import base_external_dbsource |
@ -0,0 +1,43 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# Copyright 2011 Daniel Reis |
|||
# Copyright 2016 LasLabs Inc. |
|||
# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl.html). |
|||
|
|||
import logging |
|||
|
|||
from odoo import api, models |
|||
|
|||
_logger = logging.getLogger(__name__) |
|||
|
|||
try: |
|||
from odoo.addons.base_external_dbsource.models import ( |
|||
base_external_dbsource, |
|||
) |
|||
CONNECTORS = base_external_dbsource.BaseExternalDbsource.CONNECTORS |
|||
try: |
|||
import MySQLdb |
|||
CONNECTORS.append(('mysql', 'MySQL')) |
|||
assert MySQLdb |
|||
except (ImportError, AssertionError): |
|||
_logger.info('MySQL not available. Please install "mysqldb" ' |
|||
'python package.') |
|||
except ImportError: |
|||
_logger.info('base_external_dbsource Odoo module not found.') |
|||
|
|||
|
|||
class BaseExternalDbsource(models.Model): |
|||
""" It provides logic for connection to a MySQL data source. """ |
|||
|
|||
_inherit = "base.external.dbsource" |
|||
|
|||
@api.multi |
|||
def connection_close_mysql(self, connection): |
|||
return connection.close() |
|||
|
|||
@api.multi |
|||
def connection_open_mysql(self): |
|||
return self._connection_open_sqlalchemy() |
|||
|
|||
@api.multi |
|||
def execute_mysql(self, sqlquery, sqlparams, metadata): |
|||
return self._execute_sqlalchemy(sqlquery, sqlparams, metadata) |
@ -0,0 +1,3 @@ |
|||
# -*- encoding: utf-8 -*- |
|||
|
|||
from . import test_base_external_dbsource |
@ -0,0 +1,42 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# Copyright 2016 LasLabs Inc. |
|||
|
|||
import mock |
|||
|
|||
from odoo.tests import common |
|||
|
|||
|
|||
ADAPTER = ('odoo.addons.base_external_dbsource_mysql.models' |
|||
'.base_external_dbsource.MySQLdb') |
|||
|
|||
|
|||
class TestBaseExternalDbsource(common.TransactionCase): |
|||
|
|||
def setUp(self): |
|||
super(TestBaseExternalDbsource, self).setUp() |
|||
self.dbsource = self.env.ref( |
|||
'base_external_dbsource_mysql.demo_mysql', |
|||
) |
|||
|
|||
def test_connection_close_mysql(self): |
|||
""" It should close the connection """ |
|||
connection = mock.MagicMock() |
|||
res = self.dbsource.connection_close_mysql(connection) |
|||
self.assertEqual(res, connection.close()) |
|||
|
|||
def test_connection_open_mysql(self): |
|||
""" It should call SQLAlchemy open """ |
|||
with mock.patch.object( |
|||
self.dbsource, '_connection_open_sqlalchemy' |
|||
) as parent_method: |
|||
self.dbsource.connection_open_mysql() |
|||
parent_method.assert_called_once_with() |
|||
|
|||
def test_excecute_mysql(self): |
|||
""" It should pass args to SQLAlchemy execute """ |
|||
expect = 'sqlquery', 'sqlparams', 'metadata' |
|||
with mock.patch.object( |
|||
self.dbsource, '_execute_sqlalchemy' |
|||
) as parent_method: |
|||
self.dbsource.execute_mysql(*expect) |
|||
parent_method.assert_called_once_with(*expect) |
@ -0,0 +1,79 @@ |
|||
.. image:: https://img.shields.io/badge/licence-LGPL--3-blue.svg |
|||
:target: http://www.gnu.org/licenses/lgpl-3.0-standalone.html |
|||
:alt: License: LGPL-3 |
|||
|
|||
=============================== |
|||
External Database Source - ODBC |
|||
=============================== |
|||
|
|||
This module extends ``base_external_dbsource``, allowing you to connect to |
|||
foreign ODBC databases using PyODBC. |
|||
|
|||
|
|||
|
|||
Installation |
|||
============ |
|||
|
|||
* Install ``unixodbc`` and ``python-pyodbc`` packages |
|||
* Install ``base_external_dbsource_sqlite`` Odoo module |
|||
|
|||
Configuration |
|||
============= |
|||
|
|||
Database sources can be configured in Settings > Configuration -> Data sources. |
|||
|
|||
|
|||
Usage |
|||
===== |
|||
|
|||
To use this module: |
|||
|
|||
* Go to Settings > Database Structure > Database Sources |
|||
* Click on Create to enter the following information: |
|||
|
|||
* Datasource name |
|||
* Pasword |
|||
* Connector: Choose the database to which you want to connect |
|||
* Connection string: Specify how to connect to database |
|||
|
|||
.. image:: https://odoo-community.org/website/image/ir.attachment/5784_f2813bd/datas |
|||
:alt: Try me on Runbot |
|||
:target: https://runbot.odoo-community.org/runbot/149/10.0 for server-tools |
|||
|
|||
Known issues / Roadmap |
|||
====================== |
|||
|
|||
* Add X.509 authentication |
|||
|
|||
Bug Tracker |
|||
=========== |
|||
|
|||
Bugs are tracked on `GitHub Issues <https://github.com/OCA/server-tools/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. |
|||
|
|||
Credits |
|||
======= |
|||
|
|||
Contributors |
|||
------------ |
|||
|
|||
* Daniel Reis <dreis.pt@hotmail.com> |
|||
* Maxime Chambreuil <maxime.chambreuil@savoirfairelinux.com> |
|||
* Gervais Naoussi <gervaisnaoussi@gmail.com> |
|||
* Dave Lasley <dave@laslabs.com> |
|||
|
|||
Maintainer |
|||
---------- |
|||
|
|||
.. image:: https://odoo-community.org/logo.png |
|||
:alt: Odoo Community Association |
|||
:target: https://odoo-community.org |
|||
|
|||
This module is maintained by the OCA. |
|||
|
|||
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. |
|||
|
|||
To contribute to this module, please visit http://odoo-community.org. |
@ -0,0 +1,2 @@ |
|||
# -*- coding: utf-8 -*- |
|||
from . import models |
@ -0,0 +1,29 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# Copyright <2011> <Daniel Reis, Maxime Chambreuil, Savoir-faire Linux> |
|||
# Copyright 2016 LasLabs Inc. |
|||
# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl). |
|||
{ |
|||
'name': 'External Database Source - ODBC', |
|||
'version': '10.0.1.0.0', |
|||
'category': 'Tools', |
|||
'author': "Daniel Reis, " |
|||
"LasLabs, " |
|||
"Odoo Community Association (OCA)", |
|||
'website': 'https://github.com/OCA/server-tools', |
|||
'license': 'LGPL-3', |
|||
'depends': [ |
|||
'base_external_dbsource_sqlite', |
|||
], |
|||
# Uncomment this for v11 |
|||
# 'external_dependencies': [ |
|||
# 'python': [ |
|||
# 'sqlalchemy', |
|||
# 'pyodbc', |
|||
# ] |
|||
# ], |
|||
'demo': [ |
|||
'demo/base_external_dbsource.xml', |
|||
], |
|||
'installable': True, |
|||
'auto_install': True, # Remove this key for v11 |
|||
} |
@ -0,0 +1,9 @@ |
|||
<?xml version="1.0"?> |
|||
<odoo> |
|||
<record model="base.external.dbsource" id="demo_odbc"> |
|||
<field name="name">ODBC Demo</field> |
|||
<field name="conn_string">Driver='ODBC Driver 11 for SQL Server';Server=myServerAddress;Database=myDataBase;Uid=myUsername;</field> |
|||
<field name="password">password</field> |
|||
<field name="connector">pyodbc</field> |
|||
</record> |
|||
</odoo> |
@ -0,0 +1,3 @@ |
|||
# -*- coding: utf-8 -*- |
|||
|
|||
from . import base_external_dbsource |
@ -0,0 +1,42 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# Copyright 2011 Daniel Reis |
|||
# Copyright 2016 LasLabs Inc. |
|||
# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl.html). |
|||
|
|||
import logging |
|||
|
|||
from odoo import api, models |
|||
|
|||
_logger = logging.getLogger(__name__) |
|||
|
|||
try: |
|||
from odoo.addons.base_external_dbsource.models import ( |
|||
base_external_dbsource, |
|||
) |
|||
CONNECTORS = base_external_dbsource.BaseExternalDbsource.CONNECTORS |
|||
try: |
|||
import pyodbc |
|||
CONNECTORS.append(('pyodbc', 'ODBC')) |
|||
except ImportError: |
|||
_logger.info('ODBC libraries not available. Please install ' |
|||
'"unixodbc" and "python-pyodbc" packages.') |
|||
except ImportError: |
|||
_logger.info('base_external_dbsource Odoo module not found.') |
|||
|
|||
|
|||
class BaseExternalDbsource(models.Model): |
|||
""" It provides logic for connection to a ODBC data source. """ |
|||
|
|||
_inherit = "base.external.dbsource" |
|||
|
|||
@api.multi |
|||
def connection_close_pyodbc(self, connection): |
|||
return connection.close() |
|||
|
|||
@api.multi |
|||
def connection_open_pyodbc(self): |
|||
return pyodbc.connect(self.conn_string_full) |
|||
|
|||
@api.multi |
|||
def execute_pyodbc(self, sqlquery, sqlparams, metadata): |
|||
return self._execute_generic(sqlquery, sqlparams, metadata) |
@ -0,0 +1,3 @@ |
|||
# -*- encoding: utf-8 -*- |
|||
|
|||
from . import test_base_external_dbsource |
@ -0,0 +1,46 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# Copyright 2016 LasLabs Inc. |
|||
|
|||
import mock |
|||
|
|||
from odoo.tests import common |
|||
|
|||
|
|||
ADAPTER = ('odoo.addons.base_external_dbsource_odbc.models' |
|||
'.base_external_dbsource.pyodbc') |
|||
|
|||
|
|||
class TestBaseExternalDbsource(common.TransactionCase): |
|||
|
|||
def setUp(self): |
|||
super(TestBaseExternalDbsource, self).setUp() |
|||
self.dbsource = self.env.ref( |
|||
'base_external_dbsource_odbc.demo_odbc', |
|||
) |
|||
|
|||
def test_connection_close_pyodbc(self): |
|||
""" It should close the connection """ |
|||
connection = mock.MagicMock() |
|||
res = self.dbsource.connection_close_pyodbc(connection) |
|||
self.assertEqual(res, connection.close()) |
|||
|
|||
@mock.patch(ADAPTER) |
|||
def test_connection_open_pyodbc(self, pyodbc): |
|||
""" It should open the connection with the full conn string """ |
|||
self.dbsource.connection_open_pyodbc() |
|||
pyodbc.connect.assert_called_once_with( |
|||
self.dbsource.conn_string_full, |
|||
) |
|||
|
|||
@mock.patch(ADAPTER) |
|||
def test_connection_open_pyodbc_return(self, pyodbc): |
|||
""" It should return the newly opened connection """ |
|||
res = self.dbsource.connection_open_pyodbc() |
|||
self.assertEqual(res, pyodbc.connect()) |
|||
|
|||
def test_execute_pyodbc(self): |
|||
""" It should call the generic execute method w/ proper args """ |
|||
expect = 'sqlquery', 'sqlparams', 'metadata' |
|||
with mock.patch.object(self.dbsource, '_execute_generic') as execute: |
|||
self.dbsource.execute_pyodbc(*expect) |
|||
execute.assert_called_once_with(*expect) |
@ -0,0 +1,79 @@ |
|||
.. image:: https://img.shields.io/badge/licence-LGPL--3-blue.svg |
|||
:target: http://www.gnu.org/licenses/lgpl-3.0-standalone.html |
|||
:alt: License: LGPL-3 |
|||
|
|||
================================= |
|||
External Database Source - Oracle |
|||
================================= |
|||
|
|||
This module extends ``base_external_dbsource``, allowing you to connect to |
|||
foreign Oracle databases. |
|||
|
|||
Installation |
|||
============ |
|||
|
|||
* Install and configure Oracle Instant Client |
|||
* Install ``cx_Oracle`` python library |
|||
|
|||
Configuration |
|||
============= |
|||
|
|||
Database sources can be configured in Settings > Configuration -> Data sources. |
|||
|
|||
|
|||
Usage |
|||
===== |
|||
|
|||
To use this module: |
|||
|
|||
* Go to Settings > Database Structure > Database Sources |
|||
* Click on Create to enter the following information: |
|||
|
|||
* Datasource name |
|||
* Pasword |
|||
* Connector: Choose the database to which you want to connect |
|||
* Connection string: Specify how to connect to database |
|||
|
|||
.. image:: https://odoo-community.org/website/image/ir.attachment/5784_f2813bd/datas |
|||
:alt: Try me on Runbot |
|||
:target: https://runbot.odoo-community.org/runbot/149/10.0 for server-tools |
|||
|
|||
Known issues / Roadmap |
|||
====================== |
|||
|
|||
* Find a way to test (cx_Oracle requires InstantClient, which Travis doesn't have) |
|||
* Add X.509 authentication |
|||
|
|||
|
|||
Bug Tracker |
|||
=========== |
|||
|
|||
Bugs are tracked on `GitHub Issues <https://github.com/OCA/server-tools/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. |
|||
|
|||
Credits |
|||
======= |
|||
|
|||
Contributors |
|||
------------ |
|||
|
|||
* Daniel Reis <dreis.pt@hotmail.com> |
|||
* Maxime Chambreuil <maxime.chambreuil@savoirfairelinux.com> |
|||
* Gervais Naoussi <gervaisnaoussi@gmail.com> |
|||
* Dave Lasley <dave@laslabs.com> |
|||
|
|||
Maintainer |
|||
---------- |
|||
|
|||
.. image:: https://odoo-community.org/logo.png |
|||
:alt: Odoo Community Association |
|||
:target: https://odoo-community.org |
|||
|
|||
This module is maintained by the OCA. |
|||
|
|||
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. |
|||
|
|||
To contribute to this module, please visit http://odoo-community.org. |
@ -0,0 +1,2 @@ |
|||
# -*- coding: utf-8 -*- |
|||
from . import models |
@ -0,0 +1,25 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# Copyright <2011> <Daniel Reis, Maxime Chambreuil, Savoir-faire Linux> |
|||
# Copyright 2016 LasLabs Inc. |
|||
# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl). |
|||
{ |
|||
'name': 'External Database Source - Oracle', |
|||
'version': '10.0.1.0.0', |
|||
'category': 'Tools', |
|||
'author': "Daniel Reis, " |
|||
"LasLabs, " |
|||
"Odoo Community Association (OCA)", |
|||
'website': 'https://github.com/OCA/server-tools', |
|||
'license': 'LGPL-3', |
|||
'depends': [ |
|||
'base_external_dbsource', |
|||
], |
|||
# Uncomment this for v11 |
|||
# 'external_dependencies': [ |
|||
# 'python': [ |
|||
# 'cx_Oracle', |
|||
# ] |
|||
# ], |
|||
'installable': True, |
|||
'auto_install': True, # Remove this key for v11 |
|||
} |
@ -0,0 +1,3 @@ |
|||
# -*- coding: utf-8 -*- |
|||
|
|||
from . import base_external_dbsource |
@ -0,0 +1,47 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# Copyright 2011 Daniel Reis |
|||
# Copyright 2016 LasLabs Inc. |
|||
# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl.html). |
|||
|
|||
import logging |
|||
import os |
|||
|
|||
from odoo import api |
|||
from odoo import models |
|||
|
|||
_logger = logging.getLogger(__name__) |
|||
|
|||
try: |
|||
from odoo.addons.base_external_dbsource.models import ( |
|||
base_external_dbsource, |
|||
) |
|||
CONNECTORS = base_external_dbsource.BaseExternalDbsource.CONNECTORS |
|||
try: |
|||
import cx_Oracle |
|||
CONNECTORS.append(('cx_Oracle', 'Oracle')) |
|||
except ImportError: |
|||
_logger.info('Oracle libraries not available. Please install ' |
|||
'"cx_Oracle" python package.') |
|||
except ImportError: |
|||
_logger.info('base_external_dbsource Odoo module not found.') |
|||
|
|||
|
|||
class BaseExternalDbsource(models.Model): |
|||
""" It provides logic for connection to an Oracle data source. """ |
|||
|
|||
_inherit = "base.external.dbsource" |
|||
|
|||
PWD_STRING_CX_ORACLE = 'Password=%s;' |
|||
|
|||
@api.multi |
|||
def connection_close_cx_Oracle(self, connection): |
|||
return connection.close() |
|||
|
|||
@api.multi |
|||
def connection_open_cx_Oracle(self): |
|||
os.environ['NLS_LANG'] = 'AMERICAN_AMERICA.UTF8' |
|||
return cx_Oracle.connect(self.conn_string_full) |
|||
|
|||
@api.multi |
|||
def execute_cx_Oracle(self, sqlquery, sqlparams, metadata): |
|||
return self._execute_generic(sqlquery, sqlparams, metadata) |
@ -0,0 +1,77 @@ |
|||
.. image:: https://img.shields.io/badge/licence-LGPL--3-blue.svg |
|||
:target: http://www.gnu.org/licenses/lgpl-3.0-standalone.html |
|||
:alt: License: LGPL-3 |
|||
|
|||
================================= |
|||
External Database Source - SQLite |
|||
================================= |
|||
|
|||
This module extends ``base_external_dbsource``, allowing you to connect to |
|||
foreign SQLite databases using SQLAlchemy. |
|||
|
|||
|
|||
Installation |
|||
============ |
|||
|
|||
* Install ``sqlalchemy`` python library |
|||
|
|||
Configuration |
|||
============= |
|||
|
|||
Database sources can be configured in Settings > Configuration -> Data sources. |
|||
|
|||
|
|||
Usage |
|||
===== |
|||
|
|||
To use this module: |
|||
|
|||
* Go to Settings > Database Structure > Database Sources |
|||
* Click on Create to enter the following information: |
|||
|
|||
* Datasource name |
|||
* Pasword |
|||
* Connector: Choose the database to which you want to connect |
|||
* Connection string: Specify how to connect to database |
|||
|
|||
.. image:: https://odoo-community.org/website/image/ir.attachment/5784_f2813bd/datas |
|||
:alt: Try me on Runbot |
|||
:target: https://runbot.odoo-community.org/runbot/149/10.0 for server-tools |
|||
|
|||
Known issues / Roadmap |
|||
====================== |
|||
|
|||
* Add X.509 authentication |
|||
|
|||
Bug Tracker |
|||
=========== |
|||
|
|||
Bugs are tracked on `GitHub Issues <https://github.com/OCA/server-tools/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. |
|||
|
|||
Credits |
|||
======= |
|||
|
|||
Contributors |
|||
------------ |
|||
|
|||
* Daniel Reis <dreis.pt@hotmail.com> |
|||
* Maxime Chambreuil <maxime.chambreuil@savoirfairelinux.com> |
|||
* Gervais Naoussi <gervaisnaoussi@gmail.com> |
|||
* Dave Lasley <dave@laslabs.com> |
|||
|
|||
Maintainer |
|||
---------- |
|||
|
|||
.. image:: https://odoo-community.org/logo.png |
|||
:alt: Odoo Community Association |
|||
:target: https://odoo-community.org |
|||
|
|||
This module is maintained by the OCA. |
|||
|
|||
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. |
|||
|
|||
To contribute to this module, please visit http://odoo-community.org. |
@ -0,0 +1,2 @@ |
|||
# -*- coding: utf-8 -*- |
|||
from . import models |
@ -0,0 +1,28 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# Copyright <2011> <Daniel Reis, Maxime Chambreuil, Savoir-faire Linux> |
|||
# Copyright 2016 LasLabs Inc. |
|||
# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl). |
|||
{ |
|||
'name': 'External Database Source - SQLite', |
|||
'version': '10.0.1.0.0', |
|||
'category': 'Tools', |
|||
'author': "Daniel Reis, " |
|||
"LasLabs, " |
|||
"Odoo Community Association (OCA)", |
|||
'website': 'https://github.com/OCA/server-tools', |
|||
'license': 'LGPL-3', |
|||
'depends': [ |
|||
'base_external_dbsource', |
|||
], |
|||
# Uncomment this for v11 |
|||
# 'external_dependencies': [ |
|||
# 'python': [ |
|||
# 'sqlalchemy', |
|||
# ] |
|||
# ], |
|||
'demo': [ |
|||
'demo/base_external_dbsource.xml', |
|||
], |
|||
'installable': True, |
|||
'auto_install': True, # Remove this key for v11 |
|||
} |
@ -0,0 +1,9 @@ |
|||
<?xml version="1.0"?> |
|||
<odoo> |
|||
<record model="base.external.dbsource" id="demo_sqlite"> |
|||
<field name="name">SQLite Demo</field> |
|||
<field name="conn_string">Data Source=:memory:;Version=3;New=True;</field> |
|||
<field name="password">password</field> |
|||
<field name="connector">sqlite</field> |
|||
</record> |
|||
</odoo> |
@ -0,0 +1,3 @@ |
|||
# -*- coding: utf-8 -*- |
|||
|
|||
from . import base_external_dbsource |
@ -0,0 +1,59 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# Copyright 2011 Daniel Reis |
|||
# Copyright 2016 LasLabs Inc. |
|||
# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl.html). |
|||
|
|||
import logging |
|||
|
|||
from odoo import api, models |
|||
|
|||
_logger = logging.getLogger(__name__) |
|||
|
|||
try: |
|||
from odoo.addons.base_external_dbsource.models import ( |
|||
base_external_dbsource, |
|||
) |
|||
CONNECTORS = base_external_dbsource.BaseExternalDbsource.CONNECTORS |
|||
try: |
|||
import sqlalchemy |
|||
CONNECTORS.append(('sqlite', 'SQLite')) |
|||
except ImportError: |
|||
_logger.info('SQLAlchemy library not available. Please ' |
|||
'install "sqlalchemy" python package.') |
|||
except ImportError: |
|||
_logger.info('base_external_dbsource Odoo module not found.') |
|||
|
|||
|
|||
class BaseExternalDbsource(models.Model): |
|||
""" It provides logic for connection to a SQLite data source. """ |
|||
|
|||
_inherit = "base.external.dbsource" |
|||
|
|||
PWD_STRING_SQLITE = 'Password=%s;' |
|||
|
|||
@api.multi |
|||
def connection_close_sqlite(self, connection): |
|||
return connection.close() |
|||
|
|||
@api.multi |
|||
def connection_open_sqlite(self): |
|||
return self._connection_open_sqlalchemy() |
|||
|
|||
@api.multi |
|||
def execute_sqlite(self, sqlquery, sqlparams, metadata): |
|||
return self._execute_sqlalchemy(sqlquery, sqlparams, metadata) |
|||
|
|||
@api.multi |
|||
def _connection_open_sqlalchemy(self): |
|||
return sqlalchemy.create_engine(self.conn_string_full).connect() |
|||
|
|||
@api.multi |
|||
def _execute_sqlalchemy(self, sqlquery, sqlparams, metadata): |
|||
rows, cols = list(), list() |
|||
for record in self: |
|||
with record.connection_open() as connection: |
|||
cur = connection.execute(sqlquery, sqlparams) |
|||
if metadata: |
|||
cols = cur.keys() |
|||
rows = [r for r in cur] |
|||
return rows, cols |
@ -0,0 +1,3 @@ |
|||
# -*- encoding: utf-8 -*- |
|||
|
|||
from . import test_base_external_dbsource |
@ -0,0 +1,42 @@ |
|||
# -*- coding: utf-8 -*- |
|||
# Copyright 2016 LasLabs Inc. |
|||
|
|||
import mock |
|||
|
|||
from odoo.tests import common |
|||
|
|||
|
|||
ADAPTER = ('odoo.addons.base_external_dbsource_sqlite.models' |
|||
'.base_external_dbsource.sqlalchemy') |
|||
|
|||
|
|||
class TestBaseExternalDbsource(common.TransactionCase): |
|||
|
|||
def setUp(self): |
|||
super(TestBaseExternalDbsource, self).setUp() |
|||
self.dbsource = self.env.ref( |
|||
'base_external_dbsource_sqlite.demo_sqlite', |
|||
) |
|||
|
|||
def test_connection_close_sqlite(self): |
|||
""" It should close the connection """ |
|||
connection = mock.MagicMock() |
|||
res = self.dbsource.connection_close_sqlite(connection) |
|||
self.assertEqual(res, connection.close()) |
|||
|
|||
def test_connection_open_sqlite(self): |
|||
""" It should call SQLAlchemy open """ |
|||
with mock.patch.object( |
|||
self.dbsource, '_connection_open_sqlalchemy' |
|||
) as parent_method: |
|||
self.dbsource.connection_open_sqlite() |
|||
parent_method.assert_called_once_with() |
|||
|
|||
def test_excecute_sqlite(self): |
|||
""" It should pass args to SQLAlchemy execute """ |
|||
expect = 'sqlquery', 'sqlparams', 'metadata' |
|||
with mock.patch.object( |
|||
self.dbsource, '_execute_sqlalchemy' |
|||
) as parent_method: |
|||
self.dbsource.execute_sqlite(*expect) |
|||
parent_method.assert_called_once_with(*expect) |
Write
Preview
Loading…
Cancel
Save
Reference in new issue