mirror of
				https://github.com/usmannasir/cyberpanel.git
				synced 2025-10-26 15:56:34 +01:00 
			
		
		
		
	
		
			
	
	
		
			1794 lines
		
	
	
		
			88 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
		
		
			
		
	
	
			1794 lines
		
	
	
		
			88 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
|   | #!/usr/local/CyberCP/bin/python | ||
|  | import os.path | ||
|  | import sys | ||
|  | 
 | ||
|  | import paramiko | ||
|  | 
 | ||
|  | sys.path.append('/usr/local/CyberCP') | ||
|  | os.environ.setdefault("DJANGO_SETTINGS_MODULE", "CyberCP.settings") | ||
|  | import django | ||
|  | django.setup() | ||
|  | from plogical.getSystemInformation import SystemInformation | ||
|  | from IncBackups.IncBackupsControl import IncJobs | ||
|  | from IncBackups.models import BackupJob | ||
|  | from random import randint | ||
|  | import argparse | ||
|  | import json | ||
|  | from websiteFunctions.models import GitLogs, Websites, GDrive, GDriveJobLogs | ||
|  | from websiteFunctions.website import WebsiteManager | ||
|  | import time | ||
|  | import datetime | ||
|  | import google.oauth2.credentials | ||
|  | from google.oauth2.credentials import Credentials | ||
|  | from googleapiclient.discovery import build | ||
|  | from googleapiclient.http import MediaFileUpload | ||
|  | from plogical.backupSchedule import backupSchedule | ||
|  | import requests | ||
|  | import socket | ||
|  | from websiteFunctions.models import NormalBackupJobs, NormalBackupJobLogs | ||
|  | from boto3.s3.transfer import TransferConfig | ||
|  | 
 | ||
|  | try: | ||
|  |     from s3Backups.models import BackupPlan, BackupLogs | ||
|  |     import boto3 | ||
|  |     from plogical.virtualHostUtilities import virtualHostUtilities | ||
|  |     from plogical.mailUtilities import mailUtilities | ||
|  |     from plogical.CyberCPLogFileWriter import CyberCPLogFileWriter as logging | ||
|  |     from plogical.processUtilities import ProcessUtilities | ||
|  | except: | ||
|  |     pass | ||
|  | import threading as multi | ||
|  | 
 | ||
|  | class IncScheduler(multi.Thread): | ||
|  |     logPath = '/home/cyberpanel/incbackuplogs' | ||
|  |     gitFolder = '/home/cyberpanel/git' | ||
|  | 
 | ||
|  |     timeFormat = time.strftime("%m.%d.%Y_%H-%M-%S") | ||
|  | 
 | ||
|  |     ### Normal scheduled backups constants | ||
|  | 
 | ||
|  |     frequency = 'frequency' | ||
|  |     allSites = 'allSites' | ||
|  |     currentStatus = 'currentStatus' | ||
|  |     lastRun = 'lastRun' | ||
|  | 
 | ||
|  |     def __init__(self, function, extraArgs): | ||
|  |         multi.Thread.__init__(self) | ||
|  |         self.function = function | ||
|  |         self.data = extraArgs | ||
|  | 
 | ||
|  |     def run(self): | ||
|  |         if self.function == "startBackup": | ||
|  |             IncScheduler.startBackup(self.data['freq']) | ||
|  |         elif self.function == "CalculateAndUpdateDiskUsage": | ||
|  |             IncScheduler.CalculateAndUpdateDiskUsage() | ||
|  | 
 | ||
|  |     @staticmethod | ||
|  |     def startBackup(type): | ||
|  |         try: | ||
|  |             logging.statusWriter(IncScheduler.logPath, 'Starting Incremental Backup job..', 1) | ||
|  |             tempPath = "/home/cyberpanel/" + str(randint(1000, 9999)) | ||
|  |             for job in BackupJob.objects.all(): | ||
|  |                 logging.statusWriter(IncScheduler.logPath, 'Job Description:\n\n Destination: %s, Frequency: %s.\n ' % ( | ||
|  |                     job.destination, job.frequency), 1) | ||
|  |                 if job.frequency == type: | ||
|  | 
 | ||
|  | 
 | ||
|  |                     ### now run backups | ||
|  |                     for web in job.jobsites_set.all(): | ||
|  | 
 | ||
|  |                         logging.statusWriter(IncScheduler.logPath, 'Backing up %s.' % (web.website), 1) | ||
|  | 
 | ||
|  |                         extraArgs = {} | ||
|  |                         extraArgs['website'] = web.website | ||
|  |                         extraArgs['tempPath'] = tempPath | ||
|  |                         extraArgs['backupDestinations'] = job.destination | ||
|  | 
 | ||
|  |                         if job.websiteData == 1: | ||
|  |                             extraArgs['websiteData'] = True | ||
|  |                         else: | ||
|  |                             extraArgs['websiteData'] = False | ||
|  | 
 | ||
|  |                         if job.websiteDatabases == 1: | ||
|  |                             extraArgs['websiteDatabases'] = True | ||
|  |                         else: | ||
|  |                             extraArgs['websiteDatabases'] = False | ||
|  | 
 | ||
|  |                         if job.websiteDataEmails == 1: | ||
|  |                             extraArgs['websiteEmails'] = True | ||
|  |                         else: | ||
|  |                             extraArgs['websiteEmails'] = False | ||
|  | 
 | ||
|  |                         extraArgs['websiteSSLs'] = False | ||
|  | 
 | ||
|  |                         startJob = IncJobs('createBackup', extraArgs) | ||
|  |                         startJob.start() | ||
|  | 
 | ||
|  |                         ### Checking status | ||
|  | 
 | ||
|  |                         while True: | ||
|  |                             if os.path.exists(tempPath): | ||
|  |                                 result = open(tempPath, 'r').read() | ||
|  | 
 | ||
|  |                                 if result.find("Completed") > -1: | ||
|  | 
 | ||
|  |                                     ### Removing Files | ||
|  | 
 | ||
|  |                                     os.remove(tempPath) | ||
|  | 
 | ||
|  |                                     logging.statusWriter(IncScheduler.logPath, 'Backed up %s.' % (web.website), 1) | ||
|  |                                     break | ||
|  |                                 elif result.find("[5009]") > -1: | ||
|  |                                     ## removing status file, so that backup can re-runn | ||
|  |                                     try: | ||
|  |                                         os.remove(tempPath) | ||
|  |                                     except: | ||
|  |                                         pass | ||
|  | 
 | ||
|  |                                     logging.statusWriter(IncScheduler.logPath, | ||
|  |                                                          'Failed backup for %s, error: %s.' % (web.website, result), 1) | ||
|  |                                     break | ||
|  | 
 | ||
|  |         except BaseException as msg: | ||
|  |             logging.writeToFile( "%s [startBackup]"%str(msg)) | ||
|  | 
 | ||
|  |     @staticmethod | ||
|  |     def git(type): | ||
|  |         try: | ||
|  |             for website in os.listdir(IncScheduler.gitFolder): | ||
|  |                 finalText = '' | ||
|  |                 web = Websites.objects.get(domain=website) | ||
|  | 
 | ||
|  |                 message = '[%s Cron] Checking if %s has any pending commits on %s.' % ( | ||
|  |                     type, website, time.strftime("%m.%d.%Y_%H-%M-%S")) | ||
|  |                 finalText = '%s\n' % (message) | ||
|  |                 GitLogs(owner=web, type='INFO', message=message).save() | ||
|  | 
 | ||
|  |                 finalPathInside = '%s/%s' % (IncScheduler.gitFolder, website) | ||
|  | 
 | ||
|  |                 for file in os.listdir(finalPathInside): | ||
|  | 
 | ||
|  |                     try: | ||
|  | 
 | ||
|  |                         ## | ||
|  |                         finalPathConf = '%s/%s' % (finalPathInside, file) | ||
|  | 
 | ||
|  |                         gitConf = json.loads(open(finalPathConf, 'r').read()) | ||
|  | 
 | ||
|  |                         data = {} | ||
|  |                         data['domain'] = gitConf['domain'] | ||
|  |                         data['folder'] = gitConf['folder'] | ||
|  |                         data['commitMessage'] = 'Auto commit by CyberPanel %s cron on %s' % ( | ||
|  |                             type, time.strftime('%m-%d-%Y_%H-%M-%S')) | ||
|  | 
 | ||
|  |                         if gitConf['autoCommit'] == type: | ||
|  | 
 | ||
|  |                             wm = WebsiteManager() | ||
|  |                             resp = wm.commitChanges(1, data) | ||
|  |                             resp = json.loads(resp.content) | ||
|  | 
 | ||
|  |                             if resp['status'] == 1: | ||
|  |                                 message = 'Folder: %s, Status: %s' % (gitConf['folder'], resp['commandStatus']) | ||
|  |                                 finalText = '%s\n%s' % (finalText, message) | ||
|  |                                 GitLogs(owner=web, type='INFO', message=message).save() | ||
|  |                             else: | ||
|  |                                 message = 'Folder: %s, Status: %s' % (gitConf['folder'], resp['commandStatus']) | ||
|  |                                 finalText = '%s\n%s' % (finalText, message) | ||
|  |                                 GitLogs(owner=web, type='ERROR', message=message).save() | ||
|  | 
 | ||
|  |                         if gitConf['autoPush'] == type: | ||
|  | 
 | ||
|  |                             wm = WebsiteManager() | ||
|  |                             resp = wm.gitPush(1, data) | ||
|  |                             resp = json.loads(resp.content) | ||
|  | 
 | ||
|  |                             if resp['status'] == 1: | ||
|  |                                 GitLogs(owner=web, type='INFO', message=resp['commandStatus']).save() | ||
|  |                                 finalText = '%s\n%s' % (finalText, resp['commandStatus']) | ||
|  |                             else: | ||
|  |                                 GitLogs(owner=web, type='ERROR', message=resp['commandStatus']).save() | ||
|  |                                 finalText = '%s\n%s' % (finalText, resp['commandStatus']) | ||
|  |                     except BaseException as msg: | ||
|  |                         message = 'File: %s, Status: %s' % (file, str(msg)) | ||
|  |                         finalText = '%s\n%s' % (finalText, message) | ||
|  | 
 | ||
|  |                 message = '[%s Cron] Finished checking for %s on %s.' % ( | ||
|  |                     type, website, time.strftime("%m.%d.%Y_%H-%M-%S")) | ||
|  |                 finalText = '%s\n%s' % (finalText, message) | ||
|  |                 logging.SendEmail(web.adminEmail, web.adminEmail, finalText, 'Git report for %s.' % (web.domain)) | ||
|  |                 GitLogs(owner=web, type='INFO', message=message).save() | ||
|  | 
 | ||
|  |         except BaseException as msg: | ||
|  |             logging.writeToFile('%s. [IncScheduler.git:90]' % (str(msg))) | ||
|  | 
 | ||
|  |     @staticmethod | ||
|  |     def checkDiskUsage(): | ||
|  |         sender_email = 'root@%s' % (socket.gethostname()) | ||
|  | 
 | ||
|  |         try: | ||
|  | 
 | ||
|  |             import psutil, math | ||
|  |             from websiteFunctions.models import Administrator | ||
|  |             admin = Administrator.objects.get(pk=1) | ||
|  | 
 | ||
|  |             diskUsage = math.floor(psutil.disk_usage('/')[3]) | ||
|  | 
 | ||
|  |             from plogical.acl import ACLManager | ||
|  |             message = '%s - Disk Usage Warning - CyberPanel' % (ACLManager.fetchIP()) | ||
|  | 
 | ||
|  |             if diskUsage >= 50 and diskUsage <= 60: | ||
|  | 
 | ||
|  |                 finalText = 'Current disk usage at "/" is %s percent. No action required.' % (str(diskUsage)) | ||
|  |                 logging.SendEmail(sender_email, admin.email, finalText, message) | ||
|  | 
 | ||
|  |             elif diskUsage >= 60 and diskUsage <= 80: | ||
|  | 
 | ||
|  |                 finalText = 'Current disk usage at "/" is %s percent. We recommend clearing log directory by running \n\n rm -rf /usr/local/lsws/logs/*. \n\n When disk usage go above 80 percent we will automatically run this command.' % ( | ||
|  |                     str(diskUsage)) | ||
|  |                 logging.SendEmail(sender_email, admin.email, finalText, message) | ||
|  | 
 | ||
|  |             elif diskUsage > 80: | ||
|  | 
 | ||
