diff --git a/auto_database_backup/README.rst b/auto_database_backup/README.rst new file mode 100644 index 000000000..9a8b710ff --- /dev/null +++ b/auto_database_backup/README.rst @@ -0,0 +1,50 @@ +.. image:: https://img.shields.io/badge/licence-LGPL--3-green.svg + :target: https://www.gnu.org/licenses/lgpl-3.0-standalone.html + :alt: License: LGPL-3 + +Automatic Database Backup To Local Server, Remote Server, Google Drive, Dropbox, Onedrive, Nextcloud and Amazon S3 +================================================================================================================== +* Generate Database Backups and store to multiple locations + +Configuration +============ +- www.odoo.com/documentation/17.0/setup/install.html +- Install our custom addon + +License +------- +General Public License, Version 3 (LGPL v3). +(https://www.gnu.org/licenses/lgpl-3.0-standalone.html) + +Company +------- +* `Cybrosys Techno Solutions `__ + +Credits +------- +* Developers : (v15) Midilaj, + (v16) Midilaj, + (v16 Amazon S3,NextCloud) Anfas Faisal K, + (v17) Aslam A K + Contact : odoo@cybrosys.com + +Contacts +-------- +* Mail Contact : odoo@cybrosys.com + +Bug Tracker +----------- +Bugs are tracked on GitHub Issues. In case of trouble, please check there if your issue has already been reported. + +Maintainer +========== +.. image:: https://cybrosys.com/images/logo.png + :target: https://cybrosys.com + +This module is maintained by Cybrosys Technologies. + +For support and more information, please visit `Our Website `__ + +Further information +=================== +HTML Description: ``__ \ No newline at end of file diff --git a/auto_database_backup/__init__.py b/auto_database_backup/__init__.py new file mode 100644 index 000000000..8ecfebba9 --- /dev/null +++ b/auto_database_backup/__init__.py @@ -0,0 +1,24 @@ +# -*- coding: utf-8 -*- +############################################################################### +# +# Cybrosys Technologies Pvt. Ltd. +# +# Copyright (C) 2023-TODAY Cybrosys Technologies() +# Author: Cybrosys Techno Solutions (odoo@cybrosys.com) +# +# You can modify it under the terms of the GNU LESSER +# GENERAL PUBLIC LICENSE (LGPL v3), Version 3. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU LESSER GENERAL PUBLIC LICENSE (LGPL v3) for more details. +# +# You should have received a copy of the GNU LESSER GENERAL PUBLIC LICENSE +# (LGPL v3) along with this program. +# If not, see . +# +############################################################################### +from . import controllers +from . import models +from . import wizard diff --git a/auto_database_backup/__manifest__.py b/auto_database_backup/__manifest__.py new file mode 100644 index 000000000..09073fbbd --- /dev/null +++ b/auto_database_backup/__manifest__.py @@ -0,0 +1,52 @@ +# -*- coding: utf-8 -*- +############################################################################### +# +# Cybrosys Technologies Pvt. Ltd. +# +# Copyright (C) 2023-TODAY Cybrosys Technologies() +# Author: Cybrosys Techno Solutions (odoo@cybrosys.com) +# +# You can modify it under the terms of the GNU LESSER +# GENERAL PUBLIC LICENSE (LGPL v3), Version 3. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU LESSER GENERAL PUBLIC LICENSE (LGPL v3) for more details. +# +# You should have received a copy of the GNU LESSER GENERAL PUBLIC LICENSE +# (LGPL v3) along with this program. +# If not, see . +# +############################################################################### +{ + 'name': "Automatic Database Backup To Local Server, Remote Server," + "Google Drive, Dropbox, Onedrive, Nextcloud and Amazon S3", + 'version': '17.0.1.0.0', + 'category': 'Extra Tools', + 'summary': 'Generate automatic backup of databases and store to local, ' + 'google drive, dropbox, nextcloud, amazon S3, onedrive or ' + 'remote server', + 'description': 'This module has been developed for creating database ' + 'backups automatically and store it to the different ' + 'locations.', + 'author': "Cybrosys Techno Solutions", + 'maintainer': 'Cybrosys Techno Solutions', + 'company': 'Cybrosys Techno Solutions', + 'website': "https://www.cybrosys.com", + 'depends': ['base', 'mail'], + 'data': [ + 'security/ir.model.access.csv', + 'data/ir_cron_data.xml', + 'data/mail_template_data.xml', + 'views/db_backup_configure_views.xml', + 'wizard/dropbox_auth_code_views.xml', + ], + 'external_dependencies': { + 'python': ['dropbox', 'pyncclient', 'boto3', 'nextcloud-api-wrapper','paramiko']}, + 'images': ['static/description/banner.gif'], + 'license': 'LGPL-3', + 'installable': True, + 'auto_install': False, + 'application': False, +} diff --git a/auto_database_backup/controllers/__init__.py b/auto_database_backup/controllers/__init__.py new file mode 100644 index 000000000..be06f5727 --- /dev/null +++ b/auto_database_backup/controllers/__init__.py @@ -0,0 +1,22 @@ +# -*- coding: utf-8 -*- +############################################################################### +# +# Cybrosys Technologies Pvt. Ltd. +# +# Copyright (C) 2023-TODAY Cybrosys Technologies() +# Author: Cybrosys Techno Solutions (odoo@cybrosys.com) +# +# You can modify it under the terms of the GNU LESSER +# GENERAL PUBLIC LICENSE (LGPL v3), Version 3. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU LESSER GENERAL PUBLIC LICENSE (LGPL v3) for more details. +# +# You should have received a copy of the GNU LESSER GENERAL PUBLIC LICENSE +# (LGPL v3) along with this program. +# If not, see . +# +############################################################################### +from . import auto_database_backup diff --git a/auto_database_backup/controllers/auto_database_backup.py b/auto_database_backup/controllers/auto_database_backup.py new file mode 100644 index 000000000..43fef84d3 --- /dev/null +++ b/auto_database_backup/controllers/auto_database_backup.py @@ -0,0 +1,54 @@ +# -*- coding: utf-8 -*- +############################################################################### +# +# Cybrosys Technologies Pvt. Ltd. +# +# Copyright (C) 2023-TODAY Cybrosys Technologies() +# Author: Cybrosys Techno Solutions (odoo@cybrosys.com) +# +# You can modify it under the terms of the GNU LESSER +# GENERAL PUBLIC LICENSE (LGPL v3), Version 3. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU LESSER GENERAL PUBLIC LICENSE (LGPL v3) for more details. +# +# You should have received a copy of the GNU LESSER GENERAL PUBLIC LICENSE +# (LGPL v3) along with this program. +# If not, see . +# +############################################################################### +import json +from odoo import http +from odoo.http import request + + +class OnedriveAuth(http.Controller): + """Controller for handling authentication with OneDrive and Google Drive.""" + @http.route('/onedrive/authentication', type='http', auth="public") + def oauth2callback(self, **kw): + """ + Callback function for OneDrive authentication. + + :param kw: A dictionary of keyword arguments. + :return: A redirect response. + """ + state = json.loads(kw['state']) + backup_config = request.env['db.backup.configure'].sudo().browse( + state.get('backup_config_id')) + backup_config.get_onedrive_tokens(kw.get('code')) + backup_config.hide_active = True + backup_config.active = True + return request.redirect(state.get('url_return')) + + @http.route('/google_drive/authentication', type='http', auth="public") + def gdrive_oauth2callback(self, **kw): + """Callback function for Google Drive authentication.""" + state = json.loads(kw['state']) + backup_config = request.env['db.backup.configure'].sudo().browse( + state.get('backup_config_id')) + backup_config.get_gdrive_tokens(kw.get('code')) + backup_config.hide_active = True + backup_config.active = True + return request.redirect(state.get('url_return')) diff --git a/auto_database_backup/data/ir_cron_data.xml b/auto_database_backup/data/ir_cron_data.xml new file mode 100644 index 000000000..15332fada --- /dev/null +++ b/auto_database_backup/data/ir_cron_data.xml @@ -0,0 +1,16 @@ + + + + + + Backup : Automatic Database Backup + + code + model._schedule_auto_backup() + 1 + days + -1 + False + + + diff --git a/auto_database_backup/data/mail_template_data.xml b/auto_database_backup/data/mail_template_data.xml new file mode 100644 index 000000000..7b4334d63 --- /dev/null +++ b/auto_database_backup/data/mail_template_data.xml @@ -0,0 +1,192 @@ + + + + + + Database Backup: Notification Successful + + Database Backup Successful: {{ object.db_name }} + {{ object.user_id.email_formatted }} + +
+

+ Dear , + +
+
+ Backup of the database + + + + has been successfully generated and stored to + + Local + + + Google Drive + + + FTP Server + + + SFTP Server + + + Dropbox + + + Onedrive + + + NextCloud + + + Amazon S3 + + . +
+
+ Database Name: + +
+ Destination: + + Local + + + Google Drive + + + FTP Server + + + SFTP Server + + + Dropbox + + + Onedrive + + + NextCloud + + + Amazon S3 + + +
+ Backup Path: + + + + + + + + + + + + + + + + + + + +
+ Backup Type: + +
+ Backup FileName: + +
+

+
+
+
+ + + Database Backup: Notification Failed + + Database Backup Failed: {{ object.db_name }} + {{ object.user_id.email_formatted }} + +
+

+ Dear , + +
+
+ Backup generation of the database + + + + has been Failed. +
+
+ Database Name: +
+ Destination: + + Local + + + Google Drive + + + FTP Server + + + SFTP Server + + + Dropbox + + + Onedrive + + + NextCloud + + + Amazon S3 + + +
+ Backup Path: + + + + + + + + + + + + + + + + + + + +
+ Backup Type: +
+
+ Error Message: +
+ +
+

