shopify-cli 1.3.1 → 1.7.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (301) hide show
  1. checksums.yaml +4 -4
  2. data/.github/CODEOWNERS +2 -2
  3. data/.github/CONTRIBUTING.md +9 -1
  4. data/.github/PULL_REQUEST_TEMPLATE.md +10 -1
  5. data/.github/workflows/release.yml +61 -0
  6. data/.github/workflows/triage.yml +22 -0
  7. data/.gitignore +0 -1
  8. data/.rubocop.yml +61 -8
  9. data/.rubocop_todo.yml +11 -0
  10. data/.travis.yml +1 -0
  11. data/CHANGELOG.md +30 -0
  12. data/Gemfile +12 -11
  13. data/Gemfile.lock +39 -37
  14. data/README.md +39 -7
  15. data/RELEASING.md +19 -29
  16. data/Rakefile +32 -28
  17. data/bin/load_shopify.rb +6 -6
  18. data/bin/shopify +2 -2
  19. data/dev.yml +2 -2
  20. data/docs/_config.yml +1 -18
  21. data/docs/app/node/commands/index.md +2 -80
  22. data/docs/app/node/index.md +2 -33
  23. data/docs/app/rails/commands/index.md +2 -78
  24. data/docs/app/rails/index.md +2 -34
  25. data/docs/core/index.md +2 -84
  26. data/docs/getting-started/index.md +2 -25
  27. data/docs/getting-started/install/index.md +1 -118
  28. data/docs/getting-started/migrate/index.md +2 -94
  29. data/docs/getting-started/uninstall/index.md +2 -35
  30. data/docs/getting-started/upgrade/index.md +2 -39
  31. data/docs/help/start-app/index.md +2 -4
  32. data/docs/index.md +2 -24
  33. data/ext/shopify-cli/extconf.rb +7 -7
  34. data/install.sh +1 -1
  35. data/lib/docgen/markdown.rb +11 -11
  36. data/lib/{project_types/extension/graphql → graphql}/get_app_by_api_key.graphql +0 -0
  37. data/lib/project_types/extension/cli.rb +64 -47
  38. data/lib/project_types/extension/commands/build.rb +3 -3
  39. data/lib/project_types/extension/commands/create.rb +16 -9
  40. data/lib/project_types/extension/commands/extension_command.rb +8 -5
  41. data/lib/project_types/extension/commands/push.rb +8 -8
  42. data/lib/project_types/extension/commands/register.rb +19 -30
  43. data/lib/project_types/extension/commands/serve.rb +2 -2
  44. data/lib/project_types/extension/commands/tunnel.rb +12 -12
  45. data/lib/project_types/extension/extension_project.rb +4 -4
  46. data/lib/project_types/extension/extension_project_keys.rb +4 -4
  47. data/lib/project_types/extension/features/argo.rb +117 -0
  48. data/lib/project_types/extension/features/argo_config.rb +5 -5
  49. data/lib/project_types/extension/features/argo_dependencies.rb +5 -5
  50. data/lib/project_types/extension/features/argo_setup.rb +2 -2
  51. data/lib/project_types/extension/features/argo_setup_steps.rb +4 -4
  52. data/lib/project_types/extension/forms/create.rb +28 -34
  53. data/lib/project_types/extension/forms/questions/ask_app.rb +53 -0
  54. data/lib/project_types/extension/forms/questions/ask_name.rb +40 -0
  55. data/lib/project_types/extension/forms/questions/ask_type.rb +36 -0
  56. data/lib/project_types/extension/messages/messages.rb +53 -52
  57. data/lib/project_types/extension/models/lazy_specification_handler.rb +12 -0
  58. data/lib/project_types/extension/models/specification.rb +35 -0
  59. data/lib/project_types/extension/models/specification_handlers/checkout_post_purchase.rb +19 -0
  60. data/lib/project_types/extension/models/specification_handlers/default.rb +67 -0
  61. data/lib/project_types/extension/models/specifications.rb +77 -0
  62. data/lib/project_types/extension/tasks/configure_features.rb +52 -0
  63. data/lib/project_types/extension/tasks/converters/app_converter.rb +6 -6
  64. data/lib/project_types/extension/tasks/converters/registration_converter.rb +6 -6
  65. data/lib/project_types/extension/tasks/converters/validation_error_converter.rb +4 -4
  66. data/lib/project_types/extension/tasks/converters/version_converter.rb +7 -7
  67. data/lib/project_types/extension/tasks/create_extension.rb +4 -4
  68. data/lib/project_types/extension/tasks/fetch_specifications.rb +38 -0
  69. data/lib/project_types/extension/tasks/get_app.rb +4 -4
  70. data/lib/project_types/extension/tasks/get_apps.rb +3 -3
  71. data/lib/project_types/extension/tasks/update_draft.rb +4 -4
  72. data/lib/project_types/extension/tasks/user_errors.rb +4 -4
  73. data/lib/project_types/node/cli.rb +19 -16
  74. data/lib/project_types/node/commands/connect.rb +15 -0
  75. data/lib/project_types/node/commands/create.rb +46 -38
  76. data/lib/project_types/node/commands/deploy.rb +4 -4
  77. data/lib/project_types/node/commands/deploy/heroku.rb +24 -24
  78. data/lib/project_types/node/commands/generate.rb +9 -18
  79. data/lib/project_types/node/commands/open.rb +2 -2
  80. data/lib/project_types/node/commands/populate.rb +6 -6
  81. data/lib/project_types/node/commands/populate/customer.rb +5 -5
  82. data/lib/project_types/node/commands/populate/draft_order.rb +5 -5
  83. data/lib/project_types/node/commands/populate/product.rb +5 -5
  84. data/lib/project_types/node/commands/serve.rb +9 -9
  85. data/lib/project_types/node/commands/tunnel.rb +7 -7
  86. data/lib/project_types/node/forms/create.rb +7 -7
  87. data/lib/project_types/node/messages/messages.rb +19 -53
  88. data/lib/project_types/rails/cli.rb +21 -18
  89. data/lib/project_types/rails/commands/connect.rb +15 -0
  90. data/lib/project_types/rails/commands/create.rb +60 -54
  91. data/lib/project_types/rails/commands/deploy.rb +4 -4
  92. data/lib/project_types/rails/commands/deploy/heroku.rb +30 -30
  93. data/lib/project_types/rails/commands/generate.rb +7 -7
  94. data/lib/project_types/rails/commands/generate/webhook.rb +6 -6
  95. data/lib/project_types/rails/commands/open.rb +2 -2
  96. data/lib/project_types/rails/commands/populate.rb +6 -6
  97. data/lib/project_types/rails/commands/populate/customer.rb +5 -5
  98. data/lib/project_types/rails/commands/populate/draft_order.rb +5 -5
  99. data/lib/project_types/rails/commands/populate/product.rb +5 -5
  100. data/lib/project_types/rails/commands/serve.rb +11 -11
  101. data/lib/project_types/rails/commands/tunnel.rb +7 -7
  102. data/lib/project_types/rails/forms/create.rb +24 -24
  103. data/lib/project_types/rails/gem.rb +24 -24
  104. data/lib/project_types/rails/messages/messages.rb +12 -9
  105. data/lib/project_types/rails/ruby.rb +2 -2
  106. data/lib/project_types/script/cli.rb +42 -38
  107. data/lib/project_types/script/commands/create.rb +13 -10
  108. data/lib/project_types/script/commands/disable.rb +3 -3
  109. data/lib/project_types/script/commands/enable.rb +19 -9
  110. data/lib/project_types/script/commands/push.rb +10 -17
  111. data/lib/project_types/script/config/extension_points.yml +17 -12
  112. data/lib/project_types/script/errors.rb +38 -0
  113. data/lib/project_types/script/forms/create.rb +29 -5
  114. data/lib/project_types/script/graphql/app_script_update_or_create.graphql +12 -1
  115. data/lib/project_types/script/layers/application/build_script.rb +18 -19
  116. data/lib/project_types/script/layers/application/create_script.rb +13 -11
  117. data/lib/project_types/script/layers/application/disable_script.rb +2 -2
  118. data/lib/project_types/script/layers/application/enable_script.rb +2 -2
  119. data/lib/project_types/script/layers/application/extension_points.rb +24 -0
  120. data/lib/project_types/script/layers/application/project_dependencies.rb +4 -4
  121. data/lib/project_types/script/layers/application/push_script.rb +12 -18
  122. data/lib/project_types/script/layers/domain/errors.rb +7 -0
  123. data/lib/project_types/script/layers/domain/extension_point.rb +62 -7
  124. data/lib/project_types/script/layers/domain/metadata.rb +55 -0
  125. data/lib/project_types/script/layers/domain/push_package.rb +29 -6
  126. data/lib/project_types/script/layers/infrastructure/assemblyscript_project_creator.rb +19 -54
  127. data/lib/project_types/script/layers/infrastructure/assemblyscript_task_runner.rb +49 -18
  128. data/lib/project_types/script/layers/infrastructure/errors.rb +17 -1
  129. data/lib/project_types/script/layers/infrastructure/extension_point_repository.rb +12 -6
  130. data/lib/project_types/script/layers/infrastructure/project_creator.rb +2 -1
  131. data/lib/project_types/script/layers/infrastructure/push_package_repository.rb +20 -13
  132. data/lib/project_types/script/layers/infrastructure/rust_project_creator.rb +72 -0
  133. data/lib/project_types/script/layers/infrastructure/rust_task_runner.rb +59 -0
  134. data/lib/project_types/script/layers/infrastructure/script_service.rb +26 -16
  135. data/lib/project_types/script/layers/infrastructure/task_runner.rb +4 -3
  136. data/lib/project_types/script/messages/messages.rb +67 -10
  137. data/lib/project_types/script/script_project.rb +47 -16
  138. data/lib/project_types/script/ui/error_handler.rb +115 -45
  139. data/lib/project_types/script/ui/printing_spinner.rb +1 -1
  140. data/lib/project_types/script/ui/strict_spinner.rb +1 -1
  141. data/lib/project_types/theme/cli.rb +40 -0
  142. data/lib/project_types/theme/commands/connect.rb +54 -0
  143. data/lib/project_types/theme/commands/create.rb +48 -0
  144. data/lib/project_types/theme/commands/deploy.rb +38 -0
  145. data/lib/project_types/theme/commands/generate.rb +20 -0
  146. data/lib/project_types/theme/commands/generate/env.rb +79 -0
  147. data/lib/project_types/theme/commands/push.rb +55 -0
  148. data/lib/project_types/theme/commands/serve.rb +31 -0
  149. data/lib/project_types/theme/forms/connect.rb +34 -0
  150. data/lib/project_types/theme/forms/create.rb +22 -0
  151. data/lib/project_types/theme/messages/messages.rb +147 -0
  152. data/lib/project_types/theme/tasks/ensure_themekit_installed.rb +78 -0
  153. data/lib/project_types/theme/themekit.rb +113 -0
  154. data/lib/rubygems_plugin.rb +3 -3
  155. data/lib/shopify-cli/admin_api.rb +52 -12
  156. data/lib/shopify-cli/admin_api/populate_resource_command.rb +17 -17
  157. data/lib/shopify-cli/admin_api/schema.rb +3 -3
  158. data/lib/shopify-cli/api.rb +38 -37
  159. data/lib/shopify-cli/command.rb +1 -1
  160. data/lib/shopify-cli/commands.rb +9 -9
  161. data/lib/shopify-cli/commands/config.rb +28 -28
  162. data/lib/shopify-cli/commands/connect.rb +35 -18
  163. data/lib/shopify-cli/commands/create.rb +5 -5
  164. data/lib/shopify-cli/commands/help.rb +6 -6
  165. data/lib/shopify-cli/commands/logout.rb +3 -3
  166. data/lib/shopify-cli/commands/system.rb +40 -31
  167. data/lib/shopify-cli/commands/version.rb +2 -2
  168. data/lib/shopify-cli/context.rb +43 -22
  169. data/lib/shopify-cli/core.rb +4 -4
  170. data/lib/shopify-cli/core/entry_point.rb +6 -6
  171. data/lib/shopify-cli/core/executor.rb +1 -1
  172. data/lib/shopify-cli/core/help_resolver.rb +2 -2
  173. data/lib/shopify-cli/core/monorail.rb +21 -19
  174. data/lib/shopify-cli/db.rb +2 -2
  175. data/lib/shopify-cli/feature.rb +1 -3
  176. data/lib/shopify-cli/form.rb +1 -1
  177. data/lib/shopify-cli/git.rb +17 -17
  178. data/lib/shopify-cli/helpers.rb +1 -1
  179. data/lib/shopify-cli/helpers/haikunator.rb +1 -1
  180. data/lib/shopify-cli/heroku.rb +28 -28
  181. data/lib/shopify-cli/http_request.rb +27 -0
  182. data/lib/shopify-cli/js_deps.rb +13 -13
  183. data/lib/shopify-cli/js_system.rb +5 -5
  184. data/lib/shopify-cli/lazy_delegator.rb +55 -0
  185. data/lib/shopify-cli/messages/messages.rb +24 -10
  186. data/lib/shopify-cli/method_object.rb +104 -0
  187. data/lib/shopify-cli/oauth.rb +25 -25
  188. data/lib/shopify-cli/oauth/servlet.rb +9 -9
  189. data/lib/shopify-cli/options.rb +3 -3
  190. data/lib/shopify-cli/packager.rb +24 -24
  191. data/lib/shopify-cli/partners_api.rb +38 -16
  192. data/lib/shopify-cli/partners_api/organizations.rb +10 -10
  193. data/lib/shopify-cli/process_supervision.rb +8 -8
  194. data/lib/shopify-cli/project.rb +27 -23
  195. data/lib/shopify-cli/project_type.rb +21 -5
  196. data/lib/shopify-cli/resolve_constant.rb +25 -0
  197. data/lib/shopify-cli/resources.rb +1 -1
  198. data/lib/shopify-cli/resources/env_file.rb +9 -9
  199. data/lib/shopify-cli/result.rb +432 -0
  200. data/lib/shopify-cli/shopifolk.rb +84 -0
  201. data/lib/shopify-cli/sub_command.rb +1 -1
  202. data/lib/shopify-cli/task.rb +9 -1
  203. data/lib/shopify-cli/tasks.rb +7 -7
  204. data/lib/shopify-cli/tasks/create_api_client.rb +17 -6
  205. data/lib/shopify-cli/tasks/ensure_dev_store.rb +11 -11
  206. data/lib/shopify-cli/tasks/ensure_env.rb +18 -15
  207. data/lib/shopify-cli/tasks/ensure_loopback_url.rb +4 -4
  208. data/lib/shopify-cli/tasks/select_org_and_shop.rb +29 -24
  209. data/lib/shopify-cli/tasks/update_dashboard_urls.rb +10 -10
  210. data/lib/shopify-cli/transform_data_structure.rb +86 -0
  211. data/lib/shopify-cli/tunnel.rb +36 -30
  212. data/lib/shopify-cli/version.rb +1 -1
  213. data/lib/shopify_cli.rb +57 -51
  214. data/shopify-cli.gemspec +6 -6
  215. data/shopify.fish +1 -1
  216. data/shopify.sh +1 -1
  217. data/vendor/deps/cli-kit/REVISION +1 -1
  218. data/vendor/deps/cli-kit/lib/cli/kit/logger.rb +2 -2
  219. data/vendor/deps/cli-kit/lib/cli/kit/system.rb +3 -3
  220. data/vendor/deps/cli-ui/REVISION +1 -1
  221. data/vendor/deps/cli-ui/lib/cli/ui.rb +26 -22
  222. data/vendor/deps/cli-ui/lib/cli/ui/ansi.rb +4 -6
  223. data/vendor/deps/cli-ui/lib/cli/ui/frame.rb +3 -3
  224. data/vendor/deps/cli-ui/lib/cli/ui/frame/frame_stack.rb +8 -9
  225. data/vendor/deps/cli-ui/lib/cli/ui/frame/frame_style.rb +1 -1
  226. data/vendor/deps/cli-ui/lib/cli/ui/glyph.rb +1 -0
  227. data/vendor/deps/cli-ui/lib/cli/ui/printer.rb +15 -3
  228. data/vendor/deps/cli-ui/lib/cli/ui/prompt/interactive_options.rb +4 -11
  229. data/vendor/deps/cli-ui/lib/cli/ui/spinner.rb +3 -5
  230. data/vendor/deps/cli-ui/lib/cli/ui/terminal.rb +10 -10
  231. data/vendor/deps/cli-ui/lib/cli/ui/version.rb +1 -1
  232. data/vendor/deps/cli-ui/lib/cli/ui/wrap.rb +56 -0
  233. data/vendor/deps/webrick/.gitignore +9 -0
  234. data/vendor/deps/webrick/Gemfile +3 -0
  235. data/vendor/deps/webrick/LICENSE.txt +22 -0
  236. data/vendor/deps/webrick/README.md +61 -0
  237. data/vendor/deps/webrick/Rakefile +10 -0
  238. data/vendor/deps/webrick/lib/webrick.rb +232 -0
  239. data/vendor/deps/webrick/lib/webrick/accesslog.rb +157 -0
  240. data/vendor/deps/webrick/lib/webrick/cgi.rb +313 -0
  241. data/vendor/deps/webrick/lib/webrick/compat.rb +36 -0
  242. data/vendor/deps/webrick/lib/webrick/config.rb +158 -0
  243. data/vendor/deps/webrick/lib/webrick/cookie.rb +172 -0
  244. data/vendor/deps/webrick/lib/webrick/htmlutils.rb +30 -0
  245. data/vendor/deps/webrick/lib/webrick/httpauth.rb +96 -0
  246. data/vendor/deps/webrick/lib/webrick/httpauth/authenticator.rb +117 -0
  247. data/vendor/deps/webrick/lib/webrick/httpauth/basicauth.rb +116 -0
  248. data/vendor/deps/webrick/lib/webrick/httpauth/digestauth.rb +395 -0
  249. data/vendor/deps/webrick/lib/webrick/httpauth/htdigest.rb +132 -0
  250. data/vendor/deps/webrick/lib/webrick/httpauth/htgroup.rb +97 -0
  251. data/vendor/deps/webrick/lib/webrick/httpauth/htpasswd.rb +158 -0
  252. data/vendor/deps/webrick/lib/webrick/httpauth/userdb.rb +53 -0
  253. data/vendor/deps/webrick/lib/webrick/httpproxy.rb +354 -0
  254. data/vendor/deps/webrick/lib/webrick/httprequest.rb +636 -0
  255. data/vendor/deps/webrick/lib/webrick/httpresponse.rb +564 -0
  256. data/vendor/deps/webrick/lib/webrick/https.rb +152 -0
  257. data/vendor/deps/webrick/lib/webrick/httpserver.rb +294 -0
  258. data/vendor/deps/webrick/lib/webrick/httpservlet.rb +23 -0
  259. data/vendor/deps/webrick/lib/webrick/httpservlet/abstract.rb +152 -0
  260. data/vendor/deps/webrick/lib/webrick/httpservlet/cgi_runner.rb +47 -0
  261. data/vendor/deps/webrick/lib/webrick/httpservlet/cgihandler.rb +126 -0
  262. data/vendor/deps/webrick/lib/webrick/httpservlet/erbhandler.rb +88 -0
  263. data/vendor/deps/webrick/lib/webrick/httpservlet/filehandler.rb +552 -0
  264. data/vendor/deps/webrick/lib/webrick/httpservlet/prochandler.rb +47 -0
  265. data/vendor/deps/webrick/lib/webrick/httpstatus.rb +194 -0
  266. data/vendor/deps/webrick/lib/webrick/httputils.rb +512 -0
  267. data/vendor/deps/webrick/lib/webrick/httpversion.rb +76 -0
  268. data/vendor/deps/webrick/lib/webrick/log.rb +156 -0
  269. data/vendor/deps/webrick/lib/webrick/server.rb +381 -0
  270. data/vendor/deps/webrick/lib/webrick/ssl.rb +215 -0
  271. data/vendor/deps/webrick/lib/webrick/utils.rb +265 -0
  272. data/vendor/deps/webrick/lib/webrick/version.rb +18 -0
  273. data/vendor/deps/webrick/webrick.gemspec +74 -0
  274. data/vendor/gen/template/bin/update-deps +9 -9
  275. metadata +84 -29
  276. data/docs/Gemfile +0 -5
  277. data/docs/Gemfile.lock +0 -258
  278. data/docs/_data/nav.yml +0 -35
  279. data/docs/_includes/footer.html +0 -15
  280. data/docs/_includes/head.html +0 -19
  281. data/docs/_includes/sidebar_nav.html +0 -22
  282. data/docs/_includes/toc.html +0 -112
  283. data/docs/_layouts/default.html +0 -79
  284. data/docs/css/docs.css +0 -157
  285. data/docs/images/header.png +0 -0
  286. data/docs/installing-ruby.md +0 -28
  287. data/lib/project_types/extension/features/argo/admin.rb +0 -20
  288. data/lib/project_types/extension/features/argo/base.rb +0 -129
  289. data/lib/project_types/extension/features/argo/checkout.rb +0 -20
  290. data/lib/project_types/extension/forms/register.rb +0 -47
  291. data/lib/project_types/extension/models/type.rb +0 -81
  292. data/lib/project_types/extension/models/types/checkout_post_purchase.rb +0 -23
  293. data/lib/project_types/extension/models/types/product_subscription.rb +0 -24
  294. data/lib/project_types/node/commands/generate/billing.rb +0 -39
  295. data/lib/project_types/node/commands/generate/page.rb +0 -59
  296. data/lib/project_types/node/commands/generate/webhook.rb +0 -37
  297. data/lib/project_types/script/layers/domain/script.rb +0 -18
  298. data/lib/project_types/script/layers/infrastructure/assemblyscript_tsconfig.rb +0 -38
  299. data/lib/project_types/script/layers/infrastructure/script_repository.rb +0 -59
  300. data/lib/project_types/script/templates/ts/as-pect.config.js +0 -27
  301. data/lib/project_types/script/templates/ts/as-pect.d.ts +0 -1
