mirror of
				https://github.com/usmannasir/cyberpanel.git
				synced 2025-10-26 15:56:34 +01:00 
			
		
		
		
	
		
			
	
	
		
			190 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
		
		
			
		
	
	
			190 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
|   | import os | ||
|  | import shutil | ||
|  | import pathlib | ||
|  | import stat | ||
|  | 
 | ||
|  | 
 | ||
|  | def mkdir_p(path, exist_ok=True): | ||
|  |     """
 | ||
|  |     Creates the directory and paths leading up to it like unix mkdir -p . | ||
|  |     Defaults to exist_ok so if it exists were not throwing fatal errors | ||
|  |     https://docs.python.org/3.7/library/os.html#os.makedirs | ||
|  |     """
 | ||
|  |     if not os.path.exists(path): | ||
|  |         print('creating directory: ' + path) | ||
|  |         os.makedirs(path, exist_ok) | ||
|  | 
 | ||
|  | 
 | ||
|  | def chmod_digit(file_path, perms): | ||
|  |     """
 | ||
|  |     Helper function to chmod like you would in unix without having to preface 0o or converting to octal yourself. | ||
|  |     Credits: https://stackoverflow.com/a/60052847/1621381 | ||
|  |     """
 | ||
|  |     try: | ||
|  |         os.chmod(file_path, int(str(perms), base=8)) | ||
|  |     except: | ||
|  |         print(f'Could not chmod : {file_path} to {perms}') | ||
|  |         pass | ||
|  | 
 | ||
|  | 
 | ||
|  | def touch(filepath: str, exist_ok=True): | ||
|  |     """
 | ||
|  |     Touches a file like unix `touch somefile` would. | ||
|  |     """
 | ||
|  |     try: | ||
|  |         pathlib.Path(filepath).touch(exist_ok) | ||
|  |     except FileExistsError: | ||
|  |         print('Could touch : ' + filepath) | ||
|  |         pass | ||
|  | 
 | ||
|  | 
 | ||
|  | def symlink(src, dst): | ||
|  |     """
 | ||
|  |     Symlink a path to another if the src exists. | ||
|  |     """
 | ||
|  |     try: | ||
|  |         if os.access(src, os.R_OK): | ||
|  |             os.symlink(src, dst) | ||
|  |     except: | ||
|  |         print(f'Could not symlink Source: {src} > Destination: {dst}') | ||
|  |         pass | ||
|  | 
 | ||
|  | 
 | ||
|  | def chown(path, user, group=-1): | ||
|  |     """
 | ||
|  |     Chown file/path to user/group provided. Passing -1 to user or group will leave it unchanged. | ||
|  |     Useful if just changing user or group vs both. | ||
|  |     """
 | ||
|  |     try: | ||
|  |         shutil.chown(path, user, group) | ||
|  |     except PermissionError: | ||
|  |         print(f'Could not change permissions for: {path} to {user}:{group}') | ||
|  |         pass | ||
|  | 
 | ||
|  | 
 | ||
|  | def recursive_chown(path, owner, group=-1): | ||
|  |     """
 | ||
|  |     Recursively chown a path and contents to owner. | ||
|  |     https://docs.python.org/3/library/shutil.html | ||
|  |     """
 | ||
|  |     for dirpath, dirnames, filenames in os.walk(path): | ||
|  |         try: | ||
|  |             shutil.chown(dirpath, owner, group) | ||
|  |         except PermissionError: | ||
|  |             print('Could not change permissions for: ' + dirpath + ' to: ' + owner) | ||
|  |             pass | ||
|  |         for filename in filenames: | ||
|  |             try: | ||
|  |                 shutil.chown(os.path.join(dirpath, filename), owner, group) | ||
|  |             except PermissionError: | ||
|  |                 print('Could not change permissions for: ' + os.path.join(dirpath, filename) + ' to: ' + owner) | ||
|  |                 pass | ||
|  | 
 | ||
|  | 
 | ||
|  | def recursive_permissions(path, dir_mode=755, file_mode=644, topdir=True): | ||
|  |     """
 | ||
|  |     Recursively chmod a path and contents to mode. | ||
|  |     Defaults to chmod top level directory but can be optionally | ||
|  |     toggled off when you want to chmod only contents of like a user's homedir vs homedir itself | ||
|  |     https://docs.python.org/3.6/library/os.html#os.walk | ||
|  |     """
 | ||
|  | 
 | ||
|  |     # Here we are converting the integers to string and then to octal. | ||
|  |     # so this function doesn't need to be called with 0o prefixed for the file and dir mode | ||
|  |     dir_mode = int(str(dir_mode), base=8) | ||
|  |     file_mode = int(str(file_mode), base=8) | ||
|  | 
 | ||