+
+
+
+
+
diff --git a/auto_database_backup/doc/RELEASE_NOTES.md b/auto_database_backup/doc/RELEASE_NOTES.md new file mode 100644 index 000000000..8284f55eb --- /dev/null +++ b/auto_database_backup/doc/RELEASE_NOTES.md @@ -0,0 +1,7 @@ +## Module + +#### 06.11.2023 +#### Version 17.0.1.0.0 +#### ADD + +- Initial commit for auto_database_backup diff --git a/auto_database_backup/models/__init__.py b/auto_database_backup/models/__init__.py new file mode 100644 index 000000000..d441b67c3 --- /dev/null +++ b/auto_database_backup/models/__init__.py @@ -0,0 +1,22 @@ +# -*- coding: utf-8 -*- +############################################################################### +# +# Cybrosys Technologies Pvt. Ltd. +# +# Copyright (C) 2023-TODAY Cybrosys Technologies() +# Author: Cybrosys Techno Solutions (odoo@cybrosys.com) +# +# You can modify it under the terms of the GNU LESSER +# GENERAL PUBLIC LICENSE (LGPL v3), Version 3. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU LESSER GENERAL PUBLIC LICENSE (LGPL v3) for more details. +# +# You should have received a copy of the GNU LESSER GENERAL PUBLIC LICENSE +# (LGPL v3) along with this program. +# If not, see . +# +############################################################################### +from . import db_backup_configure diff --git a/auto_database_backup/models/db_backup_configure.py b/auto_database_backup/models/db_backup_configure.py new file mode 100644 index 000000000..8c93cb2a1 --- /dev/null +++ b/auto_database_backup/models/db_backup_configure.py @@ -0,0 +1,1011 @@ +# -*- coding: utf-8 -*- +############################################################################### +# +# Cybrosys Technologies Pvt. Ltd. +# +# Copyright (C) 2023-TODAY Cybrosys Technologies() +# Author: Cybrosys Techno Solutions (odoo@cybrosys.com) +# +# You can modify it under the terms of the GNU LESSER +# GENERAL PUBLIC LICENSE (LGPL v3), Version 3. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU LESSER GENERAL PUBLIC LICENSE (LGPL v3) for more details. +# +# You should have received a copy of the GNU LESSER GENERAL PUBLIC LICENSE +# (LGPL v3) along with this program. +# If not, see . +# +############################################################################### +import boto3 +import dropbox +import errno +import ftplib +import json +import logging +import nextcloud_client +import os +import paramiko +import requests +import tempfile +import odoo +from datetime import timedelta +from nextcloud import NextCloud +from requests.auth import HTTPBasicAuth +from werkzeug import urls +from odoo import api, fields, models, _ +from odoo.exceptions import UserError, ValidationError +from odoo.http import request +from odoo.service import db + +_logger = logging.getLogger(__name__) +ONEDRIVE_SCOPE = ['offline_access openid Files.ReadWrite.All'] +MICROSOFT_GRAPH_END_POINT = "https://graph.microsoft.com" +GOOGLE_AUTH_ENDPOINT = 'https://accounts.google.com/o/oauth2/auth' +GOOGLE_TOKEN_ENDPOINT = 'https://accounts.google.com/o/oauth2/token' +GOOGLE_API_BASE_URL = 'https://www.googleapis.com' + + +class DbBackupConfigure(models.Model): + """DbBackupConfigure class provides an interface to manage database + backups of Local Server, Remote Server, Google Drive, Dropbox, Onedrive, + Nextcloud and Amazon S3""" + _name = 'db.backup.configure' + _description = 'Automatic Database Backup' + + name = fields.Char(string='Name', required=True, help='Add the name') + db_name = fields.Char(string='Database Name', required=True, + help='Name of the database') + master_pwd = fields.Char(string='Master Password', required=True, + help='Master password') + backup_format = fields.Selection([ + ('zip', 'Zip'), + ('dump', 'Dump') + ], string='Backup Format', default='zip', required=True, + help='Format of the backup') + backup_destination = fields.Selection([ + ('local', 'Local Storage'), + ('google_drive', 'Google Drive'), + ('ftp', 'FTP'), + ('sftp', 'SFTP'), + ('dropbox', 'Dropbox'), + ('onedrive', 'Onedrive'), + ('next_cloud', 'Next Cloud'), + ('amazon_s3', 'Amazon S3') + ], string='Backup Destination', help='Destination of the backup') + backup_path = fields.Char(string='Backup Path', + help='Local storage directory path') + sftp_host = fields.Char(string='SFTP Host', help='SFTP host details') + sftp_port = fields.Char(string='SFTP Port', default=22, + help='SFTP port details') + sftp_user = fields.Char(string='SFTP User', copy=False, + help='SFTP user details') + sftp_password = fields.Char(string='SFTP Password', copy=False, + help='SFTP password') + sftp_path = fields.Char(string='SFTP Path', help='SFTP path details') + ftp_host = fields.Char(string='FTP Host', help='FTP host details') + ftp_port = fields.Char(string='FTP Port', default=21, + help='FTP port details') + ftp_user = fields.Char(string='FTP User', copy=False, + help='FTP user details') + ftp_password = fields.Char(string='FTP Password', copy=False, + help='FTP password') + ftp_path = fields.Char(string='FTP Path', help='FTP path details') + dropbox_client_key = fields.Char(string='Dropbox Client ID', copy=False, + help='Client id of the dropbox') + dropbox_client_secret = fields.Char(string='Dropbox Client Secret', + copy=False, + help='Client secret id of the dropbox') + dropbox_refresh_token = fields.Char(string='Dropbox Refresh Token', + copy=False, + help='Refresh token for the dropbox') + is_dropbox_token_generated = fields.Boolean( + string='Dropbox Token Generated', + compute='_compute_is_dropbox_token_generated', + copy=False, help='Is the dropbox token generated or not?') + dropbox_folder = fields.Char(string='Dropbox Folder', + help='Dropbox folder') + active = fields.Boolean(default=False, string='Active', + help='Activate the Scheduled Action or not') + hide_active = fields.Boolean(string="Hide Active", + help="Make active field to readonly") + auto_remove = fields.Boolean(string='Remove Old Backups', + help='Remove old backups') + days_to_remove = fields.Integer(string='Remove After', + help='Automatically delete stored backups' + ' after this specified number of days') + google_drive_folder_key = fields.Char(string='Drive Folder ID', + help='Folder id of the drive') + notify_user = fields.Boolean(string='Notify User', + help='Send an email notification to user when' + 'the backup operation is successful' + 'or failed') + user_id = fields.Many2one('res.users', string='User', + help='Name of the user') + backup_filename = fields.Char(string='Backup Filename', + help='For Storing generated backup filename') + generated_exception = fields.Char(string='Exception', + help='Exception Encountered while Backup' + 'generation') + onedrive_client_key = fields.Char(string='Onedrive Client ID', copy=False, + help='Client ID of the onedrive') + onedrive_client_secret = fields.Char(string='Onedrive Client Secret', + copy=False, help='Client secret id of' + ' the onedrive') + onedrive_access_token = fields.Char(string='Onedrive Access Token', + copy=False, + help='Access token for one drive') + onedrive_refresh_token = fields.Char(string='Onedrive Refresh Token', + copy=False, + help='Refresh token for one drive') + onedrive_token_validity = fields.Datetime(string='Onedrive Token Validity', + copy=False, + help='Token validity date') + onedrive_folder_key = fields.Char(string='Folder ID', + help='Folder id of the onedrive') + is_onedrive_token_generated = fields.Boolean( + string='onedrive Tokens Generated', + compute='_compute_is_onedrive_token_generated', + copy=False, help='Whether to generate onedrive token?') + gdrive_refresh_token = fields.Char(string='Google drive Refresh Token', + copy=False, + help='Refresh token for google drive') + gdrive_access_token = fields.Char(string='Google Drive Access Token', + copy=False, + help='Access token for google drive') + is_google_drive_token_generated = fields.Boolean( + string='Google drive Token Generated', + compute='_compute_is_google_drive_token_generated', copy=False, + help='Google drive token generated or not') + gdrive_client_key = fields.Char(string='Google Drive Client ID', + copy=False, + help='Client id of the google drive') + gdrive_client_secret = fields.Char(string='Google Drive Client Secret', + copy=False, + help='Client secret id of the google' + ' drive') + gdrive_token_validity = fields.Datetime( + string='Google Drive Token Validity', copy=False, + help='Token validity of the google drive') + onedrive_redirect_uri = fields.Char(string='Onedrive Redirect URI', + compute='_compute_redirect_uri', + help='Redirect URI of the onedrive') + gdrive_redirect_uri = fields.Char(string='Google Drive Redirect URI', + compute='_compute_redirect_uri', + help='Redirect URI of the google drive') + domain = fields.Char(string='Domain Name', help="Field used to store the " + "name of a domain") + next_cloud_user_name = fields.Char(string='User Name', + help="Field used to store the user name" + " for a Nextcloud account.") + next_cloud_password = fields.Char(string='Password', + help="Field used to store the password" + " for a Nextcloud account.") + nextcloud_folder_key = fields.Char(string='Next Cloud Folder Id', + help="Field used to store the unique " + "identifier for a Nextcloud " + "folder.") + aws_access_key = fields.Char(string="Amazon S3 Access Key", + help="Field used to store the Access Key" + " for an Amazon S3 bucket.") + aws_secret_access_key = fields.Char(string='Amazon S3 Secret Key', + help="Field used to store the Secret" + " Key for an Amazon S3 bucket.") + bucket_file_name = fields.Char(string='Bucket Name', + help="Field used to store the name of an" + " Amazon S3 bucket.") + aws_folder_name = fields.Char(string='File Name', + help="field used to store the name of a" + " folder in an Amazon S3 bucket.") + + def action_s3cloud(self): + """If it has aws_secret_access_key, which will perform s3cloud + operations for connection test""" + if self.aws_access_key and self.aws_secret_access_key: + try: + bo3 = boto3.client( + 's3', + aws_access_key_id=self.aws_access_key, + aws_secret_access_key=self.aws_secret_access_key) + response = bo3.list_buckets() + for bucket in response['Buckets']: + if self.bucket_file_name == bucket['Name']: + self.active = True + self.hide_active = True + return { + 'type': 'ir.actions.client', + 'tag': 'display_notification', + 'params': { + 'type': 'success', + 'title': _("Connection Test Succeeded!"), + 'message': _( + "Everything seems properly set up!"), + 'sticky': False, + } + } + raise UserError( + _("Bucket not found. Please check the bucket name and" + " try again.")) + except Exception: + self.active = False + self.hide_active = False + return { + 'type': 'ir.actions.client', + 'tag': 'display_notification', + 'params': { + 'type': 'danger', + 'title': _("Connection Test Failed!"), + 'message': _("An error occurred while testing the " + "connection."), + 'sticky': False, + } + } + + def action_nextcloud(self): + """If it has next_cloud_password, domain, and next_cloud_user_name + which will perform an action for nextcloud connection test""" + if self.domain and self.next_cloud_password and \ + self.next_cloud_user_name: + try: + ncx = NextCloud(self.domain, + auth=HTTPBasicAuth(self.next_cloud_user_name, + self.next_cloud_password)) + + data = ncx.list_folders('/').__dict__ + if data['raw'].status_code == 207: + self.active = True + self.hide_active = True + return { + 'type': 'ir.actions.client', + 'tag': 'display_notification', + 'params': { + 'type': 'success', + 'title': _("Connection Test Succeeded!"), + 'message': _("Everything seems properly set up!"), + 'sticky': False, + } + } + else: + self.active = False + self.hide_active = False + return { + 'type': 'ir.actions.client', + 'tag': 'display_notification', + 'params': { + 'type': 'danger', + 'title': _("Connection Test Failed!"), + 'message': _("An error occurred while testing the " + "connection."), + 'sticky': False, + } + } + except Exception: + self.active = False + self.hide_active = False + return { + 'type': 'ir.actions.client', + 'tag': 'display_notification', + 'params': { + 'type': 'danger', + 'title': _("Connection Test Failed!"), + 'message': _("An error occurred while testing the " + "connection."), + 'sticky': False, + } + } + + @api.depends('onedrive_redirect_uri', 'gdrive_redirect_uri') + def _compute_redirect_uri(self): + """Compute the redirect URI for onedrive and Google Drive""" + for rec in self: + base_url = request.env['ir.config_parameter'].get_param( + 'web.base.url') + rec.onedrive_redirect_uri = base_url + '/onedrive/authentication' + rec.gdrive_redirect_uri = base_url + '/google_drive/authentication' + + @api.depends('onedrive_access_token', 'onedrive_refresh_token') + def _compute_is_onedrive_token_generated(self): + """Set true if onedrive tokens are generated""" + for rec in self: + rec.is_onedrive_token_generated = bool( + rec.onedrive_access_token) and bool(rec.onedrive_refresh_token) + + @api.depends('dropbox_refresh_token') + def _compute_is_dropbox_token_generated(self): + """Set True if the dropbox refresh token is generated""" + for rec in self: + rec.is_dropbox_token_generated = bool(rec.dropbox_refresh_token) + + @api.depends('gdrive_access_token', 'gdrive_refresh_token') + def _compute_is_google_drive_token_generated(self): + """Set True if the Google Drive refresh token is generated""" + for rec in self: + rec.is_google_drive_token_generated = bool( + rec.gdrive_access_token) and bool(rec.gdrive_refresh_token) + + def action_get_dropbox_auth_code(self): + """Open a wizards to set up dropbox Authorization code""" + return { + 'type': 'ir.actions.act_window', + 'name': 'Dropbox Authorization Wizard', + 'res_model': 'dropbox.auth.code', + 'view_mode': 'form', + 'target': 'new', + 'context': {'dropbox_auth': True} + } + + def action_get_onedrive_auth_code(self): + """Generate onedrive authorization code""" + AUTHORITY = \ + 'https://login.microsoftonline.com/common/oauth2/v2.0/authorize' + action = self.env["ir.actions.act_window"].sudo()._for_xml_id( + "auto_database_backup.db_backup_configure_action") + base_url = request.env['ir.config_parameter'].get_param('web.base.url') + url_return = base_url + \ + '/web#id=%d&action=%d&view_type=form&model=%s' % ( + self.id, action['id'], 'db.backup.configure') + state = { + 'backup_config_id': self.id, + 'url_return': url_return + } + encoded_params = urls.url_encode({ + 'response_type': 'code', + 'client_id': self.onedrive_client_key, + 'state': json.dumps(state), + 'scope': ONEDRIVE_SCOPE, + 'redirect_uri': base_url + '/onedrive/authentication', + 'prompt': 'consent', + 'access_type': 'offline' + }) + auth_url = "%s?%s" % (AUTHORITY, encoded_params) + return { + 'type': 'ir.actions.act_url', + 'target': 'self', + 'url': auth_url, + } + + def action_get_gdrive_auth_code(self): + """Generate google drive authorization code""" + action = self.env["ir.actions.act_window"].sudo()._for_xml_id( + "auto_database_backup.db_backup_configure_action") + base_url = request.env['ir.config_parameter'].get_param('web.base.url') + url_return = base_url + \ + '/web#id=%d&action=%d&view_type=form&model=%s' % ( + self.id, action['id'], 'db.backup.configure') + state = { + 'backup_config_id': self.id, + 'url_return': url_return + } + encoded_params = urls.url_encode({ + 'response_type': 'code', + 'client_id': self.gdrive_client_key, + 'scope': 'https://www.googleapis.com/auth/drive https://www.googleapis.com/auth/drive.file', + 'redirect_uri': base_url + '/google_drive/authentication', + 'access_type': 'offline', + 'state': json.dumps(state), + 'approval_prompt': 'force', + }) + auth_url = "%s?%s" % (GOOGLE_AUTH_ENDPOINT, encoded_params) + return { + 'type': 'ir.actions.act_url', + 'target': 'self', + 'url': auth_url, + } + + def generate_onedrive_refresh_token(self): + """Generate onedrive access token from refresh token if expired""" + base_url = request.env['ir.config_parameter'].get_param('web.base.url') + headers = {"Content-type": "application/x-www-form-urlencoded"} + data = { + 'client_id': self.onedrive_client_key, + 'client_secret': self.onedrive_client_secret, + 'scope': ONEDRIVE_SCOPE, + 'grant_type': "refresh_token", + 'redirect_uri': base_url + '/onedrive/authentication', + 'refresh_token': self.onedrive_refresh_token + } + try: + res = requests.post( + "https://login.microsoftonline.com/common/oauth2/v2.0/token", + data=data, headers=headers) + res.raise_for_status() + response = res.content and res.json() or {} + if response: + expires_in = response.get('expires_in') + self.write({ + 'onedrive_access_token': response.get('access_token'), + 'onedrive_refresh_token': response.get('refresh_token'), + 'onedrive_token_validity': fields.Datetime.now() + timedelta( + seconds=expires_in) if expires_in else False, + }) + except requests.HTTPError as error: + _logger.exception("Bad microsoft onedrive request : %s !", + error.response.content) + raise error + + def get_onedrive_tokens(self, authorize_code): + """Generate onedrive tokens from authorization code.""" + headers = {"content-type": "application/x-www-form-urlencoded"} + base_url = request.env['ir.config_parameter'].get_param('web.base.url') + data = { + 'code': authorize_code, + 'client_id': self.onedrive_client_key, + 'client_secret': self.onedrive_client_secret, + 'grant_type': 'authorization_code', + 'scope': ONEDRIVE_SCOPE, + 'redirect_uri': base_url + '/onedrive/authentication' + } + try: + res = requests.post( + "https://login.microsoftonline.com/common/oauth2/v2.0/token", + data=data, headers=headers) + res.raise_for_status() + response = res.content and res.json() or {} + if response: + expires_in = response.get('expires_in') + self.write({ + 'onedrive_access_token': response.get('access_token'), + 'onedrive_refresh_token': response.get('refresh_token'), + 'onedrive_token_validity': fields.Datetime.now() + timedelta( + seconds=expires_in) if expires_in else False, + }) + except requests.HTTPError as error: + _logger.exception("Bad microsoft onedrive request : %s !", + error.response.content) + raise error + + def generate_gdrive_refresh_token(self): + """Generate Google Drive access token from refresh token if expired""" + headers = {"content-type": "application/x-www-form-urlencoded"} + data = { + 'refresh_token': self.gdrive_refresh_token, + 'client_id': self.gdrive_client_key, + 'client_secret': self.gdrive_client_secret, + 'grant_type': 'refresh_token', + } + try: + res = requests.post(GOOGLE_TOKEN_ENDPOINT, data=data, + headers=headers) + res.raise_for_status() + response = res.content and res.json() or {} + if response: + expires_in = response.get('expires_in') + self.write({ + 'gdrive_access_token': response.get('access_token'), + 'gdrive_token_validity': fields.Datetime.now() + timedelta( + seconds=expires_in) if expires_in else False, + }) + except requests.HTTPError as error: + error_key = error.response.json().get("error", "nc") + error_msg = _( + "An error occurred while generating the token. Your" + "authorization code may be invalid or has already expired [%s]." + "You should check your Client ID and secret on the Google APIs" + " plateform or try to stop and restart your calendar" + " synchronisation.", + error_key) + raise UserError(error_msg) + + def get_gdrive_tokens(self, authorize_code): + """Generate onedrive tokens from authorization code.""" + base_url = request.env['ir.config_parameter'].get_param('web.base.url') + headers = {"content-type": "application/x-www-form-urlencoded"} + data = { + 'code': authorize_code, + 'client_id': self.gdrive_client_key, + 'client_secret': self.gdrive_client_secret, + 'grant_type': 'authorization_code', + 'redirect_uri': base_url + '/google_drive/authentication' + } + try: + res = requests.post(GOOGLE_TOKEN_ENDPOINT, params=data, + headers=headers) + res.raise_for_status() + response = res.content and res.json() or {} + if response: + expires_in = response.get('expires_in') + self.write({ + 'gdrive_access_token': response.get('access_token'), + 'gdrive_refresh_token': response.get('refresh_token'), + 'gdrive_token_validity': fields.Datetime.now() + timedelta( + seconds=expires_in) if expires_in else False, + }) + except requests.HTTPError: + error_msg = _( + "Something went wrong during your token generation. Maybe your" + " Authorization Code is invalid") + raise UserError(error_msg) + + def get_dropbox_auth_url(self): + """Return dropbox authorization url""" + dbx_auth = dropbox.oauth.DropboxOAuth2FlowNoRedirect( + self.dropbox_client_key, + self.dropbox_client_secret, + token_access_type='offline') + return dbx_auth.start() + + def set_dropbox_refresh_token(self, auth_code): + """Generate and set the dropbox refresh token from authorization code""" + dbx_auth = dropbox.oauth.DropboxOAuth2FlowNoRedirect( + self.dropbox_client_key, + self.dropbox_client_secret, + token_access_type='offline') + outh_result = dbx_auth.finish(auth_code) + self.dropbox_refresh_token = outh_result.refresh_token + + @api.constrains('db_name') + def _check_db_credentials(self): + """Validate entered database name and master password""" + database_list = db.list_dbs() + if self.db_name not in database_list: + raise ValidationError(_("Invalid Database Name!")) + try: + odoo.service.db.check_super(self.master_pwd) + except Exception: + raise ValidationError(_("Invalid Master Password!")) + + def action_sftp_connection(self): + """Test the sftp and ftp connection using entered credentials""" + if self.backup_destination == 'sftp': + client = paramiko.SSHClient() + client.set_missing_host_key_policy(paramiko.AutoAddPolicy()) + try: + client.connect(hostname=self.sftp_host, + username=self.sftp_user, + password=self.sftp_password, + port=self.sftp_port) + sftp = client.open_sftp() + sftp.close() + except Exception as e: + raise UserError(_("SFTP Exception: %s", e)) + finally: + client.close() + elif self.backup_destination == 'ftp': + try: + ftp_server = ftplib.FTP() + ftp_server.connect(self.ftp_host, int(self.ftp_port)) + ftp_server.login(self.ftp_user, self.ftp_password) + ftp_server.quit() + except Exception as e: + raise UserError(_("FTP Exception: %s", e)) + self.hide_active = True + self.active = True + return { + 'type': 'ir.actions.client', + 'tag': 'display_notification', + 'params': { + 'title': _("Connection Test Succeeded!"), + 'message': _("Everything seems properly set up!"), + 'sticky': False, + } + } + + @api.onchange('backup_destination') + def _onchange_back_up_local(self): + """ + On change handler for the 'backup_destination' field. This method is + triggered when the value of 'backup_destination' is changed. If the + chosen backup destination is 'local', it sets the 'hide_active' field + to True which make active field to readonly to False. + """ + if self.backup_destination == 'local': + self.hide_active = True + + def _schedule_auto_backup(self): + """Function for generating and storing backup. + Database backup for all the active records in backup configuration + model will be created.""" + records = self.search([]) + mail_template_success = self.env.ref( + 'auto_database_backup.mail_template_data_db_backup_successful') + mail_template_failed = self.env.ref( + 'auto_database_backup.mail_template_data_db_backup_failed') + for rec in records: + backup_time = fields.datetime.utcnow().strftime( + "%Y-%m-%d_%H-%M-%S") + backup_filename = "%s_%s.%s" % ( + rec.db_name, backup_time, rec.backup_format) + rec.backup_filename = backup_filename + # Local backup + if rec.backup_destination == 'local': + try: + if not os.path.isdir(rec.backup_path): + os.makedirs(rec.backup_path) + backup_file = os.path.join(rec.backup_path, + backup_filename) + f = open(backup_file, "wb") + odoo.service.db.dump_db(rec.db_name, f, rec.backup_format) + f.close() + # Remove older backups + if rec.auto_remove: + for filename in os.listdir(rec.backup_path): + file = os.path.join(rec.backup_path, filename) + create_time = fields.datetime.fromtimestamp( + os.path.getctime(file)) + backup_duration = fields.datetime.utcnow() - create_time + if backup_duration.days >= rec.days_to_remove: + os.remove(file) + if rec.notify_user: + mail_template_success.send_mail(rec.id, + force_send=True) + except Exception as e: + rec.generated_exception = e + _logger.info('FTP Exception: %s', e) + if rec.notify_user: + mail_template_failed.send_mail(rec.id, force_send=True) + # FTP backup + elif rec.backup_destination == 'ftp': + try: + ftp_server = ftplib.FTP() + ftp_server.connect(rec.ftp_host, int(rec.ftp_port)) + ftp_server.login(rec.ftp_user, rec.ftp_password) + ftp_server.encoding = "utf-8" + temp = tempfile.NamedTemporaryFile( + suffix='.%s' % rec.backup_format) + try: + ftp_server.cwd(rec.ftp_path) + except ftplib.error_perm: + ftp_server.mkd(rec.ftp_path) + ftp_server.cwd(rec.ftp_path) + with open(temp.name, "wb+") as tmp: + odoo.service.db.dump_db(rec.db_name, tmp, + rec.backup_format) + ftp_server.storbinary('STOR %s' % backup_filename, + open(temp.name, "rb")) + if rec.auto_remove: + files = ftp_server.nlst() + for file in files: + create_time = fields.datetime.strptime( + ftp_server.sendcmd('MDTM ' + file)[4:], + "%Y%m%d%H%M%S") + diff_days = ( + fields.datetime.now() - create_time).days + if diff_days >= rec.days_to_remove: + ftp_server.delete(file) + ftp_server.quit() + if rec.notify_user: + mail_template_success.send_mail(rec.id, + force_send=True) + except Exception as e: + rec.generated_exception = e + _logger.info('FTP Exception: %s', e) + if rec.notify_user: + mail_template_failed.send_mail(rec.id, force_send=True) + # SFTP backup + elif rec.backup_destination == 'sftp': + client = paramiko.SSHClient() + client.set_missing_host_key_policy(paramiko.AutoAddPolicy()) + try: + client.connect(hostname=rec.sftp_host, + username=rec.sftp_user, + password=rec.sftp_password, + port=rec.sftp_port) + sftp = client.open_sftp() + temp = tempfile.NamedTemporaryFile( + suffix='.%s' % rec.backup_format) + with open(temp.name, "wb+") as tmp: + odoo.service.db.dump_db(rec.db_name, tmp, + rec.backup_format) + try: + sftp.chdir(rec.sftp_path) + except IOError as e: + if e.errno == errno.ENOENT: + sftp.mkdir(rec.sftp_path) + sftp.chdir(rec.sftp_path) + sftp.put(temp.name, backup_filename) + if rec.auto_remove: + files = sftp.listdir() + expired = list(filter( + lambda fl: (fields.datetime.now() + - fields.datetime.fromtimestamp( + sftp.stat(fl).st_mtime)).days >= + rec.days_to_remove, files)) + for file in expired: + sftp.unlink(file) + sftp.close() + if rec.notify_user: + mail_template_success.send_mail(rec.id, + force_send=True) + except Exception as e: + rec.generated_exception = e + _logger.info('SFTP Exception: %s', e) + if rec.notify_user: + mail_template_failed.send_mail(rec.id, force_send=True) + finally: + client.close() + # Google Drive backup + elif rec.backup_destination == 'google_drive': + try: + if rec.gdrive_token_validity <= fields.Datetime.now(): + rec.generate_gdrive_refresh_token() + temp = tempfile.NamedTemporaryFile( + suffix='.%s' % rec.backup_format) + with open(temp.name, "wb+") as tmp: + odoo.service.db.dump_db(rec.db_name, tmp, + rec.backup_format) + try: + headers = { + "Authorization": "Bearer %s" % rec.gdrive_access_token} + para = { + "name": backup_filename, + "parents": [rec.google_drive_folder_key], + } + files = { + 'data': ('metadata', json.dumps(para), + 'application/json; charset=UTF-8'), + 'file': open(temp.name, "rb") + } + requests.post( + "https://www.googleapis.com/upload/drive/v3/files?uploadType=multipart", + headers=headers, + files=files + ) + if rec.auto_remove: + query = "parents = '%s'" % rec.google_drive_folder_key + files_req = requests.get( + "https://www.googleapis.com/drive/v3/files?q=%s" % query, + headers=headers) + files = files_req.json()['files'] + for file in files: + file_date_req = requests.get( + "https://www.googleapis.com/drive/v3/files/%s?fields=createdTime" % + file['id'], headers=headers) + create_time = file_date_req.json()[ + 'createdTime'][ + :19].replace('T', ' ') + diff_days = ( + fields.datetime.now() - fields.datetime.strptime( + create_time, '%Y-%m-%d %H:%M:%S')).days + if diff_days >= rec.days_to_remove: + requests.delete( + "https://www.googleapis.com/drive/v3/files/%s" % + file['id'], headers=headers) + if rec.notify_user: + mail_template_success.send_mail(rec.id, + force_send=True) + except Exception as e: + + rec.generated_exception = e + _logger.info('Google Drive Exception: %s', e) + if rec.notify_user: + mail_template_failed.send_mail(rec.id, + force_send=True) + except Exception: + if rec.notify_user: + mail_template_failed.send_mail(rec.id, force_send=True) + raise ValidationError( + 'Please check the credentials before activation') + else: + raise ValidationError('Please check connection') + # Dropbox backup + elif rec.backup_destination == 'dropbox': + temp = tempfile.NamedTemporaryFile( + suffix='.%s' % rec.backup_format) + with open(temp.name, "wb+") as tmp: + odoo.service.db.dump_db(rec.db_name, tmp, + rec.backup_format) + try: + dbx = dropbox.Dropbox( + app_key=rec.dropbox_client_key, + app_secret=rec.dropbox_client_secret, + oauth2_refresh_token=rec.dropbox_refresh_token) + dropbox_destination = (rec.dropbox_folder + '/' + + backup_filename) + dbx.files_upload(temp.read(), dropbox_destination) + if rec.auto_remove: + files = dbx.files_list_folder(rec.dropbox_folder) + file_entries = files.entries + expired_files = list(filter( + lambda fl: (fields.datetime.now() - + fl.client_modified).days >= + rec.days_to_remove, + file_entries)) + for file in expired_files: + dbx.files_delete_v2(file.path_display) + if rec.notify_user: + mail_template_success.send_mail(rec.id, + force_send=True) + except Exception as error: + rec.generated_exception = error + _logger.info('Dropbox Exception: %s', error) + if rec.notify_user: + mail_template_failed.send_mail(rec.id, force_send=True) + # Onedrive Backup + elif rec.backup_destination == 'onedrive': + if rec.onedrive_token_validity <= fields.Datetime.now(): + rec.generate_onedrive_refresh_token() + temp = tempfile.NamedTemporaryFile( + suffix='.%s' % rec.backup_format) + with open(temp.name, "wb+") as tmp: + odoo.service.db.dump_db(rec.db_name, tmp, + rec.backup_format) + headers = { + 'Authorization': 'Bearer %s' % rec.onedrive_access_token, + 'Content-Type': 'application/json'} + upload_session_url = MICROSOFT_GRAPH_END_POINT + "/v1.0/me/drive/items/%s:/%s:/createUploadSession" % ( + rec.onedrive_folder_key, backup_filename) + try: + upload_session = requests.post(upload_session_url, + headers=headers) + upload_url = upload_session.json().get('uploadUrl') + requests.put(upload_url, data=temp.read()) + if rec.auto_remove: + list_url = MICROSOFT_GRAPH_END_POINT + "/v1.0/me/drive/items/%s/children" % rec.onedrive_folder_key + response = requests.get(list_url, headers=headers) + files = response.json().get('value') + for file in files: + create_time = file['createdDateTime'][:19].replace( + 'T', + ' ') + diff_days = ( + fields.datetime.now() - fields.datetime.strptime( + create_time, '%Y-%m-%d %H:%M:%S')).days + if diff_days >= rec.days_to_remove: + delete_url = MICROSOFT_GRAPH_END_POINT + "/v1.0/me/drive/items/%s" % \ + file['id'] + requests.delete(delete_url, headers=headers) + if rec.notify_user: + mail_template_success.send_mail(rec.id, + force_send=True) + except Exception as error: + rec.generated_exception = error + _logger.info('Onedrive Exception: %s', error) + if rec.notify_user: + mail_template_failed.send_mail(rec.id, force_send=True) + # NextCloud Backup + elif rec.backup_destination == 'next_cloud': + try: + if rec.domain and rec.next_cloud_password and \ + rec.next_cloud_user_name: + try: + # Connect to NextCloud using the provided username + # and password + ncx = NextCloud(rec.domain, + auth=HTTPBasicAuth( + rec.next_cloud_user_name, + rec.next_cloud_password)) + # Connect to NextCloud again to perform additional + # operations + nc = nextcloud_client.Client(rec.domain) + nc.login(rec.next_cloud_user_name, + rec.next_cloud_password) + # Get the folder name from the NextCloud folder ID + folder_name = rec.nextcloud_folder_key + # If auto_remove is enabled, remove backup files + # older than specified days + if rec.auto_remove: + folder_path = "/" + folder_name + for item in nc.list(folder_path): + backup_file_name = item.path.split("/")[-1] + backup_date_str = \ + backup_file_name.split("_")[ + 2] + backup_date = fields.datetime.strptime( + backup_date_str, '%Y-%m-%d').date() + if (fields.date.today() - backup_date).days \ + >= rec.days_to_remove: + nc.delete(item.path) + # If notify_user is enabled, send a success email + # notification + if rec.notify_user: + mail_template_success.send_mail(rec.id, + force_send=True) + except Exception as error: + rec.generated_exception = error + _logger.info('NextCloud Exception: %s', error) + if rec.notify_user: + # If an exception occurs, send a failed email + # notification + mail_template_failed.send_mail(rec.id, + force_send=True) + # Get the list of folders in the root directory of NextCloud + data = ncx.list_folders('/').__dict__ + folders = [ + [file_name['href'].split('/')[-2], + file_name['file_id']] + for file_name in data['data'] if + file_name['href'].endswith('/')] + # If the folder name is not found in the list of folders, + # create the folder + if folder_name not in [file[0] for file in folders]: + nc.mkdir(folder_name) + # Dump the database to a temporary file + temp = tempfile.NamedTemporaryFile( + suffix='.%s' % rec.backup_format) + with open(temp.name, "wb+") as tmp: + odoo.service.db.dump_db(rec.db_name, tmp, + rec.backup_format) + backup_file_path = temp.name + remote_file_path = f"/{folder_name}/{rec.db_name}_" \ + f"{backup_time}.{rec.backup_format}" + nc.put_file(remote_file_path, backup_file_path) + else: + # Dump the database to a temporary file + temp = tempfile.NamedTemporaryFile( + suffix='.%s' % rec.backup_format) + with open(temp.name, "wb+") as tmp: + odoo.service.db.dump_db(rec.db_name, tmp, + rec.backup_format) + backup_file_path = temp.name + remote_file_path = f"/{folder_name}/{rec.db_name}_" \ + f"{backup_time}.{rec.backup_format}" + nc.put_file(remote_file_path, backup_file_path) + except Exception: + raise ValidationError('Please check connection') + # Amazon S3 Backup + elif rec.backup_destination == 'amazon_s3': + if rec.aws_access_key and rec.aws_secret_access_key: + try: + # Create a boto3 client for Amazon S3 with provided + # access key id and secret access key + bo3 = boto3.client( + 's3', + aws_access_key_id=rec.aws_access_key, + aws_secret_access_key=rec.aws_secret_access_key) + # If auto_remove is enabled, remove the backups that + # are older than specified days from the S3 bucket + if rec.auto_remove: + folder_path = rec.aws_folder_name + response = bo3.list_objects( + Bucket=rec.bucket_file_name, + Prefix=folder_path) + today = fields.date.today() + for file in response['Contents']: + file_path = file['Key'] + last_modified = file['LastModified'] + date = last_modified.date() + age_in_days = (today - date).days + if age_in_days >= rec.days_to_remove: + bo3.delete_object( + Bucket=rec.bucket_file_name, + Key=file_path) + # Create a boto3 resource for Amazon S3 with provided + # access key id and secret access key + s3 = boto3.resource( + 's3', + aws_access_key_id=rec.aws_access_key, + aws_secret_access_key=rec.aws_secret_access_key) + # Create a folder in the specified bucket, if it + # doesn't already exist + s3.Object(rec.bucket_file_name, + rec.aws_folder_name + '/').put() + bucket = s3.Bucket(rec.bucket_file_name) + # Get all the prefixes in the bucket + prefixes = set() + for obj in bucket.objects.all(): + key = obj.key + if key.endswith('/'): + prefix = key[:-1] # Remove the trailing slash + prefixes.add(prefix) + # If the specified folder is present in the bucket, + # take a backup of the database and upload it to the + # S3 bucket + if rec.aws_folder_name in prefixes: + temp = tempfile.NamedTemporaryFile( + suffix='.%s' % rec.backup_format) + with open(temp.name, "wb+") as tmp: + odoo.service.db.dump_db(rec.db_name, tmp, + rec.backup_format) + backup_file_path = temp.name + remote_file_path = f"{rec.aws_folder_name}/{rec.db_name}_" \ + f"{backup_time}.{rec.backup_format}" + s3.Object(rec.bucket_file_name, + remote_file_path).upload_file( + backup_file_path) + # If notify_user is enabled, send an email to the + # user notifying them about the successful backup + if rec.notify_user: + mail_template_success.send_mail(rec.id, + force_send=True) + except Exception as error: + # If any error occurs, set the 'generated_exception' + # field to the error message and log the error + rec.generated_exception = error + _logger.info('Amazon S3 Exception: %s', error) + # If notify_user is enabled, send an email to the user + # notifying them about the failed backup + if rec.notify_user: + mail_template_failed.send_mail(rec.id, + force_send=True) diff --git a/auto_database_backup/security/ir.model.access.csv b/auto_database_backup/security/ir.model.access.csv new file mode 100644 index 000000000..8d4ff9c61 --- /dev/null +++ b/auto_database_backup/security/ir.model.access.csv @@ -0,0 +1,3 @@ +id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink +access_db_backup_configure_user,access.db.backup.configure.user,model_db_backup_configure,base.group_user,1,1,1,1 +access_dropbox_auth_code_user,access.dropbox.auth.code.user,model_dropbox_auth_code,base.group_user,1,1,1,1 diff --git a/auto_database_backup/static/description/assets/icons/capture (1).png b/auto_database_backup/static/description/assets/icons/capture (1).png new file mode 100644 index 000000000..8824deafc Binary files /dev/null and b/auto_database_backup/static/description/assets/icons/capture (1).png differ diff --git a/auto_database_backup/static/description/assets/icons/check.png b/auto_database_backup/static/description/assets/icons/check.png new file mode 100644 index 000000000..c8e85f51d Binary files /dev/null and b/auto_database_backup/static/description/assets/icons/check.png differ diff --git a/auto_database_backup/static/description/assets/icons/chevron.png b/auto_database_backup/static/description/assets/icons/chevron.png new file mode 100644 index 000000000..2089293d6 Binary files /dev/null and b/auto_database_backup/static/description/assets/icons/chevron.png differ diff --git a/auto_database_backup/static/description/assets/icons/cogs.png b/auto_database_backup/static/description/assets/icons/cogs.png new file mode 100644 index 000000000..95d0bad62 Binary files /dev/null and b/auto_database_backup/static/description/assets/icons/cogs.png differ diff --git a/auto_database_backup/static/description/assets/icons/consultation.png b/auto_database_backup/static/description/assets/icons/consultation.png new file mode 100644 index 000000000..8319d4baa Binary files /dev/null and b/auto_database_backup/static/description/assets/icons/consultation.png differ diff --git a/auto_database_backup/static/description/assets/icons/ecom-black.png b/auto_database_backup/static/description/assets/icons/ecom-black.png new file mode 100644 index 000000000..a9385ff13 Binary files /dev/null and b/auto_database_backup/static/description/assets/icons/ecom-black.png differ diff --git a/auto_database_backup/static/description/assets/icons/education-black.png b/auto_database_backup/static/description/assets/icons/education-black.png new file mode 100644 index 000000000..3eb09b27b Binary files /dev/null and b/auto_database_backup/static/description/assets/icons/education-black.png differ diff --git a/auto_database_backup/static/description/assets/icons/hotel-black.png b/auto_database_backup/static/description/assets/icons/hotel-black.png new file mode 100644 index 000000000..130f613be Binary files /dev/null and b/auto_database_backup/static/description/assets/icons/hotel-black.png differ diff --git a/auto_database_backup/static/description/assets/icons/img.png b/auto_database_backup/static/description/assets/icons/img.png new file mode 100644 index 000000000..70197f477 Binary files /dev/null and b/auto_database_backup/static/description/assets/icons/img.png differ diff --git a/auto_database_backup/static/description/assets/icons/license.png b/auto_database_backup/static/description/assets/icons/license.png new file mode 100644 index 000000000..a5869797e Binary files /dev/null and b/auto_database_backup/static/description/assets/icons/license.png differ diff --git a/auto_database_backup/static/description/assets/icons/lifebuoy.png b/auto_database_backup/static/description/assets/icons/lifebuoy.png new file mode 100644 index 000000000..658d56ccc Binary files /dev/null and b/auto_database_backup/static/description/assets/icons/lifebuoy.png differ diff --git a/auto_database_backup/static/description/assets/icons/manufacturing-black.png b/auto_database_backup/static/description/assets/icons/manufacturing-black.png new file mode 100644 index 000000000..697eb0e9f Binary files /dev/null and b/auto_database_backup/static/description/assets/icons/manufacturing-black.png differ diff --git a/auto_database_backup/static/description/assets/icons/photo-capture.png b/auto_database_backup/static/description/assets/icons/photo-capture.png new file mode 100644 index 000000000..06c111758 Binary files /dev/null and b/auto_database_backup/static/description/assets/icons/photo-capture.png differ diff --git a/auto_database_backup/static/description/assets/icons/pos-black.png b/auto_database_backup/static/description/assets/icons/pos-black.png new file mode 100644 index 000000000..97c0f90c1 Binary files /dev/null and b/auto_database_backup/static/description/assets/icons/pos-black.png differ diff --git a/auto_database_backup/static/description/assets/icons/puzzle.png b/auto_database_backup/static/description/assets/icons/puzzle.png new file mode 100644 index 000000000..65cf854e7 Binary files /dev/null and b/auto_database_backup/static/description/assets/icons/puzzle.png differ diff --git a/auto_database_backup/static/description/assets/icons/restaurant-black.png b/auto_database_backup/static/description/assets/icons/restaurant-black.png new file mode 100644 index 000000000..4a35eb939 Binary files /dev/null and b/auto_database_backup/static/description/assets/icons/restaurant-black.png differ diff --git a/auto_database_backup/static/description/assets/icons/service-black.png b/auto_database_backup/static/description/assets/icons/service-black.png new file mode 100644 index 000000000..301ab51cb Binary files /dev/null and b/auto_database_backup/static/description/assets/icons/service-black.png differ diff --git a/auto_database_backup/static/description/assets/icons/trading-black.png b/auto_database_backup/static/description/assets/icons/trading-black.png new file mode 100644 index 000000000..9398ba2f1 Binary files /dev/null and b/auto_database_backup/static/description/assets/icons/trading-black.png differ diff --git a/auto_database_backup/static/description/assets/icons/training.png b/auto_database_backup/static/description/assets/icons/training.png new file mode 100644 index 000000000..884ca024d Binary files /dev/null and b/auto_database_backup/static/description/assets/icons/training.png differ diff --git a/auto_database_backup/static/description/assets/icons/update.png b/auto_database_backup/static/description/assets/icons/update.png new file mode 100644 index 000000000..ecbc5a01a Binary files /dev/null and b/auto_database_backup/static/description/assets/icons/update.png differ diff --git a/auto_database_backup/static/description/assets/icons/user.png b/auto_database_backup/static/description/assets/icons/user.png new file mode 100644 index 000000000..6ffb23d9f Binary files /dev/null and b/auto_database_backup/static/description/assets/icons/user.png differ diff --git a/auto_database_backup/static/description/assets/icons/wrench.png b/auto_database_backup/static/description/assets/icons/wrench.png new file mode 100644 index 000000000..6c04dea0f Binary files /dev/null and b/auto_database_backup/static/description/assets/icons/wrench.png differ diff --git a/auto_database_backup/static/description/assets/misc/Cybrosys R.png b/auto_database_backup/static/description/assets/misc/Cybrosys R.png new file mode 100644 index 000000000..da4058087 Binary files /dev/null and b/auto_database_backup/static/description/assets/misc/Cybrosys R.png differ diff --git a/auto_database_backup/static/description/assets/misc/email.svg b/auto_database_backup/static/description/assets/misc/email.svg new file mode 100644 index 000000000..15291cdc3 --- /dev/null +++ b/auto_database_backup/static/description/assets/misc/email.svg @@ -0,0 +1,33 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/auto_database_backup/static/description/assets/misc/phone.svg b/auto_database_backup/static/description/assets/misc/phone.svg new file mode 100644 index 000000000..b7bd7f251 --- /dev/null +++ b/auto_database_backup/static/description/assets/misc/phone.svg @@ -0,0 +1,3 @@ + + + diff --git a/auto_database_backup/static/description/assets/misc/star (1) 2.svg b/auto_database_backup/static/description/assets/misc/star (1) 2.svg new file mode 100644 index 000000000..5ae9f507a --- /dev/null +++ b/auto_database_backup/static/description/assets/misc/star (1) 2.svg @@ -0,0 +1,9 @@ + + + + + + + + + diff --git a/auto_database_backup/static/description/assets/misc/support (1) 1.svg b/auto_database_backup/static/description/assets/misc/support (1) 1.svg new file mode 100644 index 000000000..7d37a8f30 --- /dev/null +++ b/auto_database_backup/static/description/assets/misc/support (1) 1.svg @@ -0,0 +1,9 @@ + + + + + + + + + diff --git a/auto_database_backup/static/description/assets/misc/support-email.svg b/auto_database_backup/static/description/assets/misc/support-email.svg new file mode 100644 index 000000000..eb70370d6 --- /dev/null +++ b/auto_database_backup/static/description/assets/misc/support-email.svg @@ -0,0 +1,6 @@ + + + + + + diff --git a/auto_database_backup/static/description/assets/misc/tick-mark.svg b/auto_database_backup/static/description/assets/misc/tick-mark.svg new file mode 100644 index 000000000..2dbb40187 --- /dev/null +++ b/auto_database_backup/static/description/assets/misc/tick-mark.svg @@ -0,0 +1,17 @@ + + + + + + + + + + + + + + + + + diff --git a/auto_database_backup/static/description/assets/misc/whatsapp 1.svg b/auto_database_backup/static/description/assets/misc/whatsapp 1.svg new file mode 100644 index 000000000..0bfaf8fc6 --- /dev/null +++ b/auto_database_backup/static/description/assets/misc/whatsapp 1.svg @@ -0,0 +1,9 @@ + + + + + + + + + diff --git a/auto_database_backup/static/description/assets/misc/whatsapp.svg b/auto_database_backup/static/description/assets/misc/whatsapp.svg new file mode 100644 index 000000000..b618aea1d --- /dev/null +++ b/auto_database_backup/static/description/assets/misc/whatsapp.svg @@ -0,0 +1,33 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/auto_database_backup/static/description/assets/modules/1.png b/auto_database_backup/static/description/assets/modules/1.png new file mode 100644 index 000000000..ba1058c42 Binary files /dev/null and b/auto_database_backup/static/description/assets/modules/1.png differ diff --git a/auto_database_backup/static/description/assets/modules/2.png b/auto_database_backup/static/description/assets/modules/2.png new file mode 100644 index 000000000..6949185dd Binary files /dev/null and b/auto_database_backup/static/description/assets/modules/2.png differ diff --git a/auto_database_backup/static/description/assets/modules/3.png b/auto_database_backup/static/description/assets/modules/3.png new file mode 100644 index 000000000..4e506f79b Binary files /dev/null and b/auto_database_backup/static/description/assets/modules/3.png differ diff --git a/auto_database_backup/static/description/assets/modules/4.png b/auto_database_backup/static/description/assets/modules/4.png new file mode 100644 index 000000000..e78427938 Binary files /dev/null and b/auto_database_backup/static/description/assets/modules/4.png differ diff --git a/auto_database_backup/static/description/assets/modules/5.png b/auto_database_backup/static/description/assets/modules/5.png new file mode 100755 index 000000000..272ec20f9 Binary files /dev/null and b/auto_database_backup/static/description/assets/modules/5.png differ diff --git a/auto_database_backup/static/description/assets/modules/6.png b/auto_database_backup/static/description/assets/modules/6.png new file mode 100644 index 000000000..7d5c3154f Binary files /dev/null and b/auto_database_backup/static/description/assets/modules/6.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/1.png b/auto_database_backup/static/description/assets/screenshots/1.png new file mode 100644 index 000000000..b7490a49c Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/1.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/10.png b/auto_database_backup/static/description/assets/screenshots/10.png new file mode 100644 index 000000000..6f914485d Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/10.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/11.png b/auto_database_backup/static/description/assets/screenshots/11.png new file mode 100644 index 000000000..32ad3ad0f Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/11.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/12.png b/auto_database_backup/static/description/assets/screenshots/12.png new file mode 100644 index 000000000..e8a1e785b Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/12.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/13.png b/auto_database_backup/static/description/assets/screenshots/13.png new file mode 100644 index 000000000..a392ba881 Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/13.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/14.png b/auto_database_backup/static/description/assets/screenshots/14.png new file mode 100644 index 000000000..773a4bced Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/14.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/15.png b/auto_database_backup/static/description/assets/screenshots/15.png new file mode 100644 index 000000000..58937c0ea Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/15.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/16.png b/auto_database_backup/static/description/assets/screenshots/16.png new file mode 100644 index 000000000..6705babdb Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/16.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/17.png b/auto_database_backup/static/description/assets/screenshots/17.png new file mode 100644 index 000000000..b13c778ab Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/17.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/18.png b/auto_database_backup/static/description/assets/screenshots/18.png new file mode 100644 index 000000000..93706bd3b Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/18.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/19.png b/auto_database_backup/static/description/assets/screenshots/19.png new file mode 100644 index 000000000..bea461c43 Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/19.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/2.png b/auto_database_backup/static/description/assets/screenshots/2.png new file mode 100644 index 000000000..12b3d80a6 Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/2.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/20.png b/auto_database_backup/static/description/assets/screenshots/20.png new file mode 100644 index 000000000..2d10e67d2 Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/20.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/21.png b/auto_database_backup/static/description/assets/screenshots/21.png new file mode 100644 index 000000000..9c69cedd6 Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/21.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/22.png b/auto_database_backup/static/description/assets/screenshots/22.png new file mode 100644 index 000000000..f087126aa Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/22.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/23.png b/auto_database_backup/static/description/assets/screenshots/23.png new file mode 100644 index 000000000..1b86ecc73 Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/23.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/24.png b/auto_database_backup/static/description/assets/screenshots/24.png new file mode 100644 index 000000000..c9e894c5e Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/24.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/25.png b/auto_database_backup/static/description/assets/screenshots/25.png new file mode 100644 index 000000000..a7a4a58ca Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/25.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/26.png b/auto_database_backup/static/description/assets/screenshots/26.png new file mode 100644 index 000000000..6cf91ec4a Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/26.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/27.png b/auto_database_backup/static/description/assets/screenshots/27.png new file mode 100644 index 000000000..6c174dbb9 Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/27.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/28.png b/auto_database_backup/static/description/assets/screenshots/28.png new file mode 100644 index 000000000..bd0b7f59a Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/28.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/29.png b/auto_database_backup/static/description/assets/screenshots/29.png new file mode 100644 index 000000000..7738ec9f0 Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/29.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/3.png b/auto_database_backup/static/description/assets/screenshots/3.png new file mode 100644 index 000000000..2cf9f727a Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/3.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/30.png b/auto_database_backup/static/description/assets/screenshots/30.png new file mode 100644 index 000000000..d543390f1 Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/30.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/31.png b/auto_database_backup/static/description/assets/screenshots/31.png new file mode 100644 index 000000000..a0fbb2f36 Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/31.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/32.png b/auto_database_backup/static/description/assets/screenshots/32.png new file mode 100644 index 000000000..2548aa71e Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/32.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/33.png b/auto_database_backup/static/description/assets/screenshots/33.png new file mode 100644 index 000000000..b213b1145 Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/33.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/34.png b/auto_database_backup/static/description/assets/screenshots/34.png new file mode 100644 index 000000000..32db0e515 Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/34.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/35.png b/auto_database_backup/static/description/assets/screenshots/35.png new file mode 100644 index 000000000..75fd2e39c Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/35.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/36.png b/auto_database_backup/static/description/assets/screenshots/36.png new file mode 100644 index 000000000..65b31b43d Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/36.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/37.png b/auto_database_backup/static/description/assets/screenshots/37.png new file mode 100644 index 000000000..52357df9a Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/37.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/38.png b/auto_database_backup/static/description/assets/screenshots/38.png new file mode 100644 index 000000000..5fcd336d6 Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/38.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/39.png b/auto_database_backup/static/description/assets/screenshots/39.png new file mode 100644 index 000000000..d259da8a9 Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/39.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/4.png b/auto_database_backup/static/description/assets/screenshots/4.png new file mode 100644 index 000000000..8d730f854 Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/4.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/40.png b/auto_database_backup/static/description/assets/screenshots/40.png new file mode 100644 index 000000000..270d27035 Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/40.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/41.png b/auto_database_backup/static/description/assets/screenshots/41.png new file mode 100644 index 000000000..22bbce54c Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/41.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/42.png b/auto_database_backup/static/description/assets/screenshots/42.png new file mode 100644 index 000000000..c50a20cc8 Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/42.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/43.png b/auto_database_backup/static/description/assets/screenshots/43.png new file mode 100644 index 000000000..415d521c4 Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/43.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/44.png b/auto_database_backup/static/description/assets/screenshots/44.png new file mode 100644 index 000000000..284490127 Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/44.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/45.png b/auto_database_backup/static/description/assets/screenshots/45.png new file mode 100644 index 000000000..5e91106df Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/45.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/46.png b/auto_database_backup/static/description/assets/screenshots/46.png new file mode 100644 index 000000000..f9345a6d8 Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/46.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/47.png b/auto_database_backup/static/description/assets/screenshots/47.png new file mode 100644 index 000000000..1f435f9ae Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/47.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/48.png b/auto_database_backup/static/description/assets/screenshots/48.png new file mode 100644 index 000000000..034cf59f6 Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/48.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/49.png b/auto_database_backup/static/description/assets/screenshots/49.png new file mode 100644 index 000000000..05e7392a5 Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/49.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/5.png b/auto_database_backup/static/description/assets/screenshots/5.png new file mode 100644 index 000000000..38edc89ae Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/5.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/50.png b/auto_database_backup/static/description/assets/screenshots/50.png new file mode 100644 index 000000000..5e8a6bb8e Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/50.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/51.png b/auto_database_backup/static/description/assets/screenshots/51.png new file mode 100644 index 000000000..6a886eee0 Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/51.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/52.png b/auto_database_backup/static/description/assets/screenshots/52.png new file mode 100644 index 000000000..274a561ec Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/52.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/6.png b/auto_database_backup/static/description/assets/screenshots/6.png new file mode 100644 index 000000000..150ba0dca Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/6.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/7.png b/auto_database_backup/static/description/assets/screenshots/7.png new file mode 100644 index 000000000..7c86f058b Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/7.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/8.png b/auto_database_backup/static/description/assets/screenshots/8.png new file mode 100644 index 000000000..9c78554f1 Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/8.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/9.png b/auto_database_backup/static/description/assets/screenshots/9.png new file mode 100644 index 000000000..1864565b8 Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/9.png differ diff --git a/auto_database_backup/static/description/assets/screenshots/hero.gif b/auto_database_backup/static/description/assets/screenshots/hero.gif new file mode 100644 index 000000000..0b90ff3f2 Binary files /dev/null and b/auto_database_backup/static/description/assets/screenshots/hero.gif differ diff --git a/auto_database_backup/static/description/banner.gif b/auto_database_backup/static/description/banner.gif new file mode 100644 index 000000000..8ac32db03 Binary files /dev/null and b/auto_database_backup/static/description/banner.gif differ diff --git a/auto_database_backup/static/description/icon.png b/auto_database_backup/static/description/icon.png new file mode 100644 index 000000000..6f21cf7ec Binary files /dev/null and b/auto_database_backup/static/description/icon.png differ diff --git a/auto_database_backup/static/description/index.html b/auto_database_backup/static/description/index.html new file mode 100644 index 000000000..3d4e93bc3 --- /dev/null +++ b/auto_database_backup/static/description/index.html @@ -0,0 +1,1266 @@ + + + + + + + Odoo App 3 Index + + + + + + + + +
+
+
+
+
+ +
+
+
+ Community +
+
+ Enterprise +
+
+
+
+
+
+

