shopify-cli 1.4.1 → 1.8.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (303) 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 +3 -2
  5. data/.github/workflows/release.yml +0 -1
  6. data/.github/workflows/triage.yml +22 -0
  7. data/.rubocop.yml +21 -7
  8. data/.rubocop_todo.yml +2 -15
  9. data/.travis.yml +1 -1
  10. data/CHANGELOG.md +28 -0
  11. data/Gemfile +12 -11
  12. data/Gemfile.lock +17 -14
  13. data/README.md +39 -7
  14. data/RELEASING.md +5 -13
  15. data/Rakefile +32 -28
  16. data/bin/load_shopify.rb +6 -6
  17. data/bin/shopify +2 -2
  18. data/dev.yml +2 -2
  19. data/docs/_config.yml +1 -18
  20. data/docs/app/node/commands/index.md +2 -80
  21. data/docs/app/node/index.md +2 -33
  22. data/docs/app/rails/commands/index.md +2 -78
  23. data/docs/app/rails/index.md +2 -34
  24. data/docs/core/index.md +2 -84
  25. data/docs/getting-started/index.md +2 -25
  26. data/docs/getting-started/install/index.md +1 -118
  27. data/docs/getting-started/migrate/index.md +2 -94
  28. data/docs/getting-started/uninstall/index.md +2 -35
  29. data/docs/getting-started/upgrade/index.md +2 -39
  30. data/docs/help/start-app/index.md +2 -4
  31. data/docs/index.md +2 -24
  32. data/ext/shopify-cli/extconf.rb +7 -7
  33. data/install.sh +1 -1
  34. data/lib/docgen/markdown.rb +12 -12
  35. data/lib/graphql/fetch_specifications.graphql +14 -0
  36. data/lib/{project_types/extension/graphql → graphql}/get_app_by_api_key.graphql +0 -0
  37. data/lib/project_types/extension/cli.rb +54 -47
  38. data/lib/project_types/extension/commands/build.rb +3 -3
  39. data/lib/project_types/extension/commands/create.rb +17 -10
  40. data/lib/project_types/extension/commands/extension_command.rb +12 -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 +31 -3
  44. data/lib/project_types/extension/commands/tunnel.rb +12 -12
  45. data/lib/project_types/extension/extension_project.rb +8 -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 +47 -0
  56. data/lib/project_types/extension/messages/messages.rb +55 -52
  57. data/lib/project_types/extension/models/lazy_specification_handler.rb +12 -0
  58. data/lib/project_types/extension/models/specification.rb +37 -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 +88 -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 +18 -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 +44 -41
  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 +17 -8
  87. data/lib/project_types/node/messages/messages.rb +20 -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 +58 -57
  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 +35 -25
  103. data/lib/project_types/rails/gem.rb +24 -24
  104. data/lib/project_types/rails/messages/messages.rb +13 -9
  105. data/lib/project_types/rails/ruby.rb +2 -2
  106. data/lib/project_types/script/cli.rb +44 -38
  107. data/lib/project_types/script/commands/create.rb +15 -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 +30 -12
  112. data/lib/project_types/script/errors.rb +22 -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 +19 -19
  116. data/lib/project_types/script/layers/application/create_script.rb +45 -12
  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 +15 -18
  122. data/lib/project_types/script/layers/domain/config_ui.rb +16 -0
  123. data/lib/project_types/script/layers/domain/errors.rb +23 -0
  124. data/lib/project_types/script/layers/domain/extension_point.rb +67 -7
  125. data/lib/project_types/script/layers/domain/metadata.rb +55 -0
  126. data/lib/project_types/script/layers/domain/push_package.rb +29 -6
  127. data/lib/project_types/script/layers/infrastructure/assemblyscript_project_creator.rb +45 -55
  128. data/lib/project_types/script/layers/infrastructure/assemblyscript_task_runner.rb +41 -45
  129. data/lib/project_types/script/layers/infrastructure/config_ui_repository.rb +46 -0
  130. data/lib/project_types/script/layers/infrastructure/errors.rb +32 -5
  131. data/lib/project_types/script/layers/infrastructure/extension_point_repository.rb +12 -6
  132. data/lib/project_types/script/layers/infrastructure/project_creator.rb +2 -1
  133. data/lib/project_types/script/layers/infrastructure/push_package_repository.rb +20 -13
  134. data/lib/project_types/script/layers/infrastructure/rust_project_creator.rb +72 -0
  135. data/lib/project_types/script/layers/infrastructure/rust_task_runner.rb +59 -0
  136. data/lib/project_types/script/layers/infrastructure/script_service.rb +39 -17
  137. data/lib/project_types/script/layers/infrastructure/task_runner.rb +4 -3
  138. data/lib/project_types/script/messages/messages.rb +76 -10
  139. data/lib/project_types/script/script_project.rb +26 -16
  140. data/lib/project_types/script/ui/error_handler.rb +135 -50
  141. data/lib/project_types/script/ui/printing_spinner.rb +1 -1
  142. data/lib/project_types/script/ui/strict_spinner.rb +1 -1
  143. data/lib/project_types/theme/cli.rb +40 -0
  144. data/lib/project_types/theme/commands/connect.rb +54 -0
  145. data/lib/project_types/theme/commands/create.rb +48 -0
  146. data/lib/project_types/theme/commands/deploy.rb +38 -0
  147. data/lib/project_types/theme/commands/generate.rb +20 -0
  148. data/lib/project_types/theme/commands/generate/env.rb +79 -0
  149. data/lib/project_types/theme/commands/push.rb +55 -0
  150. data/lib/project_types/theme/commands/serve.rb +31 -0
  151. data/lib/project_types/theme/forms/connect.rb +34 -0
  152. data/lib/project_types/theme/forms/create.rb +22 -0
  153. data/lib/project_types/theme/messages/messages.rb +147 -0
  154. data/lib/project_types/theme/tasks/ensure_themekit_installed.rb +78 -0
  155. data/lib/project_types/theme/themekit.rb +113 -0
  156. data/lib/rubygems_plugin.rb +3 -3
  157. data/lib/shopify-cli/admin_api.rb +52 -12
  158. data/lib/shopify-cli/admin_api/populate_resource_command.rb +17 -17
  159. data/lib/shopify-cli/admin_api/schema.rb +3 -3
  160. data/lib/shopify-cli/api.rb +36 -31
  161. data/lib/shopify-cli/command.rb +1 -1
  162. data/lib/shopify-cli/commands.rb +9 -9
  163. data/lib/shopify-cli/commands/config.rb +28 -28
  164. data/lib/shopify-cli/commands/connect.rb +35 -18
  165. data/lib/shopify-cli/commands/create.rb +5 -5
  166. data/lib/shopify-cli/commands/help.rb +6 -6
  167. data/lib/shopify-cli/commands/logout.rb +3 -3
  168. data/lib/shopify-cli/commands/system.rb +33 -33
  169. data/lib/shopify-cli/commands/version.rb +2 -2
  170. data/lib/shopify-cli/context.rb +43 -22
  171. data/lib/shopify-cli/core.rb +4 -4
  172. data/lib/shopify-cli/core/entry_point.rb +5 -5
  173. data/lib/shopify-cli/core/executor.rb +1 -1
  174. data/lib/shopify-cli/core/help_resolver.rb +2 -2
  175. data/lib/shopify-cli/core/monorail.rb +17 -16
  176. data/lib/shopify-cli/db.rb +2 -2
  177. data/lib/shopify-cli/feature.rb +1 -3
  178. data/lib/shopify-cli/form.rb +1 -1
  179. data/lib/shopify-cli/git.rb +17 -17
  180. data/lib/shopify-cli/helpers.rb +1 -1
  181. data/lib/shopify-cli/helpers/haikunator.rb +1 -1
  182. data/lib/shopify-cli/heroku.rb +28 -28
  183. data/lib/shopify-cli/http_request.rb +21 -9
  184. data/lib/shopify-cli/js_deps.rb +13 -13
  185. data/lib/shopify-cli/js_system.rb +5 -5
  186. data/lib/shopify-cli/lazy_delegator.rb +55 -0
  187. data/lib/shopify-cli/messages/messages.rb +21 -10
  188. data/lib/shopify-cli/method_object.rb +104 -0
  189. data/lib/shopify-cli/oauth.rb +25 -25
  190. data/lib/shopify-cli/oauth/servlet.rb +9 -9
  191. data/lib/shopify-cli/options.rb +3 -3
  192. data/lib/shopify-cli/packager.rb +24 -24
  193. data/lib/shopify-cli/partners_api.rb +38 -16
  194. data/lib/shopify-cli/partners_api/organizations.rb +10 -10
  195. data/lib/shopify-cli/process_supervision.rb +8 -8
  196. data/lib/shopify-cli/project.rb +27 -23
  197. data/lib/shopify-cli/project_type.rb +21 -5
  198. data/lib/shopify-cli/resolve_constant.rb +25 -0
  199. data/lib/shopify-cli/resources.rb +1 -1
  200. data/lib/shopify-cli/resources/env_file.rb +9 -9
  201. data/lib/shopify-cli/result.rb +432 -0
  202. data/lib/shopify-cli/shopifolk.rb +35 -18
  203. data/lib/shopify-cli/sub_command.rb +1 -1
  204. data/lib/shopify-cli/task.rb +9 -1
  205. data/lib/shopify-cli/tasks.rb +7 -7
  206. data/lib/shopify-cli/tasks/create_api_client.rb +13 -4
  207. data/lib/shopify-cli/tasks/ensure_dev_store.rb +12 -12
  208. data/lib/shopify-cli/tasks/ensure_env.rb +18 -15
  209. data/lib/shopify-cli/tasks/ensure_loopback_url.rb +4 -4
  210. data/lib/shopify-cli/tasks/select_org_and_shop.rb +28 -24
  211. data/lib/shopify-cli/tasks/update_dashboard_urls.rb +10 -10
  212. data/lib/shopify-cli/transform_data_structure.rb +86 -0
  213. data/lib/shopify-cli/tunnel.rb +36 -30
  214. data/lib/shopify-cli/version.rb +1 -1
  215. data/lib/shopify_cli.rb +57 -52
  216. data/shopify-cli.gemspec +6 -6
  217. data/shopify.fish +1 -1
  218. data/shopify.sh +1 -1
  219. data/vendor/deps/cli-kit/REVISION +1 -1
  220. data/vendor/deps/cli-kit/lib/cli/kit/logger.rb +2 -2
  221. data/vendor/deps/cli-kit/lib/cli/kit/system.rb +3 -3
  222. data/vendor/deps/cli-ui/REVISION +1 -1
  223. data/vendor/deps/cli-ui/lib/cli/ui.rb +26 -22
  224. data/vendor/deps/cli-ui/lib/cli/ui/ansi.rb +4 -6
  225. data/vendor/deps/cli-ui/lib/cli/ui/frame.rb +3 -3
  226. data/vendor/deps/cli-ui/lib/cli/ui/frame/frame_stack.rb +8 -9
  227. data/vendor/deps/cli-ui/lib/cli/ui/frame/frame_style.rb +1 -1
  228. data/vendor/deps/cli-ui/lib/cli/ui/glyph.rb +1 -0
  229. data/vendor/deps/cli-ui/lib/cli/ui/printer.rb +15 -3
  230. data/vendor/deps/cli-ui/lib/cli/ui/prompt/interactive_options.rb +4 -11
  231. data/vendor/deps/cli-ui/lib/cli/ui/spinner.rb +3 -5
  232. data/vendor/deps/cli-ui/lib/cli/ui/terminal.rb +10 -10
  233. data/vendor/deps/cli-ui/lib/cli/ui/version.rb +1 -1
  234. data/vendor/deps/cli-ui/lib/cli/ui/wrap.rb +56 -0
  235. data/vendor/deps/webrick/.gitignore +9 -0
  236. data/vendor/deps/webrick/Gemfile +3 -0
  237. data/vendor/deps/webrick/LICENSE.txt +22 -0
  238. data/vendor/deps/webrick/README.md +61 -0
  239. data/vendor/deps/webrick/Rakefile +10 -0
  240. data/vendor/deps/webrick/lib/webrick.rb +232 -0
  241. data/vendor/deps/webrick/lib/webrick/accesslog.rb +157 -0
  242. data/vendor/deps/webrick/lib/webrick/cgi.rb +313 -0
  243. data/vendor/deps/webrick/lib/webrick/compat.rb +36 -0
  244. data/vendor/deps/webrick/lib/webrick/config.rb +158 -0
  245. data/vendor/deps/webrick/lib/webrick/cookie.rb +172 -0
  246. data/vendor/deps/webrick/lib/webrick/htmlutils.rb +30 -0
  247. data/vendor/deps/webrick/lib/webrick/httpauth.rb +96 -0
  248. data/vendor/deps/webrick/lib/webrick/httpauth/authenticator.rb +117 -0
  249. data/vendor/deps/webrick/lib/webrick/httpauth/basicauth.rb +116 -0
  250. data/vendor/deps/webrick/lib/webrick/httpauth/digestauth.rb +395 -0
  251. data/vendor/deps/webrick/lib/webrick/httpauth/htdigest.rb +132 -0
  252. data/vendor/deps/webrick/lib/webrick/httpauth/htgroup.rb +97 -0
  253. data/vendor/deps/webrick/lib/webrick/httpauth/htpasswd.rb +158 -0
  254. data/vendor/deps/webrick/lib/webrick/httpauth/userdb.rb +53 -0
  255. data/vendor/deps/webrick/lib/webrick/httpproxy.rb +354 -0
  256. data/vendor/deps/webrick/lib/webrick/httprequest.rb +636 -0
  257. data/vendor/deps/webrick/lib/webrick/httpresponse.rb +564 -0
  258. data/vendor/deps/webrick/lib/webrick/https.rb +152 -0
  259. data/vendor/deps/webrick/lib/webrick/httpserver.rb +294 -0
  260. data/vendor/deps/webrick/lib/webrick/httpservlet.rb +23 -0
  261. data/vendor/deps/webrick/lib/webrick/httpservlet/abstract.rb +152 -0
  262. data/vendor/deps/webrick/lib/webrick/httpservlet/cgi_runner.rb +47 -0
  263. data/vendor/deps/webrick/lib/webrick/httpservlet/cgihandler.rb +126 -0
  264. data/vendor/deps/webrick/lib/webrick/httpservlet/erbhandler.rb +88 -0
  265. data/vendor/deps/webrick/lib/webrick/httpservlet/filehandler.rb +552 -0
  266. data/vendor/deps/webrick/lib/webrick/httpservlet/prochandler.rb +47 -0
  267. data/vendor/deps/webrick/lib/webrick/httpstatus.rb +194 -0
  268. data/vendor/deps/webrick/lib/webrick/httputils.rb +512 -0
  269. data/vendor/deps/webrick/lib/webrick/httpversion.rb +76 -0
  270. data/vendor/deps/webrick/lib/webrick/log.rb +156 -0
  271. data/vendor/deps/webrick/lib/webrick/server.rb +381 -0
  272. data/vendor/deps/webrick/lib/webrick/ssl.rb +215 -0
  273. data/vendor/deps/webrick/lib/webrick/utils.rb +265 -0
  274. data/vendor/deps/webrick/lib/webrick/version.rb +18 -0
  275. data/vendor/deps/webrick/webrick.gemspec +74 -0
  276. data/vendor/gen/template/bin/update-deps +9 -9
  277. metadata +83 -29
  278. data/docs/Gemfile +0 -5
  279. data/docs/Gemfile.lock +0 -258
  280. data/docs/_data/nav.yml +0 -35
  281. data/docs/_includes/footer.html +0 -15
  282. data/docs/_includes/head.html +0 -19
  283. data/docs/_includes/sidebar_nav.html +0 -22
  284. data/docs/_includes/toc.html +0 -112
  285. data/docs/_layouts/default.html +0 -79
  286. data/docs/css/docs.css +0 -157
  287. data/docs/images/header.png +0 -0
  288. data/docs/installing-ruby.md +0 -28
  289. data/lib/project_types/extension/features/argo/admin.rb +0 -20
  290. data/lib/project_types/extension/features/argo/base.rb +0 -129
  291. data/lib/project_types/extension/features/argo/checkout.rb +0 -20
  292. data/lib/project_types/extension/forms/register.rb +0 -47
  293. data/lib/project_types/extension/models/type.rb +0 -81
  294. data/lib/project_types/extension/models/types/checkout_post_purchase.rb +0 -23
  295. data/lib/project_types/extension/models/types/product_subscription.rb +0 -24
  296. data/lib/project_types/node/commands/generate/billing.rb +0 -39
  297. data/lib/project_types/node/commands/generate/page.rb +0 -59
  298. data/lib/project_types/node/commands/generate/webhook.rb +0 -37
  299. data/lib/project_types/script/layers/domain/script.rb +0 -18
  300. data/lib/project_types/script/layers/infrastructure/assemblyscript_tsconfig.rb +0 -38
  301. data/lib/project_types/script/layers/infrastructure/script_repository.rb +0 -59
  302. data/lib/project_types/script/templates/ts/as-pect.config.js +0 -27
  303. data/lib/project_types/script/templates/ts/as-pect.d.ts +0 -1
