File: //usr/local/CyberCP/plogical/cPanelImporter.py
#!/usr/local/CyberCP/bin/python
import os.path
import sys
import django
sys.path.append('/usr/local/CyberCP')
os.environ.setdefault("DJANGO_SETTINGS_MODULE", "CyberCP.settings")
django.setup()
import argparse
from plogical.processUtilities import ProcessUtilities
from random import randint
from plogical.CyberCPLogFileWriter import CyberCPLogFileWriter as logging
import re
import shutil
from dns.models import Domains, Records
from manageServices.models import PDNSStatus
from loginSystem.models import Administrator
from plogical.dnsUtilities import DNS
import MySQLdb as mysql
import MySQLdb.cursors as cursors
import shlex
import subprocess
from databases.models import Databases
from websiteFunctions.models import Websites, ChildDomains as CDomains
from plogical.vhost import vhost
from plogical.virtualHostUtilities import virtualHostUtilities
from plogical.mailUtilities import mailUtilities
from mailServer.models import EUsers
import time
class ChildDomains:
    def __init__(self, domain, addon):
        self.domain = domain
        self.addon = addon
class cPanelImporter:
    MailDir = 1
    MdBox = 0
    mainBackupPath = '/home/backup/'
    def __init__(self, backupFile, logFile):
        self.backupFile = backupFile
        self.fileName = backupFile.split('/')[-1].replace('.tar.gz', '')
        self.logFile = logFile
        self.PHPVersion = ''
        self.email = ''
        self.mainDomain = ''
        self.homeDir = ''
        self.documentRoot = ''
        self.mailFormat = 1
        self.externalApp = ''
        ## New
        self.MainSite = []
        self.OtherDomains = []
        self.OtherDomainNames = []
        self.InheritPHP = ''
    def LoadDomains(self):
        try:
            ###
            CompletPathToExtractedArchive = cPanelImporter.mainBackupPath + self.fileName
            ### Find Domain Name
            import json
            UserData = json.loads(open('%s/userdata/cache.json' % (CompletPathToExtractedArchive), 'r').read())
            for key, value in UserData.items():
                if value[2] == 'main':
                    self.MainSite = value
                    self.PHPVersion = value[9]
                    self.InheritPHP = self.PHPDecider(None)
                    if os.path.exists(ProcessUtilities.debugPath):
                        print(f'Main domain: {self.MainSite}, MainSite php version or version to be used for sites with no PHP {self.InheritPHP}')
                else:
                    self.OtherDomainNames.append(key)
                    self.OtherDomains.append(value)
                    if os.path.exists(ProcessUtilities.debugPath):
                        print(f'Other domains: {key}, Value {str(value)}')
        except BaseException as msg:
            print(str(msg))
    def PHPDecider(self, domainName):
        if self.PHPVersion == 'inherit' or not self.PHPVersion:
            self.PHPVersion = self.InheritPHP or 'PHP 7.4'
        else:
            version_number = ''.join(filter(str.isdigit, self.PHPVersion))
            if len(version_number) == 2:  # Ensure there are exactly two digits
                self.PHPVersion = f'PHP {version_number[0]}.{version_number[1]}'
            ### if the PHP Version extracted from file is not available then change it to next available
            try:
                from plogical.phpUtilities import phpUtilities
                if domainName !=None:
                    completePathToConfigFile = f'/usr/local/lsws/conf/vhosts/{domainName}/vhost.conf'
                else:
                    completePathToConfigFile = None
                phpVersion = phpUtilities.FindIfSaidPHPIsAvaiableOtherwiseMaketheNextOneAvailableToUse(completePathToConfigFile, self.PHPVersion)
                if phpVersion != self.PHPVersion:
                    logging.statusWriter(self.logFile, f'PHP version for {self.mainDomain} has been changed from {self.PHPVersion} to {phpVersion}.', 1)
                    self.PHPVersion = phpVersion
            except:
                pass
            return self.PHPVersion
    def SetupSSL(self, path, domain):
        data = open(path, 'r').readlines()
        Key = []
        Cert = []
        KeyCheck = 1
        CertCheck = 0
        for items in data:
            if KeyCheck == 1 and items.find('-----END RSA PRIVATE KEY-----') > -1:
                KeyCheck = 0
                CertCheck = 1
                Key.append(items)
                continue
            else:
                Key.append(items)
            if CertCheck == 1:
                Cert.append(items)
        KeyPath = '/home/cyberpanel/%s' % (str(randint(1000, 9999)))
        writeToFile = open(KeyPath, 'w')
        for items in Key:
            writeToFile.writelines(items)
        writeToFile.close()
        ##
        CertPath = '/home/cyberpanel/%s' % (str(randint(1000, 9999)))
        writeToFile = open(CertPath, 'w')
        for items in Cert:
            writeToFile.writelines(items)
        writeToFile.close()
        virtualHostUtilities.saveSSL(domain, KeyPath, CertPath)
    def ExtractBackup(self):
        try:
            message = 'Extracting main cPanel archive file: %s' % (self.backupFile)
            logging.statusWriter(self.logFile, message, 1)
            if not os.path.exists(cPanelImporter.mainBackupPath):
                os.mkdir(cPanelImporter.mainBackupPath)
            os.chdir(cPanelImporter.mainBackupPath)
            command = 'tar -xf %s --directory %s' % (self.backupFile, cPanelImporter.mainBackupPath)
            ProcessUtilities.normalExecutioner(command)
            message = '%s successfully extracted.' % (self.backupFile)
            logging.statusWriter(self.logFile, message, 1)
            return 1
        except BaseException as msg:
            message = 'Failed to extract backup for file %s, error message: %s. [ExtractBackup]' % (
            self.backupFile, str(msg))
            logging.statusWriter(self.logFile, message, 1)
            return 0
    def CreateMainWebsite(self):
        try:
            message = 'Creating main account from archive file: %s' % (self.backupFile)
            logging.statusWriter(self.logFile, message, 1)
            ## Paths
            DomainName = self.MainSite[3]
            self.mainDomain = DomainName
            CompletPathToExtractedArchive = cPanelImporter.mainBackupPath + self.fileName
            DomainMeta = '%s/userdata/%s' % (CompletPathToExtractedArchive, DomainName)
            ### Find Domain Name
            message = 'Detected main domain for this file is: %s.' % (DomainName)
            logging.statusWriter(self.logFile, message, 1)
            ## Find PHP Version
            message = 'Finding PHP version for %s.' % (DomainName)
            logging.statusWriter(self.logFile, message, 1)
            self.PHPVersion = self.MainSite[9]
            self.PHPDecider(None)
            message = 'PHP version of %s is %s.' % (DomainName, self.PHPVersion)
            logging.statusWriter(self.logFile, message, 1)
            ## Find Email
            message = 'Finding Server Admin email for %s.' % (DomainName)
            logging.statusWriter(self.logFile, message, 1)
            data = open(DomainMeta, 'r').readlines()
            for items in data:
                if items.find('serveradmin') > -1:
                    self.email = items.split(' ')[-1].replace('\n', '')
                    break
            message = 'Server Admin email for %s is %s.' % (DomainName, self.email)
            logging.statusWriter(self.logFile, message, 1)
            ## Create Site
            message = 'Calling core to create %s.' % (DomainName)
            logging.statusWriter(self.logFile, message, 1)
            self.externalApp = "".join(re.findall("[a-zA-Z]+", DomainName))[:7]
            try:
                counter = 0
                while True:
                    tWeb = Websites.objects.get(externalApp=self.externalApp)
                    self.externalApp = '%s%s' % (tWeb.externalApp, str(counter))
                    counter = counter + 1
                    print(self.externalApp)
            except BaseException as msg:
                logging.statusWriter(self.logFile, str(msg), 1)
                time.sleep(2)
            result = virtualHostUtilities.createVirtualHost(DomainName, self.email, self.PHPVersion, self.externalApp, 1, 0,
                                                            0, 'admin', 'Default', 0)
            if result[0] == 1:
                pass
            else:
                message = f'Failed to create main site %s from archive file: %s. Error {str(result)}' % (DomainName, self.backupFile)
                logging.statusWriter(self.logFile, message, 1)
                return 0
            message = 'Successfully created %s from core.' % (DomainName)
            logging.statusWriter(self.logFile, message, 1)
            ### Let see if there is SSL
            message = 'Detecting SSL for %s.' % (DomainName)
            logging.statusWriter(self.logFile, message, 1)
            SSLPath = '%s/apache_tls/%s' % (CompletPathToExtractedArchive, DomainName)
            if os.path.exists(SSLPath):
                message = 'SSL found for %s, setting up.' % (DomainName)
                logging.statusWriter(self.logFile, message, 1)
                self.SetupSSL(SSLPath, DomainName)
                message = 'SSL set up OK for %s.' % (DomainName)
                logging.statusWriter(self.logFile, message, 1)
            else:
                message = 'SSL not detected for %s, you can later issue SSL from Manage SSL in CyberPanel.' % (DomainName)
                logging.statusWriter(self.logFile, message, 1)
            ## Document root
            message = 'Restoring document root files for %s.' % (DomainName)
            logging.statusWriter(self.logFile, message, 1)
            message = 'self.MainSite[4] %s.' % (self.MainSite[4])
            logging.statusWriter(self.logFile, message, 1)
            message = 'self.MainSite[0] %s.' % (self.MainSite[0])
            logging.statusWriter(self.logFile, message, 1)
            if self.MainSite[4].find('home/') > -1:
                self.homeDir = self.MainSite[4].replace('/home/%s/' % (self.MainSite[0]), '')
            else:
                self.homeDir = self.MainSite[4].replace('/home2/%s/' % (self.MainSite[0]), '')
            message = 'self.homeDir %s.' % (self.homeDir)
            logging.statusWriter(self.logFile, message, 1)
            nowPath = '/home/%s/public_html' % (DomainName)
            if os.path.exists(nowPath):
                shutil.rmtree(nowPath)
            movePath = '%s/homedir/%s' % (
            CompletPathToExtractedArchive, self.homeDir)
            if os.path.exists(ProcessUtilities.debugPath):
                logging.statusWriter(self.logFile, f'Directory from where docRoot of main site data will be moved {movePath}')
            shutil.copytree(movePath, nowPath, symlinks=True)
            message = 'Main site %s created from archive file: %s' % (DomainName, self.backupFile)
            logging.statusWriter(self.logFile, message, 1)
            return 1
        except BaseException as msg:
            message = 'Failed to create main website from backup file %s, error message: %s.' % (
            self.backupFile, str(msg))
            logging.statusWriter(self.logFile, message, 1)
            return 0
    def CreateChildDomains(self):
        try:
            message = 'Creating child domains from archive file: %s' % (self.backupFile)
            logging.statusWriter(self.logFile, message, 1)
            CompletPathToExtractedArchive = cPanelImporter.mainBackupPath + self.fileName
            ### Find Possible Child Domains
            message = 'Finding Addon/Subdomains from backup file %s. Account main domain was %s.' % (self.backupFile, self.mainDomain)
            logging.statusWriter(self.logFile, message, 1)
            message = 'Following Addon/Subdomains found for backup file %s. Account main domain was %s.' % (
            self.backupFile, self.mainDomain)
            logging.statusWriter(self.logFile, message, 1)
            for items in self.OtherDomainNames:
                print(items)
            ## Starting Child-domains creation
            message = 'Starting Addon/Subdomains creation from backup file %s. Account main domain was %s.' % (
                self.backupFile, self.mainDomain)
            logging.statusWriter(self.logFile, message, 1)
            counter = 0
            for items in self.OtherDomainNames:
                try:
                    message = 'Creating %s.' % (items)
                    logging.statusWriter(self.logFile, message, 1)
                    path = '/home/' + self.mainDomain + '/' + items
                    ## Find PHP Version
                    self.PHPVersion = self.OtherDomains[counter][9]
                    self.PHPDecider(None)
                    message = 'Calling core to create %s.' % (items)
                    logging.statusWriter(self.logFile, message, 1)
                    result = virtualHostUtilities.createDomain(self.mainDomain, items, self.PHPVersion, path, 1, 0,
                                                               0, 'admin', 0)
                    if result[0] == 1:
                        message = 'Child domain %s created from archive file: %s' % (items, self.backupFile)
                        logging.statusWriter(self.logFile, message, 1)
                    else:
                        message = 'Failed to create Child domain %s from archive file: %s' % (items, self.backupFile)
                        logging.statusWriter(self.logFile, message, 1)
                    ## Setup SSL
                    message = 'Detecting SSL for %s.' % (items)
                    logging.statusWriter(self.logFile, message, 1)
                    SSLPath = '%s/apache_tls/%s' % (CompletPathToExtractedArchive, items)
                    if os.path.exists(SSLPath):
                        message = 'SSL found for %s, setting up.' % (items)
                        logging.statusWriter(self.logFile, message, 1)
                        self.SetupSSL(SSLPath, items)
                        message = 'SSL set up OK for %s.' % (items)
                        logging.statusWriter(self.logFile, message, 1)
                    else:
                        SSLPath = '%s/apache_tls/%s.%s' % (CompletPathToExtractedArchive, items, self.mainDomain)
                        if os.path.exists(SSLPath):
                            message = 'SSL found for %s, setting up.' % (items)
                            logging.statusWriter(self.logFile, message, 1)
                            self.SetupSSL(SSLPath, items)
                            message = 'SSL set up OK for %s.' % (items)
                            logging.statusWriter(self.logFile, message, 1)
                        else:
                            message = 'SSL not detected for %s, you can later issue SSL from Manage SSL in CyberPanel.' % (
                                items)
                            logging.statusWriter(self.logFile, message, 1)
                    ## Creating Document root for childs
                    message = 'Restoring document root files for %s.' % (items)
                    logging.statusWriter(self.logFile, message, 1)
                    message = 'self.OtherDomains[counter][4] %s.' % (self.OtherDomains[counter][4])
                    logging.statusWriter(self.logFile, message, 1)
                    message = 'self.MainSite[0] %s.' % (self.MainSite[0])
                    logging.statusWriter(self.logFile, message, 1)
                    if self.OtherDomains[counter][4].find('home/') > -1:
                        ChildDocRoot = self.OtherDomains[counter][4].replace('/home/%s/' % (self.MainSite[0]), '')
                    else:
                        ChildDocRoot = self.OtherDomains[counter][4].replace('/home2/%s/' % (self.MainSite[0]), '')
                    #ChildDocRoot = self.OtherDomains[counter][4].replace('/home/%s/' % (self.MainSite[0]), '')
                    message = 'ChildDocRoot %s.' % (ChildDocRoot)
                    logging.statusWriter(self.logFile, message, 1)
                    if os.path.exists(path):
                        shutil.rmtree(path)
                    movePath = '%s/homedir/%s' % (CompletPathToExtractedArchive, ChildDocRoot)
                    logging.statusWriter(self.logFile, 'Document root in cPanel Backup for %s is %s' % (items, movePath), 1)
                    if os.path.exists(movePath):
                        shutil.copytree(movePath, path)
                    message = 'Successfully created child domain.'
                    logging.statusWriter(self.logFile, message, 1)
                    counter = counter + 1
                except BaseException as msg:
                    message = 'Failed to create child domain from backup file %s, error message: %s. Moving on..' % (
                        self.backupFile, str(msg))
                    logging.statusWriter(self.logFile, message, 1)
                    counter = counter + 1
            return 1
        except BaseException as msg:
            message = 'Failed to create child domain from backup file %s, error message: %s.' % (
            self.backupFile, str(msg))
            logging.statusWriter(self.logFile, message, 1)
            return 0
    def createDummyChild(self, childDomain):
        path = '/home/%s/public_html/%s' % (self.mainDomain, childDomain)
        virtualHostUtilities.createDomain(self.mainDomain, childDomain, self.PHPVersion, path, 0, 0,
                                                   0, 'admin', 0)
    def CreateDNSRecords(self):
        try:
            message = 'We are going to create DNS records now, please note we will not create DKIM records. Make sure to create them from CyberPanel interface using our DKIM manager.'
            logging.statusWriter(self.logFile, message, 1)
            ipFile = "/etc/cyberpanel/machineIP"
            f = open(ipFile)
            ipData = f.read()
            ipAddress = ipData.split('\n', 1)[0]
            admin = Administrator.objects.get(userName='admin')
            CompletPathToExtractedArchive = cPanelImporter.mainBackupPath + self.fileName
            DNSZonesPath = '%s/dnszones' % (CompletPathToExtractedArchive)
            for items in os.listdir(DNSZonesPath):
                topLevelDomain = items.replace('.db', '', 1)
                message = 'Creating DNS records for %s' % (topLevelDomain)
                logging.statusWriter(self.logFile, message, 1)
                try:
                    Domains.objects.get(name=topLevelDomain).delete()
                except:
                    pass
                try:
                    pdns = PDNSStatus.objects.get(pk=1)
                    if pdns.type == 'MASTER':
                        zone = Domains(admin=admin, name=topLevelDomain, type="MASTER")
                        zone.save()
                    else:
                        zone = Domains(admin=admin, name=topLevelDomain, type="NATIVE")
                        zone.save()
                except:
                    zone = Domains(admin=admin, name=topLevelDomain, type="NATIVE")
                    zone.save()
                    pass
                content = "ns1." + topLevelDomain + " hostmaster." + topLevelDomain + " 1 10800 3600 1209600 3600"
                soaRecord = Records(domainOwner=zone,
                                    domain_id=zone.id,
                                    name=topLevelDomain,
                                    type="SOA",
                                    content=content,
                                    ttl=3600,
                                    prio=0,
                                    disabled=0,
                                    auth=1)
                soaRecord.save()
                CurrentZonePath = '%s/%s' % (DNSZonesPath, items)
                data = open(CurrentZonePath, 'r').readlines()
                SOACheck = 0
                start = 0
                for items in data:
                    try:
                        if items.find('SOA') > -1:
                            SOACheck = 1
                            continue
                        if (SOACheck == 1 and items.find(')') > -1) or (SOACheck == 1 and items.find('NS') > -1 and items.find(zone.name) > -1):
                            SOACheck = 0
                            start = 1
                            continue
                        else:
                            pass
                        if start == 1:
                            if os.path.exists(ProcessUtilities.debugPath):
                                message = f'Current DNS record we are creating for {zone.name} is {items}'
                                logging.statusWriter(self.logFile, message, 1)
                            if len(items) > 3:
                                if items.find("DKIM1") > -1:
                                    continue
                                RecordsData = items.split('\t')
                                if RecordsData[3] == 'A':
                                    RecordsData[4] = ipAddress
                                if RecordsData[0].find(topLevelDomain) > -1:
                                    if RecordsData[3] == 'MX':
                                        DNS.createDNSRecord(zone, RecordsData[0].rstrip('.'), RecordsData[3],RecordsData[5].rstrip('.').rstrip('.\n'), int(RecordsData[4]), RecordsData[1])
                                    else:
                                        DNS.createDNSRecord(zone, RecordsData[0].rstrip('.'), RecordsData[3], RecordsData[4].rstrip('.').rstrip('.\n'), 0, RecordsData[1])
                                else:
                                    if RecordsData[3] == 'MX':
                                        DNS.createDNSRecord(zone, RecordsData[0] + '.' + topLevelDomain, RecordsData[3],
                                                            RecordsData[5].rstrip('.').rstrip('.\n'), RecordsData[4],
                                                            RecordsData[1])
                                    else:
                                        DNS.createDNSRecord(zone, RecordsData[0] + '.' + topLevelDomain , RecordsData[3], RecordsData[4].rstrip('.').rstrip('.\n'), 0,
                                                            RecordsData[1])
                    except BaseException as msg:
                        message = 'Failed while creating DNS entry for %s, error message: %s.' % (topLevelDomain, str(msg))
                        logging.statusWriter(self.logFile, message, 1)
                message = 'DNS records successfully created for %s.' % (topLevelDomain)
                logging.statusWriter(self.logFile, message, 1)
            return 1
        except BaseException as msg:
            message = 'Failed to create DNS records from file %s, error message: %s.' % (
            self.backupFile, str(msg))
            logging.statusWriter(self.logFile, message, 1)
            return 0
    def setupConnection(self, db=None):
        try:
            passFile = "/etc/cyberpanel/mysqlPassword"
            f = open(passFile)
            data = f.read()
            password = data.split('\n', 1)[0]
            password = password.replace('\n', '').replace('\r', '')
            conn = mysql.connect(user='root', passwd=password, cursorclass=cursors.SSCursor)
            cursor = conn.cursor()
            return conn, cursor
        except BaseException as msg:
            message = 'Failed to connect to database, error message: %s. [ExtractBackup]' % (str(msg))
            logging.statusWriter(self.logFile, message, 1)
            return 0, 0
    def RestoreDatabases(self):
        try:
            message = 'Restoring databases from %s.' % (self.backupFile)
            logging.statusWriter(self.logFile, message, 1)
            ##
            passFile = "/etc/cyberpanel/mysqlPassword"
            try:
                import json
                jsonData = json.loads(open(passFile, 'r').read())
                mysqluser = jsonData['mysqluser']
                mysqlpassword = jsonData['mysqlpassword']
                mysqlport = jsonData['mysqlport']
                mysqlhost = jsonData['mysqlhost']
                password = mysqlpassword
            except:
                passFile = "/etc/cyberpanel/mysqlPassword"
                f = open(passFile)
                data = f.read()
                password = data.split('\n', 1)[0]
                mysqlhost = 'localhost'
                mysqlport = '3306'
                mysqluser = 'root'
            cnfPath = '/home/cyberpanel/.my.cnf'
            if not os.path.exists(cnfPath):
                cnfContent = """[mysqldump]
user=root
password=%s
max_allowed_packet=1024M
[mysql]
user=root
password=%s
""" % (password, password)
                writeToFile = open(cnfPath, 'w')
                writeToFile.write(cnfContent)
                writeToFile.close()
                os.chmod(cnfPath, 0o600)
            f = open(passFile)
            data = f.read()
            password = data.split('\n', 1)[0]
            ##
            connection, cursor = self.setupConnection()
            CompletPathToExtractedArchive = cPanelImporter.mainBackupPath + self.fileName
            DatabasesPath = '%s/mysql' % (CompletPathToExtractedArchive)
            for items in os.listdir(DatabasesPath):
                if items.find('roundcube') > -1:
                    continue
                if items.endswith('.sql'):
                    message = 'Restoring MySQL dump for %s.' % (items.replace('.sql', ''))
                    logging.statusWriter(self.logFile, message, 1)
                    try:
                        cursor.execute("CREATE DATABASE `%s`" % (items.replace('.sql', '')))
                    except BaseException as msg:
                        message = 'Failed while restoring database %s from backup file %s, error message: %s' % (items.replace('.sql', ''), self.backupFile, str(msg))
                        logging.statusWriter(self.logFile, message, 1)
                    command =  f'mysql --defaults-file=/home/cyberpanel/.my.cnf -u {mysqluser} --host={mysqlhost} --port {mysqlport} ' + items.replace('.sql', '')
                    message = f'Full command to restore DB {command}'
                    logging.statusWriter(self.logFile, message, 1)
                    cmd = shlex.split(command)
                    DBPath = "%s/%s" % (DatabasesPath, items)
                    # with open(DBPath, 'r') as f:
                    #     message = f'Full command to restore DB {cmd}'
                    #     logging.statusWriter(self.logFile, message, 1)
                    #
                    #     res = subprocess.call(cmd, stdin=f)
                    with open(DBPath, 'r') as f:
                        try:
                            # Run the command using subprocess.run, capturing stdout and stderr
                            result = subprocess.run(cmd, stdin=f, stdout=subprocess.PIPE, stderr=subprocess.PIPE,
                                                    universal_newlines=True)
                            # Log stdout and stderr
                            logging.statusWriter(self.logFile, f'STDOUT: {result.stdout}', 1)
                            logging.statusWriter(self.logFile, f'STDERR: {result.stderr}', 1)
                            # Check if the command failed
                            if result.returncode != 0:
                                logging.statusWriter(self.logFile,
                                                     f'Command failed with return code {result.returncode}', 2)
                        except Exception as e:
                            # Log any exception that occurs
                            logging.statusWriter(self.logFile, f'Exception occurred: {str(e)}', 2)
                    website = Websites.objects.get(domain=self.mainDomain)
                    ## Trying to figure out dbname
                    CommandsPath = '%s/mysql.sql' % (CompletPathToExtractedArchive)
                    data = open(CommandsPath, 'r').readlines()
                    ### temp disable if user not added, need to remove this try,catch and to ensure user gets added
                    try:
                        for inItems in data:
                            if (inItems.find('GRANT ALL PRIVILEGES') > -1 or inItems.find('GRANT USAGE') > -1) and inItems.find('localhost') > -1 and inItems.find('_test') == -1:
                                cDBName = inItems.split('`')[1].replace('\\', '')
                                logging.statusWriter(self.logFile, inItems, 1)
                                if cDBName == items.replace('.sql', ''):
                                    cDBUser = inItems.replace("`","'").replace("\\","").split("'")[1]
                                    message = 'Database user for %s is %s.' % (cDBName, cDBUser)
                                    logging.statusWriter(self.logFile, message, 1)
                                    if Databases.objects.filter(dbUser=cDBUser).count() > 0:
                                        continue
                                    break
                    except:
                        pass
                    ### temp disable if user not added, need to remove this try,catch and to ensure user gets added
                    try:
                        db = Databases(website=website, dbName=items.replace('.sql', ''), dbUser=cDBUser)
                        db.save()
                    except:
                        db = Databases(website=website, dbName=items.replace('.sql', ''), dbUser='root')
                        db.save()
                        pass
                    message = 'MySQL dump successfully restored for %s.' % (items.replace('.sql', ''))
                    logging.statusWriter(self.logFile, message, 1)
            message = 'Creating Database users from backup file %s.' % (self.backupFile)
            logging.statusWriter(self.logFile, message, 1)
            CommandsPath = '%s/mysql.sql' % (CompletPathToExtractedArchive)
            data = open(CommandsPath, 'r').readlines()
            for items in data:
                if items.find("--") > -1 or items.find("'cyberpanel'@") > -1:
                    continue
                try:
                    if os.path.exists(ProcessUtilities.debugPath):
                        message = f'Currently executing MySQL command {items}'
                        logging.statusWriter(self.logFile, message, 1)
                    cursor.execute(items)
                except BaseException as msg:
                    message = 'Failed while restoring database %s from backup file %s, error message: %s' % (
                    items.replace('.sql', ''), self.backupFile, str(msg))
                    logging.statusWriter(self.logFile, message, 1)
            connection.close()
            message = 'Databases successfully restored.'
            logging.statusWriter(self.logFile, message, 1)
            return 1
        except BaseException as msg:
            message = 'Failed to restore databases from file %s, error message: %s.' % (
            self.backupFile, str(msg))
            logging.statusWriter(self.logFile, message, 1)
            return 0
    def FixPermissions(self):
        from filemanager.filemanager import FileManager
        fm = FileManager(None, None)
        fm.fixPermissions(self.mainDomain)
        
    def createCronJobs(self):
        try:
            
            message = 'Restoring cron jobs from %s.' % (self.backupFile)
            logging.statusWriter(self.logFile, message, 1)
            
            CompletPathToExtractedArchive = cPanelImporter.mainBackupPath + self.fileName            
            cronPath = '%s/cron' % (CompletPathToExtractedArchive)
            
            
            if len(os.listdir(cronPath)) == 0:
                message = 'No Cron Job file found.'
                logging.statusWriter(self.logFile, message, 1)
                return 1
            
            if ProcessUtilities.decideDistro() == ProcessUtilities.centos or ProcessUtilities.decideDistro() == ProcessUtilities.cent8:
                localCronPath = "/var/spool/cron/" + self.externalApp
            else:
                localCronPath = "/var/spool/cron/crontabs/" + self.externalApp
                
            localCronFile = open(localCronPath, "a+")
                
            commandT = 'touch %s' % (localCronPath)
            ProcessUtilities.executioner(commandT, 'root')
            commandT = 'chown %s:%s %s' % (self.externalApp, self.externalApp, localCronPath)
            ProcessUtilities.executioner(commandT, 'root')
            # There's only single file usually but running for all found
            for item in os.listdir(cronPath):
                cronFile = open('%s/%s' % (cronPath, item), 'r')
                cronJobs = cronFile.readlines()
                
                # Filter actual jobs and remove variables and last new line character
                for job in cronJobs:
                    if len(job.split(' ')) > 1:
                        # Valid enough, add it to user
                        localCronFile.write(job)
                    
            message = 'Cron Jobs successfully restored.'
            logging.statusWriter(self.logFile, message, 1)
            return 1
        
        except BaseException as msg:
            message = 'Failed to restore Cron Jobs from file %s, error message: %s.' % (
            self.backupFile, str(msg))
            logging.statusWriter(self.logFile, message, 1)
            return 0
    def DeleteSite(self):
        vhost.deleteVirtualHostConfigurations(self.mainDomain)
    def checkIfExists(self, virtualHostName):
        if Websites.objects.filter(domain=virtualHostName).count() > 0:
            return 1
        if CDomains.objects.filter(domain=virtualHostName).count() > 0:
            return 1
        return 0
    def RestoreEmails(self):
        try:
            message = 'Restoring emails from archive file: %s' % (self.backupFile)
            logging.statusWriter(self.logFile, message, 1)
            CompletPathToExtractedArchive = cPanelImporter.mainBackupPath + self.fileName
            ### Find Mail Format
            UserData = '%s/homedir/mail' % (CompletPathToExtractedArchive)
            FormatPath = '%s/mailbox_format.cpanel' % (UserData)
            message = 'Detecting email format from %s.' % (self.backupFile)
            logging.statusWriter(self.logFile, message, 1)
            try:
                Format = open(FormatPath, 'r').read()
                if Format.find('mdbox') > -1:
                    self.mailFormat = cPanelImporter.MdBox
                    message = 'Mdbox format detected from %s.' % (self.backupFile)
                    logging.statusWriter(self.logFile, message, 1)
                else:
                    self.mailFormat = cPanelImporter.MailDir
                    message = 'Maildir format detected from %s.' % (self.backupFile)
                    logging.statusWriter(self.logFile, message, 1)
            except:
                self.mailFormat = cPanelImporter.MailDir
            ####
            if os.path.exists(ProcessUtilities.debugPath):
                logging.statusWriter(self.logFile, f'Content of user data before starting to restore emails {str(UserData)}')
            for items in os.listdir(UserData):
                FinalMailDomainPath = '%s/%s' % (UserData, items)
                if os.path.exists(ProcessUtilities.debugPath):
                    logging.statusWriter(self.logFile,
                                         f'Final email path for {items} is {str(FinalMailDomainPath)}')
                if os.path.isdir(FinalMailDomainPath):
                    if items[0] == '.':
                        continue
                    if items.find('.') > -1:
                        for it in os.listdir(FinalMailDomainPath):
                            try:
                                if self.checkIfExists(items) == 0:
                                    self.createDummyChild(items)
                                mailUtilities.createEmailAccount(items, it, 'cyberpanel')
                                finalEmailUsername = it + "@" + items
                                message = 'Starting restore for %s.' % (finalEmailUsername)
                                logging.statusWriter(self.logFile, message, 1)
                                eUser = EUsers.objects.get(email=finalEmailUsername)
                                if self.mailFormat == cPanelImporter.MailDir:
                                    eUser.mail = 'maildir:/home/vmail/%s/%s/Maildir' % (items, it)
                                    MailPath = '/home/vmail/%s/%s' % (items, it)
                                    command = 'mkdir -p %s' % (MailPath)
                                    ProcessUtilities.normalExecutioner(command)
                                    command = 'rm -rf %s/Maildir' % (MailPath)
                                    ProcessUtilities.normalExecutioner(command)
                                    MailPathInBackup = '%s/%s' % (FinalMailDomainPath, it)
                                    if os.path.exists(ProcessUtilities.debugPath):
                                        logging.statusWriter(self.logFile,
                                                             f'Mail path in backup for {items} is {str(MailPathInBackup)}')
                                    command = 'mv %s %s/Maildir' % (MailPathInBackup, MailPath)
                                    subprocess.call(command, shell=True)
                                else:
                                    eUser.mail = 'mdbox:/home/vmail/%s/%s/Mdbox' % (items, it)
                                    MailPath = '/home/vmail/%s/%s' % (items, it)
                                    command = 'mkdir -p %s' % (MailPath)
                                    ProcessUtilities.normalExecutioner(command)
                                    command = 'rm -rf %s/Mdbox' % (MailPath)
                                    ProcessUtilities.normalExecutioner(command)
                                    MailPathInBackup = '%s/%s' % (FinalMailDomainPath, it)
                                    if os.path.exists(ProcessUtilities.debugPath):
                                        logging.statusWriter(self.logFile,
                                                             f'Mail path in backup for {items} is {str(MailPathInBackup)}')
                                    command = 'mv %s %s/Mdbox' % (MailPathInBackup, MailPath)
                                    subprocess.call(command, shell=True)
                                ## Also update password
                                PasswordPath = '%s/homedir/etc/%s/shadow' % (CompletPathToExtractedArchive, items)
                                PasswordData = open(PasswordPath, 'r').readlines()
                                for i in PasswordData:
                                    if i.find(it) > -1:
                                        finalPassword = '%s%s' % ('{CRYPT}', i.split(':')[1])
                                        eUser.password = finalPassword
                                eUser.save()
                                message = 'Restore completed for %s.' % (finalEmailUsername)
                                logging.statusWriter(self.logFile, message, 1)
                            except BaseException as msg:
                                message = 'Failed to restore emails from archive file %s, For domain: %s. error message: %s. [ExtractBackup]' % (
                                    self.backupFile, items, str(msg))
                                logging.statusWriter(self.logFile, message, 1)
            command = 'chown -R vmail:vmail /home/vmail'
            ProcessUtilities.normalExecutioner(command)
            message = 'Emails successfully restored'
            logging.statusWriter(self.logFile, message, 1)
            return 1
        except BaseException as msg:
            message = 'Failed to restore emails from archive file %s, error message: %s. [ExtractBackup]' % (
            self.backupFile, str(msg))
            logging.statusWriter(self.logFile, message, 1)
            return 0
    def MainController(self):
        if self.ExtractBackup():
            pass
        else:
            return
        self.LoadDomains()
        if self.CreateMainWebsite():
            pass
        else:
            return 0
        if self.CreateChildDomains():
            pass
        else:
            return 0
        if self.CreateDNSRecords():
            pass
        else:
            return 0
        if self.RestoreDatabases():
            pass
        else:
            return 0
        if self.createCronJobs():
            pass
        else:
            pass
        self.RestoreEmails()
        self.FixPermissions()
        message = 'Backup file %s successfully restored.' % (self.backupFile)
        logging.statusWriter(self.logFile, message, 1)
        return 1
def main():
    LogFile = '/home/cyberpanel/%s' % (str(randint(1000, 9999)))
    message = 'Backup logs to be generated in %s' % (LogFile)
    print(message)
    parser = argparse.ArgumentParser(description='CyberPanel cPanel Importer')
    parser.add_argument('--path', help='Path where cPanel .tar.gz files are stored.')
    args = parser.parse_args()
    for items in os.listdir(args.path):
        if items.endswith('.tar.gz'):
            finalPath = '%s/%s' % (args.path.rstrip('/'), items)
            try:
                cI = cPanelImporter(finalPath, LogFile)
                if cI.MainController():
                    pass
                else:
                    pass
            except:
                pass
if __name__ == "__main__":
    main()