+ Automatic Database Backup To Local Server, Remote Server, + Google Drive, Dropbox, Onedrive, Nextcloud and Amazon + S3

+

+ This Module Helps To Generate Backup Of Databases. +

+
+ +
+
+
+
+
+

+ Key Highlights +

+
+
+
+
+
+ +
+
+

+ Generate Database Backups on regular + intervals

+ + +
+
+
+
+
+
+ +
+
+

+ Notify user on success and failure of backup + generation

+ + + +
+
+
+
+
+
+ +
+
+

+ Automatically remove old backups

+ + +
+
+
+
+
+
+ +
+
+

+ Store Backup to Different Locations

+

Store to + Remote Servers, Nextcloud, Amazon S3, Google + Drive, Dropbox, Onedrive +

+
+
+
+
+
+
+ +
+
+
+
+
+ +
+
+

+ Database Backup Configuration Menu.

+

+ Go to Setting --> Technical --> Backup + Configuration to configure + backups

+
+
+
+
+
+
+ +
+
+

+ Create New Database Backup + Configuration.

+

+ Enter the Database Name and Master + Password. Specify Backup Destination. + Enter the backup directory path, if + directory does not exist new + directory will be created.

+
+
+
+
+
+
+ +
+
+

+ Store Backup to + Remote + SFTP Server.