@@ -0,0 +1,117 @@
1
+ # frozen_string_literal: true
2
+ #--
3
+ # httpauth/authenticator.rb -- Authenticator mix-in module.
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: authenticator.rb,v 1.3 2003/02/20 07:15:47 gotoyuzo Exp $
10
+
11
+ module WEBrick
12
+ module HTTPAuth
13
+
14
+ ##
15
+ # Module providing generic support for both Digest and Basic
16
+ # authentication schemes.
17
+
18
+ module Authenticator
19
+
20
+ RequestField = "Authorization" # :nodoc:
21
+ ResponseField = "WWW-Authenticate" # :nodoc:
22
+ ResponseInfoField = "Authentication-Info" # :nodoc:
23
+ AuthException = HTTPStatus::Unauthorized # :nodoc:
24
+
25
+ ##
26
+ # Method of authentication, must be overridden by the including class
27
+
28
+ AuthScheme = nil
29
+
30
+ ##
31
+ # The realm this authenticator covers
32
+
33
+ attr_reader :realm
34
+
35
+ ##
36
+ # The user database for this authenticator
37
+
38
+ attr_reader :userdb
39
+
40
+ ##
41
+ # The logger for this authenticator
42
+
43
+ attr_reader :logger
44
+
45
+ private
46
+
47
+ # :stopdoc:
48
+
49
+ ##
50
+ # Initializes the authenticator from +config+
51
+
52
+ def check_init(config)
53
+ [:UserDB, :Realm].each{|sym|
54
+ unless config[sym]
55
+ raise ArgumentError, "Argument #{sym.inspect} missing."
56
+ end
57
+ }
58
+ @realm = config[:Realm]
59
+ @userdb = config[:UserDB]
60
+ @logger = config[:Logger] || Log::new($stderr)
61
+ @reload_db = config[:AutoReloadUserDB]
62
+ @request_field = self::class::RequestField
63
+ @response_field = self::class::ResponseField
64
+ @resp_info_field = self::class::ResponseInfoField
65
+ @auth_exception = self::class::AuthException
66
+ @auth_scheme = self::class::AuthScheme
67
+ end
68
+
69
+ ##
70
+ # Ensures +req+ has credentials that can be authenticated.
71
+
72
+ def check_scheme(req)
73
+ unless credentials = req[@request_field]
74
+ error("no credentials in the request.")
75
+ return nil
76
+ end
77
+ unless match = /^#{@auth_scheme}\s+/i.match(credentials)
78
+ error("invalid scheme in %s.", credentials)
79
+ info("%s: %s", @request_field, credentials) if $DEBUG
80
+ return nil
81
+ end
82
+ return match.post_match
83
+ end
84
+
85
+ def log(meth, fmt, *args)
86
+ msg = format("%s %s: ", @auth_scheme, @realm)
87
+ msg << fmt % args
88
+ @logger.__send__(meth, msg)
89
+ end
90
+
91
+ def error(fmt, *args)
92
+ if @logger.error?
93
+ log(:error, fmt, *args)
94
+ end
95
+ end
96
+
97
+ def info(fmt, *args)
98
+ if @logger.info?
99
+ log(:info, fmt, *args)
100
+ end
101
+ end
102
+
103
+ # :startdoc:
104
+ end
105
+
106
+ ##
107
+ # Module providing generic support for both Digest and Basic
108
+ # authentication schemes for proxies.
109
+
110
+ module ProxyAuthenticator
111
+ RequestField = "Proxy-Authorization" # :nodoc:
112
+ ResponseField = "Proxy-Authenticate" # :nodoc:
113
+ InfoField = "Proxy-Authentication-Info" # :nodoc:
114
+ AuthException = HTTPStatus::ProxyAuthenticationRequired # :nodoc:
115
+ end
116
+ end
117
+ end
@@ -0,0 +1,116 @@
1
+ # frozen_string_literal: true
2
+ #
3
+ # httpauth/basicauth.rb -- HTTP basic access authentication
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: basicauth.rb,v 1.5 2003/02/20 07:15:47 gotoyuzo Exp $
10
+
11
+ require_relative '../config'
12
+ require_relative '../httpstatus'
13
+ require_relative 'authenticator'
14
+
15
+ module WEBrick
16
+ module HTTPAuth
17
+
18
+ ##
19
+ # Basic Authentication for WEBrick
20
+ #
21
+ # Use this class to add basic authentication to a WEBrick servlet.
22
+ #
23
+ # Here is an example of how to set up a BasicAuth:
24
+ #
25
+ # config = { :Realm => 'BasicAuth example realm' }
26
+ #
27
+ # htpasswd = WEBrick::HTTPAuth::Htpasswd.new 'my_password_file', password_hash: :bcrypt
28
+ # htpasswd.set_passwd config[:Realm], 'username', 'password'
29
+ # htpasswd.flush
30
+ #
31
+ # config[:UserDB] = htpasswd
32
+ #
33
+ # basic_auth = WEBrick::HTTPAuth::BasicAuth.new config
34
+
35
+ class BasicAuth
36
+ include Authenticator
37
+
38
+ AuthScheme = "Basic" # :nodoc:
39
+
40
+ ##
41
+ # Used by UserDB to create a basic password entry
42
+
43
+ def self.make_passwd(realm, user, pass)
44
+ pass ||= ""
45
+ pass.crypt(Utils::random_string(2))
46
+ end
47
+
48
+ attr_reader :realm, :userdb, :logger
49
+
50
+ ##
51
+ # Creates a new BasicAuth instance.
52
+ #
53
+ # See WEBrick::Config::BasicAuth for default configuration entries
54
+ #
55
+ # You must supply the following configuration entries:
56
+ #
57
+ # :Realm:: The name of the realm being protected.
58
+ # :UserDB:: A database of usernames and passwords.
59
+ # A WEBrick::HTTPAuth::Htpasswd instance should be used.
60
+
61
+ def initialize(config, default=Config::BasicAuth)
62
+ check_init(config)
63
+ @config = default.dup.update(config)
64
+ end
65
+
66
+ ##
67
+ # Authenticates a +req+ and returns a 401 Unauthorized using +res+ if
68
+ # the authentication was not correct.
69
+
70
+ def authenticate(req, res)
71
+ unless basic_credentials = check_scheme(req)
72
+ challenge(req, res)
73
+ end
74
+ userid, password = basic_credentials.unpack("m*")[0].split(":", 2)
75
+ password ||= ""
76
+ if userid.empty?
77
+ error("user id was not given.")
78
+ challenge(req, res)
79
+ end
80
+ unless encpass = @userdb.get_passwd(@realm, userid, @reload_db)
81
+ error("%s: the user is not allowed.", userid)
82
+ challenge(req, res)
83
+ end
84
+
85
+ case encpass
86
+ when /\A\$2[aby]\$/
87
+ password_matches = BCrypt::Password.new(encpass.sub(/\A\$2[aby]\$/, '$2a$')) == password
88
+ else
89
+ password_matches = password.crypt(encpass) == encpass
90
+ end
91
+
92
+ unless password_matches
93
+ error("%s: password unmatch.", userid)
94
+ challenge(req, res)
95
+ end
96
+ info("%s: authentication succeeded.", userid)
97
+ req.user = userid
98
+ end
99
+
100
+ ##
101
+ # Returns a challenge response which asks for authentication information
102
+
103
+ def challenge(req, res)
104
+ res[@response_field] = "#{@auth_scheme} realm=\"#{@realm}\""
105
+ raise @auth_exception
106
+ end
107
+ end
108
+
109
+ ##
110
+ # Basic authentication for proxy servers. See BasicAuth for details.
111
+
112
+ class ProxyBasicAuth < BasicAuth
113
+ include ProxyAuthenticator
114
+ end
115
+ end
116
+ end
@@ -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