Skip to content

Commit

Permalink
add restoredb command
Browse files Browse the repository at this point in the history
  • Loading branch information
lmignon committed Mar 3, 2020
1 parent a3a6e61 commit 82f8529
Show file tree
Hide file tree
Showing 5 changed files with 232 additions and 3 deletions.
6 changes: 6 additions & 0 deletions CHANGES.rst
Original file line number Diff line number Diff line change
@@ -1,6 +1,12 @@
Changes
~~~~~~~

unreleased
----------

- add click-odoo-restoredb


1.8.0 (2019-10-01)
------------------
- Support Odoo SaaS versions
Expand Down
10 changes: 7 additions & 3 deletions click_odoo_contrib/backupdb.py
Original file line number Diff line number Diff line change
Expand Up @@ -15,13 +15,17 @@
from ._backup import backup
from ._dbutils import db_exists, db_management_enabled

MANIFEST_FILENAME = "manifest.json"
DBDUMP_FILENAME = "db.dump"
FILESTORE_DIRAME = "filestore"


def _dump_db(dbname, backup):
cmd = ["pg_dump", "--no-owner", dbname]
filename = "dump.sql"
if backup.format == "folder":
cmd.insert(-1, "--format=c")
filename = "db.dump"
filename = DBDUMP_FILENAME
_stdin, stdout = odoo.tools.exec_pg_command_pipe(*cmd)
backup.write(stdout, filename)

Expand All @@ -31,13 +35,13 @@ def _create_manifest(cr, dbname, backup):
with tempfile.NamedTemporaryFile(mode="w") as f:
json.dump(manifest, f, indent=4)
f.seek(0)
backup.addfile(f.name, "manifest.json")
backup.addfile(f.name, MANIFEST_FILENAME)


def _backup_filestore(dbname, backup):
filestore_source = odoo.tools.config.filestore(dbname)
if os.path.isdir(filestore_source):
backup.addtree(filestore_source, "filestore")
backup.addtree(filestore_source, FILESTORE_DIRAME)


@click.command()
Expand Down
111 changes: 111 additions & 0 deletions click_odoo_contrib/restoredb.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,111 @@
#!/usr/bin/env python
# -*- coding: utf-8 -*-
# Copyright 2019 ACSONE SA/NV (<http://acsone.eu>)
# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl.html).

import os
import shutil

import click
import click_odoo
import psycopg2
from click_odoo import OdooEnvironment, odoo

from ._dbutils import db_exists, db_management_enabled, reset_config_parameters
from .backupdb import DBDUMP_FILENAME, FILESTORE_DIRAME, MANIFEST_FILENAME


def _restore_from_folder(dbname, backup, copy=True, jobs=1):
manifest_file_path = os.path.join(backup, MANIFEST_FILENAME)
dbdump_file_path = os.path.join(backup, DBDUMP_FILENAME)
filestore_dir_path = os.path.join(backup, FILESTORE_DIRAME)
if not os.path.exists(manifest_file_path) or not os.path.exists(dbdump_file_path):
msg = (
"{} is not folder backup created by the backupdb command. "
"{} and {} files are missing.".format(
backup, MANIFEST_FILENAME, DBDUMP_FILENAME
)
)
raise click.ClickException(msg)

odoo.service.db._create_empty_database(dbname)
pg_args = ["--jobs", str(jobs), "--dbname", dbname, "--no-owner", dbdump_file_path]
if odoo.tools.exec_pg_command("pg_restore", *pg_args):
raise click.ClickException("Couldn't restore database")
if copy:
# if it's a copy of a database, force generation of a new dbuuid
reset_config_parameters(dbname)
with OdooEnvironment(dbname) as env:
if os.path.exists(filestore_dir_path):
filestore_dest = env["ir.attachment"]._filestore()
shutil.move(filestore_dir_path, filestore_dest)

if odoo.tools.config["unaccent"]:
try:
with env.cr.savepoint():
env.cr.execute("CREATE EXTENSION unaccent")
except psycopg2.Error:
pass
odoo.sql_db.close_db(dbname)


def _restore_from_file(dbname, backup, copy=True):
with db_management_enabled(), open(backup, "rb") as backup_file:
odoo.service.db.restore_db(dbname, backup_file, copy)
odoo.sql_db.close_db(dbname)