+

+ Select Backup Destination as SFTP, enter + credentials. + "TEST CONNECTION" button to check whether + the connection is + successful.

+
+
+
+
+
+
+ +
+
+

+ Store Backup to + Remote + FTP Server.

+

+ Select Backup + Destination + as FTP, enter credentials. + "TEST CONNECTION" button to check whether + the connection is + successful.

+
+
+
+
+
+
+ +
+
+

+ Store Backup to + Google + Drive

+

+ You'll need to create a + new + Google API project and enabling the Google + Drive API, Go to the + Google API Console + and log into your + account. + While creating the project, for the + Redirect URI restrictions, + copy your Odoo database URI followed by + /google_drive/authentication. Example:

+ +

+ ENABLE API AND SERVICES

+ +

+ ENALBE GOOGLE DRIVE API

+ +

+ Create Credentials, + Follow + the steps, select Website application for + the Application + Type.

+ +

+ Under the Authorized + JavaScript Origins section, click + Add URI + and type your + company's Odoo URL address. + Under the Authorized redirect URIs section, + click + Add URI and + type your company's Odoo URL address + followed + by /google_drive/authentication. + After all the steps are completed, A Client + ID and Client secret will + be given, copy the credentials +

+ + +

+ Go to the "OAuth consent screen", then Test + users and click on 'ADD USERS' then add the + user. + +

