registrar.py 4.0 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192
  1. import argparse
  2. import os
  3. import ctypes
  4. import pathlib
  5. import subprocess
  6. class RegistrarActionRequiresAdmin(PermissionError):
  7. ''' Used to denote that this action requires admin permissions '''
  8. pass
  9. class Registrar:
  10. ''' A collection of methods relating to registering and unregistering PyDeskband via regsvr32.exe '''
  11. @classmethod
  12. def is_64_bit(cls) -> bool:
  13. return '64' in subprocess.check_output([
  14. 'wmic', 'os', 'get', 'osarchitecture'
  15. ]).decode()
  16. @classmethod
  17. def is_admin(cls) -> bool:
  18. ''' Asks Windows if we are running as admin '''
  19. return bool(ctypes.windll.shell32.IsUserAnAdmin())
  20. @classmethod
  21. def get_dll_path(cls) -> pathlib.Path:
  22. ''' Returns the path to the PyDeskband dll for the OS architecture '''
  23. arch = '64' if cls.is_64_bit() else '86'
  24. dll_path = (pathlib.Path(__file__).parent / f"dlls/PyDeskband_x{arch}.dll").resolve()
  25. if not dll_path.is_file():
  26. raise FileNotFoundError(f"dll_path: {dll_path} is missing")
  27. return dll_path
  28. @classmethod
  29. def get_regsvr32_path(cls) -> pathlib.Path:
  30. ''' Returns the path to regsvr32.exe '''
  31. path = pathlib.Path(os.path.expandvars(r'%systemroot%\System32\regsvr32.exe'))
  32. if not path.is_file():
  33. raise FileNotFoundError(f"regsvr32.exe {path} is missing")
  34. return path
  35. @classmethod
  36. def register(cls) -> int:
  37. '''
  38. Attempts to register the PyDeskband DLL with the OS. Will return the exit code from that attempt. 0 typically means success.
  39. Requires admin privledges to run.
  40. Funny enough, on register, you may need to view right click and view the Toolbars list twice before the option PyDeskband option comes up.
  41. (This is even if you restart Windows Explorer). This is a known Windows behavior and not a bug with PyDeskband.
  42. '''
  43. if not cls.is_admin():
  44. raise RegistrarActionRequiresAdmin("Registering pyDeskband requires admin permissions!")
  45. return subprocess.call([cls.get_regsvr32_path(), cls.get_dll_path(), '/s'])
  46. @classmethod
  47. def unregister(cls) -> int:
  48. '''
  49. Attempts to unregister the PyDeskband DLL with the OS. Will return the exit code from that attempt. 0 typically means success.
  50. Requires admin privledges to run.
  51. '''
  52. if not cls.is_admin():
  53. raise RegistrarActionRequiresAdmin("Unregistering pyDeskband requires admin permissions!")
  54. return subprocess.call([cls.get_regsvr32_path(), '/u', cls.get_dll_path(), '/s'])
  55. @classmethod
  56. def restart_windows_explorer(cls) -> None:
  57. '''
  58. Uses the knowledge of us being on Windows to use a subprocess to restart Windows Explorer.
  59. Technically a Windows Explorer restart is not necessary though on an unregister, it will force the dll to be unloaded.
  60. '''
  61. subprocess.call('taskkill /F /IM explorer.exe && start explorer', shell=True, stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL)
  62. if __name__ == '__main__':
  63. parser = argparse.ArgumentParser(description="CLI to register/unregister PyDeskband with the registry")
  64. action = parser.add_mutually_exclusive_group()
  65. action.add_argument('-r', '--register', help='Registers the PyDeskband DLL with the OS.', action='store_true')
  66. action.add_argument('-u', '--unregister', help='Unregisters the PyDeskband DLL with the OS. Unless -x/--no-restart-explorer is given, Windows Explorer will restart after success.', action='store_true')
  67. parser.add_argument('-x', '--no-restart-explorer', help='If given, do not restart Windows Explorer after registering or unregistering.', action='store_true')
  68. args = parser.parse_args()
  69. ret_code = None
  70. if args.register:
  71. ret_code = Registrar.register()
  72. elif args.unregister:
  73. ret_code = Registrar.unregister()
  74. if ret_code is not None:
  75. if ret_code == 0 and not args.no_restart_explorer:
  76. Registrar.restart_windows_explorer()
  77. exit(ret_code)