py2docfx 0.1.16.dev2064350__py3-none-any.whl → 0.1.17__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 (388) hide show
  1. py2docfx/convert_prepare/environment.py +1 -3
  2. py2docfx/convert_prepare/get_source.py +1 -6
  3. py2docfx/convert_prepare/package_info.py +1 -1
  4. py2docfx/docfx_yaml/miss_reference.py +1 -1
  5. py2docfx/docfx_yaml/parameter_utils.py +58 -6
  6. py2docfx/docfx_yaml/tests/roots/test-translator-typing/code_with_typing.py +14 -0
  7. py2docfx/docfx_yaml/tests/roots/test-translator-typing/conf.py +18 -0
  8. py2docfx/docfx_yaml/tests/test_translator_typing.py +39 -0
  9. py2docfx/docfx_yaml/utils.py +2 -2
  10. py2docfx/venv/basevenv/Lib/site-packages/certifi/__init__.py +1 -1
  11. py2docfx/venv/basevenv/Lib/site-packages/certifi/core.py +1 -32
  12. py2docfx/venv/basevenv/Lib/site-packages/charset_normalizer/cli/__main__.py +62 -2
  13. py2docfx/venv/basevenv/Lib/site-packages/charset_normalizer/constant.py +17 -0
  14. py2docfx/venv/basevenv/Lib/site-packages/charset_normalizer/legacy.py +1 -3
  15. py2docfx/venv/basevenv/Lib/site-packages/charset_normalizer/md.py +19 -14
  16. py2docfx/venv/basevenv/Lib/site-packages/charset_normalizer/utils.py +6 -0
  17. py2docfx/venv/basevenv/Lib/site-packages/charset_normalizer/version.py +1 -1
  18. py2docfx/venv/basevenv/Lib/site-packages/packaging/__init__.py +1 -1
  19. py2docfx/venv/basevenv/Lib/site-packages/packaging/_elffile.py +1 -2
  20. py2docfx/venv/basevenv/Lib/site-packages/packaging/_manylinux.py +1 -2
  21. py2docfx/venv/basevenv/Lib/site-packages/packaging/_parser.py +1 -2
  22. py2docfx/venv/basevenv/Lib/site-packages/packaging/_tokenizer.py +5 -4
  23. py2docfx/venv/basevenv/Lib/site-packages/packaging/licenses/__init__.py +1 -1
  24. py2docfx/venv/basevenv/Lib/site-packages/packaging/markers.py +53 -22
  25. py2docfx/venv/basevenv/Lib/site-packages/packaging/metadata.py +1 -2
  26. py2docfx/venv/basevenv/Lib/site-packages/packaging/specifiers.py +1 -2
  27. py2docfx/venv/basevenv/Lib/site-packages/packaging/tags.py +39 -0
  28. py2docfx/venv/basevenv/Lib/site-packages/pkg_resources/__init__.py +6 -7
  29. py2docfx/venv/basevenv/Lib/site-packages/pygments/__init__.py +1 -1
  30. py2docfx/venv/basevenv/Lib/site-packages/pygments/lexers/_sql_builtins.py +106 -0
  31. py2docfx/venv/basevenv/Lib/site-packages/pygments/lexers/scripting.py +5 -3
  32. py2docfx/venv/basevenv/Lib/site-packages/pygments/lexers/sql.py +24 -118
  33. py2docfx/venv/basevenv/Lib/site-packages/requests/__version__.py +2 -2
  34. py2docfx/venv/basevenv/Lib/site-packages/requests/compat.py +12 -0
  35. py2docfx/venv/basevenv/Lib/site-packages/requests/models.py +3 -1
  36. py2docfx/venv/basevenv/Lib/site-packages/requests/utils.py +6 -16
  37. py2docfx/venv/basevenv/Lib/site-packages/setuptools/__init__.py +0 -38
  38. py2docfx/venv/basevenv/Lib/site-packages/setuptools/_discovery.py +33 -0
  39. py2docfx/venv/basevenv/Lib/site-packages/setuptools/_distutils/command/build_scripts.py +1 -1
  40. py2docfx/venv/basevenv/Lib/site-packages/setuptools/_distutils/command/config.py +0 -2
  41. py2docfx/venv/basevenv/Lib/site-packages/setuptools/_distutils/compilers/C/tests/test_unix.py +63 -0
  42. py2docfx/venv/basevenv/Lib/site-packages/setuptools/_distutils/compilers/C/unix.py +5 -6
  43. py2docfx/venv/basevenv/Lib/site-packages/setuptools/_entry_points.py +4 -0
  44. py2docfx/venv/basevenv/Lib/site-packages/setuptools/_normalization.py +0 -2
  45. py2docfx/venv/basevenv/Lib/site-packages/setuptools/_path.py +12 -3
  46. py2docfx/venv/basevenv/Lib/site-packages/setuptools/_reqs.py +1 -1
  47. py2docfx/venv/basevenv/Lib/site-packages/setuptools/_scripts.py +361 -0
  48. py2docfx/venv/basevenv/Lib/site-packages/setuptools/_shutil.py +6 -0
  49. py2docfx/venv/basevenv/Lib/site-packages/setuptools/build_meta.py +25 -37
  50. py2docfx/venv/basevenv/Lib/site-packages/setuptools/command/bdist_egg.py +9 -11
  51. py2docfx/venv/basevenv/Lib/site-packages/setuptools/command/bdist_wheel.py +1 -1
  52. py2docfx/venv/basevenv/Lib/site-packages/setuptools/command/build_ext.py +29 -28
  53. py2docfx/venv/basevenv/Lib/site-packages/setuptools/command/develop.py +39 -179
  54. py2docfx/venv/basevenv/Lib/site-packages/setuptools/command/easy_install.py +17 -2352
  55. py2docfx/venv/basevenv/Lib/site-packages/setuptools/command/editable_wheel.py +14 -31
  56. py2docfx/venv/basevenv/Lib/site-packages/setuptools/command/egg_info.py +0 -2
  57. py2docfx/venv/basevenv/Lib/site-packages/setuptools/command/install.py +3 -55
  58. py2docfx/venv/basevenv/Lib/site-packages/setuptools/command/install_scripts.py +5 -11
  59. py2docfx/venv/basevenv/Lib/site-packages/setuptools/command/sdist.py +1 -1
  60. py2docfx/venv/basevenv/Lib/site-packages/setuptools/compat/py310.py +11 -0
  61. py2docfx/venv/basevenv/Lib/site-packages/setuptools/dist.py +10 -12
  62. py2docfx/venv/basevenv/Lib/site-packages/setuptools/installer.py +34 -29
  63. py2docfx/venv/basevenv/Lib/site-packages/setuptools/tests/contexts.py +0 -14
  64. py2docfx/venv/basevenv/Lib/site-packages/setuptools/tests/fixtures.py +235 -0
  65. py2docfx/venv/basevenv/Lib/site-packages/setuptools/tests/integration/test_pbr.py +20 -0
  66. py2docfx/venv/basevenv/Lib/site-packages/setuptools/tests/test_build_meta.py +0 -24
  67. py2docfx/venv/basevenv/Lib/site-packages/setuptools/tests/test_develop.py +1 -64
  68. py2docfx/venv/basevenv/Lib/site-packages/setuptools/tests/test_dist.py +2 -2
  69. py2docfx/venv/basevenv/Lib/site-packages/setuptools/tests/test_dist_info.py +0 -63
  70. py2docfx/venv/basevenv/Lib/site-packages/setuptools/tests/test_editable_install.py +15 -41
  71. py2docfx/venv/basevenv/Lib/site-packages/setuptools/tests/test_egg_info.py +3 -2
  72. py2docfx/venv/basevenv/Lib/site-packages/setuptools/tests/test_scripts.py +12 -0
  73. py2docfx/venv/basevenv/Lib/site-packages/setuptools/tests/test_wheel.py +12 -36
  74. py2docfx/venv/basevenv/Lib/site-packages/setuptools/tests/test_windows_wrappers.py +3 -4
  75. py2docfx/venv/basevenv/Lib/site-packages/setuptools/wheel.py +49 -24
  76. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/__init__.py +6 -0
  77. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/arabic_stemmer.py +798 -797
  78. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/armenian_stemmer.py +212 -213
  79. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/basestemmer.py +20 -54
  80. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/basque_stemmer.py +202 -228
  81. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/catalan_stemmer.py +370 -375
  82. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/danish_stemmer.py +63 -61
  83. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/dutch_porter_stemmer.py +466 -0
  84. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/dutch_stemmer.py +1217 -343
  85. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/english_stemmer.py +295 -264
  86. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/esperanto_stemmer.py +588 -0
  87. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/estonian_stemmer.py +850 -0
  88. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/finnish_stemmer.py +144 -143
  89. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/french_stemmer.py +308 -246
  90. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/german_stemmer.py +224 -167
  91. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/greek_stemmer.py +1314 -1313
  92. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/hindi_stemmer.py +39 -38
  93. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/hungarian_stemmer.py +231 -256
  94. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/indonesian_stemmer.py +61 -64
  95. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/irish_stemmer.py +101 -106
  96. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/italian_stemmer.py +272 -272
  97. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/lithuanian_stemmer.py +233 -240
  98. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/nepali_stemmer.py +108 -134
  99. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/norwegian_stemmer.py +108 -63
  100. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/porter_stemmer.py +101 -108
  101. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/portuguese_stemmer.py +241 -250
  102. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/romanian_stemmer.py +332 -296
  103. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/russian_stemmer.py +178 -179
  104. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/serbian_stemmer.py +2331 -2332
  105. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/spanish_stemmer.py +254 -259
  106. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/swedish_stemmer.py +143 -70
  107. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/tamil_stemmer.py +638 -1491
  108. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/turkish_stemmer.py +326 -292
  109. py2docfx/venv/basevenv/Lib/site-packages/snowballstemmer/yiddish_stemmer.py +245 -246
  110. py2docfx/venv/basevenv/Lib/site-packages/urllib3/_version.py +2 -2
  111. py2docfx/venv/basevenv/Lib/site-packages/urllib3/connection.py +87 -38
  112. py2docfx/venv/basevenv/Lib/site-packages/urllib3/contrib/emscripten/fetch.py +20 -0
  113. py2docfx/venv/basevenv/Lib/site-packages/urllib3/poolmanager.py +17 -1
  114. py2docfx/venv/basevenv/Lib/site-packages/urllib3/response.py +53 -24
  115. py2docfx/venv/basevenv/Lib/site-packages/urllib3/util/request.py +12 -4
  116. py2docfx/venv/basevenv/Lib/site-packages/urllib3/util/ssl_.py +1 -1
  117. py2docfx/venv/venv1/Lib/site-packages/azure/core/_version.py +1 -1
  118. py2docfx/venv/venv1/Lib/site-packages/azure/core/pipeline/_base_async.py +1 -1
  119. py2docfx/venv/venv1/Lib/site-packages/azure/core/pipeline/transport/_aiohttp.py +1 -1
  120. py2docfx/venv/venv1/Lib/site-packages/azure/core/pipeline/transport/_base_async.py +2 -1
  121. py2docfx/venv/venv1/Lib/site-packages/azure/core/pipeline/transport/_requests_asyncio.py +1 -1
  122. py2docfx/venv/venv1/Lib/site-packages/azure/core/rest/_aiohttp.py +2 -2
  123. py2docfx/venv/venv1/Lib/site-packages/azure/core/rest/_requests_asyncio.py +1 -1
  124. py2docfx/venv/venv1/Lib/site-packages/azure/core/tracing/opentelemetry.py +13 -1
  125. py2docfx/venv/venv1/Lib/site-packages/azure/core/utils/_pipeline_transport_rest_shared_async.py +2 -1
  126. py2docfx/venv/venv1/Lib/site-packages/azure/core/utils/_utils.py +1 -1
  127. py2docfx/venv/venv1/Lib/site-packages/azure/identity/_constants.py +1 -0
  128. py2docfx/venv/venv1/Lib/site-packages/azure/identity/_credentials/default.py +23 -2
  129. py2docfx/venv/venv1/Lib/site-packages/azure/identity/_credentials/environment.py +12 -16
  130. py2docfx/venv/venv1/Lib/site-packages/azure/identity/_credentials/service_fabric.py +22 -1
  131. py2docfx/venv/venv1/Lib/site-packages/azure/identity/_credentials/user_password.py +10 -6
  132. py2docfx/venv/venv1/Lib/site-packages/azure/identity/_credentials/vscode.py +13 -4
  133. py2docfx/venv/venv1/Lib/site-packages/azure/identity/_internal/auth_code_redirect_handler.py +3 -5
  134. py2docfx/venv/venv1/Lib/site-packages/azure/identity/_internal/msal_client.py +2 -1
  135. py2docfx/venv/venv1/Lib/site-packages/azure/identity/_persistent_cache.py +3 -2
  136. py2docfx/venv/venv1/Lib/site-packages/azure/identity/_version.py +1 -1
  137. py2docfx/venv/venv1/Lib/site-packages/azure/identity/aio/_credentials/azd_cli.py +1 -1
  138. py2docfx/venv/venv1/Lib/site-packages/azure/identity/aio/_credentials/azure_cli.py +1 -1
  139. py2docfx/venv/venv1/Lib/site-packages/azure/identity/aio/_credentials/azure_powershell.py +1 -1
  140. py2docfx/venv/venv1/Lib/site-packages/azure/identity/aio/_credentials/chained.py +1 -1
  141. py2docfx/venv/venv1/Lib/site-packages/azure/identity/aio/_credentials/default.py +22 -2
  142. py2docfx/venv/venv1/Lib/site-packages/azure/identity/aio/_credentials/environment.py +2 -2
  143. py2docfx/venv/venv1/Lib/site-packages/azure/identity/aio/_credentials/service_fabric.py +17 -1
  144. py2docfx/venv/venv1/Lib/site-packages/azure/identity/aio/_credentials/vscode.py +4 -4
  145. py2docfx/venv/venv1/Lib/site-packages/certifi/__init__.py +1 -1
  146. py2docfx/venv/venv1/Lib/site-packages/certifi/core.py +1 -32
  147. py2docfx/venv/venv1/Lib/site-packages/charset_normalizer/cli/__main__.py +62 -2
  148. py2docfx/venv/venv1/Lib/site-packages/charset_normalizer/constant.py +17 -0
  149. py2docfx/venv/venv1/Lib/site-packages/charset_normalizer/legacy.py +1 -3
  150. py2docfx/venv/venv1/Lib/site-packages/charset_normalizer/md.py +19 -14
  151. py2docfx/venv/venv1/Lib/site-packages/charset_normalizer/utils.py +6 -0
  152. py2docfx/venv/venv1/Lib/site-packages/charset_normalizer/version.py +1 -1
  153. py2docfx/venv/venv1/Lib/site-packages/cryptography/__about__.py +2 -2
  154. py2docfx/venv/venv1/Lib/site-packages/cryptography/__init__.py +2 -2
  155. py2docfx/venv/venv1/Lib/site-packages/cryptography/fernet.py +2 -1
  156. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/_oid.py +35 -2
  157. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/backends/openssl/backend.py +33 -10
  158. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/bindings/_rust/__init__.pyi +14 -5
  159. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/bindings/_rust/ocsp.pyi +2 -2
  160. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/bindings/_rust/openssl/__init__.pyi +3 -0
  161. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/bindings/_rust/openssl/aead.pyi +49 -45
  162. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/bindings/_rust/openssl/ciphers.pyi +2 -2
  163. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/bindings/_rust/openssl/ed25519.pyi +2 -1
  164. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/bindings/_rust/openssl/ed448.pyi +2 -1
  165. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/bindings/_rust/openssl/hashes.pyi +10 -1
  166. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/bindings/_rust/openssl/hmac.pyi +3 -2
  167. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/bindings/_rust/openssl/kdf.pyi +8 -2
  168. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/bindings/_rust/openssl/keys.pyi +3 -2
  169. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/bindings/_rust/openssl/poly1305.pyi +6 -4
  170. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/bindings/_rust/openssl/x25519.pyi +2 -1
  171. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/bindings/_rust/openssl/x448.pyi +2 -1
  172. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/bindings/_rust/pkcs12.pyi +9 -3
  173. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/bindings/_rust/pkcs7.pyi +7 -6
  174. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/bindings/_rust/test_support.pyi +2 -1
  175. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/bindings/_rust/x509.pyi +79 -12
  176. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/bindings/openssl/_conditional.py +9 -1
  177. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/bindings/openssl/binding.py +2 -1
  178. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/decrepit/ciphers/algorithms.py +5 -0
  179. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/primitives/_cipheralgorithm.py +4 -2
  180. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/primitives/_serialization.py +1 -2
  181. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/primitives/asymmetric/dh.py +12 -0
  182. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/primitives/asymmetric/dsa.py +16 -3
  183. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/primitives/asymmetric/ec.py +47 -3
  184. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/primitives/asymmetric/ed25519.py +16 -3
  185. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/primitives/asymmetric/ed448.py +16 -3
  186. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/primitives/asymmetric/rsa.py +14 -0
  187. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/primitives/asymmetric/x25519.py +14 -1
  188. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/primitives/asymmetric/x448.py +14 -1
  189. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/primitives/ciphers/algorithms.py +6 -6
  190. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/primitives/ciphers/base.py +5 -4
  191. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/primitives/ciphers/modes.py +18 -18
  192. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/primitives/hashes.py +5 -1
  193. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/primitives/kdf/concatkdf.py +5 -4
  194. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/primitives/kdf/hkdf.py +4 -4
  195. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/primitives/kdf/kbkdf.py +7 -4
  196. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/primitives/kdf/pbkdf2.py +1 -1
  197. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/primitives/kdf/x963kdf.py +1 -1
  198. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/primitives/padding.py +7 -121
  199. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/primitives/serialization/__init__.py +2 -0
  200. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/primitives/serialization/pkcs12.py +21 -1
  201. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/primitives/serialization/pkcs7.py +48 -6
  202. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/primitives/serialization/ssh.py +68 -18
  203. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/primitives/twofactor/hotp.py +3 -2
  204. py2docfx/venv/venv1/Lib/site-packages/cryptography/hazmat/primitives/twofactor/totp.py +2 -1
  205. py2docfx/venv/venv1/Lib/site-packages/cryptography/utils.py +15 -3
  206. py2docfx/venv/venv1/Lib/site-packages/cryptography/x509/__init__.py +3 -0
  207. py2docfx/venv/venv1/Lib/site-packages/cryptography/x509/base.py +39 -6
  208. py2docfx/venv/venv1/Lib/site-packages/cryptography/x509/extensions.py +100 -49
  209. py2docfx/venv/venv1/Lib/site-packages/cryptography/x509/name.py +27 -15
  210. py2docfx/venv/venv1/Lib/site-packages/cryptography/x509/ocsp.py +60 -25
  211. py2docfx/venv/venv1/Lib/site-packages/cryptography/x509/oid.py +2 -0
  212. py2docfx/venv/venv1/Lib/site-packages/cryptography/x509/verification.py +6 -0
  213. py2docfx/venv/venv1/Lib/site-packages/google/api_core/bidi.py +17 -4
  214. py2docfx/venv/venv1/Lib/site-packages/google/api_core/client_info.py +6 -0
  215. py2docfx/venv/venv1/Lib/site-packages/google/api_core/gapic_v1/client_info.py +1 -0
  216. py2docfx/venv/venv1/Lib/site-packages/google/api_core/retry/retry_base.py +13 -4
  217. py2docfx/venv/venv1/Lib/site-packages/google/api_core/retry/retry_streaming.py +7 -6
  218. py2docfx/venv/venv1/Lib/site-packages/google/api_core/retry/retry_streaming_async.py +8 -5
  219. py2docfx/venv/venv1/Lib/site-packages/google/api_core/retry/retry_unary.py +7 -6
  220. py2docfx/venv/venv1/Lib/site-packages/google/api_core/retry/retry_unary_async.py +7 -6
  221. py2docfx/venv/venv1/Lib/site-packages/google/api_core/version.py +1 -1
  222. py2docfx/venv/venv1/Lib/site-packages/google/auth/_default.py +2 -36
  223. py2docfx/venv/venv1/Lib/site-packages/google/auth/_helpers.py +240 -0
  224. py2docfx/venv/venv1/Lib/site-packages/google/auth/aio/_helpers.py +62 -0
  225. py2docfx/venv/venv1/Lib/site-packages/google/auth/aio/transport/aiohttp.py +6 -0
  226. py2docfx/venv/venv1/Lib/site-packages/google/auth/compute_engine/_metadata.py +5 -1
  227. py2docfx/venv/venv1/Lib/site-packages/google/auth/compute_engine/credentials.py +2 -1
  228. py2docfx/venv/venv1/Lib/site-packages/google/auth/identity_pool.py +91 -2
  229. py2docfx/venv/venv1/Lib/site-packages/google/auth/impersonated_credentials.py +75 -0
  230. py2docfx/venv/venv1/Lib/site-packages/google/auth/transport/__init__.py +1 -0
  231. py2docfx/venv/venv1/Lib/site-packages/google/auth/transport/_aiohttp_requests.py +8 -1
  232. py2docfx/venv/venv1/Lib/site-packages/google/auth/transport/_http_client.py +3 -1
  233. py2docfx/venv/venv1/Lib/site-packages/google/auth/transport/requests.py +4 -1
  234. py2docfx/venv/venv1/Lib/site-packages/google/auth/transport/urllib3.py +15 -5
  235. py2docfx/venv/venv1/Lib/site-packages/google/auth/version.py +1 -1
  236. py2docfx/venv/venv1/Lib/site-packages/google/oauth2/id_token.py +12 -0
  237. py2docfx/venv/venv1/Lib/site-packages/google/oauth2/webauthn_types.py +1 -1
  238. py2docfx/venv/venv1/Lib/site-packages/google/protobuf/__init__.py +1 -1
  239. py2docfx/venv/venv1/Lib/site-packages/google/protobuf/any.py +15 -1
  240. py2docfx/venv/venv1/Lib/site-packages/google/protobuf/any_pb2.py +3 -3
  241. py2docfx/venv/venv1/Lib/site-packages/google/protobuf/api_pb2.py +3 -3
  242. py2docfx/venv/venv1/Lib/site-packages/google/protobuf/compiler/plugin_pb2.py +3 -3
  243. py2docfx/venv/venv1/Lib/site-packages/google/protobuf/descriptor.py +15 -2
  244. py2docfx/venv/venv1/Lib/site-packages/google/protobuf/descriptor_pb2.py +258 -113
  245. py2docfx/venv/venv1/Lib/site-packages/google/protobuf/descriptor_pool.py +22 -8
  246. py2docfx/venv/venv1/Lib/site-packages/google/protobuf/duration_pb2.py +3 -3
  247. py2docfx/venv/venv1/Lib/site-packages/google/protobuf/empty_pb2.py +3 -3
  248. py2docfx/venv/venv1/Lib/site-packages/google/protobuf/field_mask_pb2.py +3 -3
  249. py2docfx/venv/venv1/Lib/site-packages/google/protobuf/internal/decoder.py +106 -23
  250. py2docfx/venv/venv1/Lib/site-packages/google/protobuf/internal/field_mask.py +3 -1
  251. py2docfx/venv/venv1/Lib/site-packages/google/protobuf/internal/python_edition_defaults.py +1 -1
  252. py2docfx/venv/venv1/Lib/site-packages/google/protobuf/internal/python_message.py +21 -18
  253. py2docfx/venv/venv1/Lib/site-packages/google/protobuf/internal/testing_refleaks.py +8 -2
  254. py2docfx/venv/venv1/Lib/site-packages/google/protobuf/internal/well_known_types.py +60 -43
  255. py2docfx/venv/venv1/Lib/site-packages/google/protobuf/json_format.py +4 -5
  256. py2docfx/venv/venv1/Lib/site-packages/google/protobuf/message_factory.py +16 -0
  257. py2docfx/venv/venv1/Lib/site-packages/google/protobuf/runtime_version.py +2 -2
  258. py2docfx/venv/venv1/Lib/site-packages/google/protobuf/source_context_pb2.py +3 -3
  259. py2docfx/venv/venv1/Lib/site-packages/google/protobuf/struct_pb2.py +3 -3
  260. py2docfx/venv/venv1/Lib/site-packages/google/protobuf/text_format.py +11 -7
  261. py2docfx/venv/venv1/Lib/site-packages/google/protobuf/timestamp_pb2.py +3 -3
  262. py2docfx/venv/venv1/Lib/site-packages/google/protobuf/type_pb2.py +3 -3
  263. py2docfx/venv/venv1/Lib/site-packages/google/protobuf/wrappers_pb2.py +3 -3
  264. py2docfx/venv/venv1/Lib/site-packages/google/rpc/error_details_pb2.py +29 -23
  265. py2docfx/venv/venv1/Lib/site-packages/google/rpc/error_details_pb2.pyi +41 -2
  266. py2docfx/venv/venv1/Lib/site-packages/msal/application.py +1 -0
  267. py2docfx/venv/venv1/Lib/site-packages/msal/individual_cache.py +9 -5
  268. py2docfx/venv/venv1/Lib/site-packages/msal/managed_identity.py +4 -5
  269. py2docfx/venv/venv1/Lib/site-packages/msal/sku.py +1 -1
  270. py2docfx/venv/venv1/Lib/site-packages/msal/throttled_http_client.py +58 -30
  271. py2docfx/venv/venv1/Lib/site-packages/opencensus/__init__.py +1 -1
  272. py2docfx/venv/venv1/Lib/site-packages/opencensus/ext/__init__.py +1 -1
  273. py2docfx/venv/venv1/Lib/site-packages/opencensus/ext/azure/__init__.py +1 -1
  274. py2docfx/venv/venv1/Lib/site-packages/opencensus/ext/azure/common/__init__.py +138 -138
  275. py2docfx/venv/venv1/Lib/site-packages/opencensus/ext/azure/common/exporter.py +93 -93
  276. py2docfx/venv/venv1/Lib/site-packages/opencensus/ext/azure/common/processor.py +63 -63
  277. py2docfx/venv/venv1/Lib/site-packages/opencensus/ext/azure/common/protocol.py +206 -206
  278. py2docfx/venv/venv1/Lib/site-packages/opencensus/ext/azure/common/storage.py +205 -205
  279. py2docfx/venv/venv1/Lib/site-packages/opencensus/ext/azure/common/transport.py +355 -355
  280. py2docfx/venv/venv1/Lib/site-packages/opencensus/ext/azure/common/utils.py +79 -79
  281. py2docfx/venv/venv1/Lib/site-packages/opencensus/ext/azure/common/version.py +15 -15
  282. py2docfx/venv/venv1/Lib/site-packages/opencensus/ext/azure/log_exporter/__init__.py +314 -314
  283. py2docfx/venv/venv1/Lib/site-packages/opencensus/ext/azure/metrics_exporter/__init__.py +190 -190
  284. py2docfx/venv/venv1/Lib/site-packages/opencensus/ext/azure/metrics_exporter/standard_metrics/__init__.py +62 -62
  285. py2docfx/venv/venv1/Lib/site-packages/opencensus/ext/azure/metrics_exporter/standard_metrics/cpu.py +50 -50
  286. py2docfx/venv/venv1/Lib/site-packages/opencensus/ext/azure/metrics_exporter/standard_metrics/http_requests.py +176 -176
  287. py2docfx/venv/venv1/Lib/site-packages/opencensus/ext/azure/metrics_exporter/standard_metrics/memory.py +42 -42
  288. py2docfx/venv/venv1/Lib/site-packages/opencensus/ext/azure/metrics_exporter/standard_metrics/process.py +87 -87
  289. py2docfx/venv/venv1/Lib/site-packages/opencensus/ext/azure/statsbeat/__init__.py +1 -1
  290. py2docfx/venv/venv1/Lib/site-packages/opencensus/ext/azure/statsbeat/state.py +50 -50
  291. py2docfx/venv/venv1/Lib/site-packages/opencensus/ext/azure/statsbeat/statsbeat.py +100 -100
  292. py2docfx/venv/venv1/Lib/site-packages/opencensus/ext/azure/statsbeat/statsbeat_metrics.py +480 -480
  293. py2docfx/venv/venv1/Lib/site-packages/opencensus/ext/azure/trace_exporter/__init__.py +236 -236
  294. py2docfx/venv/venv1/Lib/site-packages/packaging/__init__.py +1 -1
  295. py2docfx/venv/venv1/Lib/site-packages/packaging/_elffile.py +1 -2
  296. py2docfx/venv/venv1/Lib/site-packages/packaging/_manylinux.py +1 -2
  297. py2docfx/venv/venv1/Lib/site-packages/packaging/_parser.py +1 -2
  298. py2docfx/venv/venv1/Lib/site-packages/packaging/_tokenizer.py +5 -4
  299. py2docfx/venv/venv1/Lib/site-packages/packaging/licenses/__init__.py +1 -1
  300. py2docfx/venv/venv1/Lib/site-packages/packaging/markers.py +53 -22
  301. py2docfx/venv/venv1/Lib/site-packages/packaging/metadata.py +1 -2
  302. py2docfx/venv/venv1/Lib/site-packages/packaging/specifiers.py +1 -2
  303. py2docfx/venv/venv1/Lib/site-packages/packaging/tags.py +39 -0
  304. py2docfx/venv/venv1/Lib/site-packages/pkg_resources/__init__.py +6 -7
  305. py2docfx/venv/venv1/Lib/site-packages/requests/__version__.py +2 -2
  306. py2docfx/venv/venv1/Lib/site-packages/requests/compat.py +12 -0
  307. py2docfx/venv/venv1/Lib/site-packages/requests/models.py +3 -1
  308. py2docfx/venv/venv1/Lib/site-packages/requests/utils.py +6 -16
  309. py2docfx/venv/venv1/Lib/site-packages/rsa/__init__.py +2 -2
  310. py2docfx/venv/venv1/Lib/site-packages/rsa/asn1.py +52 -52
  311. py2docfx/venv/venv1/Lib/site-packages/rsa/cli.py +321 -321
  312. py2docfx/venv/venv1/Lib/site-packages/rsa/common.py +184 -184
  313. py2docfx/venv/venv1/Lib/site-packages/rsa/core.py +53 -53
  314. py2docfx/venv/venv1/Lib/site-packages/rsa/key.py +858 -858
  315. py2docfx/venv/venv1/Lib/site-packages/rsa/parallel.py +96 -96
  316. py2docfx/venv/venv1/Lib/site-packages/rsa/pem.py +134 -134
  317. py2docfx/venv/venv1/Lib/site-packages/rsa/pkcs1.py +485 -485
  318. py2docfx/venv/venv1/Lib/site-packages/rsa/pkcs1_v2.py +100 -100
  319. py2docfx/venv/venv1/Lib/site-packages/rsa/prime.py +198 -198
  320. py2docfx/venv/venv1/Lib/site-packages/rsa/py.typed +1 -1
  321. py2docfx/venv/venv1/Lib/site-packages/rsa/randnum.py +95 -95
  322. py2docfx/venv/venv1/Lib/site-packages/rsa/transform.py +72 -72
  323. py2docfx/venv/venv1/Lib/site-packages/rsa/util.py +97 -97
  324. py2docfx/venv/venv1/Lib/site-packages/setuptools/__init__.py +0 -38
  325. py2docfx/venv/venv1/Lib/site-packages/setuptools/_discovery.py +33 -0
  326. py2docfx/venv/venv1/Lib/site-packages/setuptools/_distutils/command/build_scripts.py +1 -1
  327. py2docfx/venv/venv1/Lib/site-packages/setuptools/_distutils/command/config.py +0 -2
  328. py2docfx/venv/venv1/Lib/site-packages/setuptools/_distutils/compilers/C/tests/test_unix.py +63 -0
  329. py2docfx/venv/venv1/Lib/site-packages/setuptools/_distutils/compilers/C/unix.py +5 -6
  330. py2docfx/venv/venv1/Lib/site-packages/setuptools/_entry_points.py +4 -0
  331. py2docfx/venv/venv1/Lib/site-packages/setuptools/_normalization.py +0 -2
  332. py2docfx/venv/venv1/Lib/site-packages/setuptools/_path.py +12 -3
  333. py2docfx/venv/venv1/Lib/site-packages/setuptools/_reqs.py +1 -1
  334. py2docfx/venv/venv1/Lib/site-packages/setuptools/_scripts.py +361 -0
  335. py2docfx/venv/venv1/Lib/site-packages/setuptools/_shutil.py +6 -0
  336. py2docfx/venv/venv1/Lib/site-packages/setuptools/build_meta.py +25 -37
  337. py2docfx/venv/venv1/Lib/site-packages/setuptools/command/bdist_egg.py +9 -11
  338. py2docfx/venv/venv1/Lib/site-packages/setuptools/command/bdist_wheel.py +1 -1
  339. py2docfx/venv/venv1/Lib/site-packages/setuptools/command/build_ext.py +29 -28
  340. py2docfx/venv/venv1/Lib/site-packages/setuptools/command/develop.py +39 -179
  341. py2docfx/venv/venv1/Lib/site-packages/setuptools/command/easy_install.py +17 -2352
  342. py2docfx/venv/venv1/Lib/site-packages/setuptools/command/editable_wheel.py +14 -31
  343. py2docfx/venv/venv1/Lib/site-packages/setuptools/command/egg_info.py +0 -2
  344. py2docfx/venv/venv1/Lib/site-packages/setuptools/command/install.py +3 -55
  345. py2docfx/venv/venv1/Lib/site-packages/setuptools/command/install_scripts.py +5 -11
  346. py2docfx/venv/venv1/Lib/site-packages/setuptools/command/sdist.py +1 -1
  347. py2docfx/venv/venv1/Lib/site-packages/setuptools/compat/py310.py +11 -0
  348. py2docfx/venv/venv1/Lib/site-packages/setuptools/dist.py +10 -12
  349. py2docfx/venv/venv1/Lib/site-packages/setuptools/installer.py +34 -29
  350. py2docfx/venv/venv1/Lib/site-packages/setuptools/tests/contexts.py +0 -14
  351. py2docfx/venv/venv1/Lib/site-packages/setuptools/tests/fixtures.py +235 -0
  352. py2docfx/venv/venv1/Lib/site-packages/setuptools/tests/integration/test_pbr.py +20 -0
  353. py2docfx/venv/venv1/Lib/site-packages/setuptools/tests/test_build_meta.py +0 -24
  354. py2docfx/venv/venv1/Lib/site-packages/setuptools/tests/test_develop.py +1 -64
  355. py2docfx/venv/venv1/Lib/site-packages/setuptools/tests/test_dist.py +2 -2
  356. py2docfx/venv/venv1/Lib/site-packages/setuptools/tests/test_dist_info.py +0 -63
  357. py2docfx/venv/venv1/Lib/site-packages/setuptools/tests/test_editable_install.py +15 -41
  358. py2docfx/venv/venv1/Lib/site-packages/setuptools/tests/test_egg_info.py +3 -2
  359. py2docfx/venv/venv1/Lib/site-packages/setuptools/tests/test_scripts.py +12 -0
  360. py2docfx/venv/venv1/Lib/site-packages/setuptools/tests/test_wheel.py +12 -36
  361. py2docfx/venv/venv1/Lib/site-packages/setuptools/tests/test_windows_wrappers.py +3 -4
  362. py2docfx/venv/venv1/Lib/site-packages/setuptools/wheel.py +49 -24
  363. py2docfx/venv/venv1/Lib/site-packages/typing_extensions.py +357 -703
  364. py2docfx/venv/venv1/Lib/site-packages/urllib3/_version.py +2 -2
  365. py2docfx/venv/venv1/Lib/site-packages/urllib3/connection.py +87 -38
  366. py2docfx/venv/venv1/Lib/site-packages/urllib3/contrib/emscripten/fetch.py +20 -0
  367. py2docfx/venv/venv1/Lib/site-packages/urllib3/poolmanager.py +17 -1
  368. py2docfx/venv/venv1/Lib/site-packages/urllib3/response.py +53 -24
  369. py2docfx/venv/venv1/Lib/site-packages/urllib3/util/request.py +12 -4
  370. py2docfx/venv/venv1/Lib/site-packages/urllib3/util/ssl_.py +1 -1
  371. {py2docfx-0.1.16.dev2064350.dist-info → py2docfx-0.1.17.dist-info}/METADATA +1 -1
  372. {py2docfx-0.1.16.dev2064350.dist-info → py2docfx-0.1.17.dist-info}/RECORD +374 -372
  373. {py2docfx-0.1.16.dev2064350.dist-info → py2docfx-0.1.17.dist-info}/WHEEL +1 -1
  374. py2docfx/venv/basevenv/Lib/site-packages/setuptools/package_index.py +0 -1137
  375. py2docfx/venv/basevenv/Lib/site-packages/setuptools/sandbox.py +0 -536
  376. py2docfx/venv/basevenv/Lib/site-packages/setuptools/tests/server.py +0 -86
  377. py2docfx/venv/basevenv/Lib/site-packages/setuptools/tests/test_easy_install.py +0 -1476
  378. py2docfx/venv/basevenv/Lib/site-packages/setuptools/tests/test_packageindex.py +0 -267
  379. py2docfx/venv/basevenv/Lib/site-packages/setuptools/tests/test_sandbox.py +0 -134
  380. py2docfx/venv/venv1/Lib/site-packages/azure/identity/_credentials/application.py +0 -119
  381. py2docfx/venv/venv1/Lib/site-packages/azure/identity/aio/_credentials/application.py +0 -121
  382. py2docfx/venv/venv1/Lib/site-packages/setuptools/package_index.py +0 -1137
  383. py2docfx/venv/venv1/Lib/site-packages/setuptools/sandbox.py +0 -536
  384. py2docfx/venv/venv1/Lib/site-packages/setuptools/tests/server.py +0 -86
  385. py2docfx/venv/venv1/Lib/site-packages/setuptools/tests/test_easy_install.py +0 -1476
  386. py2docfx/venv/venv1/Lib/site-packages/setuptools/tests/test_packageindex.py +0 -267
  387. py2docfx/venv/venv1/Lib/site-packages/setuptools/tests/test_sandbox.py +0 -134
  388. {py2docfx-0.1.16.dev2064350.dist-info → py2docfx-0.1.17.dist-info}/top_level.txt +0 -0
