apps.py 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250
  1. __package__ = 'archivebox.plugins_pkg.pip'
  2. import os
  3. import sys
  4. import inspect
  5. from pathlib import Path
  6. from typing import List, Dict, Optional, ClassVar
  7. from pydantic import InstanceOf, Field, model_validator
  8. import django
  9. from django.db.backends.sqlite3.base import Database as django_sqlite3 # type: ignore[import-type]
  10. from django.core.checks import Error, Tags
  11. from pydantic_pkgr import BinProvider, PipProvider, BinName, BinProviderName, ProviderLookupDict, SemVer
  12. from archivebox.config import CONSTANTS, VERSION
  13. import abx
  14. from abx.archivebox.base_plugin import BasePlugin
  15. from abx.archivebox.base_configset import BaseConfigSet
  16. from abx.archivebox.base_check import BaseCheck
  17. from abx.archivebox.base_binary import BaseBinary, BaseBinProvider, env, apt, brew
  18. from abx.archivebox.base_hook import BaseHook
  19. from ...misc.logging import hint
  20. ###################### Config ##########################
  21. class PipDependencyConfigs(BaseConfigSet):
  22. USE_PIP: bool = True
  23. PIP_BINARY: str = Field(default='pip')
  24. PIP_ARGS: Optional[List[str]] = Field(default=None)
  25. PIP_EXTRA_ARGS: List[str] = []
  26. PIP_DEFAULT_ARGS: List[str] = []
  27. DEFAULT_GLOBAL_CONFIG = {
  28. }
  29. PIP_CONFIG = PipDependencyConfigs(**DEFAULT_GLOBAL_CONFIG)
  30. class SystemPipBinProvider(PipProvider, BaseBinProvider):
  31. name: BinProviderName = "sys_pip"
  32. INSTALLER_BIN: BinName = "pip"
  33. pip_venv: Optional[Path] = None # global pip scope
  34. def on_install(self, bin_name: str, **kwargs):
  35. # never modify system pip packages
  36. return 'refusing to install packages globally with system pip, use a venv instead'
  37. class SystemPipxBinProvider(PipProvider, BaseBinProvider):
  38. name: BinProviderName = "pipx"
  39. INSTALLER_BIN: BinName = "pipx"
  40. pip_venv: Optional[Path] = None # global pipx scope
  41. class VenvPipBinProvider(PipProvider, BaseBinProvider):
  42. name: BinProviderName = "venv_pip"
  43. INSTALLER_BIN: BinName = "pip"
  44. pip_venv: Optional[Path] = Path(os.environ.get("VIRTUAL_ENV", None) or '/tmp/NotInsideAVenv')
  45. class LibPipBinProvider(PipProvider, BaseBinProvider):
  46. name: BinProviderName = "lib_pip"
  47. INSTALLER_BIN: BinName = "pip"
  48. pip_venv: Optional[Path] = CONSTANTS.LIB_PIP_DIR / 'venv'
  49. SYS_PIP_BINPROVIDER = SystemPipBinProvider()
  50. PIPX_PIP_BINPROVIDER = SystemPipxBinProvider()
  51. VENV_PIP_BINPROVIDER = VenvPipBinProvider()
  52. LIB_PIP_BINPROVIDER = LibPipBinProvider()
  53. pip = LIB_PIP_BINPROVIDER
  54. class ArchiveboxBinary(BaseBinary):
  55. name: BinName = 'archivebox'
  56. binproviders_supported: List[InstanceOf[BinProvider]] = [VENV_PIP_BINPROVIDER, SYS_PIP_BINPROVIDER, apt, brew, env]
  57. provider_overrides: Dict[BinProviderName, ProviderLookupDict] = {
  58. VENV_PIP_BINPROVIDER.name: {'packages': lambda: [], 'version': lambda: VERSION},
  59. SYS_PIP_BINPROVIDER.name: {'packages': lambda: [], 'version': lambda: VERSION},
  60. apt.name: {'packages': lambda: [], 'version': lambda: VERSION},
  61. brew.name: {'packages': lambda: [], 'version': lambda: VERSION},
  62. }
  63. ARCHIVEBOX_BINARY = ArchiveboxBinary()
  64. class PythonBinary(BaseBinary):
  65. name: BinName = 'python'
  66. binproviders_supported: List[InstanceOf[BinProvider]] = [VENV_PIP_BINPROVIDER, SYS_PIP_BINPROVIDER, apt, brew, env]
  67. provider_overrides: Dict[BinProviderName, ProviderLookupDict] = {
  68. SYS_PIP_BINPROVIDER.name: {
  69. 'abspath': lambda:
  70. sys.executable,
  71. 'version': lambda:
  72. '{}.{}.{}'.format(*sys.version_info[:3]),
  73. },
  74. }
  75. PYTHON_BINARY = PythonBinary()
  76. class SqliteBinary(BaseBinary):
  77. name: BinName = 'sqlite'
  78. binproviders_supported: List[InstanceOf[BaseBinProvider]] = Field(default=[VENV_PIP_BINPROVIDER, SYS_PIP_BINPROVIDER])
  79. provider_overrides: Dict[BinProviderName, ProviderLookupDict] = {
  80. VENV_PIP_BINPROVIDER.name: {
  81. "abspath": lambda: Path(inspect.getfile(django_sqlite3)),
  82. "version": lambda: SemVer(django_sqlite3.version),
  83. },
  84. SYS_PIP_BINPROVIDER.name: {
  85. "abspath": lambda: Path(inspect.getfile(django_sqlite3)),
  86. "version": lambda: SemVer(django_sqlite3.version),
  87. },
  88. }
  89. @model_validator(mode='after')
  90. def validate_json_extension_is_available(self):
  91. # Check to make sure JSON extension is available in our Sqlite3 instance
  92. try:
  93. cursor = django_sqlite3.connect(':memory:').cursor()
  94. cursor.execute('SELECT JSON(\'{"a": "b"}\')')
  95. except django_sqlite3.OperationalError as exc:
  96. print(f'[red][X] Your SQLite3 version is missing the required JSON1 extension: {exc}[/red]')
  97. hint([
  98. 'Upgrade your Python version or install the extension manually:',
  99. 'https://code.djangoproject.com/wiki/JSON1Extension'
  100. ])
  101. return self
  102. SQLITE_BINARY = SqliteBinary()
  103. class DjangoBinary(BaseBinary):
  104. name: BinName = 'django'
  105. binproviders_supported: List[InstanceOf[BaseBinProvider]] = Field(default=[VENV_PIP_BINPROVIDER, SYS_PIP_BINPROVIDER])
  106. provider_overrides: Dict[BinProviderName, ProviderLookupDict] = {
  107. VENV_PIP_BINPROVIDER.name: {
  108. "abspath": lambda: inspect.getfile(django),
  109. "version": lambda: django.VERSION[:3],
  110. },
  111. SYS_PIP_BINPROVIDER.name: {
  112. "abspath": lambda: inspect.getfile(django),
  113. "version": lambda: django.VERSION[:3],
  114. },
  115. }
  116. DJANGO_BINARY = DjangoBinary()
  117. class PipBinary(BaseBinary):
  118. name: BinName = "pip"
  119. binproviders_supported: List[InstanceOf[BinProvider]] = [LIB_PIP_BINPROVIDER, VENV_PIP_BINPROVIDER, SYS_PIP_BINPROVIDER, apt, brew, env]
  120. PIP_BINARY = PipBinary()
  121. class CheckUserIsNotRoot(BaseCheck):
  122. label: str = 'CheckUserIsNotRoot'
  123. tag: str = Tags.database
  124. @staticmethod
  125. def check(settings, logger) -> List[Warning]:
  126. errors = []
  127. if getattr(settings, "USER", None) == 'root' or getattr(settings, "PUID", None) == 0:
  128. errors.append(
  129. Error(
  130. "Cannot run as root!",
  131. id="core.S001",
  132. hint=f'Run ArchiveBox as a non-root user with a UID greater than 500. (currently running as UID {os.getuid()}).',
  133. )
  134. )
  135. # logger.debug('[√] UID is not root')
  136. return errors
  137. class CheckPipEnvironment(BaseCheck):
  138. label: str = "CheckPipEnvironment"
  139. tag: str = Tags.database
  140. @staticmethod
  141. def check(settings, logger) -> List[Warning]:
  142. # hard errors: check python version
  143. if sys.version_info[:3] < (3, 10, 0):
  144. print('[red][X] Python version is not new enough: {sys.version} (>3.10 is required)[/red]', file=sys.stderr)
  145. print(' See https://github.com/ArchiveBox/ArchiveBox/wiki/Troubleshooting#python for help upgrading your Python installation.', file=sys.stderr)
  146. raise SystemExit(2)
  147. # hard errors: check django version
  148. if int(django.VERSION[0]) < 5:
  149. print('[red][X] Django version is not new enough: {django.VERSION[:3]} (>=5.0 is required)[/red]', file=sys.stderr)
  150. print(' Upgrade django using pip or your system package manager: pip3 install --upgrade django', file=sys.stderr)
  151. raise SystemExit(2)
  152. # soft errors: check that lib/pip virtualenv is setup properly
  153. errors = []
  154. LIB_PIP_BINPROVIDER.setup()
  155. if not LIB_PIP_BINPROVIDER.INSTALLER_BIN_ABSPATH:
  156. errors.append(
  157. Error(
  158. "Failed to setup data/lib/pip virtualenv for runtime dependencies!",
  159. id="pip.P001",
  160. hint="Make sure the data dir is writable and make sure python3-pip and python3-venv are installed & available on the host.",
  161. )
  162. )
  163. # logger.debug("[√] CheckPipEnvironment: data/lib/pip virtualenv is setup properly")
  164. return errors
  165. USER_IS_NOT_ROOT_CHECK = CheckUserIsNotRoot()
  166. PIP_ENVIRONMENT_CHECK = CheckPipEnvironment()
  167. class PipPlugin(BasePlugin):
  168. app_label: str = 'pip'
  169. verbose_name: str = 'PIP'
  170. hooks: List[InstanceOf[BaseHook]] = [
  171. PIP_CONFIG,
  172. SYS_PIP_BINPROVIDER,
  173. PIPX_PIP_BINPROVIDER,
  174. VENV_PIP_BINPROVIDER,
  175. LIB_PIP_BINPROVIDER,
  176. PIP_BINARY,
  177. ARCHIVEBOX_BINARY,
  178. PYTHON_BINARY,
  179. SQLITE_BINARY,
  180. DJANGO_BINARY,
  181. USER_IS_NOT_ROOT_CHECK,
  182. PIP_ENVIRONMENT_CHECK,
  183. ]
  184. PLUGIN = PipPlugin()
  185. # PLUGIN.register(settings)
  186. DJANGO_APP = PLUGIN.AppConfig
  187. @abx.hookimpl
  188. def register_django_checks(settings):
  189. USER_IS_NOT_ROOT_CHECK.register_with_django_check_system(settings)
  190. PIP_ENVIRONMENT_CHECK.register_with_django_check_system(settings)