quarchpy 2.1.14.dev3__py2.py3-none-any.whl → 2.1.14.dev5__py2.py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (701) hide show
  1. quarchpy/.idea/.name +1 -0
  2. quarchpy/.idea/workspace.xml +70 -11
  3. quarchpy/__pycache__/_version.cpython-311.pyc +0 -0
  4. quarchpy/__pycache__/run.cpython-311.pyc +0 -0
  5. quarchpy/_version.py +1 -1
  6. quarchpy/_version.py.bak +1 -0
  7. quarchpy/connection_specific/__pycache__/connection_QIS.cpython-311.pyc +0 -0
  8. quarchpy/connection_specific/connection_QIS.py +22 -9
  9. quarchpy/connection_specific/connection_QIS.py.bak +35 -17
  10. quarchpy/connection_specific/connection_ReST.py +26 -16
  11. quarchpy/debug/__pycache__/module_debug.cpython-311.pyc +0 -0
  12. quarchpy/debug/__pycache__/simple_terminal.cpython-311.pyc +0 -0
  13. quarchpy/debug/__pycache__/upgrade_quarchpy.cpython-311.pyc +0 -0
  14. quarchpy/device/__pycache__/scanDevices.cpython-311.pyc +0 -0
  15. quarchpy/device/scanDevices.py +2 -4
  16. quarchpy/device/scanDevices.py.bak +657 -0
  17. quarchpy/qis/qisFuncs.py +25 -42
  18. quarchpy/qis/qisFuncs.py.bak +213 -0
  19. quarchpy/qps/__pycache__/qpsFuncs.cpython-311.pyc +0 -0
  20. quarchpy/qps/qpsFuncs.py +18 -22
  21. quarchpy/qps/qpsFuncs.py.bak +224 -0
  22. quarchpy/venv/.gitignore +2 -0
  23. quarchpy/venv/Include/Python-ast.h +535 -0
  24. quarchpy/venv/Include/Python.h +181 -0
  25. quarchpy/venv/Include/abstract.h +1396 -0
  26. quarchpy/venv/Include/asdl.h +45 -0
  27. quarchpy/venv/Include/ast.h +13 -0
  28. quarchpy/venv/Include/bitset.h +32 -0
  29. quarchpy/venv/Include/boolobject.h +36 -0
  30. quarchpy/venv/Include/bufferobject.h +33 -0
  31. quarchpy/venv/Include/bytearrayobject.h +57 -0
  32. quarchpy/venv/Include/bytes_methods.h +75 -0
  33. quarchpy/venv/Include/bytesobject.h +27 -0
  34. quarchpy/venv/Include/cStringIO.h +73 -0
  35. quarchpy/venv/Include/cellobject.h +28 -0
  36. quarchpy/venv/Include/ceval.h +154 -0
  37. quarchpy/venv/Include/classobject.h +83 -0
  38. quarchpy/venv/Include/cobject.h +89 -0
  39. quarchpy/venv/Include/code.h +116 -0
  40. quarchpy/venv/Include/codecs.h +212 -0
  41. quarchpy/venv/Include/compile.h +40 -0
  42. quarchpy/venv/Include/complexobject.h +66 -0
  43. quarchpy/venv/Include/datetime.h +239 -0
  44. quarchpy/venv/Include/descrobject.h +94 -0
  45. quarchpy/venv/Include/dictobject.h +160 -0
  46. quarchpy/venv/Include/dtoa.h +15 -0
  47. quarchpy/venv/Include/enumobject.h +17 -0
  48. quarchpy/venv/Include/errcode.h +37 -0
  49. quarchpy/venv/Include/eval.h +25 -0
  50. quarchpy/venv/Include/fileobject.h +97 -0
  51. quarchpy/venv/Include/floatobject.h +140 -0
  52. quarchpy/venv/Include/frameobject.h +89 -0
  53. quarchpy/venv/Include/funcobject.h +76 -0
  54. quarchpy/venv/Include/genobject.h +40 -0
  55. quarchpy/venv/Include/graminit.h +87 -0
  56. quarchpy/venv/Include/grammar.h +94 -0
  57. quarchpy/venv/Include/import.h +71 -0
  58. quarchpy/venv/Include/intobject.h +84 -0
  59. quarchpy/venv/Include/intrcheck.h +15 -0
  60. quarchpy/venv/Include/iterobject.h +23 -0
  61. quarchpy/venv/Include/listobject.h +68 -0
  62. quarchpy/venv/Include/longintrepr.h +103 -0
  63. quarchpy/venv/Include/longobject.h +135 -0
  64. quarchpy/venv/Include/marshal.h +25 -0
  65. quarchpy/venv/Include/memoryobject.h +74 -0
  66. quarchpy/venv/Include/metagrammar.h +18 -0
  67. quarchpy/venv/Include/methodobject.h +93 -0
  68. quarchpy/venv/Include/modsupport.h +134 -0
  69. quarchpy/venv/Include/moduleobject.h +24 -0
  70. quarchpy/venv/Include/node.h +41 -0
  71. quarchpy/venv/Include/object.h +1046 -0
  72. quarchpy/venv/Include/objimpl.h +369 -0
  73. quarchpy/venv/Include/opcode.h +171 -0
  74. quarchpy/venv/Include/osdefs.h +63 -0
  75. quarchpy/venv/Include/parsetok.h +64 -0
  76. quarchpy/venv/Include/patchlevel.h +43 -0
  77. quarchpy/venv/Include/pgen.h +18 -0
  78. quarchpy/venv/Include/pgenheaders.h +43 -0
  79. quarchpy/venv/Include/py_curses.h +177 -0
  80. quarchpy/venv/Include/pyarena.h +62 -0
  81. quarchpy/venv/Include/pycapsule.h +56 -0
  82. quarchpy/venv/Include/pyconfig.h +770 -0
  83. quarchpy/venv/Include/pyctype.h +31 -0
  84. quarchpy/venv/Include/pydebug.h +41 -0
  85. quarchpy/venv/Include/pyerrors.h +329 -0
  86. quarchpy/venv/Include/pyexpat.h +50 -0
  87. quarchpy/venv/Include/pyfpe.h +176 -0
  88. quarchpy/venv/Include/pygetopt.h +18 -0
  89. quarchpy/venv/Include/pymacconfig.h +102 -0
  90. quarchpy/venv/Include/pymactoolbox.h +232 -0
  91. quarchpy/venv/Include/pymath.h +214 -0
  92. quarchpy/venv/Include/pymem.h +122 -0
  93. quarchpy/venv/Include/pyport.h +950 -0
  94. quarchpy/venv/Include/pystate.h +200 -0
  95. quarchpy/venv/Include/pystrcmp.h +23 -0
  96. quarchpy/venv/Include/pystrtod.h +45 -0
  97. quarchpy/venv/Include/pythonrun.h +181 -0
  98. quarchpy/venv/Include/pythread.h +41 -0
  99. quarchpy/venv/Include/rangeobject.h +28 -0
  100. quarchpy/venv/Include/setobject.h +99 -0
  101. quarchpy/venv/Include/sliceobject.h +50 -0
  102. quarchpy/venv/Include/stringobject.h +210 -0
  103. quarchpy/venv/Include/structmember.h +99 -0
  104. quarchpy/venv/Include/structseq.h +41 -0
  105. quarchpy/venv/Include/symtable.h +98 -0
  106. quarchpy/venv/Include/sysmodule.h +31 -0
  107. quarchpy/venv/Include/timefuncs.h +26 -0
  108. quarchpy/venv/Include/token.h +85 -0
  109. quarchpy/venv/Include/traceback.h +31 -0
  110. quarchpy/venv/Include/tupleobject.h +61 -0
  111. quarchpy/venv/Include/ucnhash.h +33 -0
  112. quarchpy/venv/Include/unicodeobject.h +1413 -0
  113. quarchpy/venv/Include/warnings.h +23 -0
  114. quarchpy/venv/Include/weakrefobject.h +82 -0
  115. quarchpy/venv/Lib/os.py +742 -0
  116. quarchpy/venv/Lib/os.pyc +0 -0
  117. quarchpy/venv/Lib/site-packages/_virtualenv.pth +1 -0
  118. quarchpy/venv/Lib/site-packages/_virtualenv.py +130 -0
  119. quarchpy/venv/Lib/site-packages/_virtualenv.pyc +0 -0
  120. quarchpy/venv/Lib/site-packages/easy_install.py +5 -0
  121. quarchpy/venv/Lib/site-packages/pip/__init__.py +18 -0
  122. quarchpy/venv/Lib/site-packages/pip/__main__.py +26 -0
  123. quarchpy/venv/Lib/site-packages/pip/_internal/__init__.py +17 -0
  124. quarchpy/venv/Lib/site-packages/pip/_internal/build_env.py +242 -0
  125. quarchpy/venv/Lib/site-packages/pip/_internal/cache.py +346 -0
  126. quarchpy/venv/Lib/site-packages/pip/_internal/cli/__init__.py +4 -0
  127. quarchpy/venv/Lib/site-packages/pip/_internal/cli/autocompletion.py +164 -0
  128. quarchpy/venv/Lib/site-packages/pip/_internal/cli/base_command.py +260 -0
  129. quarchpy/venv/Lib/site-packages/pip/_internal/cli/cmdoptions.py +971 -0
  130. quarchpy/venv/Lib/site-packages/pip/_internal/cli/command_context.py +36 -0
  131. quarchpy/venv/Lib/site-packages/pip/_internal/cli/main.py +75 -0
  132. quarchpy/venv/Lib/site-packages/pip/_internal/cli/main_parser.py +96 -0
  133. quarchpy/venv/Lib/site-packages/pip/_internal/cli/parser.py +285 -0
  134. quarchpy/venv/Lib/site-packages/pip/_internal/cli/progress_bars.py +280 -0
  135. quarchpy/venv/Lib/site-packages/pip/_internal/cli/req_command.py +436 -0
  136. quarchpy/venv/Lib/site-packages/pip/_internal/cli/spinners.py +173 -0
  137. quarchpy/venv/Lib/site-packages/pip/_internal/cli/status_codes.py +8 -0
  138. quarchpy/venv/Lib/site-packages/pip/_internal/commands/__init__.py +123 -0
  139. quarchpy/venv/Lib/site-packages/pip/_internal/commands/cache.py +234 -0
  140. quarchpy/venv/Lib/site-packages/pip/_internal/commands/check.py +51 -0
  141. quarchpy/venv/Lib/site-packages/pip/_internal/commands/completion.py +98 -0
  142. quarchpy/venv/Lib/site-packages/pip/_internal/commands/configuration.py +280 -0
  143. quarchpy/venv/Lib/site-packages/pip/_internal/commands/debug.py +230 -0
  144. quarchpy/venv/Lib/site-packages/pip/_internal/commands/download.py +143 -0
  145. quarchpy/venv/Lib/site-packages/pip/_internal/commands/freeze.py +116 -0
  146. quarchpy/venv/Lib/site-packages/pip/_internal/commands/hash.py +63 -0
  147. quarchpy/venv/Lib/site-packages/pip/_internal/commands/help.py +46 -0
  148. quarchpy/venv/Lib/site-packages/pip/_internal/commands/install.py +737 -0
  149. quarchpy/venv/Lib/site-packages/pip/_internal/commands/list.py +327 -0
  150. quarchpy/venv/Lib/site-packages/pip/_internal/commands/search.py +169 -0
  151. quarchpy/venv/Lib/site-packages/pip/_internal/commands/show.py +186 -0
  152. quarchpy/venv/Lib/site-packages/pip/_internal/commands/uninstall.py +95 -0
  153. quarchpy/venv/Lib/site-packages/pip/_internal/commands/wheel.py +198 -0
  154. quarchpy/venv/Lib/site-packages/pip/_internal/configuration.py +407 -0
  155. quarchpy/venv/Lib/site-packages/pip/_internal/distributions/__init__.py +24 -0
  156. quarchpy/venv/Lib/site-packages/pip/_internal/distributions/base.py +46 -0
  157. quarchpy/venv/Lib/site-packages/pip/_internal/distributions/installed.py +25 -0
  158. quarchpy/venv/Lib/site-packages/pip/_internal/distributions/sdist.py +105 -0
  159. quarchpy/venv/Lib/site-packages/pip/_internal/distributions/wheel.py +37 -0
  160. quarchpy/venv/Lib/site-packages/pip/_internal/exceptions.py +391 -0
  161. quarchpy/venv/Lib/site-packages/pip/_internal/index/__init__.py +2 -0
  162. quarchpy/venv/Lib/site-packages/pip/_internal/index/collector.py +667 -0
  163. quarchpy/venv/Lib/site-packages/pip/_internal/index/package_finder.py +1015 -0
  164. quarchpy/venv/Lib/site-packages/pip/_internal/locations.py +193 -0
  165. quarchpy/venv/Lib/site-packages/pip/_internal/main.py +16 -0
  166. quarchpy/venv/Lib/site-packages/pip/_internal/models/__init__.py +2 -0
  167. quarchpy/venv/Lib/site-packages/pip/_internal/models/candidate.py +39 -0
  168. quarchpy/venv/Lib/site-packages/pip/_internal/models/direct_url.py +243 -0
  169. quarchpy/venv/Lib/site-packages/pip/_internal/models/format_control.py +92 -0
  170. quarchpy/venv/Lib/site-packages/pip/_internal/models/index.py +34 -0
  171. quarchpy/venv/Lib/site-packages/pip/_internal/models/link.py +246 -0
  172. quarchpy/venv/Lib/site-packages/pip/_internal/models/scheme.py +31 -0
  173. quarchpy/venv/Lib/site-packages/pip/_internal/models/search_scope.py +135 -0
  174. quarchpy/venv/Lib/site-packages/pip/_internal/models/selection_prefs.py +50 -0
  175. quarchpy/venv/Lib/site-packages/pip/_internal/models/target_python.py +117 -0
  176. quarchpy/venv/Lib/site-packages/pip/_internal/models/wheel.py +78 -0
  177. quarchpy/venv/Lib/site-packages/pip/_internal/network/__init__.py +2 -0
  178. quarchpy/venv/Lib/site-packages/pip/_internal/network/auth.py +310 -0
  179. quarchpy/venv/Lib/site-packages/pip/_internal/network/cache.py +79 -0
  180. quarchpy/venv/Lib/site-packages/pip/_internal/network/download.py +202 -0
  181. quarchpy/venv/Lib/site-packages/pip/_internal/network/lazy_wheel.py +231 -0
  182. quarchpy/venv/Lib/site-packages/pip/_internal/network/session.py +428 -0
  183. quarchpy/venv/Lib/site-packages/pip/_internal/network/utils.py +97 -0
  184. quarchpy/venv/Lib/site-packages/pip/_internal/network/xmlrpc.py +53 -0
  185. quarchpy/venv/Lib/site-packages/pip/_internal/operations/__init__.py +0 -0
  186. quarchpy/venv/Lib/site-packages/pip/_internal/operations/build/__init__.py +0 -0
  187. quarchpy/venv/Lib/site-packages/pip/_internal/operations/build/metadata.py +38 -0
  188. quarchpy/venv/Lib/site-packages/pip/_internal/operations/build/metadata_legacy.py +77 -0
  189. quarchpy/venv/Lib/site-packages/pip/_internal/operations/build/wheel.py +47 -0
  190. quarchpy/venv/Lib/site-packages/pip/_internal/operations/build/wheel_legacy.py +113 -0
  191. quarchpy/venv/Lib/site-packages/pip/_internal/operations/check.py +155 -0
  192. quarchpy/venv/Lib/site-packages/pip/_internal/operations/freeze.py +277 -0
  193. quarchpy/venv/Lib/site-packages/pip/_internal/operations/install/__init__.py +2 -0
  194. quarchpy/venv/Lib/site-packages/pip/_internal/operations/install/editable_legacy.py +52 -0
  195. quarchpy/venv/Lib/site-packages/pip/_internal/operations/install/legacy.py +130 -0
  196. quarchpy/venv/Lib/site-packages/pip/_internal/operations/install/wheel.py +846 -0
  197. quarchpy/venv/Lib/site-packages/pip/_internal/operations/prepare.py +608 -0
  198. quarchpy/venv/Lib/site-packages/pip/_internal/pyproject.py +196 -0
  199. quarchpy/venv/Lib/site-packages/pip/_internal/req/__init__.py +103 -0
  200. quarchpy/venv/Lib/site-packages/pip/_internal/req/constructors.py +476 -0
  201. quarchpy/venv/Lib/site-packages/pip/_internal/req/req_file.py +574 -0
  202. quarchpy/venv/Lib/site-packages/pip/_internal/req/req_install.py +907 -0
  203. quarchpy/venv/Lib/site-packages/pip/_internal/req/req_set.py +204 -0
  204. quarchpy/venv/Lib/site-packages/pip/_internal/req/req_tracker.py +151 -0
  205. quarchpy/venv/Lib/site-packages/pip/_internal/req/req_uninstall.py +657 -0
  206. quarchpy/venv/Lib/site-packages/pip/_internal/resolution/__init__.py +0 -0
  207. quarchpy/venv/Lib/site-packages/pip/_internal/resolution/base.py +21 -0
  208. quarchpy/venv/Lib/site-packages/pip/_internal/resolution/legacy/__init__.py +0 -0
  209. quarchpy/venv/Lib/site-packages/pip/_internal/resolution/legacy/resolver.py +473 -0
  210. quarchpy/venv/Lib/site-packages/pip/_internal/resolution/resolvelib/__init__.py +0 -0
  211. quarchpy/venv/Lib/site-packages/pip/_internal/resolution/resolvelib/base.py +156 -0
  212. quarchpy/venv/Lib/site-packages/pip/_internal/resolution/resolvelib/candidates.py +604 -0
  213. quarchpy/venv/Lib/site-packages/pip/_internal/resolution/resolvelib/factory.py +504 -0
  214. quarchpy/venv/Lib/site-packages/pip/_internal/resolution/resolvelib/found_candidates.py +101 -0
  215. quarchpy/venv/Lib/site-packages/pip/_internal/resolution/resolvelib/provider.py +174 -0
  216. quarchpy/venv/Lib/site-packages/pip/_internal/resolution/resolvelib/reporter.py +84 -0
  217. quarchpy/venv/Lib/site-packages/pip/_internal/resolution/resolvelib/requirements.py +201 -0
  218. quarchpy/venv/Lib/site-packages/pip/_internal/resolution/resolvelib/resolver.py +296 -0
  219. quarchpy/venv/Lib/site-packages/pip/_internal/self_outdated_check.py +197 -0
  220. quarchpy/venv/Lib/site-packages/pip/_internal/utils/__init__.py +0 -0
  221. quarchpy/venv/Lib/site-packages/pip/_internal/utils/appdirs.py +44 -0
  222. quarchpy/venv/Lib/site-packages/pip/_internal/utils/compat.py +293 -0
  223. quarchpy/venv/Lib/site-packages/pip/_internal/utils/compatibility_tags.py +178 -0
  224. quarchpy/venv/Lib/site-packages/pip/_internal/utils/datetime.py +14 -0
  225. quarchpy/venv/Lib/site-packages/pip/_internal/utils/deprecation.py +104 -0
  226. quarchpy/venv/Lib/site-packages/pip/_internal/utils/direct_url_helpers.py +126 -0
  227. quarchpy/venv/Lib/site-packages/pip/_internal/utils/distutils_args.py +48 -0
  228. quarchpy/venv/Lib/site-packages/pip/_internal/utils/encoding.py +41 -0
  229. quarchpy/venv/Lib/site-packages/pip/_internal/utils/entrypoints.py +31 -0
  230. quarchpy/venv/Lib/site-packages/pip/_internal/utils/filesystem.py +224 -0
  231. quarchpy/venv/Lib/site-packages/pip/_internal/utils/filetypes.py +26 -0
  232. quarchpy/venv/Lib/site-packages/pip/_internal/utils/glibc.py +98 -0
  233. quarchpy/venv/Lib/site-packages/pip/_internal/utils/hashes.py +169 -0
  234. quarchpy/venv/Lib/site-packages/pip/_internal/utils/inject_securetransport.py +36 -0
  235. quarchpy/venv/Lib/site-packages/pip/_internal/utils/logging.py +399 -0
  236. quarchpy/venv/Lib/site-packages/pip/_internal/utils/misc.py +962 -0
  237. quarchpy/venv/Lib/site-packages/pip/_internal/utils/models.py +44 -0
  238. quarchpy/venv/Lib/site-packages/pip/_internal/utils/packaging.py +95 -0
  239. quarchpy/venv/Lib/site-packages/pip/_internal/utils/parallel.py +107 -0
  240. quarchpy/venv/Lib/site-packages/pip/_internal/utils/pkg_resources.py +44 -0
  241. quarchpy/venv/Lib/site-packages/pip/_internal/utils/setuptools_build.py +181 -0
  242. quarchpy/venv/Lib/site-packages/pip/_internal/utils/subprocess.py +299 -0
  243. quarchpy/venv/Lib/site-packages/pip/_internal/utils/temp_dir.py +284 -0
  244. quarchpy/venv/Lib/site-packages/pip/_internal/utils/typing.py +38 -0
  245. quarchpy/venv/Lib/site-packages/pip/_internal/utils/unpacking.py +281 -0
  246. quarchpy/venv/Lib/site-packages/pip/_internal/utils/urls.py +55 -0
  247. quarchpy/venv/Lib/site-packages/pip/_internal/utils/virtualenv.py +119 -0
  248. quarchpy/venv/Lib/site-packages/pip/_internal/utils/wheel.py +225 -0
  249. quarchpy/venv/Lib/site-packages/pip/_internal/vcs/__init__.py +15 -0
  250. quarchpy/venv/Lib/site-packages/pip/_internal/vcs/bazaar.py +123 -0
  251. quarchpy/venv/Lib/site-packages/pip/_internal/vcs/git.py +454 -0
  252. quarchpy/venv/Lib/site-packages/pip/_internal/vcs/mercurial.py +172 -0
  253. quarchpy/venv/Lib/site-packages/pip/_internal/vcs/subversion.py +340 -0
  254. quarchpy/venv/Lib/site-packages/pip/_internal/vcs/versioncontrol.py +735 -0
  255. quarchpy/venv/Lib/site-packages/pip/_internal/wheel_builder.py +362 -0
  256. quarchpy/venv/Lib/site-packages/pip/_vendor/__init__.py +114 -0
  257. quarchpy/venv/Lib/site-packages/pip/_vendor/appdirs.py +633 -0
  258. quarchpy/venv/Lib/site-packages/pip/_vendor/cachecontrol/__init__.py +11 -0
  259. quarchpy/venv/Lib/site-packages/pip/_vendor/cachecontrol/_cmd.py +57 -0
  260. quarchpy/venv/Lib/site-packages/pip/_vendor/cachecontrol/adapter.py +133 -0
  261. quarchpy/venv/Lib/site-packages/pip/_vendor/cachecontrol/cache.py +39 -0
  262. quarchpy/venv/Lib/site-packages/pip/_vendor/cachecontrol/caches/__init__.py +2 -0
  263. quarchpy/venv/Lib/site-packages/pip/_vendor/cachecontrol/caches/file_cache.py +146 -0
  264. quarchpy/venv/Lib/site-packages/pip/_vendor/cachecontrol/caches/redis_cache.py +33 -0
  265. quarchpy/venv/Lib/site-packages/pip/_vendor/cachecontrol/compat.py +29 -0
  266. quarchpy/venv/Lib/site-packages/pip/_vendor/cachecontrol/controller.py +376 -0
  267. quarchpy/venv/Lib/site-packages/pip/_vendor/cachecontrol/filewrapper.py +80 -0
  268. quarchpy/venv/Lib/site-packages/pip/_vendor/cachecontrol/heuristics.py +135 -0
  269. quarchpy/venv/Lib/site-packages/pip/_vendor/cachecontrol/serialize.py +188 -0
  270. quarchpy/venv/Lib/site-packages/pip/_vendor/cachecontrol/wrapper.py +29 -0
  271. quarchpy/venv/Lib/site-packages/pip/_vendor/certifi/__init__.py +3 -0
  272. quarchpy/venv/Lib/site-packages/pip/_vendor/certifi/__main__.py +12 -0
  273. quarchpy/venv/Lib/site-packages/pip/_vendor/certifi/cacert.pem +4606 -0
  274. quarchpy/venv/Lib/site-packages/pip/_vendor/certifi/core.py +60 -0
  275. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/__init__.py +39 -0
  276. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/big5freq.py +386 -0
  277. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/big5prober.py +47 -0
  278. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/chardistribution.py +233 -0
  279. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/charsetgroupprober.py +106 -0
  280. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/charsetprober.py +145 -0
  281. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/cli/__init__.py +1 -0
  282. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/cli/chardetect.py +85 -0
  283. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/codingstatemachine.py +88 -0
  284. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/compat.py +34 -0
  285. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/cp949prober.py +49 -0
  286. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/enums.py +76 -0
  287. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/escprober.py +101 -0
  288. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/escsm.py +246 -0
  289. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/eucjpprober.py +92 -0
  290. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/euckrfreq.py +195 -0
  291. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/euckrprober.py +47 -0
  292. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/euctwfreq.py +387 -0
  293. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/euctwprober.py +46 -0
  294. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/gb2312freq.py +283 -0
  295. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/gb2312prober.py +46 -0
  296. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/hebrewprober.py +292 -0
  297. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/jisfreq.py +325 -0
  298. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/jpcntx.py +233 -0
  299. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/langbulgarianmodel.py +228 -0
  300. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/langcyrillicmodel.py +333 -0
  301. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/langgreekmodel.py +225 -0
  302. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/langhebrewmodel.py +200 -0
  303. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/langhungarianmodel.py +225 -0
  304. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/langthaimodel.py +199 -0
  305. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/langturkishmodel.py +193 -0
  306. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/latin1prober.py +145 -0
  307. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/mbcharsetprober.py +91 -0
  308. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/mbcsgroupprober.py +54 -0
  309. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/mbcssm.py +572 -0
  310. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/sbcharsetprober.py +132 -0
  311. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/sbcsgroupprober.py +73 -0
  312. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/sjisprober.py +92 -0
  313. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/universaldetector.py +286 -0
  314. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/utf8prober.py +82 -0
  315. quarchpy/venv/Lib/site-packages/pip/_vendor/chardet/version.py +9 -0
  316. quarchpy/venv/Lib/site-packages/pip/_vendor/colorama/__init__.py +6 -0
  317. quarchpy/venv/Lib/site-packages/pip/_vendor/colorama/ansi.py +102 -0
  318. quarchpy/venv/Lib/site-packages/pip/_vendor/colorama/ansitowin32.py +258 -0
  319. quarchpy/venv/Lib/site-packages/pip/_vendor/colorama/initialise.py +80 -0
  320. quarchpy/venv/Lib/site-packages/pip/_vendor/colorama/win32.py +152 -0
  321. quarchpy/venv/Lib/site-packages/pip/_vendor/colorama/winterm.py +169 -0
  322. quarchpy/venv/Lib/site-packages/pip/_vendor/contextlib2.py +518 -0
  323. quarchpy/venv/Lib/site-packages/pip/_vendor/distlib/__init__.py +23 -0
  324. quarchpy/venv/Lib/site-packages/pip/_vendor/distlib/_backport/__init__.py +6 -0
  325. quarchpy/venv/Lib/site-packages/pip/_vendor/distlib/_backport/misc.py +41 -0
  326. quarchpy/venv/Lib/site-packages/pip/_vendor/distlib/_backport/shutil.py +764 -0
  327. quarchpy/venv/Lib/site-packages/pip/_vendor/distlib/_backport/sysconfig.cfg +84 -0
  328. quarchpy/venv/Lib/site-packages/pip/_vendor/distlib/_backport/sysconfig.py +786 -0
  329. quarchpy/venv/Lib/site-packages/pip/_vendor/distlib/_backport/tarfile.py +2607 -0
  330. quarchpy/venv/Lib/site-packages/pip/_vendor/distlib/compat.py +1120 -0
  331. quarchpy/venv/Lib/site-packages/pip/_vendor/distlib/database.py +1339 -0
  332. quarchpy/venv/Lib/site-packages/pip/_vendor/distlib/index.py +516 -0
  333. quarchpy/venv/Lib/site-packages/pip/_vendor/distlib/locators.py +1302 -0
  334. quarchpy/venv/Lib/site-packages/pip/_vendor/distlib/manifest.py +393 -0
  335. quarchpy/venv/Lib/site-packages/pip/_vendor/distlib/markers.py +131 -0
  336. quarchpy/venv/Lib/site-packages/pip/_vendor/distlib/metadata.py +1056 -0
  337. quarchpy/venv/Lib/site-packages/pip/_vendor/distlib/resources.py +355 -0
  338. quarchpy/venv/Lib/site-packages/pip/_vendor/distlib/scripts.py +419 -0
  339. quarchpy/venv/Lib/site-packages/pip/_vendor/distlib/t32.exe +0 -0
  340. quarchpy/venv/Lib/site-packages/pip/_vendor/distlib/t64.exe +0 -0
  341. quarchpy/venv/Lib/site-packages/pip/_vendor/distlib/util.py +1761 -0
  342. quarchpy/venv/Lib/site-packages/pip/_vendor/distlib/version.py +736 -0
  343. quarchpy/venv/Lib/site-packages/pip/_vendor/distlib/w32.exe +0 -0
  344. quarchpy/venv/Lib/site-packages/pip/_vendor/distlib/w64.exe +0 -0
  345. quarchpy/venv/Lib/site-packages/pip/_vendor/distlib/wheel.py +1018 -0
  346. quarchpy/venv/Lib/site-packages/pip/_vendor/distro.py +1230 -0
  347. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/__init__.py +35 -0
  348. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/_ihatexml.py +289 -0
  349. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/_inputstream.py +918 -0
  350. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/_tokenizer.py +1735 -0
  351. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/_trie/__init__.py +5 -0
  352. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/_trie/_base.py +40 -0
  353. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/_trie/py.py +67 -0
  354. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/_utils.py +159 -0
  355. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/constants.py +2946 -0
  356. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/filters/__init__.py +0 -0
  357. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/filters/alphabeticalattributes.py +29 -0
  358. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/filters/base.py +12 -0
  359. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/filters/inject_meta_charset.py +73 -0
  360. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/filters/lint.py +93 -0
  361. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/filters/optionaltags.py +207 -0
  362. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/filters/sanitizer.py +916 -0
  363. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/filters/whitespace.py +38 -0
  364. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/html5parser.py +2795 -0
  365. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/serializer.py +409 -0
  366. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/treeadapters/__init__.py +30 -0
  367. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/treeadapters/genshi.py +54 -0
  368. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/treeadapters/sax.py +50 -0
  369. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/treebuilders/__init__.py +88 -0
  370. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/treebuilders/base.py +417 -0
  371. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/treebuilders/dom.py +239 -0
  372. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/treebuilders/etree.py +343 -0
  373. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/treebuilders/etree_lxml.py +392 -0
  374. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/treewalkers/__init__.py +154 -0
  375. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/treewalkers/base.py +252 -0
  376. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/treewalkers/dom.py +43 -0
  377. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/treewalkers/etree.py +131 -0
  378. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/treewalkers/etree_lxml.py +215 -0
  379. quarchpy/venv/Lib/site-packages/pip/_vendor/html5lib/treewalkers/genshi.py +69 -0
  380. quarchpy/venv/Lib/site-packages/pip/_vendor/idna/__init__.py +2 -0
  381. quarchpy/venv/Lib/site-packages/pip/_vendor/idna/codec.py +118 -0
  382. quarchpy/venv/Lib/site-packages/pip/_vendor/idna/compat.py +12 -0
  383. quarchpy/venv/Lib/site-packages/pip/_vendor/idna/core.py +400 -0
  384. quarchpy/venv/Lib/site-packages/pip/_vendor/idna/idnadata.py +2050 -0
  385. quarchpy/venv/Lib/site-packages/pip/_vendor/idna/intranges.py +53 -0
  386. quarchpy/venv/Lib/site-packages/pip/_vendor/idna/package_data.py +2 -0
  387. quarchpy/venv/Lib/site-packages/pip/_vendor/idna/uts46data.py +8357 -0
  388. quarchpy/venv/Lib/site-packages/pip/_vendor/ipaddress.py +2420 -0
  389. quarchpy/venv/Lib/site-packages/pip/_vendor/msgpack/__init__.py +54 -0
  390. quarchpy/venv/Lib/site-packages/pip/_vendor/msgpack/_version.py +1 -0
  391. quarchpy/venv/Lib/site-packages/pip/_vendor/msgpack/exceptions.py +48 -0
  392. quarchpy/venv/Lib/site-packages/pip/_vendor/msgpack/ext.py +191 -0
  393. quarchpy/venv/Lib/site-packages/pip/_vendor/msgpack/fallback.py +1063 -0
  394. quarchpy/venv/Lib/site-packages/pip/_vendor/packaging/__about__.py +27 -0
  395. quarchpy/venv/Lib/site-packages/pip/_vendor/packaging/__init__.py +26 -0
  396. quarchpy/venv/Lib/site-packages/pip/_vendor/packaging/_compat.py +38 -0
  397. quarchpy/venv/Lib/site-packages/pip/_vendor/packaging/_structures.py +86 -0
  398. quarchpy/venv/Lib/site-packages/pip/_vendor/packaging/_typing.py +48 -0
  399. quarchpy/venv/Lib/site-packages/pip/_vendor/packaging/markers.py +328 -0
  400. quarchpy/venv/Lib/site-packages/pip/_vendor/packaging/requirements.py +151 -0
  401. quarchpy/venv/Lib/site-packages/pip/_vendor/packaging/specifiers.py +864 -0
  402. quarchpy/venv/Lib/site-packages/pip/_vendor/packaging/tags.py +852 -0
  403. quarchpy/venv/Lib/site-packages/pip/_vendor/packaging/utils.py +67 -0
  404. quarchpy/venv/Lib/site-packages/pip/_vendor/packaging/version.py +556 -0
  405. quarchpy/venv/Lib/site-packages/pip/_vendor/pep517/__init__.py +6 -0
  406. quarchpy/venv/Lib/site-packages/pip/_vendor/pep517/_in_process.py +280 -0
  407. quarchpy/venv/Lib/site-packages/pip/_vendor/pep517/build.py +124 -0
  408. quarchpy/venv/Lib/site-packages/pip/_vendor/pep517/check.py +203 -0
  409. quarchpy/venv/Lib/site-packages/pip/_vendor/pep517/colorlog.py +115 -0
  410. quarchpy/venv/Lib/site-packages/pip/_vendor/pep517/compat.py +34 -0
  411. quarchpy/venv/Lib/site-packages/pip/_vendor/pep517/dirtools.py +44 -0
  412. quarchpy/venv/Lib/site-packages/pip/_vendor/pep517/envbuild.py +167 -0
  413. quarchpy/venv/Lib/site-packages/pip/_vendor/pep517/meta.py +92 -0
  414. quarchpy/venv/Lib/site-packages/pip/_vendor/pep517/wrappers.py +327 -0
  415. quarchpy/venv/Lib/site-packages/pip/_vendor/pkg_resources/__init__.py +3296 -0
  416. quarchpy/venv/Lib/site-packages/pip/_vendor/pkg_resources/py31compat.py +23 -0
  417. quarchpy/venv/Lib/site-packages/pip/_vendor/progress/__init__.py +177 -0
  418. quarchpy/venv/Lib/site-packages/pip/_vendor/progress/bar.py +91 -0
  419. quarchpy/venv/Lib/site-packages/pip/_vendor/progress/counter.py +41 -0
  420. quarchpy/venv/Lib/site-packages/pip/_vendor/progress/spinner.py +43 -0
  421. quarchpy/venv/Lib/site-packages/pip/_vendor/pyparsing.py +7107 -0
  422. quarchpy/venv/Lib/site-packages/pip/_vendor/requests/__init__.py +144 -0
  423. quarchpy/venv/Lib/site-packages/pip/_vendor/requests/__version__.py +14 -0
  424. quarchpy/venv/Lib/site-packages/pip/_vendor/requests/_internal_utils.py +42 -0
  425. quarchpy/venv/Lib/site-packages/pip/_vendor/requests/adapters.py +533 -0
  426. quarchpy/venv/Lib/site-packages/pip/_vendor/requests/api.py +161 -0
  427. quarchpy/venv/Lib/site-packages/pip/_vendor/requests/auth.py +305 -0
  428. quarchpy/venv/Lib/site-packages/pip/_vendor/requests/certs.py +18 -0
  429. quarchpy/venv/Lib/site-packages/pip/_vendor/requests/compat.py +76 -0
  430. quarchpy/venv/Lib/site-packages/pip/_vendor/requests/cookies.py +549 -0
  431. quarchpy/venv/Lib/site-packages/pip/_vendor/requests/exceptions.py +123 -0
  432. quarchpy/venv/Lib/site-packages/pip/_vendor/requests/help.py +119 -0
  433. quarchpy/venv/Lib/site-packages/pip/_vendor/requests/hooks.py +34 -0
  434. quarchpy/venv/Lib/site-packages/pip/_vendor/requests/models.py +956 -0
  435. quarchpy/venv/Lib/site-packages/pip/_vendor/requests/packages.py +16 -0
  436. quarchpy/venv/Lib/site-packages/pip/_vendor/requests/sessions.py +781 -0
  437. quarchpy/venv/Lib/site-packages/pip/_vendor/requests/status_codes.py +123 -0
  438. quarchpy/venv/Lib/site-packages/pip/_vendor/requests/structures.py +105 -0
  439. quarchpy/venv/Lib/site-packages/pip/_vendor/requests/utils.py +988 -0
  440. quarchpy/venv/Lib/site-packages/pip/_vendor/resolvelib/__init__.py +26 -0
  441. quarchpy/venv/Lib/site-packages/pip/_vendor/resolvelib/compat/__init__.py +0 -0
  442. quarchpy/venv/Lib/site-packages/pip/_vendor/resolvelib/compat/collections_abc.py +6 -0
  443. quarchpy/venv/Lib/site-packages/pip/_vendor/resolvelib/providers.py +119 -0
  444. quarchpy/venv/Lib/site-packages/pip/_vendor/resolvelib/reporters.py +37 -0
  445. quarchpy/venv/Lib/site-packages/pip/_vendor/resolvelib/resolvers.py +454 -0
  446. quarchpy/venv/Lib/site-packages/pip/_vendor/resolvelib/structs.py +149 -0
  447. quarchpy/venv/Lib/site-packages/pip/_vendor/retrying.py +267 -0
  448. quarchpy/venv/Lib/site-packages/pip/_vendor/six.py +982 -0
  449. quarchpy/venv/Lib/site-packages/pip/_vendor/toml/__init__.py +25 -0
  450. quarchpy/venv/Lib/site-packages/pip/_vendor/toml/decoder.py +1057 -0
  451. quarchpy/venv/Lib/site-packages/pip/_vendor/toml/encoder.py +304 -0
  452. quarchpy/venv/Lib/site-packages/pip/_vendor/toml/ordered.py +15 -0
  453. quarchpy/venv/Lib/site-packages/pip/_vendor/toml/tz.py +24 -0
  454. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/__init__.py +85 -0
  455. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/_collections.py +337 -0
  456. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/_version.py +2 -0
  457. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/connection.py +535 -0
  458. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/connectionpool.py +1067 -0
  459. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/contrib/__init__.py +0 -0
  460. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/contrib/_appengine_environ.py +36 -0
  461. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/contrib/_securetransport/__init__.py +0 -0
  462. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/contrib/_securetransport/bindings.py +519 -0
  463. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/contrib/_securetransport/low_level.py +396 -0
  464. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/contrib/appengine.py +314 -0
  465. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/contrib/ntlmpool.py +121 -0
  466. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/contrib/pyopenssl.py +509 -0
  467. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/contrib/securetransport.py +920 -0
  468. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/contrib/socks.py +216 -0
  469. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/exceptions.py +313 -0
  470. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/fields.py +274 -0
  471. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/filepost.py +98 -0
  472. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/packages/__init__.py +5 -0
  473. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/packages/backports/__init__.py +0 -0
  474. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/packages/backports/makefile.py +51 -0
  475. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/packages/six.py +1021 -0
  476. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/packages/ssl_match_hostname/__init__.py +22 -0
  477. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/packages/ssl_match_hostname/_implementation.py +160 -0
  478. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/poolmanager.py +536 -0
  479. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/request.py +170 -0
  480. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/response.py +821 -0
  481. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/util/__init__.py +49 -0
  482. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/util/connection.py +150 -0
  483. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/util/proxy.py +56 -0
  484. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/util/queue.py +22 -0
  485. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/util/request.py +143 -0
  486. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/util/response.py +107 -0
  487. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/util/retry.py +601 -0
  488. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/util/ssl_.py +474 -0
  489. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/util/ssltransport.py +221 -0
  490. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/util/timeout.py +268 -0
  491. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/util/url.py +430 -0
  492. quarchpy/venv/Lib/site-packages/pip/_vendor/urllib3/util/wait.py +153 -0
  493. quarchpy/venv/Lib/site-packages/pip/_vendor/vendor.txt +24 -0
  494. quarchpy/venv/Lib/site-packages/pip/_vendor/webencodings/__init__.py +342 -0
  495. quarchpy/venv/Lib/site-packages/pip/_vendor/webencodings/labels.py +231 -0
  496. quarchpy/venv/Lib/site-packages/pip/_vendor/webencodings/mklabels.py +59 -0
  497. quarchpy/venv/Lib/site-packages/pip/_vendor/webencodings/tests.py +153 -0
  498. quarchpy/venv/Lib/site-packages/pip/_vendor/webencodings/x_user_defined.py +325 -0
  499. quarchpy/venv/Lib/site-packages/pip-20.3.4.dist-info/INSTALLER +1 -0
  500. quarchpy/venv/Lib/site-packages/pip-20.3.4.dist-info/LICENSE.txt +20 -0
  501. quarchpy/venv/Lib/site-packages/pip-20.3.4.dist-info/METADATA +94 -0
  502. quarchpy/venv/Lib/site-packages/pip-20.3.4.dist-info/RECORD +812 -0
  503. quarchpy/venv/Lib/site-packages/pip-20.3.4.dist-info/WHEEL +6 -0
  504. quarchpy/venv/Lib/site-packages/pip-20.3.4.dist-info/entry_points.txt +5 -0
  505. quarchpy/venv/Lib/site-packages/pip-20.3.4.dist-info/top_level.txt +1 -0
  506. quarchpy/venv/Lib/site-packages/pip-20.3.4.virtualenv +0 -0
  507. quarchpy/venv/Lib/site-packages/pkg_resources/__init__.py +3296 -0
  508. quarchpy/venv/Lib/site-packages/pkg_resources/__init__.pyc +0 -0
  509. quarchpy/venv/Lib/site-packages/pkg_resources/_vendor/__init__.py +0 -0
  510. quarchpy/venv/Lib/site-packages/pkg_resources/_vendor/__init__.pyc +0 -0
  511. quarchpy/venv/Lib/site-packages/pkg_resources/_vendor/appdirs.py +608 -0
  512. quarchpy/venv/Lib/site-packages/pkg_resources/_vendor/appdirs.pyc +0 -0
  513. quarchpy/venv/Lib/site-packages/pkg_resources/_vendor/packaging/__about__.py +21 -0
  514. quarchpy/venv/Lib/site-packages/pkg_resources/_vendor/packaging/__about__.pyc +0 -0
  515. quarchpy/venv/Lib/site-packages/pkg_resources/_vendor/packaging/__init__.py +14 -0
  516. quarchpy/venv/Lib/site-packages/pkg_resources/_vendor/packaging/__init__.pyc +0 -0
  517. quarchpy/venv/Lib/site-packages/pkg_resources/_vendor/packaging/_compat.py +30 -0
  518. quarchpy/venv/Lib/site-packages/pkg_resources/_vendor/packaging/_compat.pyc +0 -0
  519. quarchpy/venv/Lib/site-packages/pkg_resources/_vendor/packaging/_structures.py +68 -0
  520. quarchpy/venv/Lib/site-packages/pkg_resources/_vendor/packaging/_structures.pyc +0 -0
  521. quarchpy/venv/Lib/site-packages/pkg_resources/_vendor/packaging/markers.py +301 -0
  522. quarchpy/venv/Lib/site-packages/pkg_resources/_vendor/packaging/markers.pyc +0 -0
  523. quarchpy/venv/Lib/site-packages/pkg_resources/_vendor/packaging/requirements.py +127 -0
  524. quarchpy/venv/Lib/site-packages/pkg_resources/_vendor/packaging/requirements.pyc +0 -0
  525. quarchpy/venv/Lib/site-packages/pkg_resources/_vendor/packaging/specifiers.py +774 -0
  526. quarchpy/venv/Lib/site-packages/pkg_resources/_vendor/packaging/specifiers.pyc +0 -0
  527. quarchpy/venv/Lib/site-packages/pkg_resources/_vendor/packaging/utils.py +14 -0
  528. quarchpy/venv/Lib/site-packages/pkg_resources/_vendor/packaging/version.py +393 -0
  529. quarchpy/venv/Lib/site-packages/pkg_resources/_vendor/packaging/version.pyc +0 -0
  530. quarchpy/venv/Lib/site-packages/pkg_resources/_vendor/pyparsing.py +5742 -0
  531. quarchpy/venv/Lib/site-packages/pkg_resources/_vendor/pyparsing.pyc +0 -0
  532. quarchpy/venv/Lib/site-packages/pkg_resources/_vendor/six.py +868 -0
  533. quarchpy/venv/Lib/site-packages/pkg_resources/_vendor/six.pyc +0 -0
  534. quarchpy/venv/Lib/site-packages/pkg_resources/extern/__init__.py +73 -0
  535. quarchpy/venv/Lib/site-packages/pkg_resources/extern/__init__.pyc +0 -0
  536. quarchpy/venv/Lib/site-packages/pkg_resources/py31compat.py +23 -0
  537. quarchpy/venv/Lib/site-packages/pkg_resources/py31compat.pyc +0 -0
  538. quarchpy/venv/Lib/site-packages/setuptools/__init__.py +245 -0
  539. quarchpy/venv/Lib/site-packages/setuptools/_deprecation_warning.py +7 -0
  540. quarchpy/venv/Lib/site-packages/setuptools/_imp.py +73 -0
  541. quarchpy/venv/Lib/site-packages/setuptools/_vendor/__init__.py +0 -0
  542. quarchpy/venv/Lib/site-packages/setuptools/_vendor/ordered_set.py +488 -0
  543. quarchpy/venv/Lib/site-packages/setuptools/_vendor/packaging/__about__.py +27 -0
  544. quarchpy/venv/Lib/site-packages/setuptools/_vendor/packaging/__init__.py +26 -0
  545. quarchpy/venv/Lib/site-packages/setuptools/_vendor/packaging/_compat.py +31 -0
  546. quarchpy/venv/Lib/site-packages/setuptools/_vendor/packaging/_structures.py +68 -0
  547. quarchpy/venv/Lib/site-packages/setuptools/_vendor/packaging/markers.py +296 -0
  548. quarchpy/venv/Lib/site-packages/setuptools/_vendor/packaging/requirements.py +138 -0
  549. quarchpy/venv/Lib/site-packages/setuptools/_vendor/packaging/specifiers.py +749 -0
  550. quarchpy/venv/Lib/site-packages/setuptools/_vendor/packaging/tags.py +404 -0
  551. quarchpy/venv/Lib/site-packages/setuptools/_vendor/packaging/utils.py +57 -0
  552. quarchpy/venv/Lib/site-packages/setuptools/_vendor/packaging/version.py +420 -0
  553. quarchpy/venv/Lib/site-packages/setuptools/_vendor/pyparsing.py +5742 -0
  554. quarchpy/venv/Lib/site-packages/setuptools/_vendor/six.py +868 -0
  555. quarchpy/venv/Lib/site-packages/setuptools/archive_util.py +173 -0
  556. quarchpy/venv/Lib/site-packages/setuptools/build_meta.py +264 -0
  557. quarchpy/venv/Lib/site-packages/setuptools/cli-32.exe +0 -0
  558. quarchpy/venv/Lib/site-packages/setuptools/cli-64.exe +0 -0
  559. quarchpy/venv/Lib/site-packages/setuptools/cli.exe +0 -0
  560. quarchpy/venv/Lib/site-packages/setuptools/command/__init__.py +17 -0
  561. quarchpy/venv/Lib/site-packages/setuptools/command/alias.py +80 -0
  562. quarchpy/venv/Lib/site-packages/setuptools/command/bdist_egg.py +502 -0
  563. quarchpy/venv/Lib/site-packages/setuptools/command/bdist_rpm.py +43 -0
  564. quarchpy/venv/Lib/site-packages/setuptools/command/bdist_wininst.py +21 -0
  565. quarchpy/venv/Lib/site-packages/setuptools/command/build_clib.py +98 -0
  566. quarchpy/venv/Lib/site-packages/setuptools/command/build_ext.py +327 -0
  567. quarchpy/venv/Lib/site-packages/setuptools/command/build_py.py +270 -0
  568. quarchpy/venv/Lib/site-packages/setuptools/command/develop.py +221 -0
  569. quarchpy/venv/Lib/site-packages/setuptools/command/dist_info.py +36 -0
  570. quarchpy/venv/Lib/site-packages/setuptools/command/easy_install.py +2347 -0
  571. quarchpy/venv/Lib/site-packages/setuptools/command/egg_info.py +717 -0
  572. quarchpy/venv/Lib/site-packages/setuptools/command/install.py +125 -0
  573. quarchpy/venv/Lib/site-packages/setuptools/command/install_egg_info.py +62 -0
  574. quarchpy/venv/Lib/site-packages/setuptools/command/install_lib.py +121 -0
  575. quarchpy/venv/Lib/site-packages/setuptools/command/install_scripts.py +65 -0
  576. quarchpy/venv/Lib/site-packages/setuptools/command/launcher manifest.xml +15 -0
  577. quarchpy/venv/Lib/site-packages/setuptools/command/py36compat.py +136 -0
  578. quarchpy/venv/Lib/site-packages/setuptools/command/register.py +18 -0
  579. quarchpy/venv/Lib/site-packages/setuptools/command/rotate.py +66 -0
  580. quarchpy/venv/Lib/site-packages/setuptools/command/saveopts.py +22 -0
  581. quarchpy/venv/Lib/site-packages/setuptools/command/sdist.py +252 -0
  582. quarchpy/venv/Lib/site-packages/setuptools/command/setopt.py +149 -0
  583. quarchpy/venv/Lib/site-packages/setuptools/command/test.py +279 -0
  584. quarchpy/venv/Lib/site-packages/setuptools/command/upload.py +17 -0
  585. quarchpy/venv/Lib/site-packages/setuptools/command/upload_docs.py +206 -0
  586. quarchpy/venv/Lib/site-packages/setuptools/config.py +659 -0
  587. quarchpy/venv/Lib/site-packages/setuptools/dep_util.py +23 -0
  588. quarchpy/venv/Lib/site-packages/setuptools/depends.py +176 -0
  589. quarchpy/venv/Lib/site-packages/setuptools/dist.py +1274 -0
  590. quarchpy/venv/Lib/site-packages/setuptools/errors.py +16 -0
  591. quarchpy/venv/Lib/site-packages/setuptools/extension.py +57 -0
  592. quarchpy/venv/Lib/site-packages/setuptools/extern/__init__.py +73 -0
  593. quarchpy/venv/Lib/site-packages/setuptools/glob.py +174 -0
  594. quarchpy/venv/Lib/site-packages/setuptools/gui-32.exe +0 -0
  595. quarchpy/venv/Lib/site-packages/setuptools/gui-64.exe +0 -0
  596. quarchpy/venv/Lib/site-packages/setuptools/gui.exe +0 -0
  597. quarchpy/venv/Lib/site-packages/setuptools/installer.py +150 -0
  598. quarchpy/venv/Lib/site-packages/setuptools/launch.py +35 -0
  599. quarchpy/venv/Lib/site-packages/setuptools/lib2to3_ex.py +62 -0
  600. quarchpy/venv/Lib/site-packages/setuptools/monkey.py +179 -0
  601. quarchpy/venv/Lib/site-packages/setuptools/msvc.py +1679 -0
  602. quarchpy/venv/Lib/site-packages/setuptools/namespaces.py +107 -0
  603. quarchpy/venv/Lib/site-packages/setuptools/package_index.py +1136 -0
  604. quarchpy/venv/Lib/site-packages/setuptools/py27compat.py +60 -0
  605. quarchpy/venv/Lib/site-packages/setuptools/py31compat.py +32 -0
  606. quarchpy/venv/Lib/site-packages/setuptools/py33compat.py +59 -0
  607. quarchpy/venv/Lib/site-packages/setuptools/py34compat.py +13 -0
  608. quarchpy/venv/Lib/site-packages/setuptools/sandbox.py +491 -0
  609. quarchpy/venv/Lib/site-packages/setuptools/script (dev).tmpl +6 -0
  610. quarchpy/venv/Lib/site-packages/setuptools/script.tmpl +3 -0
  611. quarchpy/venv/Lib/site-packages/setuptools/site-patch.py +74 -0
  612. quarchpy/venv/Lib/site-packages/setuptools/ssl_support.py +260 -0
  613. quarchpy/venv/Lib/site-packages/setuptools/unicode_utils.py +44 -0
  614. quarchpy/venv/Lib/site-packages/setuptools/version.py +6 -0
  615. quarchpy/venv/Lib/site-packages/setuptools/wheel.py +220 -0
  616. quarchpy/venv/Lib/site-packages/setuptools/windows_support.py +29 -0
  617. quarchpy/venv/Lib/site-packages/setuptools-44.1.1.dist-info/INSTALLER +1 -0
  618. quarchpy/venv/Lib/site-packages/setuptools-44.1.1.dist-info/LICENSE +19 -0
  619. quarchpy/venv/Lib/site-packages/setuptools-44.1.1.dist-info/METADATA +86 -0
  620. quarchpy/venv/Lib/site-packages/setuptools-44.1.1.dist-info/RECORD +207 -0
  621. quarchpy/venv/Lib/site-packages/setuptools-44.1.1.dist-info/WHEEL +6 -0
  622. quarchpy/venv/Lib/site-packages/setuptools-44.1.1.dist-info/dependency_links.txt +2 -0
  623. quarchpy/venv/Lib/site-packages/setuptools-44.1.1.dist-info/entry_points.txt +69 -0
  624. quarchpy/venv/Lib/site-packages/setuptools-44.1.1.dist-info/top_level.txt +3 -0
  625. quarchpy/venv/Lib/site-packages/setuptools-44.1.1.dist-info/zip-safe +1 -0
  626. quarchpy/venv/Lib/site-packages/setuptools-44.1.1.virtualenv +0 -0
  627. quarchpy/venv/Lib/site-packages/wheel/__init__.py +1 -0
  628. quarchpy/venv/Lib/site-packages/wheel/__main__.py +19 -0
  629. quarchpy/venv/Lib/site-packages/wheel/bdist_wheel.py +492 -0
  630. quarchpy/venv/Lib/site-packages/wheel/cli/__init__.py +88 -0
  631. quarchpy/venv/Lib/site-packages/wheel/cli/convert.py +269 -0
  632. quarchpy/venv/Lib/site-packages/wheel/cli/pack.py +82 -0
  633. quarchpy/venv/Lib/site-packages/wheel/cli/unpack.py +25 -0
  634. quarchpy/venv/Lib/site-packages/wheel/macosx_libfile.py +428 -0
  635. quarchpy/venv/Lib/site-packages/wheel/metadata.py +133 -0
  636. quarchpy/venv/Lib/site-packages/wheel/pkginfo.py +43 -0
  637. quarchpy/venv/Lib/site-packages/wheel/util.py +46 -0
  638. quarchpy/venv/Lib/site-packages/wheel/vendored/__init__.py +0 -0
  639. quarchpy/venv/Lib/site-packages/wheel/vendored/packaging/__init__.py +0 -0
  640. quarchpy/venv/Lib/site-packages/wheel/vendored/packaging/_typing.py +48 -0
  641. quarchpy/venv/Lib/site-packages/wheel/vendored/packaging/tags.py +866 -0
  642. quarchpy/venv/Lib/site-packages/wheel/wheelfile.py +181 -0
  643. quarchpy/venv/Lib/site-packages/wheel-0.37.1.dist-info/INSTALLER +1 -0
  644. quarchpy/venv/Lib/site-packages/wheel-0.37.1.dist-info/LICENSE.txt +22 -0
  645. quarchpy/venv/Lib/site-packages/wheel-0.37.1.dist-info/METADATA +69 -0
  646. quarchpy/venv/Lib/site-packages/wheel-0.37.1.dist-info/RECORD +49 -0
  647. quarchpy/venv/Lib/site-packages/wheel-0.37.1.dist-info/WHEEL +6 -0
  648. quarchpy/venv/Lib/site-packages/wheel-0.37.1.dist-info/entry_points.txt +6 -0
  649. quarchpy/venv/Lib/site-packages/wheel-0.37.1.dist-info/top_level.txt +1 -0
  650. quarchpy/venv/Lib/site-packages/wheel-0.37.1.virtualenv +0 -0
  651. quarchpy/venv/Lib/site.py +190 -0
  652. quarchpy/venv/Lib/site.pyc +0 -0
  653. quarchpy/venv/Scripts/activate +83 -0
  654. quarchpy/venv/Scripts/activate.bat +39 -0
  655. quarchpy/venv/Scripts/activate.fish +100 -0
  656. quarchpy/venv/Scripts/activate.nu +41 -0
  657. quarchpy/venv/Scripts/activate.ps1 +60 -0
  658. quarchpy/venv/Scripts/activate_this.py +32 -0
  659. quarchpy/venv/Scripts/deactivate.bat +19 -0
  660. quarchpy/venv/Scripts/deactivate.nu +11 -0
  661. quarchpy/venv/Scripts/easy_install-2.7.exe +0 -0
  662. quarchpy/venv/Scripts/easy_install.exe +0 -0
  663. quarchpy/venv/Scripts/easy_install2.7.exe +0 -0
  664. quarchpy/venv/Scripts/easy_install2.exe +0 -0
  665. quarchpy/venv/Scripts/pip-2.7.exe +0 -0
  666. quarchpy/venv/Scripts/pip.exe +0 -0
  667. quarchpy/venv/Scripts/pip2.7.exe +0 -0
  668. quarchpy/venv/Scripts/pip2.exe +0 -0
  669. quarchpy/venv/Scripts/pydoc.bat +1 -0
  670. quarchpy/venv/Scripts/python.exe +0 -0
  671. quarchpy/venv/Scripts/python27.dll +0 -0
  672. quarchpy/venv/Scripts/pythonw.exe +0 -0
  673. quarchpy/venv/Scripts/wheel-2.7.exe +0 -0
  674. quarchpy/venv/Scripts/wheel.exe +0 -0
  675. quarchpy/venv/Scripts/wheel2.7.exe +0 -0
  676. quarchpy/venv/Scripts/wheel2.exe +0 -0
  677. quarchpy/venv/libs/_bsddb.lib +0 -0
  678. quarchpy/venv/libs/_ctypes.lib +0 -0
  679. quarchpy/venv/libs/_ctypes_test.lib +0 -0
  680. quarchpy/venv/libs/_elementtree.lib +0 -0
  681. quarchpy/venv/libs/_hashlib.lib +0 -0
  682. quarchpy/venv/libs/_msi.lib +0 -0
  683. quarchpy/venv/libs/_multiprocessing.lib +0 -0
  684. quarchpy/venv/libs/_socket.lib +0 -0
  685. quarchpy/venv/libs/_sqlite3.lib +0 -0
  686. quarchpy/venv/libs/_ssl.lib +0 -0
  687. quarchpy/venv/libs/_testcapi.lib +0 -0
  688. quarchpy/venv/libs/_tkinter.lib +0 -0
  689. quarchpy/venv/libs/bz2.lib +0 -0
  690. quarchpy/venv/libs/libpython27.a +0 -0
  691. quarchpy/venv/libs/pyexpat.lib +0 -0
  692. quarchpy/venv/libs/python27.lib +0 -0
  693. quarchpy/venv/libs/select.lib +0 -0
  694. quarchpy/venv/libs/unicodedata.lib +0 -0
  695. quarchpy/venv/libs/winsound.lib +0 -0
  696. quarchpy/venv/pyvenv.cfg +8 -0
  697. {quarchpy-2.1.14.dev3.dist-info → quarchpy-2.1.14.dev5.dist-info}/METADATA +1 -1
  698. quarchpy-2.1.14.dev5.dist-info/RECORD +1219 -0
  699. quarchpy-2.1.14.dev3.dist-info/RECORD +0 -535
  700. {quarchpy-2.1.14.dev3.dist-info → quarchpy-2.1.14.dev5.dist-info}/WHEEL +0 -0
  701. {quarchpy-2.1.14.dev3.dist-info → quarchpy-2.1.14.dev5.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,1339 @@
1
+ # -*- coding: utf-8 -*-
2
+ #
3
+ # Copyright (C) 2012-2017 The Python Software Foundation.
4
+ # See LICENSE.txt and CONTRIBUTORS.txt.
5
+ #
6
+ """PEP 376 implementation."""
7
+
8
+ from __future__ import unicode_literals
9
+
10
+ import base64
11
+ import codecs
12
+ import contextlib
13
+ import hashlib
14
+ import logging
15
+ import os
16
+ import posixpath
17
+ import sys
18
+ import zipimport
19
+
20
+ from . import DistlibException, resources
21
+ from .compat import StringIO
22
+ from .version import get_scheme, UnsupportedVersionError
23
+ from .metadata import (Metadata, METADATA_FILENAME, WHEEL_METADATA_FILENAME,
24
+ LEGACY_METADATA_FILENAME)
25
+ from .util import (parse_requirement, cached_property, parse_name_and_version,
26
+ read_exports, write_exports, CSVReader, CSVWriter)
27
+
28
+
29
+ __all__ = ['Distribution', 'BaseInstalledDistribution',
30
+ 'InstalledDistribution', 'EggInfoDistribution',
31
+ 'DistributionPath']
32
+
33
+
34
+ logger = logging.getLogger(__name__)
35
+
36
+ EXPORTS_FILENAME = 'pydist-exports.json'
37
+ COMMANDS_FILENAME = 'pydist-commands.json'
38
+
39
+ DIST_FILES = ('INSTALLER', METADATA_FILENAME, 'RECORD', 'REQUESTED',
40
+ 'RESOURCES', EXPORTS_FILENAME, 'SHARED')
41
+
42
+ DISTINFO_EXT = '.dist-info'
43
+
44
+
45
+ class _Cache(object):
46
+ """
47
+ A simple cache mapping names and .dist-info paths to distributions
48
+ """
49
+ def __init__(self):
50
+ """
51
+ Initialise an instance. There is normally one for each DistributionPath.
52
+ """
53
+ self.name = {}
54
+ self.path = {}
55
+ self.generated = False
56
+
57
+ def clear(self):
58
+ """
59
+ Clear the cache, setting it to its initial state.
60
+ """
61
+ self.name.clear()
62
+ self.path.clear()
63
+ self.generated = False
64
+
65
+ def add(self, dist):
66
+ """
67
+ Add a distribution to the cache.
68
+ :param dist: The distribution to add.
69
+ """
70
+ if dist.path not in self.path:
71
+ self.path[dist.path] = dist
72
+ self.name.setdefault(dist.key, []).append(dist)
73
+
74
+
75
+ class DistributionPath(object):
76
+ """
77
+ Represents a set of distributions installed on a path (typically sys.path).
78
+ """
79
+ def __init__(self, path=None, include_egg=False):
80
+ """
81
+ Create an instance from a path, optionally including legacy (distutils/
82
+ setuptools/distribute) distributions.
83
+ :param path: The path to use, as a list of directories. If not specified,
84
+ sys.path is used.
85
+ :param include_egg: If True, this instance will look for and return legacy
86
+ distributions as well as those based on PEP 376.
87
+ """
88
+ if path is None:
89
+ path = sys.path
90
+ self.path = path
91
+ self._include_dist = True
92
+ self._include_egg = include_egg
93
+
94
+ self._cache = _Cache()
95
+ self._cache_egg = _Cache()
96
+ self._cache_enabled = True
97
+ self._scheme = get_scheme('default')
98
+
99
+ def _get_cache_enabled(self):
100
+ return self._cache_enabled
101
+
102
+ def _set_cache_enabled(self, value):
103
+ self._cache_enabled = value
104
+
105
+ cache_enabled = property(_get_cache_enabled, _set_cache_enabled)
106
+
107
+ def clear_cache(self):
108
+ """
109
+ Clears the internal cache.
110
+ """
111
+ self._cache.clear()
112
+ self._cache_egg.clear()
113
+
114
+
115
+ def _yield_distributions(self):
116
+ """
117
+ Yield .dist-info and/or .egg(-info) distributions.
118
+ """
119
+ # We need to check if we've seen some resources already, because on
120
+ # some Linux systems (e.g. some Debian/Ubuntu variants) there are
121
+ # symlinks which alias other files in the environment.
122
+ seen = set()
123
+ for path in self.path:
124
+ finder = resources.finder_for_path(path)
125
+ if finder is None:
126
+ continue
127
+ r = finder.find('')
128
+ if not r or not r.is_container:
129
+ continue
130
+ rset = sorted(r.resources)
131
+ for entry in rset:
132
+ r = finder.find(entry)
133
+ if not r or r.path in seen:
134
+ continue
135
+ if self._include_dist and entry.endswith(DISTINFO_EXT):
136
+ possible_filenames = [METADATA_FILENAME,
137
+ WHEEL_METADATA_FILENAME,
138
+ LEGACY_METADATA_FILENAME]
139
+ for metadata_filename in possible_filenames:
140
+ metadata_path = posixpath.join(entry, metadata_filename)
141
+ pydist = finder.find(metadata_path)
142
+ if pydist:
143
+ break
144
+ else:
145
+ continue
146
+
147
+ with contextlib.closing(pydist.as_stream()) as stream:
148
+ metadata = Metadata(fileobj=stream, scheme='legacy')
149
+ logger.debug('Found %s', r.path)
150
+ seen.add(r.path)
151
+ yield new_dist_class(r.path, metadata=metadata,
152
+ env=self)
153
+ elif self._include_egg and entry.endswith(('.egg-info',
154
+ '.egg')):
155
+ logger.debug('Found %s', r.path)
156
+ seen.add(r.path)
157
+ yield old_dist_class(r.path, self)
158
+
159
+ def _generate_cache(self):
160
+ """
161
+ Scan the path for distributions and populate the cache with
162
+ those that are found.
163
+ """
164
+ gen_dist = not self._cache.generated
165
+ gen_egg = self._include_egg and not self._cache_egg.generated
166
+ if gen_dist or gen_egg:
167
+ for dist in self._yield_distributions():
168
+ if isinstance(dist, InstalledDistribution):
169
+ self._cache.add(dist)
170
+ else:
171
+ self._cache_egg.add(dist)
172
+
173
+ if gen_dist:
174
+ self._cache.generated = True
175
+ if gen_egg:
176
+ self._cache_egg.generated = True
177
+
178
+ @classmethod
179
+ def distinfo_dirname(cls, name, version):
180
+ """
181
+ The *name* and *version* parameters are converted into their
182
+ filename-escaped form, i.e. any ``'-'`` characters are replaced
183
+ with ``'_'`` other than the one in ``'dist-info'`` and the one
184
+ separating the name from the version number.
185
+
186
+ :parameter name: is converted to a standard distribution name by replacing
187
+ any runs of non- alphanumeric characters with a single
188
+ ``'-'``.
189
+ :type name: string
190
+ :parameter version: is converted to a standard version string. Spaces
191
+ become dots, and all other non-alphanumeric characters
192
+ (except dots) become dashes, with runs of multiple
193
+ dashes condensed to a single dash.
194
+ :type version: string
195
+ :returns: directory name
196
+ :rtype: string"""
197
+ name = name.replace('-', '_')
198
+ return '-'.join([name, version]) + DISTINFO_EXT
199
+
200
+ def get_distributions(self):
201
+ """
202
+ Provides an iterator that looks for distributions and returns
203
+ :class:`InstalledDistribution` or
204
+ :class:`EggInfoDistribution` instances for each one of them.
205
+
206
+ :rtype: iterator of :class:`InstalledDistribution` and
207
+ :class:`EggInfoDistribution` instances
208
+ """
209
+ if not self._cache_enabled:
210
+ for dist in self._yield_distributions():
211
+ yield dist
212
+ else:
213
+ self._generate_cache()
214
+
215
+ for dist in self._cache.path.values():
216
+ yield dist
217
+
218
+ if self._include_egg:
219
+ for dist in self._cache_egg.path.values():
220
+ yield dist
221
+
222
+ def get_distribution(self, name):
223
+ """
224
+ Looks for a named distribution on the path.
225
+
226
+ This function only returns the first result found, as no more than one
227
+ value is expected. If nothing is found, ``None`` is returned.
228
+
229
+ :rtype: :class:`InstalledDistribution`, :class:`EggInfoDistribution`
230
+ or ``None``
231
+ """
232
+ result = None
233
+ name = name.lower()
234
+ if not self._cache_enabled:
235
+ for dist in self._yield_distributions():
236
+ if dist.key == name:
237
+ result = dist
238
+ break
239
+ else:
240
+ self._generate_cache()
241
+
242
+ if name in self._cache.name:
243
+ result = self._cache.name[name][0]
244
+ elif self._include_egg and name in self._cache_egg.name:
245
+ result = self._cache_egg.name[name][0]
246
+ return result
247
+
248
+ def provides_distribution(self, name, version=None):
249
+ """
250
+ Iterates over all distributions to find which distributions provide *name*.
251
+ If a *version* is provided, it will be used to filter the results.
252
+
253
+ This function only returns the first result found, since no more than
254
+ one values are expected. If the directory is not found, returns ``None``.
255
+
256
+ :parameter version: a version specifier that indicates the version
257
+ required, conforming to the format in ``PEP-345``
258
+
259
+ :type name: string
260
+ :type version: string
261
+ """
262
+ matcher = None
263
+ if version is not None:
264
+ try:
265
+ matcher = self._scheme.matcher('%s (%s)' % (name, version))
266
+ except ValueError:
267
+ raise DistlibException('invalid name or version: %r, %r' %
268
+ (name, version))
269
+
270
+ for dist in self.get_distributions():
271
+ # We hit a problem on Travis where enum34 was installed and doesn't
272
+ # have a provides attribute ...
273
+ if not hasattr(dist, 'provides'):
274
+ logger.debug('No "provides": %s', dist)
275
+ else:
276
+ provided = dist.provides
277
+
278
+ for p in provided:
279
+ p_name, p_ver = parse_name_and_version(p)
280
+ if matcher is None:
281
+ if p_name == name:
282
+ yield dist
283
+ break
284
+ else:
285
+ if p_name == name and matcher.match(p_ver):
286
+ yield dist
287
+ break
288
+
289
+ def get_file_path(self, name, relative_path):
290
+ """
291
+ Return the path to a resource file.
292
+ """
293
+ dist = self.get_distribution(name)
294
+ if dist is None:
295
+ raise LookupError('no distribution named %r found' % name)
296
+ return dist.get_resource_path(relative_path)
297
+
298
+ def get_exported_entries(self, category, name=None):
299
+ """
300
+ Return all of the exported entries in a particular category.
301
+
302
+ :param category: The category to search for entries.
303
+ :param name: If specified, only entries with that name are returned.
304
+ """
305
+ for dist in self.get_distributions():
306
+ r = dist.exports
307
+ if category in r:
308
+ d = r[category]
309
+ if name is not None:
310
+ if name in d:
311
+ yield d[name]
312
+ else:
313
+ for v in d.values():
314
+ yield v
315
+
316
+
317
+ class Distribution(object):
318
+ """
319
+ A base class for distributions, whether installed or from indexes.
320
+ Either way, it must have some metadata, so that's all that's needed
321
+ for construction.
322
+ """
323
+
324
+ build_time_dependency = False
325
+ """
326
+ Set to True if it's known to be only a build-time dependency (i.e.
327
+ not needed after installation).
328
+ """
329
+
330
+ requested = False
331
+ """A boolean that indicates whether the ``REQUESTED`` metadata file is
332
+ present (in other words, whether the package was installed by user
333
+ request or it was installed as a dependency)."""
334
+
335
+ def __init__(self, metadata):
336
+ """
337
+ Initialise an instance.
338
+ :param metadata: The instance of :class:`Metadata` describing this
339
+ distribution.
340
+ """
341
+ self.metadata = metadata
342
+ self.name = metadata.name
343
+ self.key = self.name.lower() # for case-insensitive comparisons
344
+ self.version = metadata.version
345
+ self.locator = None
346
+ self.digest = None
347
+ self.extras = None # additional features requested
348
+ self.context = None # environment marker overrides
349
+ self.download_urls = set()
350
+ self.digests = {}
351
+
352
+ @property
353
+ def source_url(self):
354
+ """
355
+ The source archive download URL for this distribution.
356
+ """
357
+ return self.metadata.source_url
358
+
359
+ download_url = source_url # Backward compatibility
360
+
361
+ @property
362
+ def name_and_version(self):
363
+ """
364
+ A utility property which displays the name and version in parentheses.
365
+ """
366
+ return '%s (%s)' % (self.name, self.version)
367
+
368
+ @property
369
+ def provides(self):
370
+ """
371
+ A set of distribution names and versions provided by this distribution.
372
+ :return: A set of "name (version)" strings.
373
+ """
374
+ plist = self.metadata.provides
375
+ s = '%s (%s)' % (self.name, self.version)
376
+ if s not in plist:
377
+ plist.append(s)
378
+ return plist
379
+
380
+ def _get_requirements(self, req_attr):
381
+ md = self.metadata
382
+ logger.debug('Getting requirements from metadata %r', md.todict())
383
+ reqts = getattr(md, req_attr)
384
+ return set(md.get_requirements(reqts, extras=self.extras,
385
+ env=self.context))
386
+
387
+ @property
388
+ def run_requires(self):
389
+ return self._get_requirements('run_requires')
390
+
391
+ @property
392
+ def meta_requires(self):
393
+ return self._get_requirements('meta_requires')
394
+
395
+ @property
396
+ def build_requires(self):
397
+ return self._get_requirements('build_requires')
398
+
399
+ @property
400
+ def test_requires(self):
401
+ return self._get_requirements('test_requires')
402
+
403
+ @property
404
+ def dev_requires(self):
405
+ return self._get_requirements('dev_requires')
406
+
407
+ def matches_requirement(self, req):
408
+ """
409
+ Say if this instance matches (fulfills) a requirement.
410
+ :param req: The requirement to match.
411
+ :rtype req: str
412
+ :return: True if it matches, else False.
413
+ """
414
+ # Requirement may contain extras - parse to lose those
415
+ # from what's passed to the matcher
416
+ r = parse_requirement(req)
417
+ scheme = get_scheme(self.metadata.scheme)
418
+ try:
419
+ matcher = scheme.matcher(r.requirement)
420
+ except UnsupportedVersionError:
421
+ # XXX compat-mode if cannot read the version
422
+ logger.warning('could not read version %r - using name only',
423
+ req)
424
+ name = req.split()[0]
425
+ matcher = scheme.matcher(name)
426
+
427
+ name = matcher.key # case-insensitive
428
+
429
+ result = False
430
+ for p in self.provides:
431
+ p_name, p_ver = parse_name_and_version(p)
432
+ if p_name != name:
433
+ continue
434
+ try:
435
+ result = matcher.match(p_ver)
436
+ break
437
+ except UnsupportedVersionError:
438
+ pass
439
+ return result
440
+
441
+ def __repr__(self):
442
+ """
443
+ Return a textual representation of this instance,
444
+ """
445
+ if self.source_url:
446
+ suffix = ' [%s]' % self.source_url
447
+ else:
448
+ suffix = ''
449
+ return '<Distribution %s (%s)%s>' % (self.name, self.version, suffix)
450
+
451
+ def __eq__(self, other):
452
+ """
453
+ See if this distribution is the same as another.
454
+ :param other: The distribution to compare with. To be equal to one
455
+ another. distributions must have the same type, name,
456
+ version and source_url.
457
+ :return: True if it is the same, else False.
458
+ """
459
+ if type(other) is not type(self):
460
+ result = False
461
+ else:
462
+ result = (self.name == other.name and
463
+ self.version == other.version and
464
+ self.source_url == other.source_url)
465
+ return result
466
+
467
+ def __hash__(self):
468
+ """
469
+ Compute hash in a way which matches the equality test.
470
+ """
471
+ return hash(self.name) + hash(self.version) + hash(self.source_url)
472
+
473
+
474
+ class BaseInstalledDistribution(Distribution):
475
+ """
476
+ This is the base class for installed distributions (whether PEP 376 or
477
+ legacy).
478
+ """
479
+
480
+ hasher = None
481
+
482
+ def __init__(self, metadata, path, env=None):
483
+ """
484
+ Initialise an instance.
485
+ :param metadata: An instance of :class:`Metadata` which describes the
486
+ distribution. This will normally have been initialised
487
+ from a metadata file in the ``path``.
488
+ :param path: The path of the ``.dist-info`` or ``.egg-info``
489
+ directory for the distribution.
490
+ :param env: This is normally the :class:`DistributionPath`
491
+ instance where this distribution was found.
492
+ """
493
+ super(BaseInstalledDistribution, self).__init__(metadata)
494
+ self.path = path
495
+ self.dist_path = env
496
+
497
+ def get_hash(self, data, hasher=None):
498
+ """
499
+ Get the hash of some data, using a particular hash algorithm, if
500
+ specified.
501
+
502
+ :param data: The data to be hashed.
503
+ :type data: bytes
504
+ :param hasher: The name of a hash implementation, supported by hashlib,
505
+ or ``None``. Examples of valid values are ``'sha1'``,
506
+ ``'sha224'``, ``'sha384'``, '``sha256'``, ``'md5'`` and
507
+ ``'sha512'``. If no hasher is specified, the ``hasher``
508
+ attribute of the :class:`InstalledDistribution` instance
509
+ is used. If the hasher is determined to be ``None``, MD5
510
+ is used as the hashing algorithm.
511
+ :returns: The hash of the data. If a hasher was explicitly specified,
512
+ the returned hash will be prefixed with the specified hasher
513
+ followed by '='.
514
+ :rtype: str
515
+ """
516
+ if hasher is None:
517
+ hasher = self.hasher
518
+ if hasher is None:
519
+ hasher = hashlib.md5
520
+ prefix = ''
521
+ else:
522
+ hasher = getattr(hashlib, hasher)
523
+ prefix = '%s=' % self.hasher
524
+ digest = hasher(data).digest()
525
+ digest = base64.urlsafe_b64encode(digest).rstrip(b'=').decode('ascii')
526
+ return '%s%s' % (prefix, digest)
527
+
528
+
529
+ class InstalledDistribution(BaseInstalledDistribution):
530
+ """
531
+ Created with the *path* of the ``.dist-info`` directory provided to the
532
+ constructor. It reads the metadata contained in ``pydist.json`` when it is
533
+ instantiated., or uses a passed in Metadata instance (useful for when
534
+ dry-run mode is being used).
535
+ """
536
+
537
+ hasher = 'sha256'
538
+
539
+ def __init__(self, path, metadata=None, env=None):
540
+ self.modules = []
541
+ self.finder = finder = resources.finder_for_path(path)
542
+ if finder is None:
543
+ raise ValueError('finder unavailable for %s' % path)
544
+ if env and env._cache_enabled and path in env._cache.path:
545
+ metadata = env._cache.path[path].metadata
546
+ elif metadata is None:
547
+ r = finder.find(METADATA_FILENAME)
548
+ # Temporary - for Wheel 0.23 support
549
+ if r is None:
550
+ r = finder.find(WHEEL_METADATA_FILENAME)
551
+ # Temporary - for legacy support
552
+ if r is None:
553
+ r = finder.find(LEGACY_METADATA_FILENAME)
554
+ if r is None:
555
+ raise ValueError('no %s found in %s' % (METADATA_FILENAME,
556
+ path))
557
+ with contextlib.closing(r.as_stream()) as stream:
558
+ metadata = Metadata(fileobj=stream, scheme='legacy')
559
+
560
+ super(InstalledDistribution, self).__init__(metadata, path, env)
561
+
562
+ if env and env._cache_enabled:
563
+ env._cache.add(self)
564
+
565
+ r = finder.find('REQUESTED')
566
+ self.requested = r is not None
567
+ p = os.path.join(path, 'top_level.txt')
568
+ if os.path.exists(p):
569
+ with open(p, 'rb') as f:
570
+ data = f.read().decode('utf-8')
571
+ self.modules = data.splitlines()
572
+
573
+ def __repr__(self):
574
+ return '<InstalledDistribution %r %s at %r>' % (
575
+ self.name, self.version, self.path)
576
+
577
+ def __str__(self):
578
+ return "%s %s" % (self.name, self.version)
579
+
580
+ def _get_records(self):
581
+ """
582
+ Get the list of installed files for the distribution
583
+ :return: A list of tuples of path, hash and size. Note that hash and
584
+ size might be ``None`` for some entries. The path is exactly
585
+ as stored in the file (which is as in PEP 376).
586
+ """
587
+ results = []
588
+ r = self.get_distinfo_resource('RECORD')
589
+ with contextlib.closing(r.as_stream()) as stream:
590
+ with CSVReader(stream=stream) as record_reader:
591
+ # Base location is parent dir of .dist-info dir
592
+ #base_location = os.path.dirname(self.path)
593
+ #base_location = os.path.abspath(base_location)
594
+ for row in record_reader:
595
+ missing = [None for i in range(len(row), 3)]
596
+ path, checksum, size = row + missing
597
+ #if not os.path.isabs(path):
598
+ # path = path.replace('/', os.sep)
599
+ # path = os.path.join(base_location, path)
600
+ results.append((path, checksum, size))
601
+ return results
602
+
603
+ @cached_property
604
+ def exports(self):
605
+ """
606
+ Return the information exported by this distribution.
607
+ :return: A dictionary of exports, mapping an export category to a dict
608
+ of :class:`ExportEntry` instances describing the individual
609
+ export entries, and keyed by name.
610
+ """
611
+ result = {}
612
+ r = self.get_distinfo_resource(EXPORTS_FILENAME)
613
+ if r:
614
+ result = self.read_exports()
615
+ return result
616
+
617
+ def read_exports(self):
618
+ """
619
+ Read exports data from a file in .ini format.
620
+
621
+ :return: A dictionary of exports, mapping an export category to a list
622
+ of :class:`ExportEntry` instances describing the individual
623
+ export entries.
624
+ """
625
+ result = {}
626
+ r = self.get_distinfo_resource(EXPORTS_FILENAME)
627
+ if r:
628
+ with contextlib.closing(r.as_stream()) as stream:
629
+ result = read_exports(stream)
630
+ return result
631
+
632
+ def write_exports(self, exports):
633
+ """
634
+ Write a dictionary of exports to a file in .ini format.
635
+ :param exports: A dictionary of exports, mapping an export category to
636
+ a list of :class:`ExportEntry` instances describing the
637
+ individual export entries.
638
+ """
639
+ rf = self.get_distinfo_file(EXPORTS_FILENAME)
640
+ with open(rf, 'w') as f:
641
+ write_exports(exports, f)
642
+
643
+ def get_resource_path(self, relative_path):
644
+ """
645
+ NOTE: This API may change in the future.
646
+
647
+ Return the absolute path to a resource file with the given relative
648
+ path.
649
+
650
+ :param relative_path: The path, relative to .dist-info, of the resource
651
+ of interest.
652
+ :return: The absolute path where the resource is to be found.
653
+ """
654
+ r = self.get_distinfo_resource('RESOURCES')
655
+ with contextlib.closing(r.as_stream()) as stream:
656
+ with CSVReader(stream=stream) as resources_reader:
657
+ for relative, destination in resources_reader:
658
+ if relative == relative_path:
659
+ return destination
660
+ raise KeyError('no resource file with relative path %r '
661
+ 'is installed' % relative_path)
662
+
663
+ def list_installed_files(self):
664
+ """
665
+ Iterates over the ``RECORD`` entries and returns a tuple
666
+ ``(path, hash, size)`` for each line.
667
+
668
+ :returns: iterator of (path, hash, size)
669
+ """
670
+ for result in self._get_records():
671
+ yield result
672
+
673
+ def write_installed_files(self, paths, prefix, dry_run=False):
674
+ """
675
+ Writes the ``RECORD`` file, using the ``paths`` iterable passed in. Any
676
+ existing ``RECORD`` file is silently overwritten.
677
+
678
+ prefix is used to determine when to write absolute paths.
679
+ """
680
+ prefix = os.path.join(prefix, '')
681
+ base = os.path.dirname(self.path)
682
+ base_under_prefix = base.startswith(prefix)
683
+ base = os.path.join(base, '')
684
+ record_path = self.get_distinfo_file('RECORD')
685
+ logger.info('creating %s', record_path)
686
+ if dry_run:
687
+ return None
688
+ with CSVWriter(record_path) as writer:
689
+ for path in paths:
690
+ if os.path.isdir(path) or path.endswith(('.pyc', '.pyo')):
691
+ # do not put size and hash, as in PEP-376
692
+ hash_value = size = ''
693
+ else:
694
+ size = '%d' % os.path.getsize(path)
695
+ with open(path, 'rb') as fp:
696
+ hash_value = self.get_hash(fp.read())
697
+ if path.startswith(base) or (base_under_prefix and
698
+ path.startswith(prefix)):
699
+ path = os.path.relpath(path, base)
700
+ writer.writerow((path, hash_value, size))
701
+
702
+ # add the RECORD file itself
703
+ if record_path.startswith(base):
704
+ record_path = os.path.relpath(record_path, base)
705
+ writer.writerow((record_path, '', ''))
706
+ return record_path
707
+
708
+ def check_installed_files(self):
709
+ """
710
+ Checks that the hashes and sizes of the files in ``RECORD`` are
711
+ matched by the files themselves. Returns a (possibly empty) list of
712
+ mismatches. Each entry in the mismatch list will be a tuple consisting
713
+ of the path, 'exists', 'size' or 'hash' according to what didn't match
714
+ (existence is checked first, then size, then hash), the expected
715
+ value and the actual value.
716
+ """
717
+ mismatches = []
718
+ base = os.path.dirname(self.path)
719
+ record_path = self.get_distinfo_file('RECORD')
720
+ for path, hash_value, size in self.list_installed_files():
721
+ if not os.path.isabs(path):
722
+ path = os.path.join(base, path)
723
+ if path == record_path:
724
+ continue
725
+ if not os.path.exists(path):
726
+ mismatches.append((path, 'exists', True, False))
727
+ elif os.path.isfile(path):
728
+ actual_size = str(os.path.getsize(path))
729
+ if size and actual_size != size:
730
+ mismatches.append((path, 'size', size, actual_size))
731
+ elif hash_value:
732
+ if '=' in hash_value:
733
+ hasher = hash_value.split('=', 1)[0]
734
+ else:
735
+ hasher = None
736
+
737
+ with open(path, 'rb') as f:
738
+ actual_hash = self.get_hash(f.read(), hasher)
739
+ if actual_hash != hash_value:
740
+ mismatches.append((path, 'hash', hash_value, actual_hash))
741
+ return mismatches
742
+
743
+ @cached_property
744
+ def shared_locations(self):
745
+ """
746
+ A dictionary of shared locations whose keys are in the set 'prefix',
747
+ 'purelib', 'platlib', 'scripts', 'headers', 'data' and 'namespace'.
748
+ The corresponding value is the absolute path of that category for
749
+ this distribution, and takes into account any paths selected by the
750
+ user at installation time (e.g. via command-line arguments). In the
751
+ case of the 'namespace' key, this would be a list of absolute paths
752
+ for the roots of namespace packages in this distribution.
753
+
754
+ The first time this property is accessed, the relevant information is
755
+ read from the SHARED file in the .dist-info directory.
756
+ """
757
+ result = {}
758
+ shared_path = os.path.join(self.path, 'SHARED')
759
+ if os.path.isfile(shared_path):
760
+ with codecs.open(shared_path, 'r', encoding='utf-8') as f:
761
+ lines = f.read().splitlines()
762
+ for line in lines:
763
+ key, value = line.split('=', 1)
764
+ if key == 'namespace':
765
+ result.setdefault(key, []).append(value)
766
+ else:
767
+ result[key] = value
768
+ return result
769
+
770
+ def write_shared_locations(self, paths, dry_run=False):
771
+ """
772
+ Write shared location information to the SHARED file in .dist-info.
773
+ :param paths: A dictionary as described in the documentation for
774
+ :meth:`shared_locations`.
775
+ :param dry_run: If True, the action is logged but no file is actually
776
+ written.
777
+ :return: The path of the file written to.
778
+ """
779
+ shared_path = os.path.join(self.path, 'SHARED')
780
+ logger.info('creating %s', shared_path)
781
+ if dry_run:
782
+ return None
783
+ lines = []
784
+ for key in ('prefix', 'lib', 'headers', 'scripts', 'data'):
785
+ path = paths[key]
786
+ if os.path.isdir(paths[key]):
787
+ lines.append('%s=%s' % (key, path))
788
+ for ns in paths.get('namespace', ()):
789
+ lines.append('namespace=%s' % ns)
790
+
791
+ with codecs.open(shared_path, 'w', encoding='utf-8') as f:
792
+ f.write('\n'.join(lines))
793
+ return shared_path
794
+
795
+ def get_distinfo_resource(self, path):
796
+ if path not in DIST_FILES:
797
+ raise DistlibException('invalid path for a dist-info file: '
798
+ '%r at %r' % (path, self.path))
799
+ finder = resources.finder_for_path(self.path)
800
+ if finder is None:
801
+ raise DistlibException('Unable to get a finder for %s' % self.path)
802
+ return finder.find(path)
803
+
804
+ def get_distinfo_file(self, path):
805
+ """
806
+ Returns a path located under the ``.dist-info`` directory. Returns a
807
+ string representing the path.
808
+
809
+ :parameter path: a ``'/'``-separated path relative to the
810
+ ``.dist-info`` directory or an absolute path;
811
+ If *path* is an absolute path and doesn't start
812
+ with the ``.dist-info`` directory path,
813
+ a :class:`DistlibException` is raised
814
+ :type path: str
815
+ :rtype: str
816
+ """
817
+ # Check if it is an absolute path # XXX use relpath, add tests
818
+ if path.find(os.sep) >= 0:
819
+ # it's an absolute path?
820
+ distinfo_dirname, path = path.split(os.sep)[-2:]
821
+ if distinfo_dirname != self.path.split(os.sep)[-1]:
822
+ raise DistlibException(
823
+ 'dist-info file %r does not belong to the %r %s '
824
+ 'distribution' % (path, self.name, self.version))
825
+
826
+ # The file must be relative
827
+ if path not in DIST_FILES:
828
+ raise DistlibException('invalid path for a dist-info file: '
829
+ '%r at %r' % (path, self.path))
830
+
831
+ return os.path.join(self.path, path)
832
+
833
+ def list_distinfo_files(self):
834
+ """
835
+ Iterates over the ``RECORD`` entries and returns paths for each line if
836
+ the path is pointing to a file located in the ``.dist-info`` directory
837
+ or one of its subdirectories.
838
+
839
+ :returns: iterator of paths
840
+ """
841
+ base = os.path.dirname(self.path)
842
+ for path, checksum, size in self._get_records():
843
+ # XXX add separator or use real relpath algo
844
+ if not os.path.isabs(path):
845
+ path = os.path.join(base, path)
846
+ if path.startswith(self.path):
847
+ yield path
848
+
849
+ def __eq__(self, other):
850
+ return (isinstance(other, InstalledDistribution) and
851
+ self.path == other.path)
852
+
853
+ # See http://docs.python.org/reference/datamodel#object.__hash__
854
+ __hash__ = object.__hash__
855
+
856
+
857
+ class EggInfoDistribution(BaseInstalledDistribution):
858
+ """Created with the *path* of the ``.egg-info`` directory or file provided
859
+ to the constructor. It reads the metadata contained in the file itself, or
860
+ if the given path happens to be a directory, the metadata is read from the
861
+ file ``PKG-INFO`` under that directory."""
862
+
863
+ requested = True # as we have no way of knowing, assume it was
864
+ shared_locations = {}
865
+
866
+ def __init__(self, path, env=None):
867
+ def set_name_and_version(s, n, v):
868
+ s.name = n
869
+ s.key = n.lower() # for case-insensitive comparisons
870
+ s.version = v
871
+
872
+ self.path = path
873
+ self.dist_path = env
874
+ if env and env._cache_enabled and path in env._cache_egg.path:
875
+ metadata = env._cache_egg.path[path].metadata
876
+ set_name_and_version(self, metadata.name, metadata.version)
877
+ else:
878
+ metadata = self._get_metadata(path)
879
+
880
+ # Need to be set before caching
881
+ set_name_and_version(self, metadata.name, metadata.version)
882
+
883
+ if env and env._cache_enabled:
884
+ env._cache_egg.add(self)
885
+ super(EggInfoDistribution, self).__init__(metadata, path, env)
886
+
887
+ def _get_metadata(self, path):
888
+ requires = None
889
+
890
+ def parse_requires_data(data):
891
+ """Create a list of dependencies from a requires.txt file.
892
+
893
+ *data*: the contents of a setuptools-produced requires.txt file.
894
+ """
895
+ reqs = []
896
+ lines = data.splitlines()
897
+ for line in lines:
898
+ line = line.strip()
899
+ if line.startswith('['):
900
+ logger.warning('Unexpected line: quitting requirement scan: %r',
901
+ line)
902
+ break
903
+ r = parse_requirement(line)
904
+ if not r:
905
+ logger.warning('Not recognised as a requirement: %r', line)
906
+ continue
907
+ if r.extras:
908
+ logger.warning('extra requirements in requires.txt are '
909
+ 'not supported')
910
+ if not r.constraints:
911
+ reqs.append(r.name)
912
+ else:
913
+ cons = ', '.join('%s%s' % c for c in r.constraints)
914
+ reqs.append('%s (%s)' % (r.name, cons))
915
+ return reqs
916
+
917
+ def parse_requires_path(req_path):
918
+ """Create a list of dependencies from a requires.txt file.
919
+
920
+ *req_path*: the path to a setuptools-produced requires.txt file.
921
+ """
922
+
923
+ reqs = []
924
+ try:
925
+ with codecs.open(req_path, 'r', 'utf-8') as fp:
926
+ reqs = parse_requires_data(fp.read())
927
+ except IOError:
928
+ pass
929
+ return reqs
930
+
931
+ tl_path = tl_data = None
932
+ if path.endswith('.egg'):
933
+ if os.path.isdir(path):
934
+ p = os.path.join(path, 'EGG-INFO')
935
+ meta_path = os.path.join(p, 'PKG-INFO')
936
+ metadata = Metadata(path=meta_path, scheme='legacy')
937
+ req_path = os.path.join(p, 'requires.txt')
938
+ tl_path = os.path.join(p, 'top_level.txt')
939
+ requires = parse_requires_path(req_path)
940
+ else:
941
+ # FIXME handle the case where zipfile is not available
942
+ zipf = zipimport.zipimporter(path)
943
+ fileobj = StringIO(
944
+ zipf.get_data('EGG-INFO/PKG-INFO').decode('utf8'))
945
+ metadata = Metadata(fileobj=fileobj, scheme='legacy')
946
+ try:
947
+ data = zipf.get_data('EGG-INFO/requires.txt')
948
+ tl_data = zipf.get_data('EGG-INFO/top_level.txt').decode('utf-8')
949
+ requires = parse_requires_data(data.decode('utf-8'))
950
+ except IOError:
951
+ requires = None
952
+ elif path.endswith('.egg-info'):
953
+ if os.path.isdir(path):
954
+ req_path = os.path.join(path, 'requires.txt')
955
+ requires = parse_requires_path(req_path)
956
+ path = os.path.join(path, 'PKG-INFO')
957
+ tl_path = os.path.join(path, 'top_level.txt')
958
+ metadata = Metadata(path=path, scheme='legacy')
959
+ else:
960
+ raise DistlibException('path must end with .egg-info or .egg, '
961
+ 'got %r' % path)
962
+
963
+ if requires:
964
+ metadata.add_requirements(requires)
965
+ # look for top-level modules in top_level.txt, if present
966
+ if tl_data is None:
967
+ if tl_path is not None and os.path.exists(tl_path):
968
+ with open(tl_path, 'rb') as f:
969
+ tl_data = f.read().decode('utf-8')
970
+ if not tl_data:
971
+ tl_data = []
972
+ else:
973
+ tl_data = tl_data.splitlines()
974
+ self.modules = tl_data
975
+ return metadata
976
+
977
+ def __repr__(self):
978
+ return '<EggInfoDistribution %r %s at %r>' % (
979
+ self.name, self.version, self.path)
980
+
981
+ def __str__(self):
982
+ return "%s %s" % (self.name, self.version)
983
+
984
+ def check_installed_files(self):
985
+ """
986
+ Checks that the hashes and sizes of the files in ``RECORD`` are
987
+ matched by the files themselves. Returns a (possibly empty) list of
988
+ mismatches. Each entry in the mismatch list will be a tuple consisting
989
+ of the path, 'exists', 'size' or 'hash' according to what didn't match
990
+ (existence is checked first, then size, then hash), the expected
991
+ value and the actual value.
992
+ """
993
+ mismatches = []
994
+ record_path = os.path.join(self.path, 'installed-files.txt')
995
+ if os.path.exists(record_path):
996
+ for path, _, _ in self.list_installed_files():
997
+ if path == record_path:
998
+ continue
999
+ if not os.path.exists(path):
1000
+ mismatches.append((path, 'exists', True, False))
1001
+ return mismatches
1002
+
1003
+ def list_installed_files(self):
1004
+ """
1005
+ Iterates over the ``installed-files.txt`` entries and returns a tuple
1006
+ ``(path, hash, size)`` for each line.
1007
+
1008
+ :returns: a list of (path, hash, size)
1009
+ """
1010
+
1011
+ def _md5(path):
1012
+ f = open(path, 'rb')
1013
+ try:
1014
+ content = f.read()
1015
+ finally:
1016
+ f.close()
1017
+ return hashlib.md5(content).hexdigest()
1018
+
1019
+ def _size(path):
1020
+ return os.stat(path).st_size
1021
+
1022
+ record_path = os.path.join(self.path, 'installed-files.txt')
1023
+ result = []
1024
+ if os.path.exists(record_path):
1025
+ with codecs.open(record_path, 'r', encoding='utf-8') as f:
1026
+ for line in f:
1027
+ line = line.strip()
1028
+ p = os.path.normpath(os.path.join(self.path, line))
1029
+ # "./" is present as a marker between installed files
1030
+ # and installation metadata files
1031
+ if not os.path.exists(p):
1032
+ logger.warning('Non-existent file: %s', p)
1033
+ if p.endswith(('.pyc', '.pyo')):
1034
+ continue
1035
+ #otherwise fall through and fail
1036
+ if not os.path.isdir(p):
1037
+ result.append((p, _md5(p), _size(p)))
1038
+ result.append((record_path, None, None))
1039
+ return result
1040
+
1041
+ def list_distinfo_files(self, absolute=False):
1042
+ """
1043
+ Iterates over the ``installed-files.txt`` entries and returns paths for
1044
+ each line if the path is pointing to a file located in the
1045
+ ``.egg-info`` directory or one of its subdirectories.
1046
+
1047
+ :parameter absolute: If *absolute* is ``True``, each returned path is
1048
+ transformed into a local absolute path. Otherwise the
1049
+ raw value from ``installed-files.txt`` is returned.
1050
+ :type absolute: boolean
1051
+ :returns: iterator of paths
1052
+ """
1053
+ record_path = os.path.join(self.path, 'installed-files.txt')
1054
+ if os.path.exists(record_path):
1055
+ skip = True
1056
+ with codecs.open(record_path, 'r', encoding='utf-8') as f:
1057
+ for line in f:
1058
+ line = line.strip()
1059
+ if line == './':
1060
+ skip = False
1061
+ continue
1062
+ if not skip:
1063
+ p = os.path.normpath(os.path.join(self.path, line))
1064
+ if p.startswith(self.path):
1065
+ if absolute:
1066
+ yield p
1067
+ else:
1068
+ yield line
1069
+
1070
+ def __eq__(self, other):
1071
+ return (isinstance(other, EggInfoDistribution) and
1072
+ self.path == other.path)
1073
+
1074
+ # See http://docs.python.org/reference/datamodel#object.__hash__
1075
+ __hash__ = object.__hash__
1076
+
1077
+ new_dist_class = InstalledDistribution
1078
+ old_dist_class = EggInfoDistribution
1079
+
1080
+
1081
+ class DependencyGraph(object):
1082
+ """
1083
+ Represents a dependency graph between distributions.
1084
+
1085
+ The dependency relationships are stored in an ``adjacency_list`` that maps
1086
+ distributions to a list of ``(other, label)`` tuples where ``other``
1087
+ is a distribution and the edge is labeled with ``label`` (i.e. the version
1088
+ specifier, if such was provided). Also, for more efficient traversal, for
1089
+ every distribution ``x``, a list of predecessors is kept in
1090
+ ``reverse_list[x]``. An edge from distribution ``a`` to
1091
+ distribution ``b`` means that ``a`` depends on ``b``. If any missing
1092
+ dependencies are found, they are stored in ``missing``, which is a
1093
+ dictionary that maps distributions to a list of requirements that were not
1094
+ provided by any other distributions.
1095
+ """
1096
+
1097
+ def __init__(self):
1098
+ self.adjacency_list = {}
1099
+ self.reverse_list = {}
1100
+ self.missing = {}
1101
+
1102
+ def add_distribution(self, distribution):
1103
+ """Add the *distribution* to the graph.
1104
+
1105
+ :type distribution: :class:`distutils2.database.InstalledDistribution`
1106
+ or :class:`distutils2.database.EggInfoDistribution`
1107
+ """
1108
+ self.adjacency_list[distribution] = []
1109
+ self.reverse_list[distribution] = []
1110
+ #self.missing[distribution] = []
1111
+
1112
+ def add_edge(self, x, y, label=None):
1113
+ """Add an edge from distribution *x* to distribution *y* with the given
1114
+ *label*.
1115
+
1116
+ :type x: :class:`distutils2.database.InstalledDistribution` or
1117
+ :class:`distutils2.database.EggInfoDistribution`
1118
+ :type y: :class:`distutils2.database.InstalledDistribution` or
1119
+ :class:`distutils2.database.EggInfoDistribution`
1120
+ :type label: ``str`` or ``None``
1121
+ """
1122
+ self.adjacency_list[x].append((y, label))
1123
+ # multiple edges are allowed, so be careful
1124
+ if x not in self.reverse_list[y]:
1125
+ self.reverse_list[y].append(x)
1126
+
1127
+ def add_missing(self, distribution, requirement):
1128
+ """
1129
+ Add a missing *requirement* for the given *distribution*.
1130
+
1131
+ :type distribution: :class:`distutils2.database.InstalledDistribution`
1132
+ or :class:`distutils2.database.EggInfoDistribution`
1133
+ :type requirement: ``str``
1134
+ """
1135
+ logger.debug('%s missing %r', distribution, requirement)
1136
+ self.missing.setdefault(distribution, []).append(requirement)
1137
+
1138
+ def _repr_dist(self, dist):
1139
+ return '%s %s' % (dist.name, dist.version)
1140
+
1141
+ def repr_node(self, dist, level=1):
1142
+ """Prints only a subgraph"""
1143
+ output = [self._repr_dist(dist)]
1144
+ for other, label in self.adjacency_list[dist]:
1145
+ dist = self._repr_dist(other)
1146
+ if label is not None:
1147
+ dist = '%s [%s]' % (dist, label)
1148
+ output.append(' ' * level + str(dist))
1149
+ suboutput = self.repr_node(other, level + 1)
1150
+ subs = suboutput.split('\n')
1151
+ output.extend(subs[1:])
1152
+ return '\n'.join(output)
1153
+
1154
+ def to_dot(self, f, skip_disconnected=True):
1155
+ """Writes a DOT output for the graph to the provided file *f*.
1156
+
1157
+ If *skip_disconnected* is set to ``True``, then all distributions
1158
+ that are not dependent on any other distribution are skipped.
1159
+
1160
+ :type f: has to support ``file``-like operations
1161
+ :type skip_disconnected: ``bool``
1162
+ """
1163
+ disconnected = []
1164
+
1165
+ f.write("digraph dependencies {\n")
1166
+ for dist, adjs in self.adjacency_list.items():
1167
+ if len(adjs) == 0 and not skip_disconnected:
1168
+ disconnected.append(dist)
1169
+ for other, label in adjs:
1170
+ if not label is None:
1171
+ f.write('"%s" -> "%s" [label="%s"]\n' %
1172
+ (dist.name, other.name, label))
1173
+ else:
1174
+ f.write('"%s" -> "%s"\n' % (dist.name, other.name))
1175
+ if not skip_disconnected and len(disconnected) > 0:
1176
+ f.write('subgraph disconnected {\n')
1177
+ f.write('label = "Disconnected"\n')
1178
+ f.write('bgcolor = red\n')
1179
+
1180
+ for dist in disconnected:
1181
+ f.write('"%s"' % dist.name)
1182
+ f.write('\n')
1183
+ f.write('}\n')
1184
+ f.write('}\n')
1185
+
1186
+ def topological_sort(self):
1187
+ """
1188
+ Perform a topological sort of the graph.
1189
+ :return: A tuple, the first element of which is a topologically sorted
1190
+ list of distributions, and the second element of which is a
1191
+ list of distributions that cannot be sorted because they have
1192
+ circular dependencies and so form a cycle.
1193
+ """
1194
+ result = []
1195
+ # Make a shallow copy of the adjacency list
1196
+ alist = {}
1197
+ for k, v in self.adjacency_list.items():
1198
+ alist[k] = v[:]
1199
+ while True:
1200
+ # See what we can remove in this run
1201
+ to_remove = []
1202
+ for k, v in list(alist.items())[:]:
1203
+ if not v:
1204
+ to_remove.append(k)
1205
+ del alist[k]
1206
+ if not to_remove:
1207
+ # What's left in alist (if anything) is a cycle.
1208
+ break
1209
+ # Remove from the adjacency list of others
1210
+ for k, v in alist.items():
1211
+ alist[k] = [(d, r) for d, r in v if d not in to_remove]
1212
+ logger.debug('Moving to result: %s',
1213
+ ['%s (%s)' % (d.name, d.version) for d in to_remove])
1214
+ result.extend(to_remove)
1215
+ return result, list(alist.keys())
1216
+
1217
+ def __repr__(self):
1218
+ """Representation of the graph"""
1219
+ output = []
1220
+ for dist, adjs in self.adjacency_list.items():
1221
+ output.append(self.repr_node(dist))
1222
+ return '\n'.join(output)
1223
+
1224
+
1225
+ def make_graph(dists, scheme='default'):
1226
+ """Makes a dependency graph from the given distributions.
1227
+
1228
+ :parameter dists: a list of distributions
1229
+ :type dists: list of :class:`distutils2.database.InstalledDistribution` and
1230
+ :class:`distutils2.database.EggInfoDistribution` instances
1231
+ :rtype: a :class:`DependencyGraph` instance
1232
+ """
1233
+ scheme = get_scheme(scheme)
1234
+ graph = DependencyGraph()
1235
+ provided = {} # maps names to lists of (version, dist) tuples
1236
+
1237
+ # first, build the graph and find out what's provided
1238
+ for dist in dists:
1239
+ graph.add_distribution(dist)
1240
+
1241
+ for p in dist.provides:
1242
+ name, version = parse_name_and_version(p)
1243
+ logger.debug('Add to provided: %s, %s, %s', name, version, dist)
1244
+ provided.setdefault(name, []).append((version, dist))
1245
+
1246
+ # now make the edges
1247
+ for dist in dists:
1248
+ requires = (dist.run_requires | dist.meta_requires |
1249
+ dist.build_requires | dist.dev_requires)
1250
+ for req in requires:
1251
+ try:
1252
+ matcher = scheme.matcher(req)
1253
+ except UnsupportedVersionError:
1254
+ # XXX compat-mode if cannot read the version
1255
+ logger.warning('could not read version %r - using name only',
1256
+ req)
1257
+ name = req.split()[0]
1258
+ matcher = scheme.matcher(name)
1259
+
1260
+ name = matcher.key # case-insensitive
1261
+
1262
+ matched = False
1263
+ if name in provided:
1264
+ for version, provider in provided[name]:
1265
+ try:
1266
+ match = matcher.match(version)
1267
+ except UnsupportedVersionError:
1268
+ match = False
1269
+
1270
+ if match:
1271
+ graph.add_edge(dist, provider, req)
1272
+ matched = True
1273
+ break
1274
+ if not matched:
1275
+ graph.add_missing(dist, req)
1276
+ return graph
1277
+
1278
+
1279
+ def get_dependent_dists(dists, dist):
1280
+ """Recursively generate a list of distributions from *dists* that are
1281
+ dependent on *dist*.
1282
+
1283
+ :param dists: a list of distributions
1284
+ :param dist: a distribution, member of *dists* for which we are interested
1285
+ """
1286
+ if dist not in dists:
1287
+ raise DistlibException('given distribution %r is not a member '
1288
+ 'of the list' % dist.name)
1289
+ graph = make_graph(dists)
1290
+
1291
+ dep = [dist] # dependent distributions
1292
+ todo = graph.reverse_list[dist] # list of nodes we should inspect
1293
+
1294
+ while todo:
1295
+ d = todo.pop()
1296
+ dep.append(d)
1297
+ for succ in graph.reverse_list[d]:
1298
+ if succ not in dep:
1299
+ todo.append(succ)
1300
+
1301
+ dep.pop(0) # remove dist from dep, was there to prevent infinite loops
1302
+ return dep
1303
+
1304
+
1305
+ def get_required_dists(dists, dist):
1306
+ """Recursively generate a list of distributions from *dists* that are
1307
+ required by *dist*.
1308
+
1309
+ :param dists: a list of distributions
1310
+ :param dist: a distribution, member of *dists* for which we are interested
1311
+ """
1312
+ if dist not in dists:
1313
+ raise DistlibException('given distribution %r is not a member '
1314
+ 'of the list' % dist.name)
1315
+ graph = make_graph(dists)
1316
+
1317
+ req = [] # required distributions
1318
+ todo = graph.adjacency_list[dist] # list of nodes we should inspect
1319
+
1320
+ while todo:
1321
+ d = todo.pop()[0]
1322
+ req.append(d)
1323
+ for pred in graph.adjacency_list[d]:
1324
+ if pred not in req:
1325
+ todo.append(pred)
1326
+
1327
+ return req
1328
+
1329
+
1330
+ def make_dist(name, version, **kwargs):
1331
+ """
1332
+ A convenience method for making a dist given just a name and version.
1333
+ """
1334
+ summary = kwargs.pop('summary', 'Placeholder for summary')
1335
+ md = Metadata(**kwargs)
1336
+ md.name = name
1337
+ md.version = version
1338
+ md.summary = summary or 'Placeholder for summary'
1339
+ return Distribution(md)