|  |     if topdir: | ||
|  |         # Set chmod on top level path | ||
|  |         try: | ||
|  |             os.chmod(path, dir_mode) | ||
|  |         except: | ||
|  |             print('Could not chmod :' + path + ' to ' + str(dir_mode)) | ||
|  |     for root, dirs, files in os.walk(path): | ||
|  |         for d in dirs: | ||
|  |             try: | ||
|  |                 os.chmod(os.path.join(root, d), dir_mode) | ||
|  |             except: | ||
|  |                 print('Could not chmod :' + os.path.join(root, d) + ' to ' + str(dir_mode)) | ||
|  |                 pass | ||
|  |         for f in files: | ||
|  |             try: | ||
|  |                 os.chmod(os.path.join(root, f), file_mode) | ||
|  |             except: | ||
|  |                 print('Could not chmod :' + path + ' to ' + str(file_mode)) | ||
|  |                 pass | ||
|  | 
 | ||
|  | 
 | ||
|  | # Left intentionally here for reference. | ||
|  | # Set recursive chown for a path | ||
|  | # recursive_chown(my_path, 'root', 'root') | ||
|  | # for changing group recursively without affecting user | ||
|  | # recursive_chown('/usr/local/lscp/cyberpanel/snappymail/data', -1, 'lscpd') | ||
|  | 
 | ||
|  | # explicitly set permissions for directories/folders to 0755 and files to 0644 | ||
|  | # recursive_permissions(my_path, 755, 644) | ||
|  | 
 | ||
|  | # Fix permissions and use default values | ||
|  | # recursive_permissions(my_path) | ||
|  | # ========================================================= | ||
|  | # Below is a helper class for getting and working with permissions | ||
|  | # Original credits to : https://github.com/keysemble/perfm | ||
|  | 
 | ||
|  | def perm_octal_digit(rwx): | ||
|  |     digit = 0 | ||
|  |     if rwx[0] == 'r': | ||
|  |         digit += 4 | ||
|  |     if rwx[1] == 'w': | ||
|  |         digit += 2 | ||
|  |     if rwx[2] == 'x': | ||
|  |         digit += 1 | ||
|  |     return digit | ||
|  | 
 | ||
|  | 
 | ||
|  | class FilePerm: | ||
|  |     def __init__(self, filepath): | ||
|  |         filemode = stat.filemode(os.stat(filepath).st_mode) | ||
|  |         permissions = [filemode[-9:][i:i + 3] for i in range(0, len(filemode[-9:]), 3)] | ||
|  |         self.filepath = filepath | ||
|  |         self.access_dict = dict(zip(['user', 'group', 'other'], [list(perm) for perm in permissions])) | ||
|  | 
 | ||
|  |     def mode(self): | ||
|  |         mode = 0 | ||
|  |         for shift, digit in enumerate(self.octal()[::-1]): | ||
|  |             mode += digit << (shift * 3) | ||
|  |         return mode | ||
|  | 
 | ||
|  |     def digits(self): | ||
|  |         """Get the octal chmod equivalent value 755 in single string""" | ||
|  |         return "".join(map(str, self.octal())) | ||
|  | 
 | ||
|  |     def octal(self): | ||
|  |         """Get the octal value in a list [7, 5, 5]""" | ||
|  |         return [perm_octal_digit(p) for p in self.access_dict.values()] | ||
|  | 
 | ||
|  |     def access_bits(self, access): | ||
|  |         if access in self.access_dict.keys(): | ||
|  |             r, w, x = self.access_dict[access] | ||
|  |             return [r == 'r', w == 'w', x == 'x'] | ||
|  | 
 | ||
|  |     def update_bitwise(self, settings): | ||
|  |         def perm_list(read=False, write=False, execute=False): | ||
|  |             pl = ['-', '-', '-'] | ||
|  |             if read: | ||
|  |                 pl[0] = 'r' | ||
|  |             if write: | ||
|  |                 pl[1] = 'w' | ||
|  |             if execute: | ||
|  |                 pl[2] = 'x' | ||
|  |             return pl | ||
|  | 
 | ||
|  |         self.access_dict = dict( | ||
|  |             [(access, perm_list(read=r, write=w, execute=x)) for access, [r, w, x] in settings.items()]) | ||
|  |         os.chmod(self.filepath, self.mode()) | ||
|  | 
 | ||
|  | # project_directory = os.path.abspath(os.path.dirname(sys.argv[0])) | ||
|  | # home_directory = os.path.expanduser('~') | ||
|  | # print(f'Path: {home_directory}  Mode: {FilePerm(home_directory).mode()}  Octal: {FilePerm(home_directory).octal()} ' | ||
|  | #      f'Digits: {FilePerm(home_directory).digits()}') | ||
|  | # Example: Output | ||
|  | # Path: /home/cooluser  Mode: 493  Octal: [7, 5, 5] Digits: 755 |