|  |                 finalText = 'Current disk usage at "/" is %s percent. We are going to run below command to free up space, If disk usage is still high, manual action is required by the system administrator. \n\n rm -rf /usr/local/lsws/logs/*.' % ( | ||
|  |                     str(diskUsage)) | ||
|  |                 logging.SendEmail(sender_email, admin.email, finalText, message) | ||
|  | 
 | ||
|  |                 command = 'rm -rf /usr/local/lsws/logs/*' | ||
|  |                 import subprocess | ||
|  |                 subprocess.call(command, shell=True) | ||
|  | 
 | ||
|  |         except BaseException as msg: | ||
|  |             logging.writeToFile('[IncScheduler:193:checkDiskUsage] %s.' % str(msg)) | ||
|  | 
 | ||
|  |     @staticmethod | ||
|  |     def runGoogleDriveBackups(type): | ||
|  | 
 | ||
|  |         ipFile = "/etc/cyberpanel/machineIP" | ||
|  |         f = open(ipFile) | ||
|  |         ipData = f.read() | ||
|  |         ipAddress = ipData.split('\n', 1)[0] | ||
|  | 
 | ||
|  |         backupRunTime = time.strftime("%m.%d.%Y_%H-%M-%S") | ||
|  |         backupLogPath = "/usr/local/lscp/logs/local_backup_log." + backupRunTime | ||
|  | 
 | ||
|  |         for items in GDrive.objects.all(): | ||
|  |             try: | ||
|  |                 if items.runTime == type: | ||
|  |                     gDriveData = json.loads(items.auth) | ||
|  |                     try: | ||
|  |                         credentials = google.oauth2.credentials.Credentials(gDriveData['token'], | ||
|  |                                                                             gDriveData['refresh_token'], | ||
|  |                                                                             gDriveData['token_uri'], None, None, | ||
|  |                                                                             gDriveData['scopes']) | ||
|  | 
 | ||
|  |                         drive = build('drive', 'v3', credentials=credentials) | ||
|  |                         drive.files().list(pageSize=10, fields="files(id, name)").execute() | ||
|  |                     except BaseException as msg: | ||
|  |                         try: | ||
|  |                             import requests | ||
|  |                             finalData = json.dumps({'refresh_token': gDriveData['refresh_token']}) | ||
|  |                             r = requests.post("https://platform.cyberpersons.com/refreshToken", data=finalData | ||
|  |                                               ) | ||
|  | 
 | ||
|  |                             gDriveData['token'] = json.loads(r.text)['access_token'] | ||
|  | 
 | ||
|  |                             credentials = google.oauth2.credentials.Credentials(gDriveData['token'], | ||
|  |                                                                                 gDriveData['refresh_token'], | ||
|  |                                                                                 gDriveData['token_uri'], | ||
|  |                                                                                 None, | ||
|  |                                                                                 None, | ||
|  |                                                                                 gDriveData['scopes']) | ||
|  | 
 | ||
|  |                             drive = build('drive', 'v3', credentials=credentials) | ||
|  |                             drive.files().list(pageSize=5, fields="files(id, name)").execute() | ||
|  | 
 | ||
|  |                             items.auth = json.dumps(gDriveData) | ||
|  |                             items.save() | ||
|  |                         except BaseException as msg: | ||
|  |                             GDriveJobLogs(owner=items, status=backupSchedule.ERROR, | ||
|  |                                           message='Connection to this account failed. Delete and re-setup this account. Error: %s' % ( | ||
|  |                                               str(msg))).save() | ||
|  |                             continue | ||
|  | 
 | ||
|  |                     try: | ||
|  |                         folderIDIP = gDriveData['folderIDIP'] | ||
|  |                     except: | ||
|  | 
 | ||
|  |                         ## Create CyberPanel Folder | ||
|  | 
 | ||
|  |                         file_metadata = { | ||
|  |                             'name': '%s-%s' % (items.name, ipAddress), | ||
|  |                             'mimeType': 'application/vnd.google-apps.folder' | ||
|  |                         } | ||
|  |                         file = drive.files().create(body=file_metadata, | ||
|  |                                                     fields='id').execute() | ||
|  |                         folderIDIP = file.get('id') | ||
|  | 
 | ||
|  |                         gDriveData['folderIDIP'] = folderIDIP | ||
|  | 
 | ||
|  |                         items.auth = json.dumps(gDriveData) | ||
|  |                         items.save() | ||
|  | 
 | ||
|  |                     ### Current folder to store files | ||
|  | 
 | ||
|  |                     file_metadata = { | ||
|  |                         'name': time.strftime("%m.%d.%Y_%H-%M-%S"), | ||
|  |                         'mimeType': 'application/vnd.google-apps.folder', | ||
|  |                         'parents': [folderIDIP] | ||
|  |                     } | ||
|  |                     file = drive.files().create(body=file_metadata, | ||
|  |                                                 fields='id').execute() | ||
|  |                     folderID = file.get('id') | ||
|  | 
 | ||
|  |                     ### | ||
|  | 
 | ||
|  |                     GDriveJobLogs(owner=items, status=backupSchedule.INFO, message='Starting backup job..').save() | ||
|  | 
 | ||
|  |                     for website in items.gdrivesites_set.all(): | ||
|  | 
 | ||
|  |                         ### If  this website dont exists continue | ||
|  | 
 | ||
|  |                         try: | ||
|  |                             Websites.objects.get(domain=website.domain) | ||
|  |                         except: | ||
|  |                             continue | ||
|  | 
 | ||
|  |                         ## | ||
|  | 
 | ||
|  |                         try: | ||
|  |                             GDriveJobLogs(owner=items, status=backupSchedule.INFO, | ||
|  |                                           message='Local backup creation started for %s..' % (website.domain)).save() | ||
|  | 
 | ||
|  |                             retValues = backupSchedule.createLocalBackup(website.domain, backupLogPath) | ||
|  | 
 | ||
|  |                             if retValues[0] == 0: | ||
|  |                                 GDriveJobLogs(owner=items, status=backupSchedule.ERROR, | ||
|  |                                               message='[ERROR] Backup failed for %s, error: %s moving on..' % ( | ||
|  |                                                   website.domain, retValues[1])).save() | ||
|  |                                 continue | ||
|  | 
 | ||
|  |                             completeFileToSend = retValues[1] + ".tar.gz" | ||
|  |                             fileName = completeFileToSend.split('/')[-1] | ||
|  | 
 | ||
|  |                             file_metadata = { | ||
|  |                                 'name': '%s' % (fileName), | ||
|  |                                 'parents': [folderID] | ||
|  |                             } | ||
|  |                             media = MediaFileUpload(completeFileToSend, mimetype='application/gzip', resumable=True) | ||
|  |                             try: | ||
|  |                                 drive.files().create(body=file_metadata, media_body=media, fields='id').execute() | ||
|  |                             except: | ||
|  |                                 import requests | ||
|  |                                 finalData = json.dumps({'refresh_token': gDriveData['refresh_token']}) | ||
|  |                                 r = requests.post("https://platform.cyberpersons.com/refreshToken", data=finalData | ||
|  |                                                   ) | ||
|  |                                 gDriveData['token'] = json.loads(r.text)['access_token'] | ||
|  | 
 | ||
|  |                                 credentials = google.oauth2.credentials.Credentials(gDriveData['token'], | ||
|  |                                                                                     gDriveData['refresh_token'], | ||
|  |                                                                                     gDriveData['token_uri'], | ||
|  |                                                                                     None, | ||
|  |                                                                                     None, | ||
|  |                                                                                     gDriveData['scopes']) | ||
|  | 
 | ||
|  |                                 drive = build('drive', 'v3', credentials=credentials) | ||
|  |                                 drive.files().create(body=file_metadata, media_body=media, fields='id').execute() | ||
|  | 
 | ||
|  |                                 items.auth = json.dumps(gDriveData) | ||
|  |                                 items.save() | ||
|  | 
 | ||
|  |                             GDriveJobLogs(owner=items, status=backupSchedule.INFO, | ||
|  |                                           message='Backup for %s successfully sent to Google Drive.' % ( | ||
|  |                                               website.domain)).save() | ||
|  | 
 | ||
|  |                             os.remove(completeFileToSend) | ||
|  |                         except BaseException as msg: | ||
|  |                             GDriveJobLogs(owner=items, status=backupSchedule.ERROR, | ||
|  |                                           message='[Site] Site backup failed, Error message: %s.' % (str(msg))).save() | ||
|  | 
 | ||
|  |                     GDriveJobLogs(owner=items, status=backupSchedule.INFO, | ||
|  |                                   message='Job Completed').save() | ||
|  | 
 | ||
|  |                     print("job com[leted") | ||
|  | 
 | ||
|  |                     # logging.writeToFile('job completed') | ||
|  | 
 | ||
|  |                     url = "https://platform.cyberpersons.com/CyberpanelAdOns/Adonpermission" | ||
|  |                     data = { | ||
|  |                         "name": "backups-retention", | ||
|  |                         "IP": ipAddress | ||
|  |                     } | ||
|  | 
 | ||
|  |                     import requests | ||
|  |                     response = requests.post(url, data=json.dumps(data)) | ||
|  |                     Status = response.json()['status'] | ||
|  | 
 | ||
|  |                     if (Status == 1) or ProcessUtilities.decideServer() == ProcessUtilities.ent: | ||
|  |                         try: | ||
|  | 
 | ||
|  |                             page_token = None | ||
|  |                             while True: | ||
|  |                                 response = drive.files().list(q="name='%s-%s'" % (items.name, ipAddress), | ||
|  |                                                               spaces='drive', | ||
|  |                                                               fields='nextPageToken, files(id, name)', | ||
|  |                                                               pageToken=page_token).execute() | ||
|  |                                 for file in response.get('files', []): | ||
|  |                                     # Process change | ||
|  |                                     # print('Fetch Main folder ID: %s (%s)' % (file.get('name'), file.get('id'))) | ||
|  |                                     # logging.writeToFile('Fetch Main folder ID: %s (%s)' % (file.get('name'), file.get('id'))) | ||
|  |                                     mainfolder_id = file.get('id') | ||
|  |                                 page_token = response.get('nextPageToken', None) | ||
|  |                                 if page_token is None: | ||
|  |                                     break | ||
|  |                             # print("new job started ") | ||
|  |                             try: | ||
|  |                                 page_token = None | ||
|  |                                 while True: | ||
|  |                                     response = drive.files().list(q="'%s' in parents" % (mainfolder_id), | ||
|  |                                                                   spaces='drive', | ||
|  |                                                                   fields='nextPageToken, files(id, name, createdTime)', | ||
|  |                                                                   pageToken=page_token).execute() | ||
|  |                                     for file in response.get('files', []): | ||
|  |                                         # Process change | ||
|  |                                         # print('Fetch all folders in main folder: %s (%s) time:-%s' % (file.get('name'), file.get('id'), file.get('createdTime'))) | ||
|  |                                         # logging.writeToFile('Fetch all folders in main folder: %s (%s) time:-%s' % (file.get('name'), file.get('id'),file.get('createdTime'))) | ||
|  |                                         ab = file.get('createdTime')[:10] | ||
|  |                                         print(f'File from gdrive {file.get("name")}') | ||
|  |                                         filename = file.get("name") | ||
|  |                                         fileDeleteID = file.get('id') | ||
|  |                                         timestamp = time.mktime(datetime.datetime.strptime(ab, "%Y-%m-%d").timetuple()) | ||
|  | 
 | ||
|  |                                         print(f'Folder creation time on gdrive {timestamp}') | ||
|  |                                         logging.writeToFile(f'Folder creation time on gdrive {timestamp}') | ||
|  | 
 | ||
|  |                                         CUrrenttimestamp = time.time() | ||
|  |                                         try: | ||
|  |                                             timerrtention = gDriveData['FileRetentiontime'] | ||
|  |                                             print(f'Retention time {timerrtention}') | ||
|  |                                             logging.writeToFile(f'Retention time {timerrtention}') | ||
|  |                                         except: | ||
|  |                                             print(f'Retention time not defined.') | ||
|  |                                             timerrtention = '6m' | ||
|  | 
 | ||