@click.command()
@click_odoo.env_options(
default_log_level="warn", with_database=False, with_rollback=False
)
@click.option(
"--copy/--move",
default=True,
help="This database is a copy.\nIn order "
"to avoid conflicts between databases, Odoo needs to know if this"
"database was moved or copied. If you don't know, set is a copy.",
)
@click.option(
"--force",
is_flag=True,
show_default=True,
help="Don't report error if destination database already exists. If "
"force and destination database exists, it will be dropped before "
"restore.",
)
@click.option(
"--jobs",
help="Uses this many parallel jobs to restore. (Only used to restore"
"folder backup)",
type=int,
default=1,
)
@click.argument("dbname", nargs=1)
@click.argument(
"backup",
nargs=1,
type=click.Path(
exists=True, file_okay=True, dir_okay=True, readable=True, resolve_path=True
),
)
def main(env, dbname, backup, copy, force, jobs):
""" Restore an Odoo database backup.
This script allows you to restore databses created by using the Odoo
web interface or the backupdb script. This
avoids timeout and file size limitation problems when
databases are too large.
"""
if db_exists(dbname):
msg = "Destination database already exists: {}".format(dbname)
if not force:
raise click.ClickException(msg)
msg = "{} -> drop".format(msg)
click.echo(click.style(msg, fg="yellow"))
with db_management_enabled():
odoo.service.db.exp_drop(dbname)
if os.path.isfile(backup):
_restore_from_file(dbname, backup, copy)
else:
_restore_from_folder(dbname, backup, copy, jobs)
1 change: 1 addition & 0 deletions setup.py
Original file line number Diff line number Diff line change
Expand Up @@ -44,6 +44,7 @@
click-odoo-dropdb=click_odoo_contrib.dropdb:main
click-odoo-initdb=click_odoo_contrib.initdb:main
click-odoo-backupdb=click_odoo_contrib.backupdb:main
click-odoo-restoredb=click_odoo_contrib.restoredb:main
click-odoo-makepot=click_odoo_contrib.makepot:main
""",
)
107 changes: 107 additions & 0 deletions tests/test_restoredb.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,107 @@
# Copyright 2018 ACSONE SA/NV (<http://acsone.eu>)
# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl.html).

import operator
import os
import shutil
import subprocess
from collections import defaultdict

import click_odoo
import pytest
from click.testing import CliRunner
from click_odoo import odoo

from click_odoo_contrib._dbutils import db_exists
from click_odoo_contrib.backupdb import main as backupdb
from click_odoo_contrib.restoredb import main as restoredb

TEST_DBNAME = "click-odoo-contrib-testrestoredb"

_DEFAULT_IR_CONFIG_PARAMETERS = ["database.uuid", "database.create_date"]


def _createdb(dbname):
subprocess.check_call(["createdb", dbname])


def _dropdb(dbname):
subprocess.check_call(["dropdb", "--if-exists", dbname])


def _dropdb_odoo(dbname):
_dropdb(dbname)
filestore_dir = odoo.tools.config.filestore(dbname)
if os.path.isdir(filestore_dir):
shutil.rmtree(filestore_dir)


def _check_default_params(db1, db2, operator):
params_by_db = defaultdict(dict)
for db in (db1, db2):
with click_odoo.OdooEnvironment(database=db) as env:
IrConfigParameters = env["ir.config_parameter"]
for key in _DEFAULT_IR_CONFIG_PARAMETERS:
params_by_db[db][key] = IrConfigParameters.get_param(key)
params1 = params_by_db[db1]
params2 = params_by_db[db2]
assert set(params1.keys()) == set(params2.keys())
for k, v in params1.items():
assert operator(v, params2[k])


@pytest.fixture(params=["folder", "zip"])
def backup(request, odoodb, odoocfg, tmp_path):
if request.param == "folder":
name = "backup"
else:
name = "backup.zip"
path = tmp_path.joinpath(name)
posix_path = path.as_posix()
CliRunner().invoke(
backupdb, ["--format={}".format(request.param), odoodb, posix_path]
)
yield posix_path, odoodb


def test_db_restore(backup):
assert not db_exists(TEST_DBNAME)
backup_path, original_db = backup
try:
result = CliRunner().invoke(restoredb, [TEST_DBNAME, backup_path])
assert result.exit_code == 0
assert db_exists(TEST_DBNAME)
# default restore mode is copy -> default params are not preserved
_check_default_params(TEST_DBNAME, original_db, operator.ne)
finally:
_dropdb_odoo(TEST_DBNAME)


def test_db_restore_target_exists(backup):
_createdb(TEST_DBNAME)
backup_path, original_db = backup
try:
result = CliRunner().invoke(restoredb, [TEST_DBNAME, backup_path])
assert result.exit_code != 0, result.output
assert "Destination database already exists" in result.output
finally:
_dropdb_odoo(TEST_DBNAME)
try:
result = CliRunner().invoke(restoredb, ["--force", TEST_DBNAME, backup_path])
assert result.exit_code == 0
assert db_exists(TEST_DBNAME)
finally:
_dropdb_odoo(TEST_DBNAME)


def test_db_restore_move(backup):
assert not db_exists(TEST_DBNAME)
backup_path, original_db = backup
try:
result = CliRunner().invoke(restoredb, ["--move", TEST_DBNAME, backup_path])
assert result.exit_code == 0
assert db_exists(TEST_DBNAME)
# when database is moved, default params are preserved
_check_default_params(TEST_DBNAME, original_db, operator.eq)
finally:
_dropdb_odoo(TEST_DBNAME)

0 comments on commit 82f8529

Please sign in to comment.