+ +

+ Configure Backup, Copy + Client ID and Client Secret from Google + Drive API Credentials + page into their respective fields.

+ +

+ Setup Token, it will + be + redirected to an authorization page.

+ +
+ +

+ Reset the token if + required by clicking on the "Reset + Token"

+
+
+
+
+
+
+ +
+
+

+ Store Backup to Dropbox.

+

+ To get the app key and secret key go to the + App + Console. + Create a new app +

+ + + Once you created the App , you can get the App + key and App Secret as seen in the screenshot + + Choose your app's permission + (files.content.write and files.content.read + permissions required).then click on Submit + + Choose Dropbox as that of the backup + destination. Enter the app secret and key and + dropbox Folder. + +

+ Click on the Setup Token

+ +

+ Get the Authorization + Code + and click confirm.

+ +

+ Reset the refresh token + if + required by clicking on the "Reset + Token"

+
+
+
+
+
+
+ +
+
+

+ Store Backup to Onedrive.

+

+ Select Backup + Destination + as OneDrive. Enter the App key and App + secret. + you'll need to register a new app in the Microsoft + Azure + portal. + While registering the app for the Redirect + URI restrictions, + copy your Odoo database URI followed by + /onedrive/authentication. Example:

+ +

+ Copy the Client ID

+ +