|  |                                         if (timerrtention == '1d'): | ||
|  |                                             new = CUrrenttimestamp - float(86400) | ||
|  |                                             print(f'New time {new}') | ||
|  |                                             if (new >= timestamp): | ||
|  |                                                 print(f'New time {new}, Folder created time {timestamp}') | ||
|  |                                                 logging.writeToFile(f'New time {new}, Folder created time {timestamp}') | ||
|  |                                                 resp = drive.files().delete(fileId=fileDeleteID).execute() | ||
|  |                                                 logging.writeToFile('Delete file %s ' % filename) | ||
|  |                                         elif (timerrtention == '1w'): | ||
|  |                                             new = CUrrenttimestamp - float(604800) | ||
|  |                                             if (new >= timestamp): | ||
|  |                                                 resp = drive.files().delete(fileId=fileDeleteID).execute() | ||
|  |                                                 logging.writeToFile('Delete file %s ' % filename) | ||
|  |                                         elif (timerrtention == '1m'): | ||
|  |                                             new = CUrrenttimestamp - float(2592000) | ||
|  |                                             if (new >= timestamp): | ||
|  |                                                 resp = drive.files().delete(fileId=fileDeleteID).execute() | ||
|  |                                                 logging.writeToFile('Delete file %s ' % filename) | ||
|  |                                         elif (timerrtention == '6m'): | ||
|  |                                             new = CUrrenttimestamp - float(15552000) | ||
|  |                                             if (new >= timestamp): | ||
|  |                                                 resp = drive.files().delete(fileId=fileDeleteID).execute() | ||
|  |                                                 logging.writeToFile('Delete file %s ' % filename) | ||
|  |                                     page_token = response.get('nextPageToken', None) | ||
|  |                                     if page_token is None: | ||
|  |                                         break | ||
|  | 
 | ||
|  |                             # logging.writeToFile('Createtime list - %s'%Createtime) | ||
|  | 
 | ||
|  |                             except BaseException as msg: | ||
|  |                                 print('An error occurred fetch child: %s' % msg) | ||
|  |                                 logging.writeToFile('An error occurred fetch child: %s' % msg) | ||
|  |                         except BaseException as msg: | ||
|  |                             logging.writeToFile('job not completed [ERROR:]..%s' % msg) | ||
|  | 
 | ||
|  |             except BaseException as msg: | ||
|  |                 GDriveJobLogs(owner=items, status=backupSchedule.ERROR, | ||
|  |                               message='[Completely] Job failed, Error message: %s.' % (str(msg))).save() | ||
|  | 
 | ||
|  |     @staticmethod | ||
|  |     def startNormalBackups(type): | ||
|  | 
 | ||
|  |         from plogical.processUtilities import ProcessUtilities | ||
|  |         from plogical.backupSchedule import backupSchedule | ||
|  |         import socket | ||
|  | 
 | ||
|  |         ## SFTP Destination Config sample | ||
|  |         ## {"type": "SFTP", "ip": "ip", "username": "root", "port": "22", "path": "/home/backup"} | ||
|  | 
 | ||
|  |         ## Local Destination config sample | ||
|  |         ## {"type": "local", "path": "/home/backup"} | ||
|  | 
 | ||
|  |         ## Backup jobs config | ||
|  | 
 | ||
|  |         ## {"frequency": "Daily", "allSites": "Selected Only"} | ||
|  |         ## {"frequency": "Daily"} | ||
|  | 
 | ||
|  |         for backupjob in NormalBackupJobs.objects.all(): | ||
|  | 
 | ||
|  |             jobConfig = json.loads(backupjob.config) | ||
|  |             destinationConfig = json.loads(backupjob.owner.config) | ||
|  | 
 | ||
|  |             currentTime = time.strftime("%m.%d.%Y_%H-%M-%S") | ||
|  |             print(destinationConfig['type']) | ||
|  | 
 | ||
|  |             if destinationConfig['type'] == 'local': | ||
|  | 
 | ||
|  | 
 | ||
|  |                 if jobConfig[IncScheduler.frequency] == type: | ||
|  | 
 | ||
|  |                     finalPath = '%s/%s' % (destinationConfig['path'].rstrip('/'), currentTime) | ||
|  |                     command = 'mkdir -p %s' % (finalPath) | ||
|  |                     ProcessUtilities.executioner(command) | ||
|  | 
 | ||
|  |                     ### Check if an old job prematurely killed, then start from there. | ||
|  |                     try: | ||
|  |                         oldJobContinue = 1 | ||
|  |                         pid = jobConfig['pid'] | ||
|  |                         stuckDomain = jobConfig['website'] | ||
|  |                         finalPath = jobConfig['finalPath'] | ||
|  |                         jobConfig['pid'] = str(os.getpid()) | ||
|  | 
 | ||
|  |                         command = 'ps aux' | ||
|  |                         result = ProcessUtilities.outputExecutioner(command) | ||
|  | 
 | ||
|  |                         if result.find(pid) > -1 and result.find('IncScheduler.py') > -1: | ||
|  |                             quit(1) | ||
|  | 
 | ||
|  |                     except: | ||
|  |                         ### Save some important info in backup config | ||
|  |                         oldJobContinue = 0 | ||
|  |                         jobConfig['pid'] = str(os.getpid()) | ||
|  |                         jobConfig['finalPath'] = finalPath | ||
|  | 
 | ||
|  |                     NormalBackupJobLogs.objects.filter(owner=backupjob).delete() | ||
|  |                     NormalBackupJobLogs(owner=backupjob, status=backupSchedule.INFO, | ||
|  |                                         message='Starting %s backup on %s..' % ( | ||
|  |                                             type, time.strftime("%m.%d.%Y_%H-%M-%S"))).save() | ||
|  | 
 | ||
|  |                     if oldJobContinue: | ||
|  |                         NormalBackupJobLogs(owner=backupjob, status=backupSchedule.INFO, | ||
|  |                                             message='Will continue old killed job starting from %s.' % ( | ||
|  |                                                 stuckDomain)).save() | ||
|  | 
 | ||
|  |                     actualDomain = 0 | ||
|  |                     try: | ||
|  |                         if jobConfig[IncScheduler.allSites] == 'all': | ||
|  |                             websites = Websites.objects.all().order_by('domain') | ||
|  |                             actualDomain = 1 | ||
|  |                         else: | ||
|  |                             websites = backupjob.normalbackupsites_set.all().order_by('domain__domain') | ||
|  |                     except: | ||
|  |                         websites = backupjob.normalbackupsites_set.all().order_by('domain__domain') | ||
|  | 
 | ||
|  |                     doit = 0 | ||
|  | 
 | ||
|  |                     for site in websites: | ||
|  |                         if actualDomain: | ||
|  |                             domain = site.domain | ||
|  |                         else: | ||
|  |                             domain = site.domain.domain | ||
|  | 
 | ||
|  |                         ## Save currently backing domain in db, so that i can restart from here when prematurely killed | ||
|  | 
 | ||
|  |                         jobConfig['website'] = domain | ||
|  |                         jobConfig[IncScheduler.lastRun] = time.strftime("%d %b %Y, %I:%M %p") | ||
|  |                         jobConfig[IncScheduler.currentStatus] = 'Running..' | ||
|  |                         backupjob.config = json.dumps(jobConfig) | ||
|  |                         backupjob.save() | ||
|  | 
 | ||
|  |                         if oldJobContinue and not doit: | ||
|  |                             if domain == stuckDomain: | ||
|  |                                 doit = 1 | ||
|  |                                 continue | ||
|  |                             else: | ||
|  |                                 continue | ||
|  | 
 | ||
|  |                         retValues = backupSchedule.createLocalBackup(domain, '/dev/null') | ||
|  | 
 | ||
|  |                         if retValues[0] == 0: | ||
|  |                             NormalBackupJobLogs(owner=backupjob, status=backupSchedule.ERROR, | ||
|  |                                                 message='Backup failed for %s on %s.' % ( | ||
|  |                                                     domain, time.strftime("%m.%d.%Y_%H-%M-%S"))).save() | ||
|  | 
 | ||
|  |                             SUBJECT = "Automatic backup failed for %s on %s." % (domain, currentTime) | ||
|  |                             adminEmailPath = '/home/cyberpanel/adminEmail' | ||
|  |                             adminEmail = open(adminEmailPath, 'r').read().rstrip('\n') | ||
|  |                             sender = 'root@%s' % (socket.gethostname()) | ||
|  |                             TO = [adminEmail] | ||
|  |                             message = """\
 | ||
|  | From: %s | ||
|  | To: %s | ||
|  | Subject: %s | ||
|  | 
 | ||
|  | Automatic backup failed for %s on %s. | ||
|  | """ % (sender, ", ".join(TO), SUBJECT, domain, currentTime)
 | ||
|  | 
 | ||
|  |                             logging.SendEmail(sender, TO, message) | ||
|  |                         else: | ||
|  |                             backupPath = retValues[1] + ".tar.gz" | ||
|  | 
 | ||
|  |                             command = 'mv %s %s' % (backupPath, finalPath) | ||
|  |                             ProcessUtilities.executioner(command) | ||
|  | 
 | ||
|  |                             NormalBackupJobLogs(owner=backupjob, status=backupSchedule.INFO, | ||
|  |                                                 message='Backup completed for %s on %s.' % ( | ||
|  |                                                     domain, time.strftime("%m.%d.%Y_%H-%M-%S"))).save() | ||
|  | 
 | ||
|  |                     jobConfig = json.loads(backupjob.config) | ||
|  |                     try: | ||
|  |                         if jobConfig['pid']: | ||
|  |                             del jobConfig['pid'] | ||
|  |                     except: | ||
|  |                         pass | ||
|  |                     jobConfig[IncScheduler.currentStatus] = 'Not running' | ||
|  |                     backupjob.config = json.dumps(jobConfig) | ||
|  |                     backupjob.save() | ||
|  |             else: | ||
|  |                 if jobConfig[IncScheduler.frequency] == type: | ||
|  |                     print(jobConfig[IncScheduler.frequency]) | ||
|  |                     finalPath = '%s/%s' % (destinationConfig['path'].rstrip('/'), currentTime) | ||
|  | 
 | ||
|  |                     # import subprocess | ||
|  |                     # import shlex | ||
|  |                     # command = "ssh -o StrictHostKeyChecking=no -p " + destinationConfig[ | ||
|  |                     #     'port'] + " -i /root/.ssh/cyberpanel " + destinationConfig['username'] + "@" + \ | ||
|  |                     #           destinationConfig[ | ||
|  |                     #               'ip'] + " mkdir -p %s" % (finalPath) | ||
|  |                     # subprocess.call(shlex.split(command)) | ||
|  | 
 | ||
|  |                     ### improved paramiko code | ||
|  |                     private_key_path = '/root/.ssh/cyberpanel' | ||
|  | 
 | ||
|  |                     # Create an SSH client | ||
|  |                     ssh = paramiko.SSHClient() | ||
|  |                     ssh.set_missing_host_key_policy(paramiko.AutoAddPolicy()) | ||
|  | 
 | ||
|  |                     # Load the private key | ||
|  |                     private_key = paramiko.RSAKey.from_private_key_file(private_key_path) | ||
|  | 
 | ||
|  |                     # Connect to the server using the private key | ||
|  |                     try: | ||
|  |                         ssh.connect(destinationConfig['ip'], port=int(destinationConfig['port']), username=destinationConfig['username'], pkey=private_key) | ||
|  |                     except BaseException as msg: | ||
|  |                         NormalBackupJobLogs(owner=backupjob, status=backupSchedule.INFO, | ||
|  |                                             message=f'Failed to make sftp connection {str(msg)}').save() | ||
|  | 
 | ||
|  |                         print(str(msg)) | ||
|  |                         continue | ||
|  | 
 | ||
|  |                     # Always try SSH commands first | ||
|  |                     ssh_commands_supported = True | ||
|  |                      | ||
|  |                     try: | ||
|  |                         command = f'find cpbackups -type f -mtime +{jobConfig["retention"]} -exec rm -f {{}} \\;' | ||
|  |                         logging.writeToFile(command) | ||
|  |                         ssh.exec_command(command) | ||
|  |                         command = 'find cpbackups -type d -empty -delete' | ||
|  |                         ssh.exec_command(command) | ||
|  | 
 | ||
|  |                     except BaseException as msg: | ||
|  |                         logging.writeToFile(f'Failed to delete old backups, Error {str(msg)}') | ||
|  |                         pass | ||
|  | 
 | ||