@@ -0,0 +1,395 @@
1
+ # frozen_string_literal: true
2
+ #
3
+ # httpauth/digestauth.rb -- HTTP digest access authentication
4
+ #
5
+ # Author: IPR -- Internet Programming with Ruby -- writers
6
+ # Copyright (c) 2003 Internet Programming with Ruby writers.
7
+ # Copyright (c) 2003 H.M.
8
+ #
9
+ # The original implementation is provided by H.M.
10
+ # URL: http://rwiki.jin.gr.jp/cgi-bin/rw-cgi.rb?cmd=view;name=
11
+ # %C7%A7%BE%DA%B5%A1%C7%BD%A4%F2%B2%FE%C2%A4%A4%B7%A4%C6%A4%DF%A4%EB
12
+ #
13
+ # $IPR: digestauth.rb,v 1.5 2003/02/20 07:15:47 gotoyuzo Exp $
14
+
15
+ require_relative '../config'
16
+ require_relative '../httpstatus'
17
+ require_relative 'authenticator'
18
+ require 'digest/md5'
19
+ require 'digest/sha1'
20
+
21
+ module WEBrick
22
+ module HTTPAuth
23
+
24
+ ##
25
+ # RFC 2617 Digest Access Authentication for WEBrick
26
+ #
27
+ # Use this class to add digest authentication to a WEBrick servlet.
28
+ #
29
+ # Here is an example of how to set up DigestAuth:
30
+ #
31
+ # config = { :Realm => 'DigestAuth example realm' }
32
+ #
33
+ # htdigest = WEBrick::HTTPAuth::Htdigest.new 'my_password_file'
34
+ # htdigest.set_passwd config[:Realm], 'username', 'password'
35
+ # htdigest.flush
36
+ #
37
+ # config[:UserDB] = htdigest
38
+ #
39
+ # digest_auth = WEBrick::HTTPAuth::DigestAuth.new config
40
+ #
41
+ # When using this as with a servlet be sure not to create a new DigestAuth
42
+ # object in the servlet's #initialize. By default WEBrick creates a new
43
+ # servlet instance for every request and the DigestAuth object must be
44
+ # used across requests.
45
+
46
+ class DigestAuth
47
+ include Authenticator
48
+
49
+ AuthScheme = "Digest" # :nodoc:
50
+
51
+ ##
52
+ # Struct containing the opaque portion of the digest authentication
53
+
54
+ OpaqueInfo = Struct.new(:time, :nonce, :nc) # :nodoc:
55
+
56
+ ##
57
+ # Digest authentication algorithm
58
+
59
+ attr_reader :algorithm
60
+
61
+ ##
62
+ # Quality of protection. RFC 2617 defines "auth" and "auth-int"
63
+
64
+ attr_reader :qop
65
+
66
+ ##
67
+ # Used by UserDB to create a digest password entry
68
+
69
+ def self.make_passwd(realm, user, pass)
70
+ pass ||= ""
71
+ Digest::MD5::hexdigest([user, realm, pass].join(":"))
72
+ end
73
+
74
+ ##
75
+ # Creates a new DigestAuth instance. Be sure to use the same DigestAuth
76
+ # instance for multiple requests as it saves state between requests in
77
+ # order to perform authentication.
78
+ #
79
+ # See WEBrick::Config::DigestAuth for default configuration entries
80
+ #
81
+ # You must supply the following configuration entries:
82
+ #
83
+ # :Realm:: The name of the realm being protected.
84
+ # :UserDB:: A database of usernames and passwords.
85
+ # A WEBrick::HTTPAuth::Htdigest instance should be used.
86
+
87
+ def initialize(config, default=Config::DigestAuth)
88
+ check_init(config)
89
+ @config = default.dup.update(config)
90
+ @algorithm = @config[:Algorithm]
91
+ @domain = @config[:Domain]
92
+ @qop = @config[:Qop]
93
+ @use_opaque = @config[:UseOpaque]
94
+ @use_next_nonce = @config[:UseNextNonce]
95
+ @check_nc = @config[:CheckNc]
96
+ @use_auth_info_header = @config[:UseAuthenticationInfoHeader]
97
+ @nonce_expire_period = @config[:NonceExpirePeriod]
98
+ @nonce_expire_delta = @config[:NonceExpireDelta]
99
+ @internet_explorer_hack = @config[:InternetExplorerHack]
100
+
101
+ case @algorithm
102
+ when 'MD5','MD5-sess'
103
+ @h = Digest::MD5
104
+ when 'SHA1','SHA1-sess' # it is a bonus feature :-)
105
+ @h = Digest::SHA1
106
+ else
107
+ msg = format('Algorithm "%s" is not supported.', @algorithm)
108
+ raise ArgumentError.new(msg)
109
+ end
110
+
111
+ @instance_key = hexdigest(self.__id__, Time.now.to_i, Process.pid)
112
+ @opaques = {}
113
+ @last_nonce_expire = Time.now
114
+ @mutex = Thread::Mutex.new
115
+ end
116
+
117
+ ##
118
+ # Authenticates a +req+ and returns a 401 Unauthorized using +res+ if
119
+ # the authentication was not correct.
120
+
121
+ def authenticate(req, res)
122
+ unless result = @mutex.synchronize{ _authenticate(req, res) }
123
+ challenge(req, res)
124
+ end
125
+ if result == :nonce_is_stale
126
+ challenge(req, res, true)
127
+ end
128
+ return true
129
+ end
130
+
131
+ ##
132
+ # Returns a challenge response which asks for authentication information
133
+
134
+ def challenge(req, res, stale=false)
135
+ nonce = generate_next_nonce(req)
136
+ if @use_opaque
137
+ opaque = generate_opaque(req)
138
+ @opaques[opaque].nonce = nonce
139
+ end
140
+
141
+ param = Hash.new
142
+ param["realm"] = HTTPUtils::quote(@realm)
143
+ param["domain"] = HTTPUtils::quote(@domain.to_a.join(" ")) if @domain
144
+ param["nonce"] = HTTPUtils::quote(nonce)
145
+ param["opaque"] = HTTPUtils::quote(opaque) if opaque
146
+ param["stale"] = stale.to_s
147
+ param["algorithm"] = @algorithm
148
+ param["qop"] = HTTPUtils::quote(@qop.to_a.join(",")) if @qop
149
+
150
+ res[@response_field] =
151
+ "#{@auth_scheme} " + param.map{|k,v| "#{k}=#{v}" }.join(", ")
152
+ info("%s: %s", @response_field, res[@response_field]) if $DEBUG
153
+ raise @auth_exception
154
+ end
155
+
156
+ private
157
+
158
+ # :stopdoc:
159
+
160
+ MustParams = ['username','realm','nonce','uri','response']
161
+ MustParamsAuth = ['cnonce','nc']
162
+
163
+ def _authenticate(req, res)
164
+ unless digest_credentials = check_scheme(req)
165
+ return false
166
+ end
167
+
168
+ auth_req = split_param_value(digest_credentials)
169
+ if auth_req['qop'] == "auth" || auth_req['qop'] == "auth-int"
170
+ req_params = MustParams + MustParamsAuth
171
+ else
172
+ req_params = MustParams
173
+ end
174
+ req_params.each{|key|
175
+ unless auth_req.has_key?(key)
176
+ error('%s: parameter missing. "%s"', auth_req['username'], key)
177
+ raise HTTPStatus::BadRequest
178
+ end
179
+ }
180
+
181
+ if !check_uri(req, auth_req)
182
+ raise HTTPStatus::BadRequest
183
+ end
184
+
185
+ if auth_req['realm'] != @realm
186
+ error('%s: realm unmatch. "%s" for "%s"',
187
+ auth_req['username'], auth_req['realm'], @realm)
188
+ return false
189
+ end
190
+
191
+ auth_req['algorithm'] ||= 'MD5'
192
+ if auth_req['algorithm'].upcase != @algorithm.upcase
193
+ error('%s: algorithm unmatch. "%s" for "%s"',
194
+ auth_req['username'], auth_req['algorithm'], @algorithm)
195
+ return false
196
+ end
197
+
198
+ if (@qop.nil? && auth_req.has_key?('qop')) ||
199
+ (@qop && (! @qop.member?(auth_req['qop'])))
200
+ error('%s: the qop is not allowed. "%s"',
201
+ auth_req['username'], auth_req['qop'])
202
+ return false
203
+ end
204
+
205
+ password = @userdb.get_passwd(@realm, auth_req['username'], @reload_db)
206
+ unless password
207
+ error('%s: the user is not allowed.', auth_req['username'])
208
+ return false
209
+ end
210
+
211
+ nonce_is_invalid = false
212
+ if @use_opaque
213
+ info("@opaque = %s", @opaque.inspect) if $DEBUG
214
+ if !(opaque = auth_req['opaque'])
215
+ error('%s: opaque is not given.', auth_req['username'])
216
+ nonce_is_invalid = true
217
+ elsif !(opaque_struct = @opaques[opaque])
218
+ error('%s: invalid opaque is given.', auth_req['username'])
219
+ nonce_is_invalid = true
220
+ elsif !check_opaque(opaque_struct, req, auth_req)
221
+ @opaques.delete(auth_req['opaque'])
222
+ nonce_is_invalid = true
223
+ end
224
+ elsif !check_nonce(req, auth_req)
225
+ nonce_is_invalid = true
226
+ end
227
+
228
+ if /-sess$/i =~ auth_req['algorithm']
229
+ ha1 = hexdigest(password, auth_req['nonce'], auth_req['cnonce'])
230
+ else
231
+ ha1 = password
232
+ end
233
+
234
+ if auth_req['qop'] == "auth" || auth_req['qop'] == nil
235
+ ha2 = hexdigest(req.request_method, auth_req['uri'])
236
+ ha2_res = hexdigest("", auth_req['uri'])
237
+ elsif auth_req['qop'] == "auth-int"
238
+ body_digest = @h.new
239
+ req.body { |chunk| body_digest.update(chunk) }
240
+ body_digest = body_digest.hexdigest
241
+ ha2 = hexdigest(req.request_method, auth_req['uri'], body_digest)
242
+ ha2_res = hexdigest("", auth_req['uri'], body_digest)
243
+ end
244
+
245
+ if auth_req['qop'] == "auth" || auth_req['qop'] == "auth-int"
246
+ param2 = ['nonce', 'nc', 'cnonce', 'qop'].map{|key|
247
+ auth_req[key]
248
+ }.join(':')
249
+ digest = hexdigest(ha1, param2, ha2)
250
+ digest_res = hexdigest(ha1, param2, ha2_res)
251
+ else
252
+ digest = hexdigest(ha1, auth_req['nonce'], ha2)
253
+ digest_res = hexdigest(ha1, auth_req['nonce'], ha2_res)
254
+ end
255
+
256
+ if digest != auth_req['response']
257
+ error("%s: digest unmatch.", auth_req['username'])
258
+ return false
259
+ elsif nonce_is_invalid
260
+ error('%s: digest is valid, but nonce is not valid.',
261
+ auth_req['username'])
262
+ return :nonce_is_stale
263
+ elsif @use_auth_info_header
264
+ auth_info = {
265
+ 'nextnonce' => generate_next_nonce(req),
266
+ 'rspauth' => digest_res
267
+ }
268
+ if @use_opaque
269
+ opaque_struct.time = req.request_time
270
+ opaque_struct.nonce = auth_info['nextnonce']
271
+ opaque_struct.nc = "%08x" % (auth_req['nc'].hex + 1)
272
+ end
273
+ if auth_req['qop'] == "auth" || auth_req['qop'] == "auth-int"
274
+ ['qop','cnonce','nc'].each{|key|
275
+ auth_info[key] = auth_req[key]
276
+ }
277
+ end
278
+ res[@resp_info_field] = auth_info.keys.map{|key|
279
+ if key == 'nc'
280
+ key + '=' + auth_info[key]
281
+ else
282
+ key + "=" + HTTPUtils::quote(auth_info[key])
283
+ end
284
+ }.join(', ')
285
+ end
286
+ info('%s: authentication succeeded.', auth_req['username'])
287
+ req.user = auth_req['username']
288
+ return true
289
+ end
290
+
291
+ def split_param_value(string)
292
+ ret = {}
293
+ string.scan(/\G\s*([\w\-.*%!]+)=\s*(?:\"((?>\\.|[^\"])*)\"|([^,\"]*))\s*,?/) do
294
+ ret[$1] = $3 || $2.gsub(/\\(.)/, "\\1")
295
+ end
296
+ ret
297
+ end
298
+
299
+ def generate_next_nonce(req)
300
+ now = "%012d" % req.request_time.to_i
301
+ pk = hexdigest(now, @instance_key)[0,32]
302
+ nonce = [now + ":" + pk].pack("m0") # it has 60 length of chars.
303
+ nonce
304
+ end
305
+
306
+ def check_nonce(req, auth_req)
307
+ username = auth_req['username']
308
+ nonce = auth_req['nonce']
309
+
310
+ pub_time, pk = nonce.unpack("m*")[0].split(":", 2)
311
+ if (!pub_time || !pk)
312
+ error("%s: empty nonce is given", username)
313
+ return false
314
+ elsif (hexdigest(pub_time, @instance_key)[0,32] != pk)
315
+ error("%s: invalid private-key: %s for %s",
316
+ username, hexdigest(pub_time, @instance_key)[0,32], pk)
317
+ return false
318
+ end
319
+
320
+ diff_time = req.request_time.to_i - pub_time.to_i
321
+ if (diff_time < 0)
322
+ error("%s: difference of time-stamp is negative.", username)
323
+ return false
324
+ elsif diff_time > @nonce_expire_period
325
+ error("%s: nonce is expired.", username)
326
+ return false
327
+ end
328
+
329
+ return true
330
+ end
331
+
332
+ def generate_opaque(req)
333
+ @mutex.synchronize{
334
+ now = req.request_time
335
+ if now - @last_nonce_expire > @nonce_expire_delta
336
+ @opaques.delete_if{|key,val|
337
+ (now - val.time) > @nonce_expire_period
338
+ }
339
+ @last_nonce_expire = now
340
+ end
341
+ begin
342
+ opaque = Utils::random_string(16)
343
+ end while @opaques[opaque]
344
+ @opaques[opaque] = OpaqueInfo.new(now, nil, '00000001')
345
+ opaque
346
+ }
347
+ end
348
+
349
+ def check_opaque(opaque_struct, req, auth_req)
350
+ if (@use_next_nonce && auth_req['nonce'] != opaque_struct.nonce)
351
+ error('%s: nonce unmatched. "%s" for "%s"',
352
+ auth_req['username'], auth_req['nonce'], opaque_struct.nonce)
353
+ return false
354
+ elsif !check_nonce(req, auth_req)
355
+ return false
356
+ end
357
+ if (@check_nc && auth_req['nc'] != opaque_struct.nc)
358
+ error('%s: nc unmatched."%s" for "%s"',
359
+ auth_req['username'], auth_req['nc'], opaque_struct.nc)
360
+ return false
361
+ end
362
+ true
363
+ end
364
+
365
+ def check_uri(req, auth_req)
366
+ uri = auth_req['uri']
367
+ if uri != req.request_uri.to_s && uri != req.unparsed_uri &&
368
+ (@internet_explorer_hack && uri != req.path)
369
+ error('%s: uri unmatch. "%s" for "%s"', auth_req['username'],
370
+ auth_req['uri'], req.request_uri.to_s)
371
+ return false
372
+ end
373
+ true
374
+ end
375
+
376
+ def hexdigest(*args)
377
+ @h.hexdigest(args.join(":"))
378
+ end
379
+
380
+ # :startdoc:
381
+ end
382
+
383
+ ##
384
+ # Digest authentication for proxy servers. See DigestAuth for details.
385
+
386
+ class ProxyDigestAuth < DigestAuth
387
+ include ProxyAuthenticator
388
+
389
+ private
390
+ def check_uri(req, auth_req) # :nodoc:
391
+ return true
392
+ end
393
+ end
394
+ end
395
+ end
@@ -0,0 +1,132 @@
1
+ # frozen_string_literal: true
2
+ #
3
+ # httpauth/htdigest.rb -- Apache compatible htdigest file
4
+ #
5
+ # Author: IPR -- Internet Programming with Ruby -- writers
6
+ # Copyright (c) 2003 Internet Programming with Ruby writers. All rights
7
+ # reserved.
8
+ #
9
+ # $IPR: htdigest.rb,v 1.4 2003/07/22 19:20:45 gotoyuzo Exp $
10
+
11
+ require_relative 'userdb'
12
+ require_relative 'digestauth'
13
+ require 'tempfile'
14
+
15
+ module WEBrick
16
+ module HTTPAuth
17
+
18
+ ##
19
+ # Htdigest accesses apache-compatible digest password files. Passwords are
20
+ # matched to a realm where they are valid. For security, the path for a
21
+ # digest password database should be stored outside of the paths available
22
+ # to the HTTP server.
23
+ #
24
+ # Htdigest is intended for use with WEBrick::HTTPAuth::DigestAuth and
25
+ # stores passwords using cryptographic hashes.
26
+ #
27
+ # htpasswd = WEBrick::HTTPAuth::Htdigest.new 'my_password_file'
28
+ # htpasswd.set_passwd 'my realm', 'username', 'password'
29
+ # htpasswd.flush
30
+
31
+ class Htdigest
32
+ include UserDB
33
+
34
+ ##
35
+ # Open a digest password database at +path+
36
+
37
+ def initialize(path)
38
+ @path = path
39
+ @mtime = Time.at(0)
40
+ @digest = Hash.new
41
+ @mutex = Thread::Mutex::new
42
+ @auth_type = DigestAuth
43
+ File.open(@path,"a").close unless File.exist?(@path)
44
+ reload
45
+ end
46
+
47
+ ##
48
+ # Reloads passwords from the database
49
+
50
+ def reload
51
+ mtime = File::mtime(@path)
52
+ if mtime > @mtime
53
+ @digest.clear
54
+ File.open(@path){|io|
55
+ while line = io.gets
56
+ line.chomp!
57
+ user, realm, pass = line.split(/:/, 3)
58
+ unless @digest[realm]
59
+ @digest[realm] = Hash.new
60
+ end
61
+ @digest[realm][user] = pass
62
+ end
63
+ }
64
+ @mtime = mtime
65
+ end
66
+ end
67
+
68
+ ##
69
+ # Flush the password database. If +output+ is given the database will
70
+ # be written there instead of to the original path.
71
+
72
+ def flush(output=nil)
73
+ output ||= @path
74
+ tmp = Tempfile.create("htpasswd", File::dirname(output))
75
+ renamed = false
76
+ begin
77
+ each{|item| tmp.puts(item.join(":")) }
78
+ tmp.close
79
+ File::rename(tmp.path, output)
80
+ renamed = true
81
+ ensure
82
+ tmp.close
83
+ File.unlink(tmp.path) if !renamed
84
+ end
85
+ end
86
+
87
+ ##
88
+ # Retrieves a password from the database for +user+ in +realm+. If
89
+ # +reload_db+ is true the database will be reloaded first.
90
+
91
+ def get_passwd(realm, user, reload_db)
92
+ reload() if reload_db
93
+ if hash = @digest[realm]
94
+ hash[user]
95
+ end
96
+ end
97
+
98
+ ##
99
+ # Sets a password in the database for +user+ in +realm+ to +pass+.
100
+
101
+ def set_passwd(realm, user, pass)
102
+ @mutex.synchronize{
103
+ unless @digest[realm]
104
+ @digest[realm] = Hash.new
105
+ end
106
+ @digest[realm][user] = make_passwd(realm, user, pass)
107
+ }
108
+ end
109
+
110
+ ##
111
+ # Removes a password from the database for +user+ in +realm+.
112
+
113
+ def delete_passwd(realm, user)
114
+ if hash = @digest[realm]
115
+ hash.delete(user)
116
+ end
117
+ end
118
+
119
+ ##
120
+ # Iterate passwords in the database.
121
+
122
+ def each # :yields: [user, realm, password_hash]
123
+ @digest.keys.sort.each{|realm|
124
+ hash = @digest[realm]
125
+ hash.keys.sort.each{|user|
126
+ yield([user, realm, hash[user]])
127
+ }
128
+ }
129
+ end
130
+ end
131
+ end
132
+ end