2
0

apps.py 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263
  1. __package__ = 'archivebox.plugins_pkg.pip'
  2. import os
  3. import sys
  4. import site
  5. from pathlib import Path
  6. from typing import List, Dict, Optional
  7. from pydantic import InstanceOf, Field, model_validator, validate_call
  8. import django
  9. import django.db.backends.sqlite3.base
  10. from django.db.backends.sqlite3.base import Database as django_sqlite3 # type: ignore[import-type]
  11. from pydantic_pkgr import BinProvider, PipProvider, BinName, BinProviderName, ProviderLookupDict, SemVer
  12. from archivebox.config import CONSTANTS, VERSION
  13. from abx.archivebox.base_plugin import BasePlugin
  14. from abx.archivebox.base_configset import BaseConfigSet
  15. from abx.archivebox.base_binary import BaseBinary, BaseBinProvider, env, apt, brew
  16. from abx.archivebox.base_hook import BaseHook
  17. from ...misc.logging import hint
  18. ###################### Config ##########################
  19. class PipDependencyConfigs(BaseConfigSet):
  20. USE_PIP: bool = True
  21. PIP_BINARY: str = Field(default='pip')
  22. PIP_ARGS: Optional[List[str]] = Field(default=None)
  23. PIP_EXTRA_ARGS: List[str] = []
  24. PIP_DEFAULT_ARGS: List[str] = []
  25. PIP_CONFIG = PipDependencyConfigs()
  26. class SystemPipBinProvider(PipProvider, BaseBinProvider):
  27. name: BinProviderName = "sys_pip"
  28. INSTALLER_BIN: BinName = "pip"
  29. pip_venv: Optional[Path] = None # global pip scope
  30. def on_install(self, bin_name: str, **kwargs):
  31. # never modify system pip packages
  32. return 'refusing to install packages globally with system pip, use a venv instead'
  33. class SystemPipxBinProvider(PipProvider, BaseBinProvider):
  34. name: BinProviderName = "pipx"
  35. INSTALLER_BIN: BinName = "pipx"
  36. pip_venv: Optional[Path] = None # global pipx scope
  37. IS_INSIDE_VENV = sys.prefix != sys.base_prefix
  38. class VenvPipBinProvider(PipProvider, BaseBinProvider):
  39. name: BinProviderName = "venv_pip"
  40. INSTALLER_BIN: BinName = "pip"
  41. pip_venv: Optional[Path] = Path(sys.prefix if IS_INSIDE_VENV else os.environ.get("VIRTUAL_ENV", '/tmp/NotInsideAVenv/lib'))
  42. def setup(self):
  43. """never attempt to create a venv here, this is just used to detect if we are inside an existing one"""
  44. return None
  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. # ensure python libraries are importable from these locations (if archivebox wasnt executed from one of these then they wont already be in sys.path)
  55. assert VENV_PIP_BINPROVIDER.pip_venv is not None
  56. assert LIB_PIP_BINPROVIDER.pip_venv is not None
  57. major, minor, patch = sys.version_info[:3]
  58. site_packages_dir = f'lib/python{major}.{minor}/site-packages'
  59. LIB_SITE_PACKAGES = (LIB_PIP_BINPROVIDER.pip_venv / site_packages_dir,)
  60. VENV_SITE_PACKAGES = (VENV_PIP_BINPROVIDER.pip_venv / site_packages_dir,)
  61. USER_SITE_PACKAGES = site.getusersitepackages()
  62. SYS_SITE_PACKAGES = site.getsitepackages()
  63. ALL_SITE_PACKAGES = (
  64. *LIB_SITE_PACKAGES,
  65. *VENV_SITE_PACKAGES,
  66. *USER_SITE_PACKAGES,
  67. *SYS_SITE_PACKAGES,
  68. )
  69. for site_packages_dir in ALL_SITE_PACKAGES:
  70. if site_packages_dir not in sys.path:
  71. sys.path.append(str(site_packages_dir))
  72. class ArchiveboxBinary(BaseBinary):
  73. name: BinName = 'archivebox'
  74. binproviders_supported: List[InstanceOf[BinProvider]] = [VENV_PIP_BINPROVIDER, SYS_PIP_BINPROVIDER, apt, brew, env]
  75. provider_overrides: Dict[BinProviderName, ProviderLookupDict] = {
  76. VENV_PIP_BINPROVIDER.name: {'packages': lambda: [], 'version': lambda: VERSION},
  77. SYS_PIP_BINPROVIDER.name: {'packages': lambda: [], 'version': lambda: VERSION},
  78. apt.name: {'packages': lambda: [], 'version': lambda: VERSION},
  79. brew.name: {'packages': lambda: [], 'version': lambda: VERSION},
  80. }
  81. @validate_call
  82. def install(self, **kwargs):
  83. return self.load() # obviously it's already installed if we are running this ;)
  84. @validate_call
  85. def load_or_install(self, **kwargs):
  86. return self.load() # obviously it's already installed if we are running this ;)
  87. ARCHIVEBOX_BINARY = ArchiveboxBinary()
  88. class PythonBinary(BaseBinary):
  89. name: BinName = 'python'
  90. binproviders_supported: List[InstanceOf[BinProvider]] = [VENV_PIP_BINPROVIDER, SYS_PIP_BINPROVIDER, apt, brew, env]
  91. provider_overrides: Dict[BinProviderName, ProviderLookupDict] = {
  92. SYS_PIP_BINPROVIDER.name: {
  93. 'abspath': lambda: sys.executable,
  94. 'version': lambda: '{}.{}.{}'.format(*sys.version_info[:3]),
  95. },
  96. }
  97. @validate_call
  98. def install(self, **kwargs):
  99. return self.load() # obviously it's already installed if we are running this ;)
  100. @validate_call
  101. def load_or_install(self, **kwargs):
  102. return self.load() # obviously it's already installed if we are running this ;)
  103. PYTHON_BINARY = PythonBinary()
  104. LOADED_SQLITE_PATH = Path(django.db.backends.sqlite3.base.__file__)
  105. LOADED_SQLITE_VERSION = SemVer(django_sqlite3.version)
  106. LOADED_SQLITE_FROM_VENV = str(LOADED_SQLITE_PATH.absolute().resolve()).startswith(str(VENV_PIP_BINPROVIDER.pip_venv.absolute().resolve()))
  107. class SqliteBinary(BaseBinary):
  108. name: BinName = 'sqlite'
  109. binproviders_supported: List[InstanceOf[BaseBinProvider]] = Field(default=[VENV_PIP_BINPROVIDER, SYS_PIP_BINPROVIDER])
  110. provider_overrides: Dict[BinProviderName, ProviderLookupDict] = {
  111. VENV_PIP_BINPROVIDER.name: {
  112. "abspath": lambda: LOADED_SQLITE_PATH if LOADED_SQLITE_FROM_VENV else None,
  113. "version": lambda: LOADED_SQLITE_VERSION if LOADED_SQLITE_FROM_VENV else None,
  114. },
  115. SYS_PIP_BINPROVIDER.name: {
  116. "abspath": lambda: LOADED_SQLITE_PATH if not LOADED_SQLITE_FROM_VENV else None,
  117. "version": lambda: LOADED_SQLITE_VERSION if not LOADED_SQLITE_FROM_VENV else None,
  118. },
  119. }
  120. @model_validator(mode='after')
  121. def validate_json_extension_is_available(self):
  122. # Check to make sure JSON extension is available in our Sqlite3 instance
  123. try:
  124. cursor = django_sqlite3.connect(':memory:').cursor()
  125. cursor.execute('SELECT JSON(\'{"a": "b"}\')')
  126. except django_sqlite3.OperationalError as exc:
  127. print(f'[red][X] Your SQLite3 version is missing the required JSON1 extension: {exc}[/red]')
  128. hint([
  129. 'Upgrade your Python version or install the extension manually:',
  130. 'https://code.djangoproject.com/wiki/JSON1Extension'
  131. ])
  132. return self
  133. @validate_call
  134. def install(self, **kwargs):
  135. return self.load() # obviously it's already installed if we are running this ;)
  136. @validate_call
  137. def load_or_install(self, **kwargs):
  138. return self.load() # obviously it's already installed if we are running this ;)
  139. SQLITE_BINARY = SqliteBinary()
  140. LOADED_DJANGO_PATH = Path(django.__file__)
  141. LOADED_DJANGO_VERSION = SemVer(django.VERSION[:3])
  142. LOADED_DJANGO_FROM_VENV = str(LOADED_DJANGO_PATH.absolute().resolve()).startswith(str(VENV_PIP_BINPROVIDER.pip_venv.absolute().resolve()))
  143. class DjangoBinary(BaseBinary):
  144. name: BinName = 'django'
  145. binproviders_supported: List[InstanceOf[BaseBinProvider]] = Field(default=[VENV_PIP_BINPROVIDER, SYS_PIP_BINPROVIDER])
  146. provider_overrides: Dict[BinProviderName, ProviderLookupDict] = {
  147. VENV_PIP_BINPROVIDER.name: {
  148. "abspath": lambda: LOADED_DJANGO_PATH if LOADED_DJANGO_FROM_VENV else None,
  149. "version": lambda: LOADED_DJANGO_VERSION if LOADED_DJANGO_FROM_VENV else None,
  150. },
  151. SYS_PIP_BINPROVIDER.name: {
  152. "abspath": lambda: LOADED_DJANGO_PATH if not LOADED_DJANGO_FROM_VENV else None,
  153. "version": lambda: LOADED_DJANGO_VERSION if not LOADED_DJANGO_FROM_VENV else None,
  154. },
  155. }
  156. @validate_call
  157. def install(self, **kwargs):
  158. return self.load() # obviously it's already installed if we are running this ;)
  159. @validate_call
  160. def load_or_install(self, **kwargs):
  161. return self.load() # obviously it's already installed if we are running this ;)
  162. DJANGO_BINARY = DjangoBinary()
  163. class PipBinary(BaseBinary):
  164. name: BinName = "pip"
  165. binproviders_supported: List[InstanceOf[BinProvider]] = [LIB_PIP_BINPROVIDER, VENV_PIP_BINPROVIDER, SYS_PIP_BINPROVIDER, apt, brew, env]
  166. @validate_call
  167. def install(self, **kwargs):
  168. return self.load() # obviously it's already installed if we are running this ;)
  169. @validate_call
  170. def load_or_install(self, **kwargs):
  171. return self.load() # obviously it's already installed if we are running this ;)
  172. PIP_BINARY = PipBinary()
  173. class PipxBinary(BaseBinary):
  174. name: BinName = "pipx"
  175. binproviders_supported: List[InstanceOf[BinProvider]] = [LIB_PIP_BINPROVIDER, VENV_PIP_BINPROVIDER, SYS_PIP_BINPROVIDER, apt, brew, env]
  176. PIPX_BINARY = PipxBinary()
  177. class PipPlugin(BasePlugin):
  178. app_label: str = 'pip'
  179. verbose_name: str = 'PIP'
  180. hooks: List[InstanceOf[BaseHook]] = [
  181. PIP_CONFIG,
  182. SYS_PIP_BINPROVIDER,
  183. PIPX_PIP_BINPROVIDER,
  184. VENV_PIP_BINPROVIDER,
  185. LIB_PIP_BINPROVIDER,
  186. PIP_BINARY,
  187. PIPX_BINARY,
  188. ARCHIVEBOX_BINARY,
  189. PYTHON_BINARY,
  190. SQLITE_BINARY,
  191. DJANGO_BINARY,
  192. ]
  193. PLUGIN = PipPlugin()
  194. # PLUGIN.register(settings)
  195. DJANGO_APP = PLUGIN.AppConfig