|  |                     # Execute the command to create the remote directory | ||
|  |                     command = f'mkdir -p {finalPath}' | ||
|  |                     try: | ||
|  |                         stdin, stdout, stderr = ssh.exec_command(command, timeout=10) | ||
|  |                         # Wait for the command to finish and check for any errors | ||
|  |                         exit_status = stdout.channel.recv_exit_status() | ||
|  |                         error_message = stderr.read().decode('utf-8') | ||
|  |                         print(error_message) | ||
|  |                          | ||
|  |                         # Check if command was rejected (SFTP-only server) | ||
|  |                         if exit_status != 0 or "not allowed" in error_message.lower() or "channel closed" in error_message.lower(): | ||
|  |                             ssh_commands_supported = False | ||
|  |                             logging.writeToFile(f'SSH command failed on {destinationConfig["ip"]}, falling back to pure SFTP mode') | ||
|  |                              | ||
|  |                             # Try creating directory via SFTP | ||
|  |                             try: | ||
|  |                                 sftp = ssh.open_sftp() | ||
|  |                                 # Try to create the directory structure | ||
|  |                                 path_parts = finalPath.strip('/').split('/') | ||
|  |                                 current_path = '' | ||
|  |                                 for part in path_parts: | ||
|  |                                     current_path = current_path + '/' + part if current_path else part | ||
|  |                                     try: | ||
|  |                                         sftp.stat(current_path) | ||
|  |                                     except FileNotFoundError: | ||
|  |                                         try: | ||
|  |                                             sftp.mkdir(current_path) | ||
|  |                                         except: | ||
|  |                                             pass | ||
|  |                                 sftp.close() | ||
|  |                             except BaseException as msg: | ||
|  |                                 logging.writeToFile(f'Failed to create directory via SFTP: {str(msg)}') | ||
|  |                                 pass | ||
|  |                         elif error_message: | ||
|  |                             NormalBackupJobLogs(owner=backupjob, status=backupSchedule.INFO, | ||
|  |                                                 message=f'Error while creating directory on remote server {error_message.strip()}').save() | ||
|  |                             continue | ||
|  |                         else: | ||
|  |                             pass | ||
|  |                     except BaseException as msg: | ||
|  |                         # SSH command failed, try SFTP | ||
|  |                         ssh_commands_supported = False | ||
|  |                         logging.writeToFile(f'SSH command failed: {str(msg)}, falling back to pure SFTP mode') | ||
|  |                          | ||
|  |                         # Try creating directory via SFTP | ||
|  |                         try: | ||
|  |                             sftp = ssh.open_sftp() | ||
|  |                             # Try to create the directory structure | ||
|  |                             path_parts = finalPath.strip('/').split('/') | ||
|  |                             current_path = '' | ||
|  |                             for part in path_parts: | ||
|  |                                 current_path = current_path + '/' + part if current_path else part | ||
|  |                                 try: | ||
|  |                                     sftp.stat(current_path) | ||
|  |                                 except FileNotFoundError: | ||
|  |                                     try: | ||
|  |                                         sftp.mkdir(current_path) | ||
|  |                                     except: | ||
|  |                                         pass | ||
|  |                             sftp.close() | ||
|  |                         except BaseException as msg: | ||
|  |                             logging.writeToFile(f'Failed to create directory via SFTP: {str(msg)}') | ||
|  |                             pass | ||
|  | 
 | ||
|  | 
 | ||
|  |                     ### Check if an old job prematurely killed, then start from there. | ||
|  |                     # try: | ||
|  |                     #     oldJobContinue = 1 | ||
|  |                     #     pid = jobConfig['pid'] | ||
|  |                     #     stuckDomain = jobConfig['website'] | ||
|  |                     #     finalPath = jobConfig['finalPath'] | ||
|  |                     #     jobConfig['pid'] = str(os.getpid()) | ||
|  |                     # | ||
|  |                     #     command = 'ps aux' | ||
|  |                     #     result = ProcessUtilities.outputExecutioner(command) | ||
|  |                     # | ||
|  |                     #     if result.find(pid) > -1 and result.find('IncScheduler.py') > -1: | ||
|  |                     #         quit(1) | ||
|  |                     # | ||
|  |                     # | ||
|  |                     # except: | ||
|  |                     #     ### Save some important info in backup config | ||
|  |                     #     oldJobContinue = 0 | ||
|  |                     #     jobConfig['pid'] = str(os.getpid()) | ||
|  |                     #     jobConfig['finalPath'] = finalPath | ||
|  | 
 | ||
|  |                     oldJobContinue = 0 | ||
|  |                     jobConfig['pid'] = str(os.getpid()) | ||
|  |                     jobConfig['finalPath'] = finalPath | ||
|  | 
 | ||
|  |                     NormalBackupJobLogs.objects.filter(owner=backupjob).delete() | ||
|  |                     NormalBackupJobLogs(owner=backupjob, status=backupSchedule.INFO, | ||
|  |                                         message='Starting %s backup on %s..' % ( | ||
|  |                                             type, time.strftime("%m.%d.%Y_%H-%M-%S"))).save() | ||
|  | 
 | ||
|  |                     if oldJobContinue: | ||
|  |                         NormalBackupJobLogs(owner=backupjob, status=backupSchedule.INFO, | ||
|  |                                             message='Will continue old killed job starting from %s.' % ( | ||
|  |                                                 stuckDomain)).save() | ||
|  | 
 | ||
|  |                     actualDomain = 0 | ||
|  |                     try: | ||
|  |                         if jobConfig[IncScheduler.allSites] == 'all': | ||
|  |                             websites = Websites.objects.all().order_by('domain') | ||
|  |                             actualDomain = 1 | ||
|  |                         else: | ||
|  |                             websites = backupjob.normalbackupsites_set.all().order_by('domain__domain') | ||
|  |                     except: | ||
|  |                         websites = backupjob.normalbackupsites_set.all().order_by('domain__domain') | ||
|  | 
 | ||
|  |                     doit = 0 | ||
|  | 
 | ||
|  |                     for site in websites: | ||
|  | 
 | ||
|  |                         if actualDomain: | ||
|  |                             domain = site.domain | ||
|  |                         else: | ||
|  |                             domain = site.domain.domain | ||
|  | 
 | ||
|  |                         ### If  this website dont exists continue | ||
|  | 
 | ||
|  |                         try: | ||
|  |                             Websites.objects.get(domain=domain) | ||
|  |                         except: | ||
|  |                             continue | ||
|  | 
 | ||
|  |                         ## | ||
|  | 
 | ||
|  |                         ## Save currently backing domain in db, so that i can restart from here when prematurely killed | ||
|  | 
 | ||
|  |                         jobConfig['website'] = domain | ||
|  |                         jobConfig[IncScheduler.lastRun] = time.strftime("%d %b %Y, %I:%M %p") | ||
|  |                         jobConfig[IncScheduler.currentStatus] = 'Running..' | ||
|  |                         backupjob.config = json.dumps(jobConfig) | ||
|  |                         backupjob.save() | ||
|  | 
 | ||
|  |                         if oldJobContinue and not doit: | ||
|  |                             if domain == stuckDomain: | ||
|  |                                 doit = 1 | ||
|  |                                 continue | ||
|  |                             else: | ||
|  |                                 continue | ||
|  | 
 | ||
|  |                         retValues = backupSchedule.createLocalBackup(domain, '/dev/null') | ||
|  | 
 | ||
|  |                         if retValues[0] == 0: | ||
|  |                             NormalBackupJobLogs(owner=backupjob, status=backupSchedule.ERROR, | ||
|  |                                                 message='Backup failed for %s on %s.' % ( | ||
|  |                                                     domain, time.strftime("%m.%d.%Y_%H-%M-%S"))).save() | ||
|  | 
 | ||
|  |                             SUBJECT = "Automatic backup failed for %s on %s." % (domain, currentTime) | ||
|  |                             adminEmailPath = '/home/cyberpanel/adminEmail' | ||
|  |                             adminEmail = open(adminEmailPath, 'r').read().rstrip('\n') | ||
|  |                             sender = 'root@%s' % (socket.gethostname()) | ||
|  |                             TO = [adminEmail] | ||
|  |                             message = """\
 | ||
|  | From: %s | ||
|  | To: %s | ||
|  | Subject: %s | ||
|  | Automatic backup failed for %s on %s. | ||
|  | """ % (sender, ", ".join(TO), SUBJECT, domain, currentTime)
 | ||
|  | 
 | ||
|  |                             logging.SendEmail(sender, TO, message) | ||
|  |                         else: | ||
|  |                             backupPath = retValues[1] + ".tar.gz" | ||
|  | 
 | ||
|  |                             # Always try scp first | ||
|  |                             command = "scp -o StrictHostKeyChecking=no -P " + destinationConfig[ | ||
|  |                                 'port'] + " -i /root/.ssh/cyberpanel " + backupPath + " " + destinationConfig[ | ||
|  |                                           'username'] + "@" + destinationConfig['ip'] + ":%s" % (finalPath) | ||
|  |                              | ||
|  |                             try: | ||
|  |                                 result = ProcessUtilities.executioner(command) | ||
|  |                                 # Check if scp failed (common with SFTP-only servers) | ||
|  |                                 if not ssh_commands_supported or result != 0: | ||
|  |                                     raise Exception("SCP failed, trying SFTP") | ||
|  |                             except: | ||
|  |                                 # If scp fails or SSH commands are not supported, use SFTP | ||
|  |                                 logging.writeToFile(f'SCP failed for {destinationConfig["ip"]}, falling back to SFTP transfer') | ||
|  |                                 try: | ||
|  |                                     sftp = ssh.open_sftp() | ||
|  |                                     remote_path = os.path.join(finalPath, os.path.basename(backupPath)) | ||
|  |                                     sftp.put(backupPath, remote_path) | ||
|  |                                     sftp.close() | ||
|  |                                     logging.writeToFile(f'Successfully transferred {backupPath} to {remote_path} via SFTP') | ||
|  |                                 except BaseException as msg: | ||
|  |                                     logging.writeToFile(f'Failed to transfer backup via SFTP: {str(msg)}') | ||
|  |                                     NormalBackupJobLogs(owner=backupjob, status=backupSchedule.ERROR, | ||
|  |                                                         message='Backup transfer failed for %s: %s' % (domain, str(msg))).save() | ||
|  |                                     continue | ||
|  | 
 | ||
|  |                             try: | ||
|  |                                 os.remove(backupPath) | ||
|  |                             except: | ||
|  |                                 pass | ||
|  | 
 | ||
|  |                             NormalBackupJobLogs(owner=backupjob, status=backupSchedule.INFO, | ||
|  |                                                 message='Backup completed for %s on %s.' % ( | ||
|  |                                                     domain, time.strftime("%m.%d.%Y_%H-%M-%S"))).save() | ||
|  | 
 | ||
|  |                     jobConfig = json.loads(backupjob.config) | ||
|  |                     try: | ||
|  |                         if jobConfig['pid']: | ||
|  |                             del jobConfig['pid'] | ||
|  |                     except: | ||
|  |                         pass | ||
|  |                     jobConfig[IncScheduler.currentStatus] = 'Not running' | ||
|  |                     backupjob.config = json.dumps(jobConfig) | ||
|  |                     backupjob.save() | ||
|  | 
 | ||
|  | 
 | ||
|  |                     ### check if todays backups are fine | ||
|  | 
 | ||
|  |                     from IncBackups.models import OneClickBackups | ||
|  | 
 | ||
|  |                     try: | ||
|  | 
 | ||
|  |                         ocb = OneClickBackups.objects.get(sftpUser=destinationConfig['username']) | ||
|  |                         from plogical.acl import ACLManager | ||
|  | 
 | ||
|  |                         for site in websites: | ||
|  | 
 | ||
|  |                             from datetime import datetime, timedelta | ||
|  | 
 | ||
|  |                             Yesterday = (datetime.now() - timedelta(days=1)).strftime("%m.%d.%Y") | ||
|  |                             print(f'date of yesterday {Yesterday}') | ||
|  | 
 | ||
|  |                             # Command to list directories under the specified path | ||
|  |                             command = f"ls -d {finalPath}/*" | ||
|  | 
 | ||
|  |                             # Try SSH command first | ||
|  |                             directories = [] | ||
|  |                             try: | ||
|  |                                 # Execute the command | ||
|  |                                 stdin, stdout, stderr = ssh.exec_command(command, timeout=10) | ||
|  | 
 | ||