+ Generate Client + Secret.

+ + +

+ Get OneDrive folder ID, + where need to store the backup files.

+ +

+ Configure the + Backup

+ +

+ Setup Token, it will + be + redirected to an authorization page.

+ + +

+ Reset the token if + required by clicking on the "Reset Token" +

+
+
+
+
+
+
+ +
+
+

+ Store Backup to Nextcloud.

+

+ To Create an account in Nextcloud go to + https://nextcloud.com/sign-up/, Enter Your + Email Address and + Sign up . + +

+ You will be redirected to the page as shown + in the screenshot, + and it will ask you enter your email and + password for the + Nextcloud.

+ +

+ To get the Domain of the Nextcloud.Go to + Settings in the + Nextcloud and Click on Mobile & desktop. + You will see server + address Copy link and paste it in your + Domain Name.

+ +

+ Select the backup destination as Nextcloud. + Enter the Domain + Name, UserName, Password and Folder Name + where you want to store + your backup on the NextCloud server. Check + the Connect button + to check if the connection is successful. + +

+ Every day, a Scheduled Action will take + place to store a backup + on the Nextcloud Server. The backup will be + stored as the folder + name provided in the Folder ID field in + Odoo. + + +

+
+
+
+
+
+ +
+
+

+ Store Backup to Amazon S3.

+

+ To Create an account in Amazon S3 go to + https://portal.aws.amazon.com/billing/signup#/start/email, + Enter + Your Email Address and Sign up . + +

+ After you created the account.You need to + get the Access Key and + Secret Key,To get these go the account + Security credentails and + go the Access Keys and create new access + keys from there you + will get Access Key and Secret Key.

+ +

+ +

+ +

+ Next You need to create a Bucket Folder in + the Amazon S3.To do + that Go the Services in the top right and + go to Storage and S3 + as shown in the Screen shot.

+ +

+ To create Bucket folder,Click on the Create + bucket Button.

+ +

+ On Creating a Bucket Folder,Check the rules + for naming the + Bucket folder,and Select the region as + well.After that click on + the create Bucket Button in the bottom of + the page.

+ +

+ You will see the Bucket Folder as shown in + the screenshot.

+ +

+ Select Backup Destination as Amazon S3. + Enter the Amazon S3 + Access Key,Amazon S3 Secret Key,Bucket + Name->Bucket folder you + have created in the Amazon S3 and the File + Name->The Folder You + want to store your backup in Amazon S3 + Bucket Folder.Test + connection button to check whether the + connection is + successful.

+ +

+ Every day, a Scheduled Action will take + place to store a backup + on the Amazon S3 Server. The backup will be + stored as the folder + name provided in the File Name field in + Odoo.

+
+
+
+
+
+
+ +
+
+

+ Automatically Remove + Old + Backups.

+

+ Enable auto remove + option, + specify number of days to remove + backups.

+
+
+
+
+
+
+ +
+
+

+ Notify User on + Success + and Failure of Backup Generation.

+

+ Enable notify user + option, + and select a user to notify. An email + notification will be sent + to the selected user on + backup successful and failure.

+
+
+
+
+
+
+ +
+
+

+ Successful backup + notification email.

+
+
+
+
+
+
+ +
+
+

+ Notification email + when + backup generation failed.

+
+
+
+
+
+
+ +
+
+ +
+
+

+ Scheduled Action For + Generating Backup.

+

+ Enable the 'Automatic + database Backup' scheduled action, and set + up the execution + interval. + Based on the scheduled action setup, + backups will be generated + on regular intervals.

+
+
+
+
+
+
+
    +
  • + Generate Database Backups on regular intervals. +
  • +
  • + Notify user on success and failure of backup generation. +
  • +
  • + Automatically remove old backups. +
  • +
  • + Store Backup to Different Locations. +
      +
    • Store to Remote Servers, Nextcloud, Amazon S3, Google Drive, Dropbox, Onedrive. +
    • +
    +
  • +
+
+
+
+
+
+
Version + 17.0.1.0.0|Released on: 6th November 2023 +
+

+ Initial Commit for Automatic Database + Backup To Local Server, Remote Server, + Google Drive, Dropbox, Onedrive, Nextcloud + and Amazon S3.

+
+
+
+
+
+
+

+ Related Products

+
+
+ +
+
+

+ Our Services

+ +
+
+
+
+
+
+
+
+ service-icon +
+
+

+ Odoo + Customization

+
+
+
+
+
+
+ service-icon +
+
+

+ Odoo + Implementation

+
+
+
+
+
+
+ service-icon +
+
+

+ Odoo + Support

+
+
+
+
+
+
+ service-icon +
+
+

+ Hire + Odoo Developer

+
+
+
+
+ +
+
+ service-icon +
+
+

+ Odoo + Integration

+
+
+
+
+
+
+ service-icon +
+
+

+ Odoo + Migration

+
+
+
+
+
+
+ service-icon +
+
+

+ Odoo + Consultancy

+
+
+
+
+
+
+ service-icon +
+
+

+ Odoo + Implementation

+
+
+
+
+
+
+ service-icon +
+
+

+ Odoo + Licensing Consultancy

+
+
+
+
+
+
+

+ Our Industries

+ +
+
+
+
+
+
+ +

Trading

+

Easily procure and sell your products

+
+
+
+
+ +

POS

+

Easy configuration and convivial experience

+
+
+
+
+ +

+ Education

+

A platform for educational management

+
+
+
+
+ +

+ Manufacturing

+

Plan, track and schedule your operations

+
+
+
+
+ +

E-commerce + & Website

+

Mobile friendly, awe-inspiring product pages

+
+
+
+
+ +

Service + Management

+

Keep track of services and invoice

+
+
+
+
+ +

+ Restaurant

+

Run your bar or restaurant methodically

+
+
+
+
+ +

Hotel + Management

+

An all-inclusive hotel management + application

+
+
+
+
+
+
+
+

+ Support

+
+
+
+
+
+
+
+ +
+ Need + Help? +

Got + questions or need help? Get in + touch.

+
odoo@cybrosys.com +
+
+
+
+
+
+
+
+ +
+ WhatsApp +

Say + hi to + us on WhatsApp!

+
+91 + 99456767686 +
+
+
+
+
+
+
+
+
+ + + + + + diff --git a/auto_database_backup/views/db_backup_configure_views.xml b/auto_database_backup/views/db_backup_configure_views.xml new file mode 100644 index 000000000..d63b0a470 --- /dev/null +++ b/auto_database_backup/views/db_backup_configure_views.xml @@ -0,0 +1,311 @@ + + + + + db.backup.configure.view.tree + db.backup.configure + + + + + + + + + + + + db.backup.configure.view.form + db.backup.configure + +
+ +
+

+ +

+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+ + Refresh token set +
+
+ + No refresh token set +
+
+
+
+ +
+
+ +
+
+
+
+ + Refresh token set +
+
+ + No refresh token set +
+
+
+
+ +
+
+ +
+
+
+
+ + Refresh token set +
+
+ + No refresh token set +
+
+
+
+ +
+
+ +
+
+
+ + + + + + + + + +