Browse Source
[ADD] auto_backup: Test coverage
[ADD] auto_backup: Test coverage
* compute_name * check_folder * action_sftp_test_connection * action_backup - sftp * action_backup_all * sftp_connection * filenamepull/614/head
Dave Lasley
8 years ago
No known key found for this signature in database
GPG Key ID: 7DDBA4BA81B934CF
8 changed files with 273 additions and 94 deletions
-
3auto_backup/__openerp__.py
-
28auto_backup/data/backup_data.yml
-
18auto_backup/data/ir_cron.xml
-
19auto_backup/data/mail_message_subtype.xml
-
2auto_backup/models/db_backup.py
-
2auto_backup/tests/__init__.py
-
63auto_backup/tests/test_auto_backup.py
-
232auto_backup/tests/test_db_backup.py
@ -1,28 +0,0 @@ |
|||||
# -*- coding: utf-8 -*- |
|
||||
# © 2016 Grupo ESOC Ingeniería de Servicios, S.L.U. - Jairo Llopis |
|
||||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html). |
|
||||
|
|
||||
# Cron job |
|
||||
- !record {model: ir.cron, id: ir_cron_backupscheduler0}: |
|
||||
name: Backup scheduler |
|
||||
user_id: base.user_root |
|
||||
interval_number: 1 |
|
||||
interval_type: days |
|
||||
numbercall: -1 |
|
||||
nextcall: !eval |
|
||||
(datetime.now() + timedelta(days=1)).strftime("%Y-%m-%d 02:00:00") |
|
||||
model: db.backup |
|
||||
function: action_backup_all |
|
||||
|
|
||||
# New message subtypes |
|
||||
- !record {model: mail.message.subtype, id: success}: |
|
||||
name: Backup successful |
|
||||
res_model: db.backup |
|
||||
default: False |
|
||||
description: Database backup succeeded. |
|
||||
|
|
||||
- !record {model: mail.message.subtype, id: failure}: |
|
||||
name: Backup failed |
|
||||
res_model: db.backup |
|
||||
default: True |
|
||||
description: Database backup failed. |
|
@ -0,0 +1,18 @@ |
|||||
|
<?xml version="1.0" encoding="UTF-8"?> |
||||
|
|
||||
|
<odoo noupdate="1"> |
||||
|
|
||||
|
<record id="ir_cron_backup_scheduler_0" model="ir.cron"> |
||||
|
<field name="name">Backup Scheduler</field> |
||||
|
<field name="user_id" ref="base.user_root" /> |
||||
|
<field name="interval_number">1</field> |
||||
|
<field name="interval_type">days</field> |
||||
|
<field name="numbercall">-1</field> |
||||
|
<field name="nextcall" |
||||
|
eval="(datetime.now() + timedelta(days=1)).strftime('%Y-%m-%d 02:00:00')" |
||||
|
/> |
||||
|
<field name="model">db.backup</field> |
||||
|
<field name="function">action_backup_all</field> |
||||
|
</record> |
||||
|
|
||||
|
</odoo> |
@ -0,0 +1,19 @@ |
|||||
|
<?xml version="1.0" encoding="UTF-8"?> |
||||
|
|
||||
|
<odoo noupdate="1"> |
||||
|
|
||||
|
<record id="mail_message_subtype_success" model="mail.message.subtype"> |
||||
|
<field name="name">Backup Successful</field> |
||||
|
<field name="description">Database backup succeeded.</field> |
||||
|
<field name="res_model">db.backup</field> |
||||
|
<field name="default" eval="False" /> |
||||
|
</record> |
||||
|
|
||||
|
<record id="mail_message_subtype_failure" model="mail.message.subtype"> |
||||
|
<field name="name">Backup Failed</field> |
||||
|
<field name="description">Database backup failed.</field> |
||||
|
<field name="res_model">db.backup</field> |
||||
|
<field name="default" eval="True" /> |
||||
|
</record> |
||||
|
|
||||
|
</odoo> |
@ -1,63 +0,0 @@ |
|||||
# -*- coding: utf-8 -*- |
|
||||
# © 2015 Agile Business Group <http://www.agilebg.com> |
|
||||
# © 2015 Alessio Gerace <alesiso.gerace@agilebg.com> |
|
||||
# © 2016 Grupo ESOC Ingeniería de Servicios, S.L.U. - Jairo Llopis |
|
||||
# Copyright 2016 LasLabs Inc. |
|
||||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html). |
|
||||
|
|
||||
import os |
|
||||
from datetime import datetime |
|
||||
from openerp.tests import common |
|
||||
from openerp import exceptions, tools |
|
||||
|
|
||||
|
|
||||
class TestsAutoBackup(common.TransactionCase): |
|
||||
|
|
||||
def setUp(self): |
|
||||
super(TestsAutoBackup, self).setUp() |
|
||||
|
|
||||
def new_record(self, method='sftp'): |
|
||||
vals = { |
|
||||
'name': u'Têst backup', |
|
||||
'method': method, |
|
||||
} |
|
||||
if method == 'sftp': |
|
||||
vals.update({ |
|
||||
'sftp_host': 'test_host', |
|
||||
'sftp_port': '222', |
|
||||
'sftp_user': 'tuser', |
|
||||
}) |
|
||||
self.vals = vals |
|
||||
self.env["db.backup"].create(vals) |
|
||||
|
|
||||
def test_local(self): |
|
||||
"""A local database is backed up.""" |
|
||||
rec_id = self.new_record('local') |
|
||||
filename = rec_id.filename(datetime.now()) |
|
||||
rec_id.action_backup() |
|
||||
generated_backup = [f for f in os.listdir(rec_id.folder) |
|
||||
if f >= filename] |
|
||||
self.assertEqual(1, len(generated_backup)) |
|
||||
|
|
||||
def test_compute_name_sftp(self): |
|
||||
""" It should create proper SFTP URI """ |
|
||||
rec_id = self.new_record() |
|
||||
self.assertEqual( |
|
||||
'sftp://%(user)@%(host):%(port)%(folder)' % { |
|
||||
'user': self.vals['sftp_user'], |
|
||||
'host': self.vals['sftp_host'], |
|
||||
'port': self.vals['sftp_port'], |
|
||||
'folder': self.vals['folder'], |
|
||||
}, |
|
||||
rec_id.name, |
|
||||
) |
|
||||
|
|
||||
def test_check_folder(self): |
|
||||
""" It should not allow recursive backups """ |
|
||||
rec_id = self.new_record() |
|
||||
with self.assertRaises(exceptions.ValidationError): |
|
||||
rec_id.write({ |
|
||||
'folder': '%s/another/path' % tools.config.filestore( |
|
||||
self.env.cr.dbname |
|
||||
), |
|
||||
}) |
|
@ -0,0 +1,232 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
# © 2015 Agile Business Group <http://www.agilebg.com> |
||||
|
# © 2015 Alessio Gerace <alesiso.gerace@agilebg.com> |
||||
|
# © 2016 Grupo ESOC Ingeniería de Servicios, S.L.U. - Jairo Llopis |
||||
|
# Copyright 2016 LasLabs Inc. |
||||
|
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html). |
||||
|
|
||||
|
import os |
||||
|
import mock |
||||
|
|
||||
|
from datetime import datetime |
||||
|
from contextlib import contextmanager |
||||
|
|
||||
|
from openerp.tests import common |
||||
|
from openerp import exceptions, tools |
||||
|
|
||||
|
try: |
||||
|
import pysftp |
||||
|
except ImportError: |
||||
|
pass |
||||
|
|
||||
|
|
||||
|
model = 'openerp.addons.auto_backup.models.db_backup' |
||||
|
|
||||
|
|
||||
|
class TestConnectionException(pysftp.ConnectionException): |
||||
|
def __init__(self): |
||||
|
super(TestConnectionException, self).__init__('test', 'test') |
||||
|
|
||||
|
|
||||
|
class TestDbBackup(common.TransactionCase): |
||||
|
|
||||
|
def setUp(self): |
||||
|
super(TestDbBackup, self).setUp() |
||||
|
self.Model = self.env["db.backup"] |
||||
|
|
||||
|
@contextmanager |
||||
|
def mock_assets(self): |
||||
|
""" It provides mocked core assets """ |
||||
|
self.path_join_val = '/this/is/a/path' |
||||
|
with mock.patch('%s.db' % model) as db: |
||||
|
with mock.patch('%s.os' % model) as os: |
||||
|
with mock.patch('%s.shutil' % model) as shutil: |
||||
|
os.path.join.return_value = self.path_join_val |
||||
|
yield { |
||||
|
'db': db, |
||||
|
'os': os, |
||||
|
'shutil': shutil, |
||||
|
} |
||||
|
|
||||
|
@contextmanager |
||||
|
def patch_filtered_sftp(self, record, mocks=None): |
||||
|
""" It patches filtered record and provides a mock """ |
||||
|
if mocks is None: |
||||
|
mocks = ['sftp_connection'] |
||||
|
mocks = {m: mock.DEFAULT for m in mocks} |
||||
|
with mock.patch.object(record, 'filtered') as filtered: |
||||
|
with mock.patch.object(record, 'backup_log'): |
||||
|
with mock.patch.multiple(record, **mocks): |
||||
|
filtered.side_effect = [], [record] |
||||
|
yield filtered |
||||
|
|
||||
|
def new_record(self, method='sftp'): |
||||
|
vals = { |
||||
|
'name': u'Têst backup', |
||||
|
'method': method, |
||||
|
} |
||||
|
if method == 'sftp': |
||||
|
vals.update({ |
||||
|
'sftp_host': 'test_host', |
||||
|
'sftp_port': '222', |
||||
|
'sftp_user': 'tuser', |
||||
|
'sftp_password': 'password', |
||||
|
'folder': '/folder/', |
||||
|
}) |
||||
|
self.vals = vals |
||||
|
return self.Model.create(vals) |
||||
|
|
||||
|
def test_compute_name_sftp(self): |
||||
|
""" It should create proper SFTP URI """ |
||||
|
rec_id = self.new_record() |
||||
|
self.assertEqual( |
||||
|
'sftp://%(user)s@%(host)s:%(port)s%(folder)s' % { |
||||
|
'user': self.vals['sftp_user'], |
||||
|
'host': self.vals['sftp_host'], |
||||
|
'port': self.vals['sftp_port'], |
||||
|
'folder': self.vals['folder'], |
||||
|
}, |
||||
|
rec_id.name, |
||||
|
) |
||||
|
|
||||
|
def test_check_folder(self): |
||||
|
""" It should not allow recursive backups """ |
||||
|
rec_id = self.new_record('local') |
||||
|
with self.assertRaises(exceptions.ValidationError): |
||||
|
rec_id.write({ |
||||
|
'folder': '%s/another/path' % tools.config.filestore( |
||||
|
self.env.cr.dbname |
||||
|
), |
||||
|
}) |
||||
|
|
||||
|
@mock.patch('%s._' % model) |
||||
|
def test_action_sftp_test_connection_success(self, _): |
||||
|
""" It should raise connection succeeded warning """ |
||||
|
rec_id = self.new_record() |
||||
|
with mock.patch.object(rec_id, 'sftp_connection'): |
||||
|
with self.assertRaises(exceptions.Warning): |
||||
|
rec_id.action_sftp_test_connection() |
||||
|
_.assert_called_once_with("Connection Test Succeeded!") |
||||
|
|
||||
|
@mock.patch('%s._' % model) |
||||
|
def test_action_sftp_test_connection_fail(self, _): |
||||
|
""" It should raise connection fail warning """ |
||||
|
rec_id = self.new_record() |
||||
|
with mock.patch.object(rec_id, 'sftp_connection') as conn: |
||||
|
conn().__enter__.side_effect = TestConnectionException |
||||
|
with self.assertRaises(exceptions.Warning): |
||||
|
rec_id.action_sftp_test_connection() |
||||
|
_.assert_called_once_with("Connection Test Failed!") |
||||
|
|
||||
|
def test_action_backup_local(self): |
||||
|
""" It should backup local database """ |
||||
|
rec_id = self.new_record('local') |
||||
|
filename = rec_id.filename(datetime.now()) |
||||
|
rec_id.action_backup() |
||||
|
generated_backup = [f for f in os.listdir(rec_id.folder) |
||||
|
if f >= filename] |
||||
|
self.assertEqual(1, len(generated_backup)) |
||||
|
|
||||
|
def test_action_backup_sftp_mkdirs(self): |
||||
|
""" It should create remote dirs """ |
||||
|
rec_id = self.new_record() |
||||
|
with self.mock_assets(): |
||||
|
with self.patch_filtered_sftp(rec_id): |
||||
|
conn = rec_id.sftp_connection().__enter__() |
||||
|
rec_id.action_backup() |
||||
|
conn.makedirs.assert_called_once_with(rec_id.folder) |
||||
|
|
||||
|
def test_action_backup_sftp_mkdirs_conn_exception(self): |
||||
|
""" It should guard from ConnectionException on remote.mkdirs """ |
||||
|
rec_id = self.new_record() |
||||
|
with self.mock_assets(): |
||||
|
with self.patch_filtered_sftp(rec_id): |
||||
|
conn = rec_id.sftp_connection().__enter__() |
||||
|
conn.makedirs.side_effect = TestConnectionException |
||||
|
rec_id.action_backup() |
||||
|
# No error was raised, test pass |
||||
|
self.assertTrue(True) |
||||
|
|
||||
|
def test_action_backup_sftp_remote_open(self): |
||||
|
""" It should open remote file w/ proper args """ |
||||
|
rec_id = self.new_record() |
||||
|
with self.mock_assets() as assets: |
||||
|
with self.patch_filtered_sftp(rec_id): |
||||
|
conn = rec_id.sftp_connection().__enter__() |
||||
|
rec_id.action_backup() |
||||
|
conn.open.assert_called_once_with( |
||||
|
assets['os'].path.join(), |
||||
|
'wb' |
||||
|
) |
||||
|
|
||||
|
def test_action_backup_sftp_remote_open(self): |
||||
|
""" It should open remote file w/ proper args """ |
||||
|
rec_id = self.new_record() |
||||
|
with self.mock_assets() as assets: |
||||
|
with self.patch_filtered_sftp(rec_id): |
||||
|
conn = rec_id.sftp_connection().__enter__() |
||||
|
rec_id.action_backup() |
||||
|
conn.open.assert_called_once_with( |
||||
|
assets['os'].path.join(), |
||||
|
'wb' |
||||
|
) |
||||
|
|
||||
|
def test_action_backup_all_search(self): |
||||
|
""" It should search all records """ |
||||
|
rec_id = self.new_record() |
||||
|
with mock.patch.object(rec_id, 'search'): |
||||
|
rec_id.action_backup_all() |
||||
|
rec_id.search.assert_called_once_with([]) |
||||
|
|
||||
|
def test_action_backup_all_return(self): |
||||
|
""" It should return result of backup operation """ |
||||
|
rec_id = self.new_record() |
||||
|
with mock.patch.object(rec_id, 'search'): |
||||
|
res = rec_id.action_backup_all() |
||||
|
self.assertEqual( |
||||
|
rec_id.search().action_backup(), res |
||||
|
) |
||||
|
|
||||
|
@mock.patch('%s.pysftp' % model) |
||||
|
def test_sftp_connection_init_passwd(self, pysftp): |
||||
|
""" It should initiate SFTP connection w/ proper args and pass """ |
||||
|
rec_id = self.new_record() |
||||
|
rec_id.sftp_connection() |
||||
|
pysftp.Connection.assert_called_once_with( |
||||
|
host=rec_id.sftp_host, |
||||
|
username=rec_id.sftp_user, |
||||
|
port=rec_id.sftp_port, |
||||
|
password=rec_id.sftp_password, |
||||
|
) |
||||
|
|
||||
|
@mock.patch('%s.pysftp' % model) |
||||
|
def test_sftp_connection_init_key(self, pysftp): |
||||
|
""" It should initiate SFTP connection w/ proper args and key """ |
||||
|
rec_id = self.new_record() |
||||
|
rec_id.write({ |
||||
|
'sftp_private_key': 'pkey', |
||||
|
'sftp_password': 'pkeypass', |
||||
|
}) |
||||
|
rec_id.sftp_connection() |
||||
|
pysftp.Connection.assert_called_once_with( |
||||
|
host=rec_id.sftp_host, |
||||
|
username=rec_id.sftp_user, |
||||
|
port=rec_id.sftp_port, |
||||
|
private_key=rec_id.sftp_private_key, |
||||
|
private_key_pass=rec_id.sftp_password, |
||||
|
) |
||||
|
|
||||
|
@mock.patch('%s.pysftp' % model) |
||||
|
def test_sftp_connection_return(self, pysftp): |
||||
|
""" It should return new sftp connection """ |
||||
|
rec_id = self.new_record() |
||||
|
res = rec_id.sftp_connection() |
||||
|
self.assertEqual( |
||||
|
pysftp.Connection(), res, |
||||
|
) |
||||
|
|
||||
|
def test_filename(self): |
||||
|
""" It should not error and should return a .dump.zip file str """ |
||||
|
now = datetime.now() |
||||
|
res = self.Model.filename(now) |
||||
|
self.assertTrue(res.endswith(".dump.zip")) |
Write
Preview
Loading…
Cancel
Save
Reference in new issue