|  |                                 # Read the results | ||
|  |                                 directories = stdout.read().decode().splitlines() | ||
|  |                             except: | ||
|  |                                 # If SSH command fails, try using SFTP | ||
|  |                                 logging.writeToFile(f'SSH ls command failed for {destinationConfig["ip"]}, trying SFTP listdir') | ||
|  |                                 try: | ||
|  |                                     sftp = ssh.open_sftp() | ||
|  |                                     # List files in the directory | ||
|  |                                     files = sftp.listdir(finalPath) | ||
|  |                                     # Format them similar to ls -d output | ||
|  |                                     directories = [f"{finalPath}/{f}" for f in files] | ||
|  |                                     sftp.close() | ||
|  |                                 except BaseException as msg: | ||
|  |                                     logging.writeToFile(f'Failed to list directory via SFTP: {str(msg)}') | ||
|  |                                     directories = [] | ||
|  | 
 | ||
|  |                             if os.path.exists(ProcessUtilities.debugPath): | ||
|  |                                 logging.writeToFile(str(directories)) | ||
|  | 
 | ||
|  |                             try: | ||
|  | 
 | ||
|  |                                 startCheck = 0 | ||
|  |                                 for directory in directories: | ||
|  |                                     if directory.find(site.domain): | ||
|  |                                         print(f'site in backup, no need to notify {site.domain}') | ||
|  |                                         startCheck = 1 | ||
|  |                                         break | ||
|  | 
 | ||
|  |                                 if startCheck: | ||
|  |                                     'send notification that backup failed' | ||
|  |                                     import requests | ||
|  | 
 | ||
|  |                                     # Define the URL of the endpoint | ||
|  |                                     url = 'http://platform.cyberpersons.com/Billing/BackupFailedNotify'  # Replace with your actual endpoint URL | ||
|  | 
 | ||
|  |                                     # Define the payload to send in the POST request | ||
|  |                                     payload = { | ||
|  |                                         'sub': ocb.subscription, | ||
|  |                                         'subject': f'Failed to backup {site.domain} on {ACLManager.fetchIP()}.', | ||
|  |                                         'message':f'Hi, \n\n Failed to create backup for {site.domain} on on {ACLManager.fetchIP()}. \n\n Please contact our support team at: http://platform.cyberpersons.com\n\nThank you.', | ||
|  |                                         # Replace with the actual SSH public key | ||
|  |                                         'sftpUser': ocb.sftpUser, | ||
|  |                                         'serverIP': ACLManager.fetchIP(),  # Replace with the actual server IP | ||
|  |                                     } | ||
|  | 
 | ||
|  |                                     # Convert the payload to JSON format | ||
|  |                                     headers = {'Content-Type': 'application/json'} | ||
|  |                                     dataRet = json.dumps(payload) | ||
|  | 
 | ||
|  |                                     # Make the POST request | ||
|  |                                     response = requests.post(url, headers=headers, data=dataRet) | ||
|  | 
 | ||
|  |                                     # # Handle the response | ||
|  |                                     # # Handle the response | ||
|  |                                     # if response.status_code == 200: | ||
|  |                                     #     response_data = response.json() | ||
|  |                                     #     if response_data.get('status') == 1: | ||
|  |                             except: | ||
|  |                                 pass | ||
|  | 
 | ||
|  |                     except: | ||
|  |                         pass | ||
|  | 
 | ||
|  | 
 | ||
|  | 
 | ||
|  |     @staticmethod | ||
|  |     def fetchAWSKeys(): | ||
|  |         path = '/home/cyberpanel/.aws' | ||
|  |         credentials = path + '/credentials' | ||
|  | 
 | ||
|  |         data = open(credentials, 'r').readlines() | ||
|  | 
 | ||
|  |         aws_access_key_id = data[1].split(' ')[2].strip(' ').strip('\n') | ||
|  |         aws_secret_access_key = data[2].split(' ')[2].strip(' ').strip('\n') | ||
|  |         region = data[3].split(' ')[2].strip(' ').strip('\n') | ||
|  | 
 | ||
|  |         return aws_access_key_id, aws_secret_access_key, region | ||
|  | 
 | ||
|  |     @staticmethod | ||
|  |     def forceRunAWSBackup(planName): | ||
|  |         try: | ||
|  | 
 | ||
|  |             plan = BackupPlan.objects.get(name=planName) | ||
|  |             bucketName = plan.bucket.strip('\n').strip(' ') | ||
|  |             runTime = time.strftime("%d:%m:%Y") | ||
|  | 
 | ||
|  |             config = TransferConfig(multipart_threshold=1024 * 25, max_concurrency=10, | ||
|  |                                     multipart_chunksize=1024 * 25, use_threads=True) | ||
|  | 
 | ||
|  |             ## | ||
|  | 
 | ||
|  |             aws_access_key_id, aws_secret_access_key, region = IncScheduler.fetchAWSKeys() | ||
|  | 
 | ||
|  |             ts = time.time() | ||
|  |             retentionSeconds = 86400 * plan.retention | ||
|  | 
 | ||
|  |             if region.find('http') > -1: | ||
|  |                 s3 = boto3.resource( | ||
|  |                     's3', | ||
|  |                     aws_access_key_id=aws_access_key_id, | ||
|  |                     aws_secret_access_key=aws_secret_access_key, | ||
|  |                     endpoint_url=region | ||
|  |                 ) | ||
|  |             else: | ||
|  |                 s3 = boto3.resource( | ||
|  |                     's3', | ||
|  |                     aws_access_key_id=aws_access_key_id, | ||
|  |                     aws_secret_access_key=aws_secret_access_key, | ||
|  |                 ) | ||
|  | 
 | ||
|  |             bucket = s3.Bucket(plan.bucket) | ||
|  | 
 | ||
|  |             for file in bucket.objects.all(): | ||
|  |                 result = float(ts - file.last_modified.timestamp()) | ||
|  |                 if result > retentionSeconds: | ||
|  |                     BackupLogs(owner=plan, level='INFO', timeStamp=time.strftime("%b %d %Y, %H:%M:%S"), | ||
|  |                                msg='File %s expired and deleted according to your retention settings.' % ( | ||
|  |                                    file.key)).save() | ||
|  |                     file.delete() | ||
|  | 
 | ||
|  |             ### | ||
|  | 
 | ||
|  |             if region.find('http') > -1: | ||
|  |                 client = boto3.client( | ||
|  |                     's3', | ||
|  |                     aws_access_key_id=aws_access_key_id, | ||
|  |                     aws_secret_access_key=aws_secret_access_key, | ||
|  |                     endpoint_url=region | ||
|  |                 ) | ||
|  |             else: | ||
|  |                 client = boto3.client( | ||
|  |                     's3', | ||
|  |                     aws_access_key_id=aws_access_key_id, | ||
|  |                     aws_secret_access_key=aws_secret_access_key, | ||
|  |                 ) | ||
|  | 
 | ||
|  |             ## | ||
|  | 
 | ||
|  |             BackupLogs(owner=plan, level='INFO', timeStamp=time.strftime("%b %d %Y, %H:%M:%S"), | ||
|  |                        msg='Starting backup process..').save() | ||
|  | 
 | ||
|  |             PlanConfig = json.loads(plan.config) | ||
|  | 
 | ||
|  |             for items in plan.websitesinplan_set.all(): | ||
|  | 
 | ||
|  |                 from plogical.backupUtilities import backupUtilities | ||
|  |                 tempStatusPath = "/home/cyberpanel/" + str(randint(1000, 9999)) | ||
|  |                 extraArgs = {} | ||
|  |                 extraArgs['domain'] = items.domain | ||
|  |                 extraArgs['tempStatusPath'] = tempStatusPath | ||
|  |                 extraArgs['data'] = int(PlanConfig['data']) | ||
|  |                 extraArgs['emails'] = int(PlanConfig['emails']) | ||
|  |                 extraArgs['databases'] = int(PlanConfig['databases']) | ||
|  |                 extraArgs['port'] = '0' | ||
|  |                 extraArgs['ip'] = '0' | ||
|  |                 extraArgs['destinationDomain'] = 'None' | ||
|  |                 extraArgs['path'] = '/home/cyberpanel/backups/%s/backup-' % ( | ||
|  |                     items.domain) + items.domain + "-" + time.strftime("%m.%d.%Y_%H-%M-%S") | ||
|  | 
 | ||
|  |                 bu = backupUtilities(extraArgs) | ||
|  |                 result, fileName = bu.CloudBackups() | ||
|  | 
 | ||
|  |                 finalResult = open(tempStatusPath, 'r').read() | ||
|  | 
 | ||
|  |                 if result == 1: | ||
|  |                     key = plan.name + '/' + items.domain + '/' + fileName.split('/')[-1] | ||
|  |                     client.upload_file( | ||
|  |                         fileName, | ||
|  |                         bucketName, | ||
|  |                         key, | ||
|  |                         Config=config | ||
|  |                     ) | ||
|  | 
 | ||
|  |                     command = 'rm -f ' + fileName | ||
|  |                     ProcessUtilities.executioner(command) | ||
|  | 
 | ||
|  |                     BackupLogs(owner=plan, level='INFO', timeStamp=time.strftime("%b %d %Y, %H:%M:%S"), | ||
|  |                                msg='Backup successful for ' + items.domain + '.').save() | ||
|  |                 else: | ||
|  |                     BackupLogs(owner=plan, level='ERROR', timeStamp=time.strftime("%b %d %Y, %H:%M:%S"), | ||
|  |                                msg='Backup failed for ' + items.domain + '. Error: ' + finalResult).save() | ||
|  | 
 | ||
|  |             plan.lastRun = runTime | ||
|  |             plan.save() | ||
|  | 
 | ||
|  |             BackupLogs(owner=plan, level='INFO', timeStamp=time.strftime("%b %d %Y, %H:%M:%S"), | ||
|  |                        msg='Backup Process Finished.').save() | ||
|  | 
 | ||
|  |         except BaseException as msg: | ||
|  |             logging.writeToFile(str(msg) + ' [S3Backups.runBackupPlan]') | ||
|  |             plan = BackupPlan.objects.get(name=planName) | ||
|  |             BackupLogs(owner=plan, timeStamp=time.strftime("%b %d %Y, %H:%M:%S"), level='ERROR', msg=str(msg)).save() | ||
|  | 
 | ||
|  |     @staticmethod | ||
|  |     def runAWSBackups(freq): | ||
|  |         try: | ||
|  |             for plan in BackupPlan.objects.all(): | ||
|  |                 if plan.freq == 'Daily' == freq: | ||
|  |                     IncScheduler.forceRunAWSBackup(plan.name) | ||
|  |         except BaseException as msg: | ||
|  |             logging.writeToFile(str(msg) + ' [S3Backups.runAWSBackups]') | ||
|  | 
 | ||
|  |     @staticmethod | ||
|  |     def CalculateAndUpdateDiskUsage(): | ||
|  |         for website in Websites.objects.all(): | ||
|  |             try: | ||
|  |                 try: | ||
|  |                     config = json.loads(website.config) | ||
|  |                 except: | ||
|  |                     config = {} | ||
|  | 
 | ||
|  |                 eDomains = website.domains_set.all() | ||
|  | 
 | ||
|  |                 for eDomain in eDomains: | ||
|  |                     for email in eDomain.eusers_set.all(): | ||
|  |                         emailPath = '/home/vmail/%s/%s' % (website.domain, email.email.split('@')[0]) | ||
|  |                         email.DiskUsage = virtualHostUtilities.getDiskUsageofPath(emailPath) | ||
|  |                         email.save() | ||
|  |                         print('Disk Usage of %s is %s' % (email.email, email.DiskUsage)) | ||
|  | 
 | ||
|  |                 config['DiskUsage'], config['DiskUsagePercentage'] = virtualHostUtilities.getDiskUsage( | ||
|  |                     "/home/" + website.domain, website.package.diskSpace) | ||
|  | 
 | ||
|  |                 # if website.package.enforceDiskLimits: | ||
|  |                 #     spaceString = f'{website.package.diskSpace}M {website.package.diskSpace}M' | ||
|  |                 #     command = f'setquota -u {website.externalApp} {spaceString} 0 0 /' | ||
|  |                 #     ProcessUtilities.executioner(command) | ||
|  |                 #     if config['DiskUsagePercentage'] >= 100: | ||
|  |                 #         command = 'chattr -R +i /home/%s/' % (website.domain) | ||
|  |                 #         ProcessUtilities.executioner(command) | ||
|  |                 # | ||
|  |                 #         command = 'chattr -R -i /home/%s/logs/' % (website.domain) | ||
|  |                 #         ProcessUtilities.executioner(command) | ||
|  |                 # | ||
|  |                 #         command = 'chattr -R -i /home/%s/.trash/' % (website.domain) | ||
|  |                 #         ProcessUtilities.executioner(command) | ||
|  |                 # | ||
|  |                 #         command = 'chattr -R -i /home/%s/backup/' % (website.domain) | ||
|  |                 #         ProcessUtilities.executioner(command) | ||
|  |                 # | ||
|  |                 #         command = 'chattr -R -i /home/%s/incbackup/' % (website.domain) | ||
|  |                 #         ProcessUtilities.executioner(command) | ||
|  |                 #     else: | ||
|  |                 #         command = 'chattr -R -i /home/%s/' % (website.domain) | ||
|  |                 #         ProcessUtilities.executioner(command) | ||
|  | 
 | ||
|  |                 ## Calculate bw usage | ||
|  | 
 | ||
|  |                 from plogical.vhost import vhost | ||
|  |                 config['bwInMB'], config['bwUsage'] = vhost.findDomainBW(website.domain, int(website.package.bandwidth)) | ||
|  | 
 | ||
|  |                 website.config = json.dumps(config) | ||
|  |                 website.save() | ||
|  | 
 | ||
|  |             except BaseException as msg: | ||
|  |                 logging.writeToFile('%s. [CalculateAndUpdateDiskUsage:753]' % (str(msg))) | ||
|  | 
 | ||
|  |     @staticmethod | ||
|  |     def WPUpdates(): | ||
|  |         from cloudAPI.models import WPDeployments | ||
|  |         for wp in WPDeployments.objects.all(): | ||
|  |             try: | ||
|  |                 try: | ||
|  |                     config = json.loads(wp.config) | ||
|  |                 except: | ||
|  |                     config = {} | ||
|  | 
 | ||
|  |                 ### Core Updates | ||
|  | 
 | ||
|  |                 if config['updates'] == 'Minor and Security Updates': | ||
|  |                     command = 'wp core update --minor --allow-root --path=/home/%s/public_html' % (config['domainName']) | ||
|  |                     ProcessUtilities.executioner(command) | ||
|  |                 elif config['updates'] == 'All (minor and major)': | ||
|  |                     command = 'wp core update --allow-root --path=/home/%s/public_html' % (config['domainName']) | ||
|  |                     ProcessUtilities.executioner(command) | ||
|  | 
 | ||
|  |                 ### Plugins, for plugins we will do minor updates only. | ||
|  | 
 | ||
|  |                 if config['pluginUpdates'] == 'Enabled': | ||
|  |                     command = 'wp plugin update --all --minor --allow-root --path=/home/%s/public_html' % ( | ||
|  |                         config['domainName']) | ||
|  |                     ProcessUtilities.executioner(command) | ||
|  | 
 | ||
|  |                 ### Themes, for plugins we will do minor updates only. | ||
|  | 
 | ||
|  |                 if config['themeUpdates'] == 'Enabled': | ||
|  |                     command = 'wp theme update --all --minor --allow-root --path=/home/%s/public_html' % ( | ||
|  |                         config['domainName']) | ||
|  |                     ProcessUtilities.executioner(command) | ||
|  | 
 | ||
|  |             except BaseException as msg: | ||
|  |                 logging.writeToFile('%s. [WPUpdates:767]' % (str(msg))) | ||
|  | 
 | ||
|  |     @staticmethod | ||
|  |     def RemoteBackup(function): | ||
|  |         try: | ||
|  |             print("....start remote backup...............") | ||
|  |             from websiteFunctions.models import RemoteBackupSchedule, RemoteBackupsites, WPSites | ||
|  |             from loginSystem.models import Administrator | ||
|  |             import json | ||
|  |             import time | ||
|  |             from plogical.applicationInstaller import ApplicationInstaller | ||
|  |             for config in RemoteBackupSchedule.objects.all(): | ||
|  |                 print("....start remote backup........site.......%s"%config.Name) | ||
|  |                 try: | ||
|  |                     configbakup = json.loads(config.config) | ||
|  |                     backuptype = configbakup['BackupType'] | ||
|  |                     print("....start remote backup........site.......%s.. and bakuptype...%s" % (config.Name, backuptype)) | ||
|  |                     if backuptype == 'Only DataBase': | ||
|  |                         Backuptype = "3" | ||
|  |                     elif backuptype == 'Only Website': | ||
|  |                         Backuptype = "2" | ||
|  |                     else: | ||
|  |                         Backuptype = "1" | ||
|  |                 except BaseException as msg: | ||
|  |                     print("....backup config type Error.%s" % str(msg)) | ||
|  |                     continue | ||
|  |                 try: | ||
|  |                     allRemoteBackupsiteobj = RemoteBackupsites.objects.filter(owner=config) | ||
|  |                     print("store site id.....%s"%str(allRemoteBackupsiteobj)) | ||
|  |                     for i in allRemoteBackupsiteobj: | ||
|  |                         try: | ||
|  |                             backupsiteID = i.WPsites | ||
|  |                             wpsite = WPSites.objects.get(pk=backupsiteID) | ||
|  |                             print("site name.....%s"%wpsite.title) | ||
|  |                             AdminID = wpsite.owner.admin_id | ||
|  |                             Admin = Administrator.objects.get(pk=AdminID) | ||
|  | 
 | ||
|  |                             Lastrun = config.lastrun | ||
|  |                             Currenttime = float(time.time()) | ||
|  | 
 | ||
|  |                             if config.timeintervel == function: | ||
|  |                                 #al = float(Currenttime) - float(1800) | ||
|  |                                 #if float(al) >= float(Lastrun): | ||
|  |                                 #if 1 == 1: | ||
|  | 
 | ||
|  |                                 extraArgs = {} | ||
|  |                                 extraArgs['adminID'] = Admin.pk | ||
|  |                                 extraArgs['WPid'] = wpsite.pk | ||
|  |                                 extraArgs['Backuptype'] = Backuptype | ||
|  |                                 extraArgs['BackupDestination'] = config.RemoteBackupConfig.configtype | ||
|  |                                 extraArgs['SFTPID'] = config.RemoteBackupConfig_id | ||
|  | 
 | ||
|  |                                 extraArgs['tempStatusPath'] = "/home/cyberpanel/" + str(randint(1000, 9999)) | ||
|  |                                 background = ApplicationInstaller('WPCreateBackup', extraArgs) | ||
|  |                                 status, msg, backupID = background.WPCreateBackup() | ||
|  |                                 if status == 1: | ||
|  |                                     filename = msg | ||
|  |                                     if config.RemoteBackupConfig.configtype == "SFTP": | ||
|  |                                         IncScheduler.SendTORemote(filename, config.RemoteBackupConfig_id) | ||
|  |                                         command = f"rm -r {filename}" | ||
|  |                                         ProcessUtilities.executioner(command) | ||
|  |                                         obj = RemoteBackupSchedule.objects.get(pk=config.id) | ||
|  |                                         obj.lastrun = time.time() | ||
|  |                                         obj.save() | ||
|  |                                     elif config.RemoteBackupConfig.configtype == "S3": | ||
|  |                                         IncScheduler.SendToS3Cloud(filename, config.RemoteBackupConfig_id, backupID, | ||
|  |                                                                    config.id) | ||
|  |                                         command = f"rm -r {filename}" | ||
|  |                                         ProcessUtilities.executioner(command) | ||
|  |                                         obj = RemoteBackupSchedule.objects.get(pk=config.id) | ||
|  |                                         obj.lastrun = time.time() | ||
|  |                                         obj.save() | ||
|  | 
 | ||
|  |                             elif config.timeintervel == function: | ||
|  |                                 #al = float(Currenttime) - float(3600) | ||
|  |                                 #if float(al) >= float(Lastrun): | ||
|  |                                 # if 1 == 1: | ||
|  | 
 | ||
|  |                                 extraArgs = {} | ||
|  |                                 extraArgs['adminID'] = Admin.pk | ||
|  |                                 extraArgs['WPid'] = wpsite.pk | ||
|  |                                 extraArgs['Backuptype'] = Backuptype | ||
|  |                                 extraArgs['BackupDestination'] = config.RemoteBackupConfig.configtype | ||
|  |                                 extraArgs['SFTPID'] = config.RemoteBackupConfig_id | ||
|  | 
 | ||
|  |                                 extraArgs['tempStatusPath'] = "/home/cyberpanel/" + str(randint(1000, 9999)) | ||
|  |                                 background = ApplicationInstaller('WPCreateBackup', extraArgs) | ||
|  |                                 status, msg, backupID = background.WPCreateBackup() | ||
|  |                                 if status == 1: | ||
|  |                                     filename = msg | ||
|  |                                     if config.RemoteBackupConfig.configtype == "SFTP": | ||
|  |                                         IncScheduler.SendTORemote(filename, config.RemoteBackupConfig_id) | ||
|  |                                         command = f"rm -r {filename}" | ||
|  |                                         ProcessUtilities.executioner(command) | ||
|  |                                         obj = RemoteBackupSchedule.objects.get(pk=config.id) | ||
|  |                                         obj.lastrun = time.time() | ||
|  |                                         obj.save() | ||
|  |                                     elif config.RemoteBackupConfig.configtype == "S3": | ||
|  |                                         IncScheduler.SendToS3Cloud(filename, config.RemoteBackupConfig_id, backupID, | ||
|  |                                                                    config.id) | ||
|  |                                         command = f"rm -r {filename}" | ||
|  |                                         ProcessUtilities.executioner(command) | ||
|  |                                         obj = RemoteBackupSchedule.objects.get(pk=config.id) | ||
|  |                                         obj.lastrun = time.time() | ||
|  |                                         obj.save() | ||
|  | 
 | ||
|  |                             elif config.timeintervel == function: | ||
|  |                                 #al = float(Currenttime) - float(21600) | ||
|  |                                 #if float(al) >= float(Lastrun): | ||
|  | 
 | ||
|  |                                 extraArgs = {} | ||
|  |                                 extraArgs['adminID'] = Admin.pk | ||
|  |                                 extraArgs['WPid'] = wpsite.pk | ||
|  |                                 extraArgs['Backuptype'] = Backuptype | ||
|  |                                 extraArgs['BackupDestination'] = "SFTP" | ||
|  |                                 extraArgs['SFTPID'] = config.RemoteBackupConfig_id | ||
|  | 
 | ||
|  |                                 extraArgs['tempStatusPath'] = "/home/cyberpanel/" + str(randint(1000, 9999)) | ||
|  |                                 background = ApplicationInstaller('WPCreateBackup', extraArgs) | ||
|  |                                 status, msg, backupID = background.WPCreateBackup() | ||
|  |                                 if status == 1: | ||
|  |                                     filename = msg | ||
|  |                                     if config.RemoteBackupConfig.configtype == "SFTP": | ||
|  |                                         IncScheduler.SendTORemote(filename, config.RemoteBackupConfig_id) | ||
|  |                                         command = f"rm -r {filename}" | ||
|  |                                         ProcessUtilities.executioner(command) | ||
|  |                                         obj = RemoteBackupSchedule.objects.get(pk=config.id) | ||
|  |                                         obj.lastrun = time.time() | ||
|  |                                         obj.save() | ||
|  |                                     elif config.RemoteBackupConfig.configtype == "S3": | ||
|  |                                         IncScheduler.SendToS3Cloud(filename, config.RemoteBackupConfig_id, backupID, | ||
|  |                                                                    config.id) | ||
|  |                                         command = f"rm -r {filename}" | ||
|  |                                         ProcessUtilities.executioner(command) | ||
|  |                                         obj = RemoteBackupSchedule.objects.get(pk=config.id) | ||
|  |                                         obj.lastrun = time.time() | ||
|  |                                         obj.save() | ||
|  | 
 | ||
|  |                             elif config.timeintervel == function: | ||
|  |                                 #al = float(Currenttime) - float(43200) | ||
|  |                                 #if float(al) >= float(Lastrun): | ||
|  |                                 extraArgs = {} | ||
|  |                                 extraArgs['adminID'] = Admin.pk | ||
|  |                                 extraArgs['WPid'] = wpsite.pk | ||
|  |                                 extraArgs['Backuptype'] = Backuptype | ||
|  |                                 extraArgs['BackupDestination'] = "SFTP" | ||
|  |                                 extraArgs['SFTPID'] = config.RemoteBackupConfig_id | ||
|  | 
 | ||
|  |                                 extraArgs['tempStatusPath'] = "/home/cyberpanel/" + str(randint(1000, 9999)) | ||
|  |                                 background = ApplicationInstaller('WPCreateBackup', extraArgs) | ||
|  |                                 status, msg, backupID = background.WPCreateBackup() | ||
|  |                                 if status == 1: | ||
|  |                                     filename = msg | ||
|  |                                     if config.RemoteBackupConfig.configtype == "SFTP": | ||
|  |                                         IncScheduler.SendTORemote(filename, config.RemoteBackupConfig_id) | ||
|  |                                         command = f"rm -r {filename}" | ||
|  |                                         ProcessUtilities.executioner(command) | ||
|  |                                         obj = RemoteBackupSchedule.objects.get(pk=config.id) | ||
|  |                                         obj.lastrun = time.time() | ||
|  |                                         obj.save() | ||
|  |                                     elif config.RemoteBackupConfig.configtype == "S3": | ||
|  |                                         IncScheduler.SendToS3Cloud(filename, config.RemoteBackupConfig_id, backupID, | ||
|  |                                                                    config.id) | ||
|  |                                         command = f"rm -r {filename}" | ||
|  |                                         ProcessUtilities.executioner(command) | ||
|  |                                         obj = RemoteBackupSchedule.objects.get(pk=config.id) | ||
|  |                                         obj.lastrun = time.time() | ||
|  |                                         obj.save() | ||
|  | 
 | ||
|  |                             elif config.timeintervel == function: | ||
|  |                                 #al = float(Currenttime) - float(86400) | ||
|  |                                 #if float(al) >= float(Lastrun): | ||
|  | 
 | ||
|  |                                 extraArgs = {} | ||
|  |                                 extraArgs['adminID'] = Admin.pk | ||
|  |                                 extraArgs['WPid'] = wpsite.pk | ||
|  |                                 extraArgs['Backuptype'] = Backuptype | ||
|  |                                 extraArgs['BackupDestination'] = "SFTP" | ||
|  |                                 extraArgs['SFTPID'] = config.RemoteBackupConfig_id | ||
|  | 
 | ||
|  |                                 extraArgs['tempStatusPath'] = "/home/cyberpanel/" + str(randint(1000, 9999)) | ||
|  |                                 background = ApplicationInstaller('WPCreateBackup', extraArgs) | ||
|  |                                 status, msg, backupID = background.WPCreateBackup() | ||
|  |                                 if status == 1: | ||
|  |                                     filename = msg | ||
|  |                                     if config.RemoteBackupConfig.configtype == "SFTP": | ||
|  |                                         IncScheduler.SendTORemote(filename, config.RemoteBackupConfig_id) | ||
|  |                                         command = f"rm -r {filename}" | ||
|  |                                         ProcessUtilities.executioner(command) | ||
|  |                                         obj = RemoteBackupSchedule.objects.get(pk=config.id) | ||
|  |                                         obj.lastrun = time.time() | ||
|  |                                         obj.save() | ||
|  |                                     elif config.RemoteBackupConfig.configtype == "S3": | ||
|  |                                         IncScheduler.SendToS3Cloud(filename, config.RemoteBackupConfig_id, backupID, | ||
|  |                                                                    config.id) | ||
|  |                                         command = f"rm -r {filename}" | ||
|  |                                         ProcessUtilities.executioner(command) | ||
|  |                                         obj = RemoteBackupSchedule.objects.get(pk=config.id) | ||
|  |                                         obj.lastrun = time.time() | ||
|  |                                         obj.save() | ||
|  | 
 | ||
|  |                             elif config.timeintervel == function: | ||
|  |                                 #al = float(Currenttime) - float(259200) | ||
|  |                                 #if float(al) >= float(Lastrun): | ||
|  | 
 | ||
|  |                                 extraArgs = {} | ||
|  |                                 extraArgs['adminID'] = Admin.pk | ||
|  |                                 extraArgs['WPid'] = wpsite.pk | ||
|  |                                 extraArgs['Backuptype'] = Backuptype | ||
|  |                                 extraArgs['BackupDestination'] = "SFTP" | ||
|  |                                 extraArgs['SFTPID'] = config.RemoteBackupConfig_id | ||
|  | 
 | ||
|  |                                 extraArgs['tempStatusPath'] = "/home/cyberpanel/" + str(randint(1000, 9999)) | ||
|  |                                 background = ApplicationInstaller('WPCreateBackup', extraArgs) | ||
|  |                                 status, msg, backupID = background.WPCreateBackup() | ||
|  |                                 if status == 1: | ||
|  |                                     filename = msg | ||
|  |                                     if config.RemoteBackupConfig.configtype == "SFTP": | ||
|  |                                         IncScheduler.SendTORemote(filename, config.RemoteBackupConfig_id) | ||
|  |                                         command = f"rm -r {filename}" | ||
|  |                                         ProcessUtilities.executioner(command) | ||
|  |                                         obj = RemoteBackupSchedule.objects.get(pk=config.id) | ||
|  |                                         obj.lastrun = time.time() | ||
|  |                                         obj.save() | ||
|  |                                     elif config.RemoteBackupConfig.configtype == "S3": | ||
|  |                                         IncScheduler.SendToS3Cloud(filename, config.RemoteBackupConfig_id, backupID, | ||
|  |                                                                    config.id) | ||
|  |                                         command = f"rm -r {filename}" | ||
|  |                                         ProcessUtilities.executioner(command) | ||
|  |                                         obj = RemoteBackupSchedule.objects.get(pk=config.id) | ||
|  |                                         obj.lastrun = time.time() | ||
|  |                                         obj.save() | ||
|  | 
 | ||
|  |                             elif config.timeintervel == function: | ||
|  |                                 #al = float(Currenttime) - float(604800) | ||
|  |                                 #if float(al) >= float(Lastrun): | ||
|  | 
 | ||
|  |                                 extraArgs = {} | ||
|  |                                 extraArgs['adminID'] = Admin.pk | ||
|  |                                 extraArgs['WPid'] = wpsite.pk | ||
|  |                                 extraArgs['Backuptype'] = Backuptype | ||
|  |                                 extraArgs['BackupDestination'] = "SFTP" | ||
|  |                                 extraArgs['SFTPID'] = config.RemoteBackupConfig_id | ||
|  | 
 | ||
|  |                                 extraArgs['tempStatusPath'] = "/home/cyberpanel/" + str(randint(1000, 9999)) | ||
|  |                                 background = ApplicationInstaller('WPCreateBackup', extraArgs) | ||
|  |                                 status, msg, backupID = background.WPCreateBackup() | ||
|  |                                 if status == 1: | ||
|  |                                     filename = msg | ||
|  |                                     if config.RemoteBackupConfig.configtype == "SFTP": | ||
|  |                                         IncScheduler.SendTORemote(filename, config.RemoteBackupConfig_id) | ||
|  |                                         command = f"rm -r {filename}" | ||
|  |                                         ProcessUtilities.executioner(command) | ||
|  |                                         obj = RemoteBackupSchedule.objects.get(pk=config.id) | ||
|  |                                         obj.lastrun = time.time() | ||
|  |                                         obj.save() | ||
|  |                                     elif config.RemoteBackupConfig.configtype == "S3": | ||
|  |                                         IncScheduler.SendToS3Cloud(filename, config.RemoteBackupConfig_id, backupID, | ||
|  |                                                                    config.id) | ||
|  |                                         command = f"rm -r {filename}" | ||
|  |                                         ProcessUtilities.executioner(command) | ||
|  |                                         obj = RemoteBackupSchedule.objects.get(pk=config.id) | ||
|  |                                         obj.lastrun = time.time() | ||
|  |                                         obj.save() | ||
|  |                         except: | ||
|  |                             pass | ||
|  | 
 | ||
|  |                 except BaseException as msg: | ||
|  |                     print("Error in Sites:%s" % str(msg)) | ||
|  |                     continue | ||
|  |         except BaseException as msg: | ||
|  |             print("Error: [RemoteBackup]: %s" % str(msg)) | ||
|  |             logging.writeToFile('%s. [RemoteBackup]' % (str(msg))) | ||
|  | 
 | ||
|  |     @staticmethod | ||
|  |     def SendTORemote(FileName, RemoteBackupID): | ||
|  |         import json | ||
|  |         from websiteFunctions.models import RemoteBackupConfig | ||
|  | 
 | ||
|  |         try: | ||
|  |             RemoteBackupOBJ = RemoteBackupConfig.objects.get(pk=RemoteBackupID) | ||
|  |             config = json.loads(RemoteBackupOBJ.config) | ||
|  |             HostName = config['Hostname'] | ||
|  |             Username = config['Username'] | ||
|  |             Password = config['Password'] | ||
|  |             Path = config['Path'] | ||
|  | 
 | ||
|  |             # Connect to the remote server using the private key | ||
|  |             ssh = paramiko.SSHClient() | ||
|  |             ssh.set_missing_host_key_policy(paramiko.AutoAddPolicy()) | ||
|  | 
 | ||
|  |             # Connect to the server using the private key | ||
|  |             ssh.connect(HostName, username=Username, password=Password) | ||
|  |             sftp = ssh.open_sftp() | ||
|  |             ssh.exec_command(f'mkdir -p {Path}') | ||
|  | 
 | ||
|  |             if os.path.exists(ProcessUtilities.debugPath): | ||
|  |                 logging.writeToFile(f"Filename: {FileName}, Path {Path}/{FileName.split('/')[-1]}. [SendTORemote]") | ||
|  | 
 | ||
|  |             sftp.put(FileName, f"{Path}/{FileName.split('/')[-1]}") | ||
|  | 
 | ||
|  |             # sftp.get(str(remotepath), str(loaclpath), | ||
|  |             #          callback=self.UpdateDownloadStatus) | ||
|  |             # | ||
|  |             # cnopts = sftp.CnOpts() | ||
|  |             # cnopts.hostkeys = None | ||
|  |             # | ||
|  |             # with pysftp.Connection(HostName, username=Username, password=Password, cnopts=cnopts) as sftp: | ||
|  |             #     print("Connection succesfully stablished ... ") | ||
|  |             # | ||
|  |             #     try: | ||
|  |             #         with sftp.cd(Path): | ||
|  |             #             sftp.put(FileName) | ||
|  |             #     except BaseException as msg: | ||
|  |             #         print(f'Error on {str(msg)}') | ||
|  |             #         sftp.mkdir(Path) | ||
|  |             #         with sftp.cd(Path): | ||
|  |             #             sftp.put(FileName) | ||
|  | 
 | ||
|  | 
 | ||
|  | 
 | ||
|  |         except BaseException as msg: | ||
|  |             print('%s. [SendTORemote]' % (str(msg))) | ||
|  |             logging.writeToFile('%s. [SendTORemote]' % (str(msg))) | ||
|  | 
 | ||
|  |     @staticmethod | ||
|  |     def SendToS3Cloud(FileName, RemoteBackupCofigID, backupID, scheduleID): | ||
|  |         import boto3 | ||
|  |         import json | ||
|  |         import time | ||
|  |         from websiteFunctions.models import RemoteBackupConfig, WPSitesBackup, RemoteBackupSchedule | ||
|  |         import plogical.randomPassword as randomPassword | ||
|  |         try: | ||
|  |             print("UPloading to S3") | ||
|  |             Backupobj = WPSitesBackup.objects.get(pk=backupID) | ||
|  |             backupConfig = json.loads(Backupobj.config) | ||
|  |             websitedomain = backupConfig['WebDomain'] | ||
|  |             RemoteBackupOBJ = RemoteBackupConfig.objects.get(pk=RemoteBackupCofigID) | ||
|  |             config = json.loads(RemoteBackupOBJ.config) | ||
|  |             provider = config['Provider'] | ||
|  |             if provider == "Backblaze": | ||
|  |                 EndURl = config['EndUrl'] | ||
|  |             elif provider == "Amazon": | ||
|  |                 EndURl = "https://s3.us-east-1.amazonaws.com" | ||
|  |             elif provider == "Wasabi": | ||
|  |                 EndURl = "https://s3.wasabisys.com" | ||
|  | 
 | ||
|  |             AccessKey = config['AccessKey'] | ||
|  |             SecertKey = config['SecertKey'] | ||
|  | 
 | ||
|  |             session = boto3.session.Session() | ||
|  | 
 | ||
|  |             client = session.client( | ||
|  |                 's3', | ||
|  |                 endpoint_url=EndURl, | ||
|  |                 aws_access_key_id=AccessKey, | ||
|  |                 aws_secret_access_key=SecertKey, | ||
|  |                 verify=False | ||
|  |             ) | ||
|  | 
 | ||
|  |             # ############Creating Bucket | ||
|  |             # BucketName = randomPassword.generate_pass().lower() | ||
|  |             # print("BucketName...%s"%BucketName) | ||
|  |             # | ||
|  |             # try: | ||
|  |             #     client.create_bucket(Bucket=BucketName) | ||
|  |             # except BaseException as msg: | ||
|  |             #     print("Error in Creating bucket...: %s" % str(msg)) | ||
|  |             #     logging.writeToFile("Create bucket error---%s:" % str(msg)) | ||
|  | 
 | ||
|  | 
 | ||
|  |             ####getting Bucket from backup schedule | ||
|  |             Scheduleobj = RemoteBackupSchedule.objects.get(pk=scheduleID) | ||
|  |             Scheduleconfig = json.loads(Scheduleobj.config) | ||
|  |             BucketName = Scheduleconfig['BucketName'] | ||
|  |             #####Uploading File | ||
|  | 
 | ||
|  |             uploadfilename = backupConfig['name'] | ||
|  |             print("uploadfilename....%s"%uploadfilename) | ||
|  | 
 | ||
|  |             try: | ||
|  |                 res = client.upload_file(Filename=FileName, Bucket=BucketName, Key=uploadfilename) | ||
|  |                 print("res of Uploading...: %s" % res) | ||
|  | 
 | ||
|  |             except BaseException as msg: | ||
|  |                 print("Error in Uploading...: %s" % msg) | ||
|  | 
 | ||
|  |             ###################### version id, this only applied to blackbaze | ||
|  |             try: | ||
|  | 
 | ||
|  |                 s3 = boto3.resource( | ||
|  |                     's3', | ||
|  |                     endpoint_url=EndURl, | ||
|  |                     aws_access_key_id=AccessKey, | ||
|  |                     aws_secret_access_key=SecertKey, | ||
|  |                 ) | ||
|  | 
 | ||
|  |                 bucket = BucketName | ||
|  |                 key = uploadfilename | ||
|  |                 versions = s3.Bucket(bucket).object_versions.filter(Prefix=key) | ||
|  |                 data = {} | ||
|  | 
 | ||
|  |                 for version in versions: | ||
|  |                     obj = version.get() | ||
|  |                     print("VersionId---%s:" % obj.get('VersionId')) | ||
|  |                     data['backupVersionId'] = obj.get('VersionId') | ||
|  | 
 | ||
|  |                 ab = os.path.getsize(FileName) | ||
|  |                 filesize = float(ab) / 1024.0 | ||
|  | 
 | ||
|  |                 backupConfig['uploadfilename'] = uploadfilename | ||
|  |                 backupConfig['backupVersionId'] = data['backupVersionId'] | ||
|  |                 backupConfig['BucketName'] = BucketName | ||
|  |                 backupConfig['Uplaodingfilesize'] = filesize | ||
|  |                 Backupobj.config = json.dumps(backupConfig) | ||
|  |                 Backupobj.save() | ||
|  | 
 | ||
|  |                 #S3 retention | ||
|  |                 #Needs a conversion table, because strings are stored instead of ints | ||
|  |                 retention_conversion = { | ||
|  |                     "3 Days" : 259200, | ||
|  |                     "1 Week" : 604800, | ||
|  |                     "3 Weeks" : 1814400, | ||
|  |                     "1 Month" : 2629743  | ||
|  |                 } | ||
|  |                 retentionSeconds = retention_conversion[Scheduleobj.fileretention] | ||
|  | 
 | ||
|  |                 bucket_obj = s3.Bucket(BucketName) | ||
|  |                 ts = time.time() | ||
|  |                 for file in bucket_obj.objects.all(): | ||
|  |                     result = float(ts - file.last_modified.timestamp()) | ||
|  |                     if result > retentionSeconds: | ||
|  |                         BackupLogs(owner=plan, level='INFO', timeStamp=time.strftime("%b %d %Y, %H:%M:%S"), | ||
|  |                                 msg='File %s expired and deleted according to your retention settings.' % ( | ||
|  |                                     file.key)).save() | ||
|  |                         file.delete() | ||
|  | 
 | ||
|  |             except BaseException as msg: | ||
|  |                 print("Version ID Error: %s"%str(msg)) | ||
|  |         except BaseException as msg: | ||
|  |             print('%s. [SendToS3Cloud]' % (str(msg))) | ||
|  |             logging.writeToFile('%s. [SendToS3Cloud]' % (str(msg))) | ||
|  | 
 | ||
|  |     @staticmethod | ||
|  |     def v2Backups(function): | ||
|  |         try: | ||
|  |             # print("....start remote backup...............") | ||
|  |             from websiteFunctions.models import Websites | ||
|  |             from loginSystem.models import Administrator | ||
|  |             import json | ||
|  |             import time | ||
|  |             if os.path.exists('/home/cyberpanel/v2backups'): | ||
|  |                 for website in Websites.objects.all(): | ||
|  |                     finalConfigPath = f'/home/cyberpanel/v2backups/{website.domain}' | ||
|  |                     if os.path.exists(finalConfigPath): | ||
|  | 
 | ||
|  |                         command = f'cat {finalConfigPath}' | ||
|  |                         RetResult = ProcessUtilities.outputExecutioner(command) | ||
|  |                         print(repr(RetResult)) | ||
|  |                         BackupConfig = json.loads(ProcessUtilities.outputExecutioner(command).rstrip('\n')) | ||
|  | 
 | ||
|  |                         for value in BackupConfig['schedules']: | ||
|  |                             try: | ||
|  | 
 | ||
|  |                                 if value['frequency'] == function: | ||
|  |                                     extra_args = {} | ||
|  |                                     extra_args['function'] = 'InitiateBackup' | ||
|  |                                     extra_args['website'] = website.domain | ||
|  |                                     extra_args['domain'] = website.domain | ||
|  |                                     extra_args['BasePath'] = '/home/backup' | ||
|  |                                     extra_args['BackendName'] = value['repo'] | ||
|  |                                     extra_args['BackupData'] = value['websiteData'] if 'websiteData' in value else False | ||
|  |                                     extra_args['BackupEmails'] = value['websiteEmails'] if 'websiteEmails' in value else False | ||
|  |                                     extra_args['BackupDatabase'] = value['websiteDatabases'] if 'websiteDatabases' in value else False | ||
|  | 
 | ||
|  |                                     from plogical.Backupsv2 import CPBackupsV2 | ||
|  |                                     background = CPBackupsV2(extra_args) | ||
|  |                                     RetStatus = background.InitiateBackup() | ||
|  | 
 | ||
|  |                                     print(RetStatus) | ||
|  | 
 | ||
|  |                                     if RetStatus == 0: | ||
|  |                                         SUBJECT = "Automatic Backupv2 failed for %s on %s." % (website.domain, time.strftime("%m.%d.%Y_%H-%M-%S")) | ||
|  |                                         adminEmailPath = '/home/cyberpanel/adminEmail' | ||
|  |                                         adminEmail = open(adminEmailPath, 'r').read().rstrip('\n') | ||
|  |                                         sender = 'root@%s' % (socket.gethostname()) | ||
|  |                                         error = ProcessUtilities.outputExecutioner(f'cat {background.StatusFile}') | ||
|  |                                         TO = [adminEmail] | ||
|  |                                         message = f"""\
 | ||
|  | From: %s | ||
|  | To: %s | ||
|  | Subject: %s | ||
|  | Automatic Backupv2 failed for %s on %s. | ||
|  | {error} | ||
|  | """ % (sender, ", ".join(TO), SUBJECT, website.domain, time.strftime("%m.%d.%Y_%H-%M-%S"))
 | ||
|  | 
 | ||
|  |                                         logging.SendEmail(sender, TO, message) | ||
|  |                                     else: | ||
|  |                                         value['lastRun'] = time.strftime("%m.%d.%Y_%H-%M-%S") | ||
|  | 
 | ||
|  |                                     if function == '1 Week': | ||
|  |                                         background.DeleteSnapshots(f"--keep-daily {value['retention']}") | ||
|  |                             except BaseException as msg: | ||
|  |                                 print("Error: [v2Backups]: %s" % str(msg)) | ||
|  |                                 logging.writeToFile('%s. [v2Backups]' % (str(msg))) | ||
|  | 
 | ||
|  |                         FinalContent = json.dumps(BackupConfig) | ||
|  |                         WriteToFile = open(finalConfigPath, 'w') | ||
|  |                         WriteToFile.write(FinalContent) | ||
|  |                         WriteToFile.close() | ||
|  | 
 | ||
|  |         except BaseException as msg: | ||
|  |             print("Error: [v2Backups]: %s" % str(msg)) | ||
|  |             logging.writeToFile('%s. [v2Backups]' % (str(msg))) | ||
|  | 
 | ||
|  |     @staticmethod | ||
|  |     def CheckHostName(): | ||
|  |         try: | ||
|  |             from loginSystem.models import Administrator | ||
|  |             admin = Administrator.objects.get(pk=1) | ||
|  |             try: | ||
|  |                 config = json.loads(admin.config) | ||
|  |             except: | ||
|  |                 config = {} | ||
|  | 
 | ||
|  |             ### probably need to add temporary dns resolver nameserver here - pending | ||
|  | 
 | ||
|  |             try: | ||
|  |                 CurrentHostName = config['hostname'] | ||
|  |                 skipRDNSCheck = config['skipRDNSCheck'] | ||
|  |             except: | ||
|  |                 CurrentHostName = mailUtilities.FetchPostfixHostname() | ||
|  |                 skipRDNSCheck = 1 | ||
|  | 
 | ||
|  |             virtualHostUtilities.OnBoardingHostName(CurrentHostName, '/home/cyberpanel/onboarding_temp_path', skipRDNSCheck) | ||
|  |         except BaseException as msg: | ||
|  |             logging.writeToFile(f'{str(msg)}. [Cron.CheckHostName]') | ||
|  | 
 | ||
|  | def main(): | ||
|  |     parser = argparse.ArgumentParser(description='CyberPanel Installer') | ||
|  |     parser.add_argument('function', help='Specific a function to call!') | ||
|  |     parser.add_argument('--planName', help='Plan name for AWS!') | ||
|  |     args = parser.parse_args() | ||
|  | 
 | ||
|  |     if args.function == 'UpdateDiskUsageForce': | ||
|  |         IncScheduler.CalculateAndUpdateDiskUsage() | ||
|  |         return 0 | ||
|  | 
 | ||
|  |     if args.function == '30 Minutes' or args.function == '1 Hour' or args.function == '6 Hours' or args.function == '12 Hours' or args.function == '1 Day' or args.function == '3 Days' or args.function == '1 Week': | ||
|  |         # IncScheduler.refresh_access_token() | ||
|  | 
 | ||
|  |         IncScheduler.RemoteBackup(args.function) | ||
|  |         IncScheduler.v2Backups(args.function) | ||
|  |         return 0 | ||
|  | 
 | ||
|  |     if args.function == 'forceRunAWSBackup': | ||
|  |         IncScheduler.forceRunAWSBackup(args.planName) | ||
|  |         return 0 | ||
|  | 
 | ||
|  |     IncScheduler.CalculateAndUpdateDiskUsage() | ||
|  |     IncScheduler.WPUpdates() | ||
|  | 
 | ||
|  |     if args.function == 'Weekly': | ||
|  |         try: | ||
|  |             IncScheduler.FixMailSSL() | ||
|  |         except: | ||
|  |             pass | ||
|  | 
 | ||
|  |     ### Run incremental backups in sep thread | ||
|  | 
 | ||
|  |     ib = IncScheduler('startBackup', {'freq': args.function}) | ||
|  |     ib.start() | ||
|  | 
 | ||
|  |     ### | ||
|  | 
 | ||
|  |     IncScheduler.startBackup(args.function) | ||
|  | 
 | ||
|  |     IncScheduler.runGoogleDriveBackups(args.function) | ||
|  |     IncScheduler.git(args.function) | ||
|  |     IncScheduler.checkDiskUsage() | ||
|  |     IncScheduler.startNormalBackups(args.function) | ||
|  |     IncScheduler.runAWSBackups(args.function) | ||
|  |     IncScheduler.CheckHostName() | ||
|  | 
 | ||
|  |     ib.join() | ||
|  | 
 | ||
|  | 
 | ||
|  | 
 | ||
|  | if __name__ == "__main__": | ||
|  |     main() |