@@ -1,1137 +0,0 @@
1
- """PyPI and direct package downloading."""
2
-
3
- from __future__ import annotations
4
-
5
- import base64
6
- import configparser
7
- import hashlib
8
- import html
9
- import http.client
10
- import io
11
- import itertools
12
- import os
13
- import re
14
- import shutil
15
- import socket
16
- import subprocess
17
- import sys
18
- import urllib.error
19
- import urllib.parse
20
- import urllib.request
21
- from fnmatch import translate
22
- from functools import wraps
23
- from typing import NamedTuple
24
-
25
- from more_itertools import unique_everseen
26
-
27
- import setuptools
28
- from pkg_resources import (
29
- BINARY_DIST,
30
- CHECKOUT_DIST,
31
- DEVELOP_DIST,
32
- EGG_DIST,
33
- SOURCE_DIST,
34
- Distribution,
35
- Environment,
36
- Requirement,
37
- find_distributions,
38
- normalize_path,
39
- parse_version,
40
- safe_name,
41
- safe_version,
42
- to_filename,
43
- )
44
- from setuptools.wheel import Wheel
45
-
46
- from .unicode_utils import _cfg_read_utf8_with_fallback, _read_utf8_with_fallback
47
-
48
- from distutils import log
49
- from distutils.errors import DistutilsError
50
-
51
- EGG_FRAGMENT = re.compile(r'^egg=([-A-Za-z0-9_.+!]+)$')
52
- HREF = re.compile(r"""href\s*=\s*['"]?([^'"> ]+)""", re.I)
53
- PYPI_MD5 = re.compile(
54
- r'<a href="([^"#]+)">([^<]+)</a>\n\s+\(<a (?:title="MD5 hash"\n\s+)'
55
- r'href="[^?]+\?:action=show_md5&amp;digest=([0-9a-f]{32})">md5</a>\)'
56
- )
57
- URL_SCHEME = re.compile('([-+.a-z0-9]{2,}):', re.I).match
58
- EXTENSIONS = ".tar.gz .tar.bz2 .tar .zip .tgz".split()
59
-
60
- __all__ = [
61
- 'PackageIndex',
62
- 'distros_for_url',
63
- 'parse_bdist_wininst',
64
- 'interpret_distro_name',
65
- ]
66
-
67
- _SOCKET_TIMEOUT = 15
68
-
69
- user_agent = f"setuptools/{setuptools.__version__} Python-urllib/{sys.version_info.major}.{sys.version_info.minor}"
70
-
71
-
72
- def parse_requirement_arg(spec):
73
- try:
74
- return Requirement.parse(spec)
75
- except ValueError as e:
76
- raise DistutilsError(
77
- f"Not a URL, existing file, or requirement spec: {spec!r}"
78
- ) from e
79
-
80
-
81
- def parse_bdist_wininst(name):
82
- """Return (base,pyversion) or (None,None) for possible .exe name"""
83
-
84
- lower = name.lower()
85
- base, py_ver, plat = None, None, None
86
-
87
- if lower.endswith('.exe'):
88
- if lower.endswith('.win32.exe'):
89
- base = name[:-10]
90
- plat = 'win32'
91
- elif lower.startswith('.win32-py', -16):
92
- py_ver = name[-7:-4]
93
- base = name[:-16]
94
- plat = 'win32'
95
- elif lower.endswith('.win-amd64.exe'):
96
- base = name[:-14]
97
- plat = 'win-amd64'
98
- elif lower.startswith('.win-amd64-py', -20):
99
- py_ver = name[-7:-4]
100
- base = name[:-20]
101
- plat = 'win-amd64'
102
- return base, py_ver, plat
103
-
104
-
105
- def egg_info_for_url(url):
106
- parts = urllib.parse.urlparse(url)
107
- _scheme, server, path, _parameters, _query, fragment = parts
108
- base = urllib.parse.unquote(path.split('/')[-1])
109
- if server == 'sourceforge.net' and base == 'download': # XXX Yuck
110
- base = urllib.parse.unquote(path.split('/')[-2])
111
- if '#' in base:
112
- base, fragment = base.split('#', 1)
113
- return base, fragment
114
-
115
-
116
- def distros_for_url(url, metadata=None):
117
- """Yield egg or source distribution objects that might be found at a URL"""
118
- base, fragment = egg_info_for_url(url)
119
- yield from distros_for_location(url, base, metadata)
120
- if fragment:
121
- match = EGG_FRAGMENT.match(fragment)
122
- if match:
123
- yield from interpret_distro_name(
124
- url, match.group(1), metadata, precedence=CHECKOUT_DIST
125
- )
126
-
127
-
128
- def distros_for_location(location, basename, metadata=None):
129
- """Yield egg or source distribution objects based on basename"""
130
- if basename.endswith('.egg.zip'):
131
- basename = basename[:-4] # strip the .zip
132
- if basename.endswith('.egg') and '-' in basename:
133
- # only one, unambiguous interpretation
134
- return [Distribution.from_location(location, basename, metadata)]
135
- if basename.endswith('.whl') and '-' in basename:
136
- wheel = Wheel(basename)
137
- if not wheel.is_compatible():
138
- return []
139
- return [
140
- Distribution(
141
- location=location,
142
- project_name=wheel.project_name,
143
- version=wheel.version,
144
- # Increase priority over eggs.
145
- precedence=EGG_DIST + 1,
146
- )
147
- ]
148
- if basename.endswith('.exe'):
149
- win_base, py_ver, platform = parse_bdist_wininst(basename)
150
- if win_base is not None:
151
- return interpret_distro_name(
152
- location, win_base, metadata, py_ver, BINARY_DIST, platform
153
- )
154
- # Try source distro extensions (.zip, .tgz, etc.)
155
- #
156
- for ext in EXTENSIONS:
157
- if basename.endswith(ext):
158
- basename = basename[: -len(ext)]
159
- return interpret_distro_name(location, basename, metadata)
160
- return [] # no extension matched
161
-
162
-
163
- def distros_for_filename(filename, metadata=None):
164
- """Yield possible egg or source distribution objects based on a filename"""
165
- return distros_for_location(
166
- normalize_path(filename), os.path.basename(filename), metadata
167
- )
168
-
169
-
170
- def interpret_distro_name(
171
- location, basename, metadata, py_version=None, precedence=SOURCE_DIST, platform=None
172
- ):
173
- """Generate the interpretation of a source distro name
174
-
175
- Note: if `location` is a filesystem filename, you should call
176
- ``pkg_resources.normalize_path()`` on it before passing it to this
177
- routine!
178
- """
179
-
180
- parts = basename.split('-')
181
- if not py_version and any(re.match(r'py\d\.\d$', p) for p in parts[2:]):
182
- # it is a bdist_dumb, not an sdist -- bail out
183
- return
184
-
185
- # find the pivot (p) that splits the name from the version.
186
- # infer the version as the first item that has a digit.
187
- for p in range(len(parts)):
188
- if parts[p][:1].isdigit():
189
- break
190
- else:
191
- p = len(parts)
192
-
193
- yield Distribution(
194
- location,
195
- metadata,
196
- '-'.join(parts[:p]),
197
- '-'.join(parts[p:]),
198
- py_version=py_version,
199
- precedence=precedence,
200
- platform=platform,
201
- )
202
-
203
-
204
- def unique_values(func):
205
- """
206
- Wrap a function returning an iterable such that the resulting iterable
207
- only ever yields unique items.
208
- """
209
-
210
- @wraps(func)
211
- def wrapper(*args, **kwargs):
212
- return unique_everseen(func(*args, **kwargs))
213
-
214
- return wrapper
215
-
216
-
217
- REL = re.compile(r"""<([^>]*\srel\s{0,10}=\s{0,10}['"]?([^'" >]+)[^>]*)>""", re.I)
218
- """
219
- Regex for an HTML tag with 'rel="val"' attributes.
220
- """
221
-
222
-
223
- @unique_values
224
- def find_external_links(url, page):
225
- """Find rel="homepage" and rel="download" links in `page`, yielding URLs"""
226
-
227
- for match in REL.finditer(page):
228
- tag, rel = match.groups()
229
- rels = set(map(str.strip, rel.lower().split(',')))
230
- if 'homepage' in rels or 'download' in rels:
231
- for match in HREF.finditer(tag):
232
- yield urllib.parse.urljoin(url, htmldecode(match.group(1)))
233
-
234
- for tag in ("<th>Home Page", "<th>Download URL"):
235
- pos = page.find(tag)
236
- if pos != -1:
237
- match = HREF.search(page, pos)
238
- if match:
239
- yield urllib.parse.urljoin(url, htmldecode(match.group(1)))
240
-
241
-
242
- class ContentChecker:
243
- """
244
- A null content checker that defines the interface for checking content
245
- """
246
-
247
- def feed(self, block):
248
- """
249
- Feed a block of data to the hash.
250
- """
251
- return
252
-
253
- def is_valid(self):
254
- """
255
- Check the hash. Return False if validation fails.
256
- """
257
- return True
258
-
259
- def report(self, reporter, template):
260
- """
261
- Call reporter with information about the checker (hash name)
262
- substituted into the template.
263
- """
264
- return
265
-
266
-
267
- class HashChecker(ContentChecker):
268
- pattern = re.compile(
269
- r'(?P<hash_name>sha1|sha224|sha384|sha256|sha512|md5)='
270
- r'(?P<expected>[a-f0-9]+)'
271
- )
272
-
273
- def __init__(self, hash_name, expected) -> None:
274
- self.hash_name = hash_name
275
- self.hash = hashlib.new(hash_name)
276
- self.expected = expected
277
-
278
- @classmethod
279
- def from_url(cls, url):
280
- "Construct a (possibly null) ContentChecker from a URL"
281
- fragment = urllib.parse.urlparse(url)[-1]
282
- if not fragment:
283
- return ContentChecker()
284
- match = cls.pattern.search(fragment)
285
- if not match:
286
- return ContentChecker()
287
- return cls(**match.groupdict())
288
-
289
- def feed(self, block):
290
- self.hash.update(block)
291
-
292
- def is_valid(self):
293
- return self.hash.hexdigest() == self.expected
294
-
295
- def report(self, reporter, template):
296
- msg = template % self.hash_name
297
- return reporter(msg)
298
-
299
-
300
- class PackageIndex(Environment):
301
- """A distribution index that scans web pages for download URLs"""
302
-
303
- def __init__(
304
- self,
305
- index_url: str = "https://pypi.org/simple/",
306
- hosts=('*',),
307
- ca_bundle=None,
308
- verify_ssl: bool = True,
309
- *args,
310
- **kw,
311
- ) -> None:
312
- super().__init__(*args, **kw)
313
- self.index_url = index_url + "/"[: not index_url.endswith('/')]
314
- self.scanned_urls: dict = {}
315
- self.fetched_urls: dict = {}
316
- self.package_pages: dict = {}
317
- self.allows = re.compile('|'.join(map(translate, hosts))).match
318
- self.to_scan: list = []
319
- self.opener = urllib.request.urlopen
320
-
321
- def add(self, dist):
322
- # ignore invalid versions
323
- try:
324
- parse_version(dist.version)
325
- except Exception:
326
- return None
327
- return super().add(dist)
328
-
329
- # FIXME: 'PackageIndex.process_url' is too complex (14)
330
- def process_url(self, url, retrieve: bool = False) -> None: # noqa: C901
331
- """Evaluate a URL as a possible download, and maybe retrieve it"""
332
- if url in self.scanned_urls and not retrieve:
333
- return
334
- self.scanned_urls[url] = True
335
- if not URL_SCHEME(url):
336
- self.process_filename(url)
337
- return
338
- else:
339
- dists = list(distros_for_url(url))
340
- if dists:
341
- if not self.url_ok(url):
342
- return
343
- self.debug("Found link: %s", url)
344
-
345
- if dists or not retrieve or url in self.fetched_urls:
346
- list(map(self.add, dists))
347
- return # don't need the actual page
348
-
349
- if not self.url_ok(url):
350
- self.fetched_urls[url] = True
351
- return
352
-
353
- self.info("Reading %s", url)
354
- self.fetched_urls[url] = True # prevent multiple fetch attempts
355
- tmpl = "Download error on %s: %%s -- Some packages may not be found!"
356
- f = self.open_url(url, tmpl % url)
357
- if f is None:
358
- return
359
- if isinstance(f, urllib.error.HTTPError) and f.code == 401:
360
- self.info(f"Authentication error: {f.msg}")
361
- self.fetched_urls[f.url] = True
362
- if 'html' not in f.headers.get('content-type', '').lower():
363
- f.close() # not html, we can't process it
364
- return
365
-
366
- base = f.url # handle redirects
367
- page = f.read()
368
- if not isinstance(page, str):
369
- # In Python 3 and got bytes but want str.
370
- if isinstance(f, urllib.error.HTTPError):
371
- # Errors have no charset, assume latin1:
372
- charset = 'latin-1'
373
- else:
374
- charset = f.headers.get_param('charset') or 'latin-1'
375
- page = page.decode(charset, "ignore")
376
- f.close()
377
- for match in HREF.finditer(page):
378
- link = urllib.parse.urljoin(base, htmldecode(match.group(1)))
379
- self.process_url(link)
380
- if url.startswith(self.index_url) and getattr(f, 'code', None) != 404:
381
- page = self.process_index(url, page)
382
-
383
- def process_filename(self, fn, nested: bool = False) -> None:
384
- # process filenames or directories
385
- if not os.path.exists(fn):
386
- self.warn("Not found: %s", fn)
387
- return
388
-
389
- if os.path.isdir(fn) and not nested:
390
- path = os.path.realpath(fn)
391
- for item in os.listdir(path):
392
- self.process_filename(os.path.join(path, item), True)
393
-
394
- dists = distros_for_filename(fn)
395
- if dists:
396
- self.debug("Found: %s", fn)
397
- list(map(self.add, dists))
398
-
399
- def url_ok(self, url, fatal: bool = False) -> bool:
400
- s = URL_SCHEME(url)
401
- is_file = s and s.group(1).lower() == 'file'
402
- if is_file or self.allows(urllib.parse.urlparse(url)[1]):
403
- return True
404
- msg = (
405
- "\nNote: Bypassing %s (disallowed host; see "
406
- "https://setuptools.pypa.io/en/latest/deprecated/"
407
- "easy_install.html#restricting-downloads-with-allow-hosts for details).\n"
408
- )
409
- if fatal:
410
- raise DistutilsError(msg % url)
411
- else:
412
- self.warn(msg, url)
413
- return False
414
-
415
- def scan_egg_links(self, search_path) -> None:
416
- dirs = filter(os.path.isdir, search_path)
417
- egg_links = (
418
- (path, entry)
419
- for path in dirs
420
- for entry in os.listdir(path)
421
- if entry.endswith('.egg-link')
422
- )
423
- list(itertools.starmap(self.scan_egg_link, egg_links))
424
-
425
- def scan_egg_link(self, path, entry) -> None:
426
- content = _read_utf8_with_fallback(os.path.join(path, entry))
427
- # filter non-empty lines
428
- lines = list(filter(None, map(str.strip, content.splitlines())))
429
-
430
- if len(lines) != 2:
431
- # format is not recognized; punt
432
- return
433
-
434
- egg_path, _setup_path = lines
435
-
436
- for dist in find_distributions(os.path.join(path, egg_path)):
437
- dist.location = os.path.join(path, *lines)
438
- dist.precedence = SOURCE_DIST
439
- self.add(dist)
440
-
441
- def _scan(self, link):
442
- # Process a URL to see if it's for a package page
443
- NO_MATCH_SENTINEL = None, None
444
- if not link.startswith(self.index_url):
445
- return NO_MATCH_SENTINEL
446
-
447
- parts = list(map(urllib.parse.unquote, link[len(self.index_url) :].split('/')))
448
- if len(parts) != 2 or '#' in parts[1]:
449
- return NO_MATCH_SENTINEL
450
-
451
- # it's a package page, sanitize and index it
452
- pkg = safe_name(parts[0])
453
- ver = safe_version(parts[1])
454
- self.package_pages.setdefault(pkg.lower(), {})[link] = True
455
- return to_filename(pkg), to_filename(ver)
456
-
457
- def process_index(self, url, page):
458
- """Process the contents of a PyPI page"""
459
-
460
- # process an index page into the package-page index
461
- for match in HREF.finditer(page):
462
- try:
463
- self._scan(urllib.parse.urljoin(url, htmldecode(match.group(1))))
464
- except ValueError:
465
- pass
466
-
467
- pkg, ver = self._scan(url) # ensure this page is in the page index
468
- if not pkg:
469
- return "" # no sense double-scanning non-package pages
470
-
471
- # process individual package page
472
- for new_url in find_external_links(url, page):
473
- # Process the found URL
474
- base, frag = egg_info_for_url(new_url)
475
- if base.endswith('.py') and not frag:
476
- if ver:
477
- new_url += f'#egg={pkg}-{ver}'
478
- else:
479
- self.need_version_info(url)
480
- self.scan_url(new_url)
481
-
482
- return PYPI_MD5.sub(
483
- lambda m: '<a href="{}#md5={}">{}</a>'.format(*m.group(1, 3, 2)), page
484
- )
485
-
486
- def need_version_info(self, url) -> None:
487
- self.scan_all(
488
- "Page at %s links to .py file(s) without version info; an index "
489
- "scan is required.",
490
- url,
491
- )
492
-
493
- def scan_all(self, msg=None, *args) -> None:
494
- if self.index_url not in self.fetched_urls:
495
- if msg:
496
- self.warn(msg, *args)
497
- self.info("Scanning index of all packages (this may take a while)")
498
- self.scan_url(self.index_url)
499
-
500
- def find_packages(self, requirement) -> None:
501
- self.scan_url(self.index_url + requirement.unsafe_name + '/')
502
-
503
- if not self.package_pages.get(requirement.key):
504
- # Fall back to safe version of the name
505
- self.scan_url(self.index_url + requirement.project_name + '/')
506
-
507
- if not self.package_pages.get(requirement.key):
508
- # We couldn't find the target package, so search the index page too
509
- self.not_found_in_index(requirement)
510
-
511
- for url in list(self.package_pages.get(requirement.key, ())):
512
- # scan each page that might be related to the desired package
513
- self.scan_url(url)
514
-
515
- def obtain(self, requirement, installer=None):
516
- self.prescan()
517
- self.find_packages(requirement)
518
- for dist in self[requirement.key]:
519
- if dist in requirement:
520
- return dist
521
- self.debug("%s does not match %s", requirement, dist)
522
- return super().obtain(requirement, installer)
523
-
524
- def check_hash(self, checker, filename, tfp) -> None:
525
- """
526
- checker is a ContentChecker
527
- """
528
- checker.report(self.debug, f"Validating %s checksum for {filename}")
529
- if not checker.is_valid():
530
- tfp.close()
531
- os.unlink(filename)
532
- raise DistutilsError(
533
- f"{checker.hash.name} validation failed for {os.path.basename(filename)}; "
534
- "possible download problem?"
535
- )
536
-
537
- def add_find_links(self, urls) -> None:
538
- """Add `urls` to the list that will be prescanned for searches"""
539
- for url in urls:
540
- if (
541
- self.to_scan is None # if we have already "gone online"
542
- or not URL_SCHEME(url) # or it's a local file/directory
543
- or url.startswith('file:')
544
- or list(distros_for_url(url)) # or a direct package link
545
- ):
546
- # then go ahead and process it now
547
- self.scan_url(url)
548
- else:
549
- # otherwise, defer retrieval till later
550
- self.to_scan.append(url)
551
-
552
- def prescan(self):
553
- """Scan urls scheduled for prescanning (e.g. --find-links)"""
554
- if self.to_scan:
555
- list(map(self.scan_url, self.to_scan))
556
- self.to_scan = None # from now on, go ahead and process immediately
557
-
558
- def not_found_in_index(self, requirement) -> None:
559
- if self[requirement.key]: # we've seen at least one distro
560
- meth, msg = self.info, "Couldn't retrieve index page for %r"
561
- else: # no distros seen for this name, might be misspelled
562
- meth, msg = self.warn, "Couldn't find index page for %r (maybe misspelled?)"
563
- meth(msg, requirement.unsafe_name)
564
- self.scan_all()
565
-
566
- def download(self, spec, tmpdir):
567
- """Locate and/or download `spec` to `tmpdir`, returning a local path
568
-
569
- `spec` may be a ``Requirement`` object, or a string containing a URL,
570
- an existing local filename, or a project/version requirement spec
571
- (i.e. the string form of a ``Requirement`` object). If it is the URL
572
- of a .py file with an unambiguous ``#egg=name-version`` tag (i.e., one
573
- that escapes ``-`` as ``_`` throughout), a trivial ``setup.py`` is
574
- automatically created alongside the downloaded file.
575
-
576
- If `spec` is a ``Requirement`` object or a string containing a
577
- project/version requirement spec, this method returns the location of
578
- a matching distribution (possibly after downloading it to `tmpdir`).
579
- If `spec` is a locally existing file or directory name, it is simply
580
- returned unchanged. If `spec` is a URL, it is downloaded to a subpath
581
- of `tmpdir`, and the local filename is returned. Various errors may be
582
- raised if a problem occurs during downloading.
583
- """
584
- if not isinstance(spec, Requirement):
585
- scheme = URL_SCHEME(spec)
586
- if scheme:
587
- # It's a url, download it to tmpdir
588
- found = self._download_url(spec, tmpdir)
589
- base, fragment = egg_info_for_url(spec)
590
- if base.endswith('.py'):
591
- found = self.gen_setup(found, fragment, tmpdir)
592
- return found
593
- elif os.path.exists(spec):
594
- # Existing file or directory, just return it
595
- return spec
596
- else:
597
- spec = parse_requirement_arg(spec)
598
- return getattr(self.fetch_distribution(spec, tmpdir), 'location', None)
599
-
600
- def fetch_distribution( # noqa: C901 # is too complex (14) # FIXME
601
- self,
602
- requirement,
603
- tmpdir,
604
- force_scan: bool = False,
605
- source: bool = False,
606
- develop_ok: bool = False,
607
- local_index=None,
608
- ) -> Distribution | None:
609
- """Obtain a distribution suitable for fulfilling `requirement`
610
-
611
- `requirement` must be a ``pkg_resources.Requirement`` instance.
612
- If necessary, or if the `force_scan` flag is set, the requirement is
613
- searched for in the (online) package index as well as the locally
614
- installed packages. If a distribution matching `requirement` is found,
615
- the returned distribution's ``location`` is the value you would have
616
- gotten from calling the ``download()`` method with the matching
617
- distribution's URL or filename. If no matching distribution is found,
618
- ``None`` is returned.
619
-
620
- If the `source` flag is set, only source distributions and source
621
- checkout links will be considered. Unless the `develop_ok` flag is
622
- set, development and system eggs (i.e., those using the ``.egg-info``
623
- format) will be ignored.
624
- """
625
- # process a Requirement
626
- self.info("Searching for %s", requirement)
627
- skipped = set()
628
- dist = None
629
-
630
- def find(req, env: Environment | None = None):
631
- if env is None:
632
- env = self
633
- # Find a matching distribution; may be called more than once
634
-
635
- for dist in env[req.key]:
636
- if dist.precedence == DEVELOP_DIST and not develop_ok:
637
- if dist not in skipped:
638
- self.warn(
639
- "Skipping development or system egg: %s",
640
- dist,
641
- )
642
- skipped.add(dist)
643
- continue
644
-
645
- test = dist in req and (dist.precedence <= SOURCE_DIST or not source)
646
- if test:
647
- loc = self.download(dist.location, tmpdir)
648
- dist.download_location = loc
649
- if os.path.exists(dist.download_location):
650
- return dist
651
-
652
- return None
653
-
654
- if force_scan:
655
- self.prescan()
656
- self.find_packages(requirement)
657
- dist = find(requirement)
658
-
659
- if not dist and local_index is not None:
660
- dist = find(requirement, local_index)
661
-
662
- if dist is None:
663
- if self.to_scan is not None:
664
- self.prescan()
665
- dist = find(requirement)
666
-
667
- if dist is None and not force_scan:
668
- self.find_packages(requirement)
669
- dist = find(requirement)
670
-
671
- if dist is None:
672
- self.warn(
673
- "No local packages or working download links found for %s%s",
674
- (source and "a source distribution of " or ""),
675
- requirement,
676
- )
677
- return None
678
- else:
679
- self.info("Best match: %s", dist)
680
- return dist.clone(location=dist.download_location)
681
-
682
- def fetch(
683
- self, requirement, tmpdir, force_scan: bool = False, source: bool = False
684
- ) -> str | None:
685
- """Obtain a file suitable for fulfilling `requirement`
686
-
687
- DEPRECATED; use the ``fetch_distribution()`` method now instead. For
688
- backward compatibility, this routine is identical but returns the
689
- ``location`` of the downloaded distribution instead of a distribution
690
- object.
691
- """
692
- dist = self.fetch_distribution(requirement, tmpdir, force_scan, source)
693
- if dist is not None:
694
- return dist.location
695
- return None
696
-
697
- def gen_setup(self, filename, fragment, tmpdir):
698
- match = EGG_FRAGMENT.match(fragment)
699
- dists = (
700
- match
701
- and [
702
- d
703
- for d in interpret_distro_name(filename, match.group(1), None)
704
- if d.version
705
- ]
706
- or []
707
- )
708
-
709
- if len(dists) == 1: # unambiguous ``#egg`` fragment
710
- basename = os.path.basename(filename)
711
-
712
- # Make sure the file has been downloaded to the temp dir.
713
- if os.path.dirname(filename) != tmpdir:
714
- dst = os.path.join(tmpdir, basename)
715
- if not (os.path.exists(dst) and os.path.samefile(filename, dst)):
716
- shutil.copy2(filename, dst)
717
- filename = dst
718
-
719
- with open(os.path.join(tmpdir, 'setup.py'), 'w', encoding="utf-8") as file:
720
- file.write(
721
- "from setuptools import setup\n"
722
- f"setup(name={dists[0].project_name!r}, version={dists[0].version!r}, py_modules=[{os.path.splitext(basename)[0]!r}])\n"
723
- )
724
- return filename
725
-
726
- elif match:
727
- raise DistutilsError(
728
- f"Can't unambiguously interpret project/version identifier {fragment!r}; "
729
- "any dashes in the name or version should be escaped using "
730
- f"underscores. {dists!r}"
731
- )
732
- else:
733
- raise DistutilsError(
734
- "Can't process plain .py files without an '#egg=name-version'"
735
- " suffix to enable automatic setup script generation."
736
- )
737
-
738
- dl_blocksize = 8192
739
-
740
- def _download_to(self, url, filename):
741
- self.info("Downloading %s", url)
742
- # Download the file
743
- fp = None
744
- try:
745
- checker = HashChecker.from_url(url)
746
- fp = self.open_url(url)
747
- if isinstance(fp, urllib.error.HTTPError):
748
- raise DistutilsError(f"Can't download {url}: {fp.code} {fp.msg}")
749
- headers = fp.info()
750
- blocknum = 0
751
- bs = self.dl_blocksize
752
- size = -1
753
- if "content-length" in headers:
754
- # Some servers return multiple Content-Length headers :(
755
- sizes = headers.get_all('Content-Length')
756
- size = max(map(int, sizes))
757
- self.reporthook(url, filename, blocknum, bs, size)
758
- with open(filename, 'wb') as tfp:
759
- while True:
760
- block = fp.read(bs)
761
- if block:
762
- checker.feed(block)
763
- tfp.write(block)
764
- blocknum += 1
765
- self.reporthook(url, filename, blocknum, bs, size)
766
- else:
767
- break
768
- self.check_hash(checker, filename, tfp)
769
- return headers
770
- finally:
771
- if fp:
772
- fp.close()
773
-
774
- def reporthook(self, url, filename, blocknum, blksize, size) -> None:
775
- pass # no-op
776
-
777
- # FIXME:
778
- def open_url(self, url, warning=None): # noqa: C901 # is too complex (12)
779
- if url.startswith('file:'):
780
- return local_open(url)
781
- try:
782
- return open_with_auth(url, self.opener)
783
- except (ValueError, http.client.InvalidURL) as v:
784
- msg = ' '.join([str(arg) for arg in v.args])
785
- if warning:
786
- self.warn(warning, msg)
787
- else:
788
- raise DistutilsError(f'{url} {msg}') from v
789
- except urllib.error.HTTPError as v:
790
- return v
791
- except urllib.error.URLError as v:
792
- if warning:
793
- self.warn(warning, v.reason)
794
- else:
795
- raise DistutilsError(f"Download error for {url}: {v.reason}") from v
796
- except http.client.BadStatusLine as v:
797
- if warning:
798
- self.warn(warning, v.line)
799
- else:
800
- raise DistutilsError(
801
- f'{url} returned a bad status line. The server might be '
802
- f'down, {v.line}'
803
- ) from v
804
- except (http.client.HTTPException, OSError) as v:
805
- if warning:
806
- self.warn(warning, v)
807
- else:
808
- raise DistutilsError(f"Download error for {url}: {v}") from v
809
-
810
- def _download_url(self, url, tmpdir):
811
- # Determine download filename
812
- #
813
- name, _fragment = egg_info_for_url(url)
814
- if name:
815
- while '..' in name:
816
- name = name.replace('..', '.').replace('\\', '_')
817
- else:
818
- name = "__downloaded__" # default if URL has no path contents
819
-
820
- if name.endswith('.egg.zip'):
821
- name = name[:-4] # strip the extra .zip before download
822
-
823
- filename = os.path.join(tmpdir, name)
824
-
825
- return self._download_vcs(url, filename) or self._download_other(url, filename)
826
-
827
- @staticmethod
828
- def _resolve_vcs(url):
829
- """
830
- >>> rvcs = PackageIndex._resolve_vcs
831
- >>> rvcs('git+http://foo/bar')
832
- 'git'
833
- >>> rvcs('hg+https://foo/bar')
834
- 'hg'
835
- >>> rvcs('git:myhost')
836
- 'git'
837
- >>> rvcs('hg:myhost')
838
- >>> rvcs('http://foo/bar')
839
- """
840
- scheme = urllib.parse.urlsplit(url).scheme
841
- pre, sep, _post = scheme.partition('+')
842
- # svn and git have their own protocol; hg does not
843
- allowed = set(['svn', 'git'] + ['hg'] * bool(sep))
844
- return next(iter({pre} & allowed), None)
845
-
846
- def _download_vcs(self, url, spec_filename):
847
- vcs = self._resolve_vcs(url)
848
- if not vcs:
849
- return None
850
- if vcs == 'svn':
851
- raise DistutilsError(
852
- f"Invalid config, SVN download is not supported: {url}"
853
- )
854
-
855
- filename, _, _ = spec_filename.partition('#')
856
- url, rev = self._vcs_split_rev_from_url(url)
857
-
858
- self.info(f"Doing {vcs} clone from {url} to {filename}")
859
- subprocess.check_call([vcs, 'clone', '--quiet', url, filename])
860
-
861
- co_commands = dict(
862
- git=[vcs, '-C', filename, 'checkout', '--quiet', rev],
863
- hg=[vcs, '--cwd', filename, 'up', '-C', '-r', rev, '-q'],
864
- )
865
- if rev is not None:
866
- self.info(f"Checking out {rev}")
867
- subprocess.check_call(co_commands[vcs])
868
-
869
- return filename
870
-
871
- def _download_other(self, url, filename):
872
- scheme = urllib.parse.urlsplit(url).scheme
873
- if scheme == 'file': # pragma: no cover
874
- return urllib.request.url2pathname(urllib.parse.urlparse(url).path)
875
- # raise error if not allowed
876
- self.url_ok(url, True)
877
- return self._attempt_download(url, filename)
878
-
879
- def scan_url(self, url) -> None:
880
- self.process_url(url, True)
881
-
882
- def _attempt_download(self, url, filename):
883
- headers = self._download_to(url, filename)
884
- if 'html' in headers.get('content-type', '').lower():
885
- return self._invalid_download_html(url, headers, filename)
886
- else:
887
- return filename
888
-
889
- def _invalid_download_html(self, url, headers, filename):
890
- os.unlink(filename)
891
- raise DistutilsError(f"Unexpected HTML page found at {url}")
892
-
893
- @staticmethod
894
- def _vcs_split_rev_from_url(url):
895
- """
896
- Given a possible VCS URL, return a clean URL and resolved revision if any.
897
-
898
- >>> vsrfu = PackageIndex._vcs_split_rev_from_url
899
- >>> vsrfu('git+https://github.com/pypa/setuptools@v69.0.0#egg-info=setuptools')
900
- ('https://github.com/pypa/setuptools', 'v69.0.0')
901
- >>> vsrfu('git+https://github.com/pypa/setuptools#egg-info=setuptools')
902
- ('https://github.com/pypa/setuptools', None)
903
- >>> vsrfu('http://foo/bar')
904
- ('http://foo/bar', None)
905
- """
906
- parts = urllib.parse.urlsplit(url)
907
-
908
- clean_scheme = parts.scheme.split('+', 1)[-1]
909
-
910
- # Some fragment identification fails
911
- no_fragment_path, _, _ = parts.path.partition('#')
912
-
913
- pre, sep, post = no_fragment_path.rpartition('@')
914
- clean_path, rev = (pre, post) if sep else (post, None)
915
-
916
- resolved = parts._replace(
917
- scheme=clean_scheme,
918
- path=clean_path,
919
- # discard the fragment
920
- fragment='',
921
- ).geturl()
922
-
923
- return resolved, rev
924
-
925
- def debug(self, msg, *args) -> None:
926
- log.debug(msg, *args)
927
-
928
- def info(self, msg, *args) -> None:
929
- log.info(msg, *args)
930
-
931
- def warn(self, msg, *args) -> None:
932
- log.warn(msg, *args)
933
-
934
-
935
- # This pattern matches a character entity reference (a decimal numeric
936
- # references, a hexadecimal numeric reference, or a named reference).
937
- entity_sub = re.compile(r'&(#(\d+|x[\da-fA-F]+)|[\w.:-]+);?').sub
938
-
939
-
940
- def decode_entity(match):
941
- what = match.group(0)
942
- return html.unescape(what)
943
-
944
-
945
- def htmldecode(text):
946
- """
947
- Decode HTML entities in the given text.
948
-
949
- >>> htmldecode(
950
- ... 'https://../package_name-0.1.2.tar.gz'
951
- ... '?tokena=A&amp;tokenb=B">package_name-0.1.2.tar.gz')
952
- 'https://../package_name-0.1.2.tar.gz?tokena=A&tokenb=B">package_name-0.1.2.tar.gz'
953
- """
954
- return entity_sub(decode_entity, text)
955
-
956
-
957
- def socket_timeout(timeout=15):
958
- def _socket_timeout(func):
959
- def _socket_timeout(*args, **kwargs):
960
- old_timeout = socket.getdefaulttimeout()
961
- socket.setdefaulttimeout(timeout)
962
- try:
963
- return func(*args, **kwargs)
964
- finally:
965
- socket.setdefaulttimeout(old_timeout)
966
-
967
- return _socket_timeout
968
-
969
- return _socket_timeout
970
-
971
-
972
- def _encode_auth(auth):
973
- """
974
- Encode auth from a URL suitable for an HTTP header.
975
- >>> str(_encode_auth('username%3Apassword'))
976
- 'dXNlcm5hbWU6cGFzc3dvcmQ='
977
-
978
- Long auth strings should not cause a newline to be inserted.
979
- >>> long_auth = 'username:' + 'password'*10
980
- >>> chr(10) in str(_encode_auth(long_auth))
981
- False
982
- """
983
- auth_s = urllib.parse.unquote(auth)
984
- # convert to bytes
985
- auth_bytes = auth_s.encode()
986
- encoded_bytes = base64.b64encode(auth_bytes)
987
- # convert back to a string
988
- encoded = encoded_bytes.decode()
989
- # strip the trailing carriage return
990
- return encoded.replace('\n', '')
991
-
992
-
993
- class Credential(NamedTuple):
994
- """
995
- A username/password pair.
996
-
997
- Displayed separated by `:`.
998
- >>> str(Credential('username', 'password'))
999
- 'username:password'
1000
- """
1001
-
1002
- username: str
1003
- password: str
1004
-
1005
- def __str__(self) -> str:
1006
- return f'{self.username}:{self.password}'
1007
-
1008
-
1009
- class PyPIConfig(configparser.RawConfigParser):
1010
- def __init__(self):
1011
- """
1012
- Load from ~/.pypirc
1013
- """
1014
- defaults = dict.fromkeys(['username', 'password', 'repository'], '')
1015
- super().__init__(defaults)
1016
-
1017
- rc = os.path.join(os.path.expanduser('~'), '.pypirc')
1018
- if os.path.exists(rc):
1019
- _cfg_read_utf8_with_fallback(self, rc)
1020
-
1021
- @property
1022
- def creds_by_repository(self):
1023
- sections_with_repositories = [
1024
- section
1025
- for section in self.sections()
1026
- if self.get(section, 'repository').strip()
1027
- ]
1028
-
1029
- return dict(map(self._get_repo_cred, sections_with_repositories))
1030
-
1031
- def _get_repo_cred(self, section):
1032
- repo = self.get(section, 'repository').strip()
1033
- return repo, Credential(
1034
- self.get(section, 'username').strip(),
1035
- self.get(section, 'password').strip(),
1036
- )
1037
-
1038
- def find_credential(self, url):
1039
- """
1040
- If the URL indicated appears to be a repository defined in this
1041
- config, return the credential for that repository.
1042
- """
1043
- for repository, cred in self.creds_by_repository.items():
1044
- if url.startswith(repository):
1045
- return cred
1046
- return None
1047
-
1048
-
1049
- def open_with_auth(url, opener=urllib.request.urlopen):
1050
- """Open a urllib2 request, handling HTTP authentication"""
1051
-
1052
- parsed = urllib.parse.urlparse(url)
1053
- scheme, netloc, path, params, query, frag = parsed
1054
-
1055
- # Double scheme does not raise on macOS as revealed by a
1056
- # failing test. We would expect "nonnumeric port". Refs #20.
1057
- if netloc.endswith(':'):
1058
- raise http.client.InvalidURL("nonnumeric port: ''")
1059
-
1060
- if scheme in ('http', 'https'):
1061
- auth, address = _splituser(netloc)
1062
- else:
1063
- auth, address = (None, None)
1064
-
1065
- if not auth:
1066
- cred = PyPIConfig().find_credential(url)
1067
- if cred:
1068
- auth = str(cred)
1069
- info = cred.username, url
1070
- log.info('Authenticating as %s for %s (from .pypirc)', *info)
1071
-
1072
- if auth:
1073
- auth = "Basic " + _encode_auth(auth)
1074
- parts = scheme, address, path, params, query, frag
1075
- new_url = urllib.parse.urlunparse(parts)
1076
- request = urllib.request.Request(new_url)
1077
- request.add_header("Authorization", auth)
1078
- else:
1079
- request = urllib.request.Request(url)
1080
-
1081
- request.add_header('User-Agent', user_agent)
1082
- fp = opener(request)
1083
-
1084
- if auth:
1085
- # Put authentication info back into request URL if same host,
1086
- # so that links found on the page will work
1087
- s2, h2, path2, param2, query2, frag2 = urllib.parse.urlparse(fp.url)
1088
- if s2 == scheme and h2 == address:
1089
- parts = s2, netloc, path2, param2, query2, frag2
1090
- fp.url = urllib.parse.urlunparse(parts)
1091
-
1092
- return fp
1093
-
1094
-
1095
- # copy of urllib.parse._splituser from Python 3.8
1096
- # See https://github.com/python/cpython/issues/80072.
1097
- def _splituser(host):
1098
- """splituser('user[:passwd]@host[:port]')
1099
- --> 'user[:passwd]', 'host[:port]'."""
1100
- user, delim, host = host.rpartition('@')
1101
- return (user if delim else None), host
1102
-
1103
-
1104
- # adding a timeout to avoid freezing package_index
1105
- open_with_auth = socket_timeout(_SOCKET_TIMEOUT)(open_with_auth)
1106
-
1107
-
1108
- def fix_sf_url(url):
1109
- return url # backward compatibility
1110
-
1111
-
1112
- def local_open(url):
1113
- """Read a local path, with special support for directories"""
1114
- _scheme, _server, path, _param, _query, _frag = urllib.parse.urlparse(url)
1115
- filename = urllib.request.url2pathname(path)
1116
- if os.path.isfile(filename):
1117
- return urllib.request.urlopen(url)
1118
- elif path.endswith('/') and os.path.isdir(filename):
1119
- files = []
1120
- for f in os.listdir(filename):
1121
- filepath = os.path.join(filename, f)
1122
- if f == 'index.html':
1123
- body = _read_utf8_with_fallback(filepath)
1124
- break
1125
- elif os.path.isdir(filepath):
1126
- f += '/'
1127
- files.append(f'<a href="{f}">{f}</a>')
1128
- else:
1129
- tmpl = "<html><head><title>{url}</title></head><body>{files}</body></html>"
1130
- body = tmpl.format(url=url, files='\n'.join(files))
1131
- status, message = 200, "OK"
1132
- else:
1133
- status, message, body = 404, "Path not found", "Not found"
1134
-
1135
- headers = {'content-type': 'text/html'}
1136
- body_stream = io.StringIO(body)
1137
- return urllib.error.HTTPError(url, status, message, headers, body_stream)