Spaces:
Runtime error
Runtime error
import subprocess, time, threading | |
from typing import List, Union | |
import os, shutil, fnmatch | |
class Channel: | |
def __init__(self,source,destination,sync_deletions=False,every=60,exclude: Union[str, List, None] = None): | |
self.source = source | |
self.destination = destination | |
self.event = threading.Event() | |
self.syncing_thread = threading.Thread(target=self._sync,args=()) | |
self.sync_deletions = sync_deletions | |
self.every = every | |
if not exclude: | |
exclude = [] | |
if isinstance(exclude,str): | |
exclude = [exclude] | |
self.exclude = exclude | |
self.command = ['rsync','-aP'] | |
def alive(self):#Check if the thread is alive | |
if self.syncing_thread.is_alive(): | |
return True | |
else: | |
return False | |
def _sync(self):#Sync constantly | |
command = self.command | |
for exclusion in self.exclude: | |
command.append(f'--exclude={exclusion}') | |
command.extend([f'{self.source}/',f'{self.destination}/']) | |
if self.sync_deletions: | |
command.append('--delete') | |
while not self.event.is_set(): | |
subprocess.run(command) | |
time.sleep(self.every) | |
def copy(self):#Sync once | |
command = self.command | |
for exclusion in self.exclude: | |
command.append(f'--exclude={exclusion}') | |
command.extend([f'{self.source}/',f'{self.destination}/']) | |
if self.sync_deletions: | |
command.append('--delete') | |
subprocess.run(command) | |
return True | |
def start(self):#Handle threads | |
if self.syncing_thread.is_alive():#Check if it's running | |
self.event.set() | |
self.syncing_thread.join() | |
if self.event.is_set(): | |
self.event.clear() | |
if self.syncing_thread._started.is_set():#If it has been started before | |
self.syncing_thread = threading.Thread(target=self._sync,args=())#Create a FRESH thread | |
self.syncing_thread.start()#Start the thread | |
return self.alive() | |
def stop(self):#Stop the thread and close the process | |
if self.alive(): | |
self.event.set() | |
self.syncing_thread.join() | |
while self.alive(): | |
if not self.alive(): | |
break | |
return not self.alive() | |
class GarbageMan: | |
def __init__(self) -> None: | |
self.thread = threading.Thread(target=self.take_out,args=()) | |
self.event = threading.Event() | |
def destroy(self, trash): | |
if not isinstance(trash,dict): | |
if os.path.isdir(os.path.join(self.path,trash)): | |
shutil.rmtree(os.path.join(self.path,trash)) | |
elif os.path.isfile(os.path.join(self.path,trash)): | |
os.remove(os.path.join(self.path,trash)) | |
else: | |
trash.Delete() | |
def take_out(self) -> None: | |
while not self.event.is_set(): | |
for object in self.garbage: | |
trash = object["title"] if isinstance(object,dict) else object | |
if fnmatch.fnmatch(trash,self.pattern): | |
self.destroy(object) | |
time.sleep(self.every) | |
def stop(self) -> None: | |
if not self.event.is_set(): | |
self.event.set() | |
self.thread.join() | |
self.event.clear() | |
if self.thread._started.is_set(): | |
self.thread = threading.Thread(target=self.take_out,args=()) | |
def start(self,path: Union[str,List],every:int=30,pattern: str='') -> None: | |
if isinstance(path,list): | |
self.path = None | |
self.garbage = path | |
elif isinstance(path,str): | |
self.path = path | |
self.garbage = os.listdir(path) | |
else: | |
return "Error" | |
self.every = every | |
self.pattern = pattern | |
if self.thread.is_alive(): | |
self.stop() | |
self.thread.start() | |
def _fake(self, trash): | |
if not isinstance(trash,dict): | |
if os.path.isdir(os.path.join(self.path,trash)): | |
with open("log.txt","a") as f: | |
f.write(f"Fake deleted dir: {trash}") | |
elif os.path.isfile(os.path.join(self.path,trash)): | |
with open("log.txt","a") as f: | |
f.write(f"Fake deleted file: {trash}") | |
else: | |
with open("log.txt","a") as f: | |
f.write(f"Fake permanently deleted: {trash['title']}") |