增加自动备份模块
This commit is contained in:
328
jikimo_auto_backup/models/db_backup.py
Normal file
328
jikimo_auto_backup/models/db_backup.py
Normal file
@@ -0,0 +1,328 @@
|
||||
|
||||
import os
|
||||
import datetime
|
||||
import time
|
||||
import shutil
|
||||
import json
|
||||
import tempfile
|
||||
|
||||
from odoo import models, fields, api, tools, _
|
||||
from odoo.exceptions import Warning, AccessDenied
|
||||
import odoo
|
||||
|
||||
import logging
|
||||
_logger = logging.getLogger(__name__)
|
||||
|
||||
try:
|
||||
import paramiko
|
||||
except ImportError:
|
||||
raise ImportError(
|
||||
'This module needs paramiko to automatically write backups to the FTP through SFTP. '
|
||||
'Please install paramiko on your system. (sudo pip3 install paramiko)')
|
||||
|
||||
|
||||
class DbBackup(models.Model):
|
||||
_name = 'db.backup'
|
||||
_description = 'Backup configuration record'
|
||||
|
||||
def _get_db_name(self):
|
||||
dbName = self._cr.dbname
|
||||
return dbName
|
||||
|
||||
# Columns for local server configuration
|
||||
host = fields.Char('Host', required=True, default='localhost')
|
||||
port = fields.Char('Port', required=True, default=8069)
|
||||
name = fields.Char('Database', required=True, help='Database you want to schedule backups for',
|
||||
default=_get_db_name)
|
||||
folder = fields.Char('Backup Directory', help='Absolute path for storing the backups', required='True',
|
||||
default='/odoo/backups')
|
||||
backup_type = fields.Selection([('zip', 'Zip'), ('dump', 'Dump')], 'Backup Type', required=True, default='zip')
|
||||
autoremove = fields.Boolean('Auto. Remove Backups',
|
||||
help='If you check this option you can choose to automaticly remove the backup '
|
||||
'after xx days')
|
||||
days_to_keep = fields.Integer('Remove after x days',
|
||||
help="Choose after how many days the backup should be deleted. For example:\n"
|
||||
"If you fill in 5 the backups will be removed after 5 days.",
|
||||
required=True)
|
||||
|
||||
# Columns for external server (SFTP)
|
||||
sftp_write = fields.Boolean('Write to external server with sftp',
|
||||
help="If you check this option you can specify the details needed to write to a remote "
|
||||
"server with SFTP.")
|
||||
sftp_path = fields.Char('Path external server',
|
||||
help='The location to the folder where the dumps should be written to. For example '
|
||||
'/odoo/backups/.\nFiles will then be written to /odoo/backups/ on your remote server.')
|
||||
sftp_host = fields.Char('IP Address SFTP Server',
|
||||
help='The IP address from your remote server. For example 192.168.0.1')
|
||||
sftp_port = fields.Integer('SFTP Port', help='The port on the FTP server that accepts SSH/SFTP calls.', default=22)
|
||||
sftp_user = fields.Char('Username SFTP Server',
|
||||
help='The username where the SFTP connection should be made with. This is the user on the '
|
||||
'external server.')
|
||||
sftp_password = fields.Char('Password User SFTP Server',
|
||||
help='The password from the user where the SFTP connection should be made with. This '
|
||||
'is the password from the user on the external server.')
|
||||
days_to_keep_sftp = fields.Integer('Remove SFTP after x days',
|
||||
help='Choose after how many days the backup should be deleted from the FTP '
|
||||
'server. For example:\nIf you fill in 5 the backups will be removed after '
|
||||
'5 days from the FTP server.',
|
||||
default=30)
|
||||
send_mail_sftp_fail = fields.Boolean('Auto. E-mail on backup fail',
|
||||
help='If you check this option you can choose to automaticly get e-mailed '
|
||||
'when the backup to the external server failed.')
|
||||
email_to_notify = fields.Char('E-mail to notify',
|
||||
help='Fill in the e-mail where you want to be notified that the backup failed on '
|
||||
'the FTP.')
|
||||
|
||||
def test_sftp_connection(self, context=None):
|
||||
self.ensure_one()
|
||||
|
||||
# Check if there is a success or fail and write messages
|
||||
message_title = ""
|
||||
message_content = ""
|
||||
error = ""
|
||||
has_failed = False
|
||||
|
||||
for rec in self:
|
||||
ip_host = rec.sftp_host
|
||||
port_host = rec.sftp_port
|
||||
username_login = rec.sftp_user
|
||||
password_login = rec.sftp_password
|
||||
|
||||
# Connect with external server over SFTP, so we know sure that everything works.
|
||||
try:
|
||||
s = paramiko.SSHClient()
|
||||
s.set_missing_host_key_policy(paramiko.AutoAddPolicy())
|
||||
s.connect(ip_host, port_host, username_login, password_login, timeout=10)
|
||||
sftp = s.open_sftp()
|
||||
sftp.close()
|
||||
message_title = _("Connection Test Succeeded!\nEverything seems properly set up for FTP back-ups!")
|
||||
except Exception as e:
|
||||
_logger.critical('There was a problem connecting to the remote ftp: %s', str(e))
|
||||
error += str(e)
|
||||
has_failed = True
|
||||
message_title = _("Connection Test Failed!")
|
||||
if len(rec.sftp_host) < 8:
|
||||
message_content += "\nYour IP address seems to be too short.\n"
|
||||
message_content += _("Here is what we got instead:\n")
|
||||
finally:
|
||||
if s:
|
||||
s.close()
|
||||
|
||||
if has_failed:
|
||||
raise Warning(message_title + '\n\n' + message_content + "%s" % str(error))
|
||||
else:
|
||||
raise Warning(message_title + '\n\n' + message_content)
|
||||
|
||||
@api.model
|
||||
def schedule_backup(self):
|
||||
conf_ids = self.search([])
|
||||
for rec in conf_ids:
|
||||
|
||||
try:
|
||||
if not os.path.isdir(rec.folder):
|
||||
os.makedirs(rec.folder)
|
||||
except:
|
||||
raise
|
||||
# Create name for dumpfile.
|
||||
bkp_file = '%s_%s.%s' % (time.strftime('%Y_%m_%d_%H_%M_%S'), rec.name, rec.backup_type)
|
||||
file_path = os.path.join(rec.folder, bkp_file)
|
||||
try:
|
||||
# try to backup database and write it away
|
||||
fp = open(file_path, 'wb')
|
||||
self._take_dump(rec.name, fp, 'db.backup', rec.backup_type)
|
||||
fp.close()
|
||||
except Exception as error:
|
||||
_logger.debug(
|
||||
"Couldn't backup database %s. Bad database administrator password for server running at "
|
||||
"http://%s:%s" % (rec.name, rec.host, rec.port))
|
||||
_logger.debug("Exact error from the exception: %s", str(error))
|
||||
continue
|
||||
|
||||
# Check if user wants to write to SFTP or not.
|
||||
if rec.sftp_write is True:
|
||||
try:
|
||||
# Store all values in variables
|
||||
dir = rec.folder
|
||||
path_to_write_to = rec.sftp_path
|
||||
ip_host = rec.sftp_host
|
||||
port_host = rec.sftp_port
|
||||
username_login = rec.sftp_user
|
||||
password_login = rec.sftp_password
|
||||
_logger.debug('sftp remote path: %s', path_to_write_to)
|
||||
|
||||
try:
|
||||
s = paramiko.SSHClient()
|
||||
s.set_missing_host_key_policy(paramiko.AutoAddPolicy())
|
||||
s.connect(ip_host, port_host, username_login, password_login, timeout=20)
|
||||
sftp = s.open_sftp()
|
||||
except Exception as error:
|
||||
_logger.critical('Error connecting to remote server! Error: %s', str(error))
|
||||
|
||||
try:
|
||||
sftp.chdir(path_to_write_to)
|
||||
except IOError:
|
||||
# Create directory and subdirs if they do not exist.
|
||||
current_directory = ''
|
||||
for dirElement in path_to_write_to.split('/'):
|
||||
current_directory += dirElement + '/'
|
||||
try:
|
||||
sftp.chdir(current_directory)
|
||||
except:
|
||||
_logger.info('(Part of the) path didn\'t exist. Creating it now at %s',
|
||||
current_directory)
|
||||
# Make directory and then navigate into it
|
||||
sftp.mkdir(current_directory, 777)
|
||||
sftp.chdir(current_directory)
|
||||
pass
|
||||
sftp.chdir(path_to_write_to)
|
||||
# Loop over all files in the directory.
|
||||
for f in os.listdir(dir):
|
||||
if rec.name in f:
|
||||
fullpath = os.path.join(dir, f)
|
||||
if os.path.isfile(fullpath):
|
||||
try:
|
||||
sftp.stat(os.path.join(path_to_write_to, f))
|
||||
_logger.debug(
|
||||
'File %s already exists on the remote FTP Server ------ skipped', fullpath)
|
||||
# This means the file does not exist (remote) yet!
|
||||
except IOError:
|
||||
try:
|
||||
sftp.put(fullpath, os.path.join(path_to_write_to, f))
|
||||
_logger.info('Copying File % s------ success', fullpath)
|
||||
except Exception as err:
|
||||
_logger.critical(
|
||||
'We couldn\'t write the file to the remote server. Error: %s', str(err))
|
||||
|
||||
# Navigate in to the correct folder.
|
||||
sftp.chdir(path_to_write_to)
|
||||
|
||||
_logger.debug("Checking expired files")
|
||||
# Loop over all files in the directory from the back-ups.
|
||||
# We will check the creation date of every back-up.
|
||||
for file in sftp.listdir(path_to_write_to):
|
||||
if rec.name in file:
|
||||
# Get the full path
|
||||
fullpath = os.path.join(path_to_write_to, file)
|
||||
# Get the timestamp from the file on the external server
|
||||
timestamp = sftp.stat(fullpath).st_mtime
|
||||
createtime = datetime.datetime.fromtimestamp(timestamp)
|
||||
now = datetime.datetime.now()
|
||||
delta = now - createtime
|
||||
# If the file is older than the days_to_keep_sftp (the days to keep that the user filled in
|
||||
# on the Odoo form it will be removed.
|
||||
if delta.days >= rec.days_to_keep_sftp:
|
||||
# Only delete files, no directories!
|
||||
if ".dump" in file or '.zip' in file:
|
||||
_logger.info("Delete too old file from SFTP servers: %s", file)
|
||||
sftp.unlink(file)
|
||||
# Close the SFTP session.
|
||||
sftp.close()
|
||||
s.close()
|
||||
except Exception as e:
|
||||
try:
|
||||
sftp.close()
|
||||
s.close()
|
||||
except:
|
||||
pass
|
||||
_logger.error('Exception! We couldn\'t back up to the FTP server. Here is what we got back '
|
||||
'instead: %s', str(e))
|
||||
# At this point the SFTP backup failed. We will now check if the user wants
|
||||
# an e-mail notification about this.
|
||||
if rec.send_mail_sftp_fail:
|
||||
try:
|
||||
ir_mail_server = self.env['ir.mail_server'].search([], order='sequence asc', limit=1)
|
||||
message = "Dear,\n\nThe backup for the server " + rec.host + " (IP: " + rec.sftp_host + \
|
||||
") failed. Please check the following details:\n\nIP address SFTP server: " + \
|
||||
rec.sftp_host + "\nUsername: " + rec.sftp_user + \
|
||||
"\n\nError details: " + tools.ustr(e) + \
|
||||
"\n\nWith kind regards"
|
||||
catch_all_domain = self.env["ir.config_parameter"].sudo().get_param("mail.catchall.domain")
|
||||
response_mail = "auto_backup@%s" % catch_all_domain if catch_all_domain else self.env.user.partner_id.email
|
||||
msg = ir_mail_server.build_email(response_mail, [rec.email_to_notify],
|
||||
"Backup from " + rec.host + "(" + rec.sftp_host +
|
||||
") failed",
|
||||
message)
|
||||
ir_mail_server.send_email(msg)
|
||||
except Exception:
|
||||
pass
|
||||
|
||||
# Remove all old files (on local server) in case this is configured..
|
||||
if rec.autoremove:
|
||||
directory = rec.folder
|
||||
# Loop over all files in the directory.
|
||||
for f in os.listdir(directory):
|
||||
fullpath = os.path.join(directory, f)
|
||||
# Only delete the ones wich are from the current database
|
||||
# (Makes it possible to save different databases in the same folder)
|
||||
if rec.name in fullpath:
|
||||
timestamp = os.stat(fullpath).st_ctime
|
||||
createtime = datetime.datetime.fromtimestamp(timestamp)
|
||||
now = datetime.datetime.now()
|
||||
delta = now - createtime
|
||||
if delta.days >= rec.days_to_keep:
|
||||
# Only delete files (which are .dump and .zip), no directories.
|
||||
if os.path.isfile(fullpath) and (".dump" in f or '.zip' in f):
|
||||
_logger.info("Delete local out-of-date file: %s", fullpath)
|
||||
os.remove(fullpath)
|
||||
|
||||
# This is more or less the same as the default Odoo function at
|
||||
# https://github.com/odoo/odoo/blob/e649200ab44718b8faefc11c2f8a9d11f2db7753/odoo/service/db.py#L209
|
||||
# The main difference is that we do not do have a wrapper for the function check_db_management_enabled here and
|
||||
# that we authenticate based on the cron its user id and by checking if we have 'db.backup' defined in the function
|
||||
# call. Since this function is called from the cron and since we have these security checks on model and on user_id
|
||||
# its pretty impossible to hack any way to take a backup. This allows us to disable the Odoo database manager
|
||||
# which is a MUCH safer way
|
||||
def _take_dump(self, db_name, stream, model, backup_format='zip'):
|
||||
"""Dump database `db` into file-like object `stream` if stream is None
|
||||
return a file object with the dump """
|
||||
|
||||
cron_user_id = self.env.ref('jikimo_auto_backup.backup_scheduler').user_id.id
|
||||
if self._name != 'db.backup' or cron_user_id != self.env.user.id:
|
||||
_logger.error('Unauthorized database operation. Backups should only be available from the cron job.')
|
||||
raise AccessDenied()
|
||||
|
||||
_logger.info('DUMP DB: %s format %s', db_name, backup_format)
|
||||
|
||||
cmd = ['pg_dump', '--no-owner']
|
||||
cmd.append(db_name)
|
||||
|
||||
if backup_format == 'zip':
|
||||
with tempfile.TemporaryDirectory() as dump_dir:
|
||||
filestore = odoo.tools.config.filestore(db_name)
|
||||
if os.path.exists(filestore):
|
||||
shutil.copytree(filestore, os.path.join(dump_dir, 'filestore'))
|
||||
with open(os.path.join(dump_dir, 'manifest.json'), 'w') as fh:
|
||||
db = odoo.sql_db.db_connect(db_name)
|
||||
with db.cursor() as cr:
|
||||
json.dump(self._dump_db_manifest(cr), fh, indent=4)
|
||||
cmd.insert(-1, '--file=' + os.path.join(dump_dir, 'dump.sql'))
|
||||
odoo.tools.exec_pg_command(*cmd)
|
||||
if stream:
|
||||
odoo.tools.osutil.zip_dir(dump_dir, stream, include_dir=False, fnct_sort=lambda file_name: file_name != 'dump.sql')
|
||||
else:
|
||||
t=tempfile.TemporaryFile()
|
||||
odoo.tools.osutil.zip_dir(dump_dir, t, include_dir=False, fnct_sort=lambda file_name: file_name != 'dump.sql')
|
||||
t.seek(0)
|
||||
return t
|
||||
else:
|
||||
cmd.insert(-1, '--format=c')
|
||||
stdin, stdout = odoo.tools.exec_pg_command_pipe(*cmd)
|
||||
if stream:
|
||||
shutil.copyfileobj(stdout, stream)
|
||||
else:
|
||||
return stdout
|
||||
|
||||
def _dump_db_manifest(self, cr):
|
||||
pg_version = "%d.%d" % divmod(cr._obj.connection.server_version / 100, 100)
|
||||
cr.execute("SELECT name, latest_version FROM ir_module_module WHERE state = 'installed'")
|
||||
modules = dict(cr.fetchall())
|
||||
manifest = {
|
||||
'odoo_dump': '1',
|
||||
'db_name': cr.dbname,
|
||||
'version': odoo.release.version,
|
||||
'version_info': odoo.release.version_info,
|
||||
'major_version': odoo.release.major_version,
|
||||
'pg_version': pg_version,
|
||||
'modules': modules,
|
||||
}
|
||||
return manifest
|
||||
Reference in New Issue
Block a user