knife 18.6.2 → 18.7.9

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 (673) hide show
  1. checksums.yaml +4 -4
  2. data/Gemfile +23 -27
  3. data/LICENSE +201 -201
  4. data/Rakefile +33 -33
  5. data/bin/knife +24 -24
  6. data/knife.gemspec +65 -64
  7. data/lib/chef/application/knife.rb +237 -237
  8. data/lib/chef/chef_fs/knife.rb +162 -162
  9. data/lib/chef/knife/acl_add.rb +57 -57
  10. data/lib/chef/knife/acl_base.rb +183 -183
  11. data/lib/chef/knife/acl_bulk_add.rb +78 -78
  12. data/lib/chef/knife/acl_bulk_remove.rb +83 -83
  13. data/lib/chef/knife/acl_remove.rb +62 -62
  14. data/lib/chef/knife/acl_show.rb +56 -56
  15. data/lib/chef/knife/bootstrap/chef_vault_handler.rb +160 -160
  16. data/lib/chef/knife/bootstrap/client_builder.rb +212 -212
  17. data/lib/chef/knife/bootstrap/templates/README.md +11 -11
  18. data/lib/chef/knife/bootstrap/templates/chef-full.erb +249 -249
  19. data/lib/chef/knife/bootstrap/templates/windows-chef-client-msi.erb +278 -278
  20. data/lib/chef/knife/bootstrap/train_connector.rb +334 -334
  21. data/lib/chef/knife/bootstrap.rb +1229 -1219
  22. data/lib/chef/knife/client_bulk_delete.rb +104 -104
  23. data/lib/chef/knife/client_create.rb +140 -140
  24. data/lib/chef/knife/client_delete.rb +62 -62
  25. data/lib/chef/knife/client_edit.rb +52 -52
  26. data/lib/chef/knife/client_key_create.rb +73 -73
  27. data/lib/chef/knife/client_key_delete.rb +80 -80
  28. data/lib/chef/knife/client_key_edit.rb +83 -83
  29. data/lib/chef/knife/client_key_list.rb +73 -73
  30. data/lib/chef/knife/client_key_show.rb +80 -80
  31. data/lib/chef/knife/client_list.rb +41 -41
  32. data/lib/chef/knife/client_reregister.rb +58 -58
  33. data/lib/chef/knife/client_show.rb +48 -48
  34. data/lib/chef/knife/config_get.rb +39 -39
  35. data/lib/chef/knife/config_get_profile.rb +37 -37
  36. data/lib/chef/knife/config_list.rb +139 -139
  37. data/lib/chef/knife/config_list_profiles.rb +37 -37
  38. data/lib/chef/knife/config_show.rb +127 -127
  39. data/lib/chef/knife/config_use.rb +61 -61
  40. data/lib/chef/knife/config_use_profile.rb +47 -47
  41. data/lib/chef/knife/configure.rb +150 -150
  42. data/lib/chef/knife/configure_client.rb +48 -48
  43. data/lib/chef/knife/cookbook_bulk_delete.rb +71 -71
  44. data/lib/chef/knife/cookbook_delete.rb +151 -151
  45. data/lib/chef/knife/cookbook_download.rb +142 -142
  46. data/lib/chef/knife/cookbook_list.rb +47 -47
  47. data/lib/chef/knife/cookbook_metadata.rb +106 -106
  48. data/lib/chef/knife/cookbook_metadata_from_file.rb +49 -49
  49. data/lib/chef/knife/cookbook_show.rb +98 -98
  50. data/lib/chef/knife/cookbook_upload.rb +313 -313
  51. data/lib/chef/knife/core/bootstrap_context.rb +274 -274
  52. data/lib/chef/knife/core/cookbook_scm_repo.rb +159 -159
  53. data/lib/chef/knife/core/cookbook_site_streaming_uploader.rb +249 -249
  54. data/lib/chef/knife/core/formatting_options.rb +49 -49
  55. data/lib/chef/knife/core/gem_glob_loader.rb +134 -134
  56. data/lib/chef/knife/core/generic_presenter.rb +238 -238
  57. data/lib/chef/knife/core/hashed_command_loader.rb +100 -100
  58. data/lib/chef/knife/core/node_editor.rb +130 -130
  59. data/lib/chef/knife/core/node_presenter.rb +133 -133
  60. data/lib/chef/knife/core/object_loader.rb +115 -115
  61. data/lib/chef/knife/core/status_presenter.rb +147 -147
  62. data/lib/chef/knife/core/subcommand_loader.rb +208 -208
  63. data/lib/chef/knife/core/text_formatter.rb +85 -85
  64. data/lib/chef/knife/core/ui.rb +339 -339
  65. data/lib/chef/knife/core/windows_bootstrap_context.rb +441 -443
  66. data/lib/chef/knife/data_bag_create.rb +81 -81
  67. data/lib/chef/knife/data_bag_delete.rb +49 -49
  68. data/lib/chef/knife/data_bag_edit.rb +74 -74
  69. data/lib/chef/knife/data_bag_from_file.rb +113 -113
  70. data/lib/chef/knife/data_bag_list.rb +42 -42
  71. data/lib/chef/knife/data_bag_secret_options.rb +122 -122
  72. data/lib/chef/knife/data_bag_show.rb +69 -69
  73. data/lib/chef/knife/delete.rb +125 -125
  74. data/lib/chef/knife/deps.rb +156 -156
  75. data/lib/chef/knife/diff.rb +83 -83
  76. data/lib/chef/knife/download.rb +85 -85
  77. data/lib/chef/knife/edit.rb +88 -88
  78. data/lib/chef/knife/environment_compare.rb +128 -128
  79. data/lib/chef/knife/environment_create.rb +52 -52
  80. data/lib/chef/knife/environment_delete.rb +44 -44
  81. data/lib/chef/knife/environment_edit.rb +44 -44
  82. data/lib/chef/knife/environment_from_file.rb +84 -84
  83. data/lib/chef/knife/environment_list.rb +41 -41
  84. data/lib/chef/knife/environment_show.rb +47 -47
  85. data/lib/chef/knife/exec.rb +99 -99
  86. data/lib/chef/knife/group_add.rb +55 -55
  87. data/lib/chef/knife/group_create.rb +49 -49
  88. data/lib/chef/knife/group_destroy.rb +53 -53
  89. data/lib/chef/knife/group_list.rb +43 -43
  90. data/lib/chef/knife/group_remove.rb +56 -56
  91. data/lib/chef/knife/group_show.rb +49 -49
  92. data/lib/chef/knife/key_create.rb +112 -112
  93. data/lib/chef/knife/key_create_base.rb +50 -50
  94. data/lib/chef/knife/key_delete.rb +55 -55
  95. data/lib/chef/knife/key_edit.rb +118 -118
  96. data/lib/chef/knife/key_edit_base.rb +55 -55
  97. data/lib/chef/knife/key_list.rb +90 -90
  98. data/lib/chef/knife/key_list_base.rb +45 -45
  99. data/lib/chef/knife/key_show.rb +53 -53
  100. data/lib/chef/knife/license.rb +52 -52
  101. data/lib/chef/knife/list.rb +177 -177
  102. data/lib/chef/knife/node_bulk_delete.rb +75 -75
  103. data/lib/chef/knife/node_create.rb +47 -47
  104. data/lib/chef/knife/node_delete.rb +46 -46
  105. data/lib/chef/knife/node_edit.rb +70 -70
  106. data/lib/chef/knife/node_environment_set.rb +53 -53
  107. data/lib/chef/knife/node_from_file.rb +51 -51
  108. data/lib/chef/knife/node_list.rb +44 -44
  109. data/lib/chef/knife/node_policy_set.rb +79 -79
  110. data/lib/chef/knife/node_run_list_add.rb +104 -104
  111. data/lib/chef/knife/node_run_list_remove.rb +67 -67
  112. data/lib/chef/knife/node_run_list_set.rb +66 -66
  113. data/lib/chef/knife/node_show.rb +63 -63
  114. data/lib/chef/knife/null.rb +12 -12
  115. data/lib/chef/knife/org_create.rb +70 -70
  116. data/lib/chef/knife/org_delete.rb +32 -32
  117. data/lib/chef/knife/org_edit.rb +48 -48
  118. data/lib/chef/knife/org_list.rb +44 -44
  119. data/lib/chef/knife/org_show.rb +31 -31
  120. data/lib/chef/knife/org_user_add.rb +62 -62
  121. data/lib/chef/knife/org_user_remove.rb +103 -103
  122. data/lib/chef/knife/raw.rb +123 -123
  123. data/lib/chef/knife/recipe_list.rb +32 -32
  124. data/lib/chef/knife/rehash.rb +50 -50
  125. data/lib/chef/knife/role_bulk_delete.rb +66 -66
  126. data/lib/chef/knife/role_create.rb +53 -53
  127. data/lib/chef/knife/role_delete.rb +46 -46
  128. data/lib/chef/knife/role_edit.rb +45 -45
  129. data/lib/chef/knife/role_env_run_list_add.rb +87 -87
  130. data/lib/chef/knife/role_env_run_list_clear.rb +55 -55
  131. data/lib/chef/knife/role_env_run_list_remove.rb +57 -57
  132. data/lib/chef/knife/role_env_run_list_replace.rb +60 -60
  133. data/lib/chef/knife/role_env_run_list_set.rb +70 -70
  134. data/lib/chef/knife/role_from_file.rb +51 -51
  135. data/lib/chef/knife/role_list.rb +42 -42
  136. data/lib/chef/knife/role_run_list_add.rb +87 -87
  137. data/lib/chef/knife/role_run_list_clear.rb +55 -55
  138. data/lib/chef/knife/role_run_list_remove.rb +56 -56
  139. data/lib/chef/knife/role_run_list_replace.rb +60 -60
  140. data/lib/chef/knife/role_run_list_set.rb +69 -69
  141. data/lib/chef/knife/role_show.rb +48 -48
  142. data/lib/chef/knife/search.rb +194 -194
  143. data/lib/chef/knife/serve.rb +65 -65
  144. data/lib/chef/knife/show.rb +72 -72
  145. data/lib/chef/knife/ssh.rb +657 -657
  146. data/lib/chef/knife/ssl_check.rb +284 -284
  147. data/lib/chef/knife/ssl_fetch.rb +162 -162
  148. data/lib/chef/knife/status.rb +95 -95
  149. data/lib/chef/knife/supermarket_download.rb +119 -119
  150. data/lib/chef/knife/supermarket_install.rb +192 -192
  151. data/lib/chef/knife/supermarket_list.rb +76 -76
  152. data/lib/chef/knife/supermarket_search.rb +53 -53
  153. data/lib/chef/knife/supermarket_share.rb +166 -166
  154. data/lib/chef/knife/supermarket_show.rb +66 -66
  155. data/lib/chef/knife/supermarket_unshare.rb +62 -62
  156. data/lib/chef/knife/tag_create.rb +52 -52
  157. data/lib/chef/knife/tag_delete.rb +60 -60
  158. data/lib/chef/knife/tag_list.rb +47 -47
  159. data/lib/chef/knife/upload.rb +87 -87
  160. data/lib/chef/knife/user_create.rb +180 -180
  161. data/lib/chef/knife/user_delete.rb +151 -151
  162. data/lib/chef/knife/user_dissociate.rb +42 -42
  163. data/lib/chef/knife/user_edit.rb +94 -94
  164. data/lib/chef/knife/user_invite_add.rb +43 -43
  165. data/lib/chef/knife/user_invite_list.rb +34 -34
  166. data/lib/chef/knife/user_invite_rescind.rb +63 -63
  167. data/lib/chef/knife/user_key_create.rb +73 -73
  168. data/lib/chef/knife/user_key_delete.rb +80 -80
  169. data/lib/chef/knife/user_key_edit.rb +83 -83
  170. data/lib/chef/knife/user_key_list.rb +73 -73
  171. data/lib/chef/knife/user_key_show.rb +80 -80
  172. data/lib/chef/knife/user_list.rb +43 -43
  173. data/lib/chef/knife/user_password.rb +70 -70
  174. data/lib/chef/knife/user_reregister.rb +59 -59
  175. data/lib/chef/knife/user_show.rb +52 -52
  176. data/lib/chef/knife/version.rb +24 -24
  177. data/lib/chef/knife/xargs.rb +282 -282
  178. data/lib/chef/knife/yaml_convert.rb +91 -91
  179. data/lib/chef/knife.rb +677 -677
  180. data/lib/chef/utils/licensing_config.rb +9 -9
  181. data/lib/chef/utils/licensing_handler.rb +72 -46
  182. data/spec/data/apt/chef-integration-test-1.0/debian/changelog +5 -5
  183. data/spec/data/apt/chef-integration-test-1.0/debian/compat +1 -1
  184. data/spec/data/apt/chef-integration-test-1.0/debian/control +13 -13
  185. data/spec/data/apt/chef-integration-test-1.0/debian/copyright +34 -34
  186. data/spec/data/apt/chef-integration-test-1.0/debian/files +1 -1
  187. data/spec/data/apt/chef-integration-test-1.0/debian/rules +13 -13
  188. data/spec/data/apt/chef-integration-test-1.0/debian/source/format +1 -1
  189. data/spec/data/apt/chef-integration-test-1.1/debian/changelog +11 -11
  190. data/spec/data/apt/chef-integration-test-1.1/debian/compat +1 -1
  191. data/spec/data/apt/chef-integration-test-1.1/debian/control +13 -13
  192. data/spec/data/apt/chef-integration-test-1.1/debian/copyright +34 -34
  193. data/spec/data/apt/chef-integration-test-1.1/debian/files +1 -1
  194. data/spec/data/apt/chef-integration-test-1.1/debian/rules +13 -13
  195. data/spec/data/apt/chef-integration-test-1.1/debian/source/format +1 -1
  196. data/spec/data/apt/chef-integration-test2-1.0/debian/changelog +5 -5
  197. data/spec/data/apt/chef-integration-test2-1.0/debian/chef-integration-test2/DEBIAN/conffiles +1 -1
  198. data/spec/data/apt/chef-integration-test2-1.0/debian/chef-integration-test2/DEBIAN/control +10 -10
  199. data/spec/data/apt/chef-integration-test2-1.0/debian/chef-integration-test2/DEBIAN/md5sums +1 -1
  200. data/spec/data/apt/chef-integration-test2-1.0/debian/chef-integration-test2.debhelper.log +45 -45
  201. data/spec/data/apt/chef-integration-test2-1.0/debian/chef-integration-test2.substvars +1 -1
  202. data/spec/data/apt/chef-integration-test2-1.0/debian/compat +1 -1
  203. data/spec/data/apt/chef-integration-test2-1.0/debian/conffiles +1 -1
  204. data/spec/data/apt/chef-integration-test2-1.0/debian/control +13 -13
  205. data/spec/data/apt/chef-integration-test2-1.0/debian/copyright +34 -34
  206. data/spec/data/apt/chef-integration-test2-1.0/debian/files +1 -1
  207. data/spec/data/apt/chef-integration-test2-1.0/debian/rules +13 -13
  208. data/spec/data/apt/chef-integration-test2-1.0/debian/source/format +1 -1
  209. data/spec/data/apt/chef-integration-test2_1.0-1.dsc +18 -18
  210. data/spec/data/apt/chef-integration-test2_1.0-1_amd64.build +91 -91
  211. data/spec/data/apt/chef-integration-test2_1.0-1_amd64.changes +31 -31
  212. data/spec/data/apt/chef-integration-test_1.0-1_amd64.changes +22 -22
  213. data/spec/data/apt/chef-integration-test_1.1-1_amd64.changes +22 -22
  214. data/spec/data/apt/var/www/apt/conf/distributions +7 -7
  215. data/spec/data/apt/var/www/apt/conf/incoming +4 -4
  216. data/spec/data/apt/var/www/apt/conf/pulls +3 -3
  217. data/spec/data/apt/var/www/apt/db/version +4 -4
  218. data/spec/data/apt/var/www/apt/dists/sid/Release +19 -19
  219. data/spec/data/apt/var/www/apt/dists/sid/main/binary-amd64/Packages +16 -16
  220. data/spec/data/apt/var/www/apt/dists/sid/main/binary-amd64/Release +5 -5
  221. data/spec/data/bootstrap/encrypted_data_bag_secret +1 -1
  222. data/spec/data/bootstrap/no_proxy.erb +2 -2
  223. data/spec/data/bootstrap/secret.erb +9 -9
  224. data/spec/data/bootstrap/test-hints.erb +12 -12
  225. data/spec/data/bootstrap/test.erb +1 -1
  226. data/spec/data/cb_version_cookbooks/tatft/README.rdoc +2 -2
  227. data/spec/data/cb_version_cookbooks/tatft/attributes/default.rb +1 -1
  228. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-600hhz-0 +1 -1
  229. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-ahd2gq-0 +1 -1
  230. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-api8ux-0 +1 -1
  231. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-b0r1m1-0 +1 -1
  232. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-bfygsi-0 +1 -1
  233. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-el14l6-0 +1 -1
  234. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-ivrl3y-0 +1 -1
  235. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-kkbs85-0 +1 -1
  236. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-ory1ux-0 +1 -1
  237. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-pgsq76-0 +1 -1
  238. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-ra8uim-0 +1 -1
  239. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-t7k1g-0 +1 -1
  240. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-t8g0sv-0 +1 -1
  241. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-ufy6g3-0 +1 -1
  242. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-x2d6j9-0 +1 -1
  243. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-xi0l6h-0 +1 -1
  244. data/spec/data/client.d_00/00-foo.rb +2 -2
  245. data/spec/data/client.d_00/01-bar.rb +1 -1
  246. data/spec/data/client.d_00/02-strings.rb +2 -2
  247. data/spec/data/client.d_00/bar +1 -1
  248. data/spec/data/client.d_01/foo/bar.rb +1 -1
  249. data/spec/data/client.d_02/foo.rb/foo.txt +1 -1
  250. data/spec/data/config.rb +6 -6
  251. data/spec/data/cookbooks/angrybash/metadata.rb +2 -2
  252. data/spec/data/cookbooks/angrybash/recipes/default.rb +8 -8
  253. data/spec/data/cookbooks/apache2/files/default/apache2_module_conf_generate.pl +2 -2
  254. data/spec/data/cookbooks/apache2/metadata.json +33 -33
  255. data/spec/data/cookbooks/apache2/metadata.rb +2 -2
  256. data/spec/data/cookbooks/apache2/recipes/default.rb +2 -2
  257. data/spec/data/cookbooks/borken/metadata.rb +2 -2
  258. data/spec/data/cookbooks/borken/recipes/default.rb +1 -1
  259. data/spec/data/cookbooks/borken/templates/default/borken.erb +1 -1
  260. data/spec/data/cookbooks/chefignore +8 -8
  261. data/spec/data/cookbooks/ignorken/files/default/not_me.rb +2 -2
  262. data/spec/data/cookbooks/ignorken/metadata.rb +2 -2
  263. data/spec/data/cookbooks/ignorken/recipes/ignoreme.rb +1 -1
  264. data/spec/data/cookbooks/ignorken/templates/ubuntu-12.10/not_me.rb +2 -2
  265. data/spec/data/cookbooks/irssi/files/default/irssi.response +2 -2
  266. data/spec/data/cookbooks/java/files/default/java.response +1 -1
  267. data/spec/data/cookbooks/java/metadata.json +33 -33
  268. data/spec/data/cookbooks/java/metadata.rb +2 -2
  269. data/spec/data/cookbooks/name-mismatch-versionnumber/README.md +4 -4
  270. data/spec/data/cookbooks/name-mismatch-versionnumber/metadata.rb +8 -8
  271. data/spec/data/cookbooks/name-mismatch-versionnumber/recipes/default.rb +8 -8
  272. data/spec/data/cookbooks/openldap/attributes/default.rb +16 -16
  273. data/spec/data/cookbooks/openldap/attributes/smokey.rb +1 -1
  274. data/spec/data/cookbooks/openldap/definitions/client.rb +5 -5
  275. data/spec/data/cookbooks/openldap/definitions/server.rb +5 -5
  276. data/spec/data/cookbooks/openldap/files/default/.dotfile +1 -1
  277. data/spec/data/cookbooks/openldap/files/default/remotedir/.a_dotdir/.a_dotfile_in_a_dotdir +1 -1
  278. data/spec/data/cookbooks/openldap/files/default/remotedir/not_a_template.erb +2 -2
  279. data/spec/data/cookbooks/openldap/files/default/remotedir/remote_dir_file1.txt +2 -2
  280. data/spec/data/cookbooks/openldap/files/default/remotedir/remote_dir_file2.txt +2 -2
  281. data/spec/data/cookbooks/openldap/files/default/remotedir/remotesubdir/.a_dotfile +1 -1
  282. data/spec/data/cookbooks/openldap/files/default/remotedir/remotesubdir/remote_subdir_file1.txt +2 -2
  283. data/spec/data/cookbooks/openldap/files/default/remotedir/remotesubdir/remote_subdir_file2.txt +2 -2
  284. data/spec/data/cookbooks/openldap/files/default/remotedir/subdir_with_no_file_just_a_subsubdir/the_subsubdir/some_file.txt +3 -3
  285. data/spec/data/cookbooks/openldap/libraries/openldap/version.rb +3 -3
  286. data/spec/data/cookbooks/openldap/libraries/openldap.rb +4 -4
  287. data/spec/data/cookbooks/openldap/metadata.rb +8 -8
  288. data/spec/data/cookbooks/openldap/recipes/default.rb +4 -4
  289. data/spec/data/cookbooks/openldap/recipes/gigantor.rb +3 -3
  290. data/spec/data/cookbooks/openldap/recipes/one.rb +15 -15
  291. data/spec/data/cookbooks/openldap/recipes/return.rb +2 -2
  292. data/spec/data/cookbooks/openldap/templates/default/helper_test.erb +1 -1
  293. data/spec/data/cookbooks/openldap/templates/default/helpers.erb +14 -14
  294. data/spec/data/cookbooks/openldap/templates/default/helpers_via_partial_test.erb +1 -1
  295. data/spec/data/cookbooks/openldap/templates/default/nested_openldap_partials.erb +1 -1
  296. data/spec/data/cookbooks/openldap/templates/default/nested_partial.erb +1 -1
  297. data/spec/data/cookbooks/openldap/templates/default/no_windows_line_endings.erb +4 -4
  298. data/spec/data/cookbooks/openldap/templates/default/openldap_nested_variable_stuff.erb +1 -1
  299. data/spec/data/cookbooks/openldap/templates/default/openldap_variable_stuff.conf.erb +1 -1
  300. data/spec/data/cookbooks/openldap/templates/default/test.erb +1 -1
  301. data/spec/data/cookbooks/preseed/files/default/preseed-file.seed +1 -1
  302. data/spec/data/cookbooks/preseed/files/default/preseed-template.seed +4 -4
  303. data/spec/data/cookbooks/preseed/metadata.rb +2 -2
  304. data/spec/data/cookbooks/preseed/templates/default/preseed-template-variables.seed +1 -1
  305. data/spec/data/cookbooks/preseed/templates/default/preseed-template.seed +1 -1
  306. data/spec/data/cookbooks/starter/chefignore +8 -8
  307. data/spec/data/cookbooks/starter/metadata.rb +2 -2
  308. data/spec/data/cookbooks/starter/recipes/default.rb +4 -4
  309. data/spec/data/cookbooks/supports-platform-constraints/metadata.rb +5 -5
  310. data/spec/data/cookbooks/wget/files/default/wget.response +2 -2
  311. data/spec/data/definitions/test.rb +4 -4
  312. data/spec/data/environment-config.rb +4 -4
  313. data/spec/data/file-providers-method-snapshot-chef-11-4.json +127 -127
  314. data/spec/data/fileedit/hosts +4 -4
  315. data/spec/data/incomplete-metadata-chef-repo/incomplete-metadata/README.md +4 -4
  316. data/spec/data/incomplete-metadata-chef-repo/incomplete-metadata/metadata.rb +13 -13
  317. data/spec/data/incomplete-metadata-chef-repo/incomplete-metadata/recipes/default.rb +8 -8
  318. data/spec/data/invalid-metadata-chef-repo/invalid-metadata/README.md +4 -4
  319. data/spec/data/invalid-metadata-chef-repo/invalid-metadata/metadata.rb +9 -9
  320. data/spec/data/invalid-metadata-chef-repo/invalid-metadata/recipes/default.rb +8 -8
  321. data/spec/data/kitchen/chefignore +6 -6
  322. data/spec/data/kitchen/openldap/attributes/default.rb +3 -3
  323. data/spec/data/kitchen/openldap/attributes/robinson.rb +2 -2
  324. data/spec/data/kitchen/openldap/definitions/client.rb +3 -3
  325. data/spec/data/kitchen/openldap/definitions/drewbarrymore.rb +2 -2
  326. data/spec/data/kitchen/openldap/recipes/gigantor.rb +2 -2
  327. data/spec/data/kitchen/openldap/recipes/ignoreme.rb +2 -2
  328. data/spec/data/kitchen/openldap/recipes/woot.rb +3 -3
  329. data/spec/data/knife_subcommand/test_explicit_category.rb +6 -6
  330. data/spec/data/knife_subcommand/test_name_mapping.rb +4 -4
  331. data/spec/data/knife_subcommand/test_yourself.rb +21 -21
  332. data/spec/data/lwrp/providers/buck_passer.rb +28 -28
  333. data/spec/data/lwrp/providers/buck_passer_2.rb +26 -26
  334. data/spec/data/lwrp/providers/embedded_resource_accesses_providers_scope.rb +28 -28
  335. data/spec/data/lwrp/providers/inline_compiler.rb +24 -24
  336. data/spec/data/lwrp/providers/monkey_name_printer.rb +5 -5
  337. data/spec/data/lwrp/providers/paint_drying_watcher.rb +7 -7
  338. data/spec/data/lwrp/providers/thumb_twiddler.rb +7 -7
  339. data/spec/data/lwrp/resources/bar.rb +4 -4
  340. data/spec/data/lwrp/resources/buck_passer.rb +6 -6
  341. data/spec/data/lwrp/resources/buck_passer_2.rb +4 -4
  342. data/spec/data/lwrp/resources/embedded_resource_accesses_providers_scope.rb +4 -4
  343. data/spec/data/lwrp/resources/foo.rb +6 -6
  344. data/spec/data/lwrp/resources/inline_compiler.rb +4 -4
  345. data/spec/data/lwrp/resources/monkey_name_printer.rb +6 -6
  346. data/spec/data/lwrp/resources/paint_drying_watcher.rb +4 -4
  347. data/spec/data/lwrp/resources/thumb_twiddler.rb +4 -4
  348. data/spec/data/lwrp/resources_with_default_attributes/nodeattr.rb +3 -3
  349. data/spec/data/lwrp_const_scoping/resources/conflict.rb +1 -1
  350. data/spec/data/lwrp_override/providers/buck_passer.rb +5 -5
  351. data/spec/data/lwrp_override/resources/foo.rb +11 -11
  352. data/spec/data/mac_users/10.9.plist.xml +560 -560
  353. data/spec/data/mac_users/10.9.shadow.xml +21 -21
  354. data/spec/data/metadata/quick_start/metadata.rb +14 -14
  355. data/spec/data/mixin/invalid_data.rb +3 -3
  356. data/spec/data/mixin/real_data.rb +2 -2
  357. data/spec/data/nested.json +2 -2
  358. data/spec/data/nodes/default.rb +15 -15
  359. data/spec/data/nodes/test.example.com.rb +17 -17
  360. data/spec/data/nodes/test.rb +15 -15
  361. data/spec/data/null_config.rb +1 -1
  362. data/spec/data/object_loader/environments/test.json +7 -7
  363. data/spec/data/object_loader/environments/test.rb +2 -2
  364. data/spec/data/object_loader/environments/test_json_class.json +8 -8
  365. data/spec/data/object_loader/nodes/test.json +7 -7
  366. data/spec/data/object_loader/nodes/test.rb +2 -2
  367. data/spec/data/object_loader/nodes/test_json_class.json +8 -8
  368. data/spec/data/object_loader/roles/test.json +7 -7
  369. data/spec/data/object_loader/roles/test.rb +2 -2
  370. data/spec/data/object_loader/roles/test_json_class.json +8 -8
  371. data/spec/data/partial_one.erb +1 -1
  372. data/spec/data/prefer_metadata_json/metadata.json +51 -51
  373. data/spec/data/prefer_metadata_json/metadata.rb +6 -6
  374. data/spec/data/recipes/test.rb +7 -7
  375. data/spec/data/root_alias_cookbooks/dup_attr/attributes/default.rb +1 -1
  376. data/spec/data/root_alias_cookbooks/dup_attr/attributes.rb +1 -1
  377. data/spec/data/root_alias_cookbooks/dup_attr/metadata.rb +2 -2
  378. data/spec/data/root_alias_cookbooks/dup_attr/recipe.rb +3 -3
  379. data/spec/data/root_alias_cookbooks/dup_recipe/attributes.rb +1 -1
  380. data/spec/data/root_alias_cookbooks/dup_recipe/metadata.rb +2 -2
  381. data/spec/data/root_alias_cookbooks/dup_recipe/recipe.rb +3 -3
  382. data/spec/data/root_alias_cookbooks/dup_recipe/recipes/default.rb +3 -3
  383. data/spec/data/root_alias_cookbooks/simple/attributes.rb +1 -1
  384. data/spec/data/root_alias_cookbooks/simple/metadata.rb +2 -2
  385. data/spec/data/root_alias_cookbooks/simple/recipe.rb +3 -3
  386. data/spec/data/rubygems.org/sexp_processor-info +49 -49
  387. data/spec/data/run_context/cookbooks/circular-dep1/attributes/default.rb +2 -2
  388. data/spec/data/run_context/cookbooks/circular-dep1/definitions/circular_dep1_res.rb +1 -1
  389. data/spec/data/run_context/cookbooks/circular-dep1/libraries/lib.rb +2 -2
  390. data/spec/data/run_context/cookbooks/circular-dep1/metadata.rb +2 -2
  391. data/spec/data/run_context/cookbooks/circular-dep1/providers/provider.rb +1 -1
  392. data/spec/data/run_context/cookbooks/circular-dep1/resources/resource.rb +2 -2
  393. data/spec/data/run_context/cookbooks/circular-dep2/attributes/default.rb +2 -2
  394. data/spec/data/run_context/cookbooks/circular-dep2/definitions/circular_dep2_res.rb +1 -1
  395. data/spec/data/run_context/cookbooks/circular-dep2/libraries/lib.rb +2 -2
  396. data/spec/data/run_context/cookbooks/circular-dep2/metadata.rb +2 -2
  397. data/spec/data/run_context/cookbooks/circular-dep2/providers/provider.rb +1 -1
  398. data/spec/data/run_context/cookbooks/circular-dep2/resources/resource.rb +2 -2
  399. data/spec/data/run_context/cookbooks/dependency1/attributes/aa_first.rb +2 -2
  400. data/spec/data/run_context/cookbooks/dependency1/attributes/default.rb +2 -2
  401. data/spec/data/run_context/cookbooks/dependency1/attributes/unparsed_file +1 -1
  402. data/spec/data/run_context/cookbooks/dependency1/attributes/zz_last.rb +2 -2
  403. data/spec/data/run_context/cookbooks/dependency1/definitions/dependency1_res.rb +1 -1
  404. data/spec/data/run_context/cookbooks/dependency1/definitions/unparsed_file +1 -1
  405. data/spec/data/run_context/cookbooks/dependency1/libraries/lib.rb +2 -2
  406. data/spec/data/run_context/cookbooks/dependency1/libraries/unparsed_file +1 -1
  407. data/spec/data/run_context/cookbooks/dependency1/providers/provider.rb +1 -1
  408. data/spec/data/run_context/cookbooks/dependency1/providers/unparsed_file +1 -1
  409. data/spec/data/run_context/cookbooks/dependency1/recipes/unparsed_file +1 -1
  410. data/spec/data/run_context/cookbooks/dependency1/resources/resource.rb +2 -2
  411. data/spec/data/run_context/cookbooks/dependency1/resources/unparsed_file +1 -1
  412. data/spec/data/run_context/cookbooks/dependency2/attributes/default.rb +2 -2
  413. data/spec/data/run_context/cookbooks/dependency2/definitions/dependency2_res.rb +1 -1
  414. data/spec/data/run_context/cookbooks/dependency2/libraries/lib.rb +2 -2
  415. data/spec/data/run_context/cookbooks/dependency2/providers/provider.rb +1 -1
  416. data/spec/data/run_context/cookbooks/dependency2/resources/resource.rb +2 -2
  417. data/spec/data/run_context/cookbooks/include/recipes/default.rb +24 -24
  418. data/spec/data/run_context/cookbooks/include/recipes/includee.rb +3 -3
  419. data/spec/data/run_context/cookbooks/no-default-attr/attributes/server.rb +2 -2
  420. data/spec/data/run_context/cookbooks/no-default-attr/definitions/no_default-attr_res.rb +1 -1
  421. data/spec/data/run_context/cookbooks/no-default-attr/providers/provider.rb +1 -1
  422. data/spec/data/run_context/cookbooks/no-default-attr/resources/resource.rb +2 -2
  423. data/spec/data/run_context/cookbooks/test/attributes/george.rb +1 -1
  424. data/spec/data/run_context/cookbooks/test/definitions/new_animals.rb +9 -9
  425. data/spec/data/run_context/cookbooks/test/definitions/new_cat.rb +5 -5
  426. data/spec/data/run_context/cookbooks/test/definitions/test_res.rb +1 -1
  427. data/spec/data/run_context/cookbooks/test/providers/provider.rb +1 -1
  428. data/spec/data/run_context/cookbooks/test/recipes/default.rb +5 -5
  429. data/spec/data/run_context/cookbooks/test/recipes/one.rb +7 -7
  430. data/spec/data/run_context/cookbooks/test/recipes/two.rb +7 -7
  431. data/spec/data/run_context/cookbooks/test/resources/resource.rb +3 -3
  432. data/spec/data/run_context/cookbooks/test-with-circular-deps/attributes/default.rb +2 -2
  433. data/spec/data/run_context/cookbooks/test-with-circular-deps/definitions/test_with-circular-deps_res.rb +1 -1
  434. data/spec/data/run_context/cookbooks/test-with-circular-deps/libraries/lib.rb +2 -2
  435. data/spec/data/run_context/cookbooks/test-with-circular-deps/metadata.rb +2 -2
  436. data/spec/data/run_context/cookbooks/test-with-circular-deps/providers/provider.rb +1 -1
  437. data/spec/data/run_context/cookbooks/test-with-circular-deps/resources/resource.rb +3 -3
  438. data/spec/data/run_context/cookbooks/test-with-deps/attributes/default.rb +2 -2
  439. data/spec/data/run_context/cookbooks/test-with-deps/definitions/test_with-deps_res.rb +1 -1
  440. data/spec/data/run_context/cookbooks/test-with-deps/libraries/lib.rb +1 -1
  441. data/spec/data/run_context/cookbooks/test-with-deps/metadata.rb +3 -3
  442. data/spec/data/run_context/cookbooks/test-with-deps/providers/provider.rb +1 -1
  443. data/spec/data/run_context/cookbooks/test-with-deps/resources/resource.rb +2 -2
  444. data/spec/data/run_context/nodes/run_context.rb +5 -5
  445. data/spec/data/sample_msu1.xml +10 -10
  446. data/spec/data/sample_msu2.xml +14 -14
  447. data/spec/data/sample_msu3.xml +16 -16
  448. data/spec/data/search_queries_to_transform.txt +98 -98
  449. data/spec/data/shef-config.rb +11 -11
  450. data/spec/data/snap_package/async_result_success.json +6 -6
  451. data/spec/data/snap_package/change_id_result.json +175 -175
  452. data/spec/data/snap_package/find_result_failure.json +10 -10
  453. data/spec/data/snap_package/find_result_success.json +70 -70
  454. data/spec/data/snap_package/get_by_name_result_failure.json +10 -10
  455. data/spec/data/snap_package/get_by_name_result_success.json +38 -38
  456. data/spec/data/snap_package/get_conf_success.json +10 -10
  457. data/spec/data/snap_package/result_failure.json +9 -9
  458. data/spec/data/ssl/5e707473.0 +18 -18
  459. data/spec/data/ssl/chef-rspec.cert +27 -27
  460. data/spec/data/ssl/chef-rspec.key +27 -27
  461. data/spec/data/ssl/key.pem +15 -15
  462. data/spec/data/ssl/private_key.pem +27 -27
  463. data/spec/data/ssl/private_key_with_whitespace.pem +32 -32
  464. data/spec/data/standalone_cookbook/chefignore +9 -9
  465. data/spec/data/standalone_cookbook/recipes/default.rb +2 -2
  466. data/spec/data/templates/failed.erb +5 -5
  467. data/spec/data/trusted_certs/example.crt +22 -22
  468. data/spec/data/trusted_certs/example_no_cn.crt +36 -36
  469. data/spec/data/trusted_certs/intermediate.pem +27 -27
  470. data/spec/data/trusted_certs/opscode.pem +57 -57
  471. data/spec/data/trusted_certs/root.pem +22 -22
  472. data/spec/data/windows_certificates/base64_test.cer +20 -20
  473. data/spec/data/windows_certificates/othertest.cer +20 -20
  474. data/spec/data/windows_certificates/test.cer +20 -20
  475. data/spec/data/windows_certificates/test.pem +20 -20
  476. data/spec/functional/configure_spec.rb +33 -33
  477. data/spec/functional/cookbook_delete_spec.rb +157 -156
  478. data/spec/functional/exec_spec.rb +55 -55
  479. data/spec/functional/rehash_spec.rb +39 -39
  480. data/spec/functional/smoke_test.rb +42 -42
  481. data/spec/functional/ssh_spec.rb +352 -352
  482. data/spec/functional/version_spec.rb +26 -26
  483. data/spec/integration/chef_fs_data_store_spec.rb +557 -557
  484. data/spec/integration/chef_repo_path_spec.rb +962 -962
  485. data/spec/integration/chef_repository_file_system_spec.rb +200 -200
  486. data/spec/integration/chefignore_spec.rb +301 -301
  487. data/spec/integration/client_bulk_delete_spec.rb +131 -131
  488. data/spec/integration/client_create_spec.rb +70 -70
  489. data/spec/integration/client_delete_spec.rb +64 -64
  490. data/spec/integration/client_key_create_spec.rb +66 -66
  491. data/spec/integration/client_key_delete_spec.rb +43 -43
  492. data/spec/integration/client_key_list_spec.rb +61 -61
  493. data/spec/integration/client_key_show_spec.rb +45 -45
  494. data/spec/integration/client_list_spec.rb +49 -49
  495. data/spec/integration/client_show_spec.rb +37 -37
  496. data/spec/integration/commands_spec.rb +55 -55
  497. data/spec/integration/common_options_spec.rb +174 -174
  498. data/spec/integration/config_list_spec.rb +220 -220
  499. data/spec/integration/config_show_spec.rb +192 -192
  500. data/spec/integration/config_use_spec.rb +198 -198
  501. data/spec/integration/cookbook_api_ipv6_spec.rb +113 -113
  502. data/spec/integration/cookbook_bulk_delete_spec.rb +65 -65
  503. data/spec/integration/cookbook_download_spec.rb +72 -72
  504. data/spec/integration/cookbook_list_spec.rb +55 -55
  505. data/spec/integration/cookbook_show_spec.rb +149 -149
  506. data/spec/integration/cookbook_upload_spec.rb +128 -128
  507. data/spec/integration/data_bag_create_spec.rb +125 -125
  508. data/spec/integration/data_bag_delete_spec.rb +59 -59
  509. data/spec/integration/data_bag_edit_spec.rb +105 -105
  510. data/spec/integration/data_bag_from_file_spec.rb +116 -116
  511. data/spec/integration/data_bag_list_spec.rb +44 -44
  512. data/spec/integration/data_bag_show_spec.rb +95 -95
  513. data/spec/integration/delete_spec.rb +1018 -1018
  514. data/spec/integration/deps_spec.rb +703 -703
  515. data/spec/integration/diff_spec.rb +605 -605
  516. data/spec/integration/download_spec.rb +1336 -1336
  517. data/spec/integration/environment_compare_spec.rb +75 -75
  518. data/spec/integration/environment_create_spec.rb +41 -41
  519. data/spec/integration/environment_delete_spec.rb +37 -37
  520. data/spec/integration/environment_from_file_spec.rb +116 -116
  521. data/spec/integration/environment_list_spec.rb +42 -42
  522. data/spec/integration/environment_show_spec.rb +77 -77
  523. data/spec/integration/list_spec.rb +1060 -1060
  524. data/spec/integration/node_bulk_delete_spec.rb +52 -52
  525. data/spec/integration/node_create_spec.rb +47 -47
  526. data/spec/integration/node_delete_spec.rb +48 -48
  527. data/spec/integration/node_environment_set_spec.rb +46 -46
  528. data/spec/integration/node_from_file_spec.rb +59 -59
  529. data/spec/integration/node_list_spec.rb +45 -45
  530. data/spec/integration/node_run_list_add_spec.rb +54 -54
  531. data/spec/integration/node_run_list_remove_spec.rb +36 -36
  532. data/spec/integration/node_run_list_set_spec.rb +41 -41
  533. data/spec/integration/node_show_spec.rb +36 -36
  534. data/spec/integration/raw_spec.rb +297 -297
  535. data/spec/integration/redirection_spec.rb +64 -64
  536. data/spec/integration/role_bulk_delete_spec.rb +52 -52
  537. data/spec/integration/role_create_spec.rb +41 -41
  538. data/spec/integration/role_delete_spec.rb +48 -48
  539. data/spec/integration/role_from_file_spec.rb +96 -96
  540. data/spec/integration/role_list_spec.rb +45 -45
  541. data/spec/integration/role_show_spec.rb +51 -51
  542. data/spec/integration/search_node_spec.rb +40 -40
  543. data/spec/integration/serve_spec.rb +92 -92
  544. data/spec/integration/show_spec.rb +197 -197
  545. data/spec/integration/upload_spec.rb +1616 -1616
  546. data/spec/knife_spec_helper.rb +241 -241
  547. data/spec/support/chef_helpers.rb +79 -79
  548. data/spec/support/key_helpers.rb +102 -102
  549. data/spec/support/platform_helpers.rb +255 -255
  550. data/spec/support/platforms/prof/gc.rb +51 -51
  551. data/spec/support/platforms/prof/win32.rb +45 -45
  552. data/spec/support/platforms/win32/spec_service.rb +57 -57
  553. data/spec/support/recipe_dsl_helper.rb +83 -83
  554. data/spec/support/shared/context/config.rb +18 -18
  555. data/spec/support/shared/functional/knife.rb +37 -37
  556. data/spec/support/shared/integration/integration_helper.rb +122 -122
  557. data/spec/support/shared/integration/knife_support.rb +192 -192
  558. data/spec/support/shared/matchers/exit_with_code.rb +32 -32
  559. data/spec/support/shared/matchers/match_environment_variable.rb +17 -17
  560. data/spec/support/shared/unit/knife_shared.rb +39 -39
  561. data/spec/support/shared/unit/mock_shellout.rb +49 -49
  562. data/spec/tiny_server.rb +193 -190
  563. data/spec/unit/application/knife_spec.rb +241 -241
  564. data/spec/unit/knife/bootstrap/chef_vault_handler_spec.rb +152 -152
  565. data/spec/unit/knife/bootstrap/client_builder_spec.rb +207 -207
  566. data/spec/unit/knife/bootstrap/train_connector_spec.rb +244 -244
  567. data/spec/unit/knife/bootstrap_spec.rb +2311 -2288
  568. data/spec/unit/knife/client_bulk_delete_spec.rb +166 -166
  569. data/spec/unit/knife/client_create_spec.rb +232 -232
  570. data/spec/unit/knife/client_delete_spec.rb +99 -99
  571. data/spec/unit/knife/client_edit_spec.rb +53 -53
  572. data/spec/unit/knife/client_list_spec.rb +34 -34
  573. data/spec/unit/knife/client_reregister_spec.rb +62 -62
  574. data/spec/unit/knife/client_show_spec.rb +52 -52
  575. data/spec/unit/knife/configure_client_spec.rb +81 -81
  576. data/spec/unit/knife/configure_spec.rb +190 -190
  577. data/spec/unit/knife/cookbook_bulk_delete_spec.rb +87 -87
  578. data/spec/unit/knife/cookbook_delete_spec.rb +239 -239
  579. data/spec/unit/knife/cookbook_download_spec.rb +255 -255
  580. data/spec/unit/knife/cookbook_list_spec.rb +88 -88
  581. data/spec/unit/knife/cookbook_metadata_from_file_spec.rb +72 -72
  582. data/spec/unit/knife/cookbook_metadata_spec.rb +182 -182
  583. data/spec/unit/knife/cookbook_show_spec.rb +253 -253
  584. data/spec/unit/knife/cookbook_upload_spec.rb +426 -426
  585. data/spec/unit/knife/core/bootstrap_context_spec.rb +287 -287
  586. data/spec/unit/knife/core/cookbook_scm_repo_spec.rb +187 -187
  587. data/spec/unit/knife/core/cookbook_site_streaming_uploader_spec.rb +198 -198
  588. data/spec/unit/knife/core/gem_glob_loader_spec.rb +242 -242
  589. data/spec/unit/knife/core/hashed_command_loader_spec.rb +112 -112
  590. data/spec/unit/knife/core/node_editor_spec.rb +211 -211
  591. data/spec/unit/knife/core/object_loader_spec.rb +81 -81
  592. data/spec/unit/knife/core/status_presenter_spec.rb +54 -54
  593. data/spec/unit/knife/core/subcommand_loader_spec.rb +64 -64
  594. data/spec/unit/knife/core/ui_spec.rb +656 -656
  595. data/spec/unit/knife/core/windows_bootstrap_context_spec.rb +282 -238
  596. data/spec/unit/knife/data_bag_create_spec.rb +175 -175
  597. data/spec/unit/knife/data_bag_edit_spec.rb +126 -126
  598. data/spec/unit/knife/data_bag_from_file_spec.rb +174 -174
  599. data/spec/unit/knife/data_bag_secret_options_spec.rb +173 -173
  600. data/spec/unit/knife/data_bag_show_spec.rb +139 -139
  601. data/spec/unit/knife/environment_compare_spec.rb +112 -112
  602. data/spec/unit/knife/environment_create_spec.rb +91 -91
  603. data/spec/unit/knife/environment_delete_spec.rb +71 -71
  604. data/spec/unit/knife/environment_edit_spec.rb +79 -79
  605. data/spec/unit/knife/environment_from_file_spec.rb +90 -90
  606. data/spec/unit/knife/environment_list_spec.rb +54 -54
  607. data/spec/unit/knife/environment_show_spec.rb +52 -52
  608. data/spec/unit/knife/key_create_spec.rb +223 -223
  609. data/spec/unit/knife/key_delete_spec.rb +133 -133
  610. data/spec/unit/knife/key_edit_spec.rb +264 -264
  611. data/spec/unit/knife/key_helper.rb +74 -74
  612. data/spec/unit/knife/key_list_spec.rb +216 -216
  613. data/spec/unit/knife/key_show_spec.rb +126 -126
  614. data/spec/unit/knife/license_spec.rb +89 -89
  615. data/spec/unit/knife/node_bulk_delete_spec.rb +94 -94
  616. data/spec/unit/knife/node_delete_spec.rb +77 -77
  617. data/spec/unit/knife/node_edit_spec.rb +116 -116
  618. data/spec/unit/knife/node_environment_set_spec.rb +61 -61
  619. data/spec/unit/knife/node_from_file_spec.rb +59 -59
  620. data/spec/unit/knife/node_list_spec.rb +62 -62
  621. data/spec/unit/knife/node_policy_set_spec.rb +122 -122
  622. data/spec/unit/knife/node_run_list_add_spec.rb +145 -145
  623. data/spec/unit/knife/node_run_list_remove_spec.rb +106 -106
  624. data/spec/unit/knife/node_run_list_set_spec.rb +115 -115
  625. data/spec/unit/knife/node_show_spec.rb +65 -65
  626. data/spec/unit/knife/org_create_spec.rb +76 -76
  627. data/spec/unit/knife/org_delete_spec.rb +41 -41
  628. data/spec/unit/knife/org_edit_spec.rb +49 -49
  629. data/spec/unit/knife/org_list_spec.rb +58 -58
  630. data/spec/unit/knife/org_show_spec.rb +45 -45
  631. data/spec/unit/knife/org_user_add_spec.rb +39 -39
  632. data/spec/unit/knife/raw_spec.rb +43 -43
  633. data/spec/unit/knife/role_bulk_delete_spec.rb +80 -80
  634. data/spec/unit/knife/role_create_spec.rb +80 -80
  635. data/spec/unit/knife/role_delete_spec.rb +67 -67
  636. data/spec/unit/knife/role_edit_spec.rb +77 -77
  637. data/spec/unit/knife/role_env_run_list_add_spec.rb +217 -217
  638. data/spec/unit/knife/role_env_run_list_clear_spec.rb +94 -94
  639. data/spec/unit/knife/role_env_run_list_remove_spec.rb +102 -102
  640. data/spec/unit/knife/role_env_run_list_replace_spec.rb +105 -105
  641. data/spec/unit/knife/role_env_run_list_set_spec.rb +99 -99
  642. data/spec/unit/knife/role_from_file_spec.rb +69 -69
  643. data/spec/unit/knife/role_list_spec.rb +54 -54
  644. data/spec/unit/knife/role_run_list_add_spec.rb +179 -179
  645. data/spec/unit/knife/role_run_list_clear_spec.rb +84 -84
  646. data/spec/unit/knife/role_run_list_remove_spec.rb +92 -92
  647. data/spec/unit/knife/role_run_list_replace_spec.rb +98 -98
  648. data/spec/unit/knife/role_run_list_set_spec.rb +89 -89
  649. data/spec/unit/knife/role_show_spec.rb +59 -59
  650. data/spec/unit/knife/search_spec.rb +147 -147
  651. data/spec/unit/knife/ssh_spec.rb +403 -403
  652. data/spec/unit/knife/ssl_check_spec.rb +256 -256
  653. data/spec/unit/knife/ssl_fetch_spec.rb +222 -222
  654. data/spec/unit/knife/status_spec.rb +112 -112
  655. data/spec/unit/knife/supermarket_download_spec.rb +152 -152
  656. data/spec/unit/knife/supermarket_install_spec.rb +203 -203
  657. data/spec/unit/knife/supermarket_list_spec.rb +70 -70
  658. data/spec/unit/knife/supermarket_search_spec.rb +85 -85
  659. data/spec/unit/knife/supermarket_share_spec.rb +208 -208
  660. data/spec/unit/knife/supermarket_unshare_spec.rb +78 -78
  661. data/spec/unit/knife/tag_create_spec.rb +23 -23
  662. data/spec/unit/knife/tag_delete_spec.rb +25 -25
  663. data/spec/unit/knife/tag_list_spec.rb +23 -23
  664. data/spec/unit/knife/user_create_spec.rb +282 -282
  665. data/spec/unit/knife/user_delete_spec.rb +171 -171
  666. data/spec/unit/knife/user_edit_spec.rb +54 -54
  667. data/spec/unit/knife/user_list_spec.rb +73 -73
  668. data/spec/unit/knife/user_password_spec.rb +64 -64
  669. data/spec/unit/knife/user_reregister_spec.rb +56 -56
  670. data/spec/unit/knife/user_show_spec.rb +91 -91
  671. data/spec/unit/knife_spec.rb +637 -637
  672. data/spec/unit/utils/licensing_handler_spec.rb +140 -0
  673. metadata +36 -21
@@ -1,656 +1,656 @@
1
- #
2
- # Author:: Adam Jacob (<adam@chef.io>)
3
- # Author:: Tim Hinderliter (<tim@chef.io>)
4
- # Author:: Daniel DeLeo (<dan@chef.io>)
5
- # Author:: John Keiser (<jkeiser@chef.io>)
6
- # Copyright:: Copyright (c) Chef Software Inc.
7
- # License:: Apache License, Version 2.0
8
- #
9
- # Licensed under the Apache License, Version 2.0 (the "License");
10
- # you may not use this file except in compliance with the License.
11
- # You may obtain a copy of the License at
12
- #
13
- # http://www.apache.org/licenses/LICENSE-2.0
14
- #
15
- # Unless required by applicable law or agreed to in writing, software
16
- # distributed under the License is distributed on an "AS IS" BASIS,
17
- # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
18
- # See the License for the specific language governing permissions and
19
- # limitations under the License.
20
- #
21
-
22
- require "knife_spec_helper"
23
-
24
- describe Chef::Knife::UI do
25
- before do
26
- @out, @err, @in = StringIO.new, StringIO.new, StringIO.new
27
- @config = {
28
- verbosity: 0,
29
- yes: nil,
30
- format: "summary",
31
- field_separator: ".",
32
- }
33
- @ui = Chef::Knife::UI.new(@out, @err, @in, @config)
34
- end
35
-
36
- class TestObject < OpenStruct
37
- def self.from_hash(hsh)
38
- new(hsh)
39
- end
40
- end
41
-
42
- describe "edit" do
43
- ruby_for_json = { "foo" => "bar" }
44
- ruby_from_json = TestObject.from_hash(ruby_for_json)
45
- json_from_ruby = "{\n \"foo\": \"bar\"\n}"
46
- json_from_editor = "{\n \"bar\": \"foo\"\n}"
47
- ruby_from_editor = TestObject.from_hash({ "bar" => "foo" })
48
- my_editor = "veeeye"
49
- temp_path = "/tmp/bar/baz"
50
-
51
- let(:subject) { @ui.edit_data(ruby_for_json, parse_output, object_class: klass) }
52
- let(:parse_output) { false }
53
- let(:klass) { nil }
54
-
55
- context "when editing is disabled" do
56
- before do
57
- @ui.config[:disable_editing] = true
58
- stub_const("Tempfile", double) # Tempfiles should never be invoked
59
- end
60
- context "when parse_output is false" do
61
- it "returns pretty json string" do
62
- expect(subject).to eql(json_from_ruby)
63
- end
64
- end
65
- context "when parse_output is true" do
66
- let(:parse_output) { true }
67
- let(:klass) { TestObject }
68
- it "returns a ruby object" do
69
- expect(subject).to eql(ruby_from_json)
70
- end
71
- context "but no object class is provided" do
72
- let(:klass) { nil }
73
- it "raises an error" do
74
- expect { subject }.to raise_error ArgumentError,
75
- /Please pass in the object class to hydrate or use #edit_hash/
76
- end
77
- end
78
- end
79
- end
80
-
81
- context "when editing is enabled" do
82
- before do
83
- @ui.config[:disable_editing] = false
84
- @ui.config[:editor] = my_editor
85
- @mock = double("Tempfile")
86
- expect(@mock).to receive(:sync=).with(true)
87
- expect(@mock).to receive(:puts).with(json_from_ruby)
88
- expect(@mock).to receive(:close)
89
- expect(@mock).to receive(:path).at_least(:once).and_return(temp_path)
90
- expect(Tempfile).to receive(:open).with([ "knife-edit-", ".json" ]).and_yield(@mock)
91
- end
92
- context "and the editor works" do
93
- before do
94
- expect(@ui).to receive(:system).with("#{my_editor} #{temp_path}").and_return(true)
95
- expect(IO).to receive(:read).with(temp_path).and_return(json_from_editor)
96
- end
97
-
98
- context "when parse_output is false" do
99
- it "returns an edited pretty json string" do
100
- expect(subject).to eql(json_from_editor)
101
- end
102
- end
103
- context "when parse_output is true" do
104
- let(:parse_output) { true }
105
- let(:klass) { TestObject }
106
- it "returns an edited ruby object" do
107
- expect(subject).to eql(ruby_from_editor)
108
- end
109
- end
110
- end
111
- context "when running the editor fails with nil" do
112
- before do
113
- expect(@ui).to receive(:system).with("#{my_editor} #{temp_path}").and_return(nil)
114
- expect(IO).not_to receive(:read)
115
- end
116
- it "throws an exception" do
117
- expect { subject }.to raise_error(RuntimeError)
118
- end
119
- end
120
- context "when running the editor fails with false" do
121
- before do
122
- expect(@ui).to receive(:system).with("#{my_editor} #{temp_path}").and_return(false)
123
- expect(IO).not_to receive(:read)
124
- end
125
- it "throws an exception" do
126
- expect { subject }.to raise_error(RuntimeError)
127
- end
128
- end
129
- end
130
- context "when editing and not stubbing Tempfile (semi-functional test)" do
131
- before do
132
- @ui.config[:disable_editing] = false
133
- @ui.config[:editor] = my_editor
134
- @tempfile = Tempfile.new([ "knife-edit-", ".json" ])
135
- expect(Tempfile).to receive(:open).with([ "knife-edit-", ".json" ]).and_yield(@tempfile)
136
- end
137
-
138
- context "and the editor works" do
139
- before do
140
- expect(@ui).to receive(:system).with("#{my_editor} #{@tempfile.path}").and_return(true)
141
- expect(IO).to receive(:read).with(@tempfile.path).and_return(json_from_editor)
142
- end
143
-
144
- context "when parse_output is false" do
145
- it "returns an edited pretty json string" do
146
- expect(subject).to eql(json_from_editor)
147
- end
148
- it "the tempfile should have mode 0600", :unix_only do
149
- # XXX: this looks odd because we're really testing Tempfile.new here
150
- expect(File.stat(@tempfile.path).mode & 0777).to eql(0600)
151
- expect(subject).to eql(json_from_editor)
152
- end
153
- end
154
-
155
- context "when parse_output is true" do
156
- let(:parse_output) { true }
157
- let(:klass) { TestObject }
158
- it "returns an edited ruby object" do
159
- expect(subject).to eql(ruby_from_editor)
160
- end
161
- it "the tempfile should have mode 0600", :unix_only do
162
- # XXX: this looks odd because we're really testing Tempfile.new here
163
- expect(File.stat(@tempfile.path).mode & 0777).to eql(0600)
164
- expect(subject).to eql(ruby_from_editor)
165
- end
166
- end
167
- end
168
- end
169
- end
170
-
171
- describe "format_list_for_display" do
172
- it "should print the full hash if --with-uri is true" do
173
- @ui.config[:with_uri] = true
174
- expect(@ui.format_list_for_display({ marcy: :playground })).to eq({ marcy: :playground })
175
- end
176
-
177
- it "should print only the keys if --with-uri is false" do
178
- @ui.config[:with_uri] = false
179
- expect(@ui.format_list_for_display({ marcy: :playground })).to eq([ :marcy ])
180
- end
181
- end
182
-
183
- shared_examples "an output mehthod handling IO exceptions" do |method|
184
- it "should throw Errno::EIO exceptions" do
185
- allow(@out).to receive(:puts).and_raise(Errno::EIO)
186
- allow(@err).to receive(:puts).and_raise(Errno::EIO)
187
- expect { @ui.send(method, "hi") }.to raise_error(Errno::EIO)
188
- end
189
-
190
- it "should ignore Errno::EPIPE exceptions (CHEF-3516)" do
191
- allow(@out).to receive(:puts).and_raise(Errno::EPIPE)
192
- allow(@err).to receive(:puts).and_raise(Errno::EPIPE)
193
- expect { @ui.send(method, "hi") }.to raise_error(SystemExit)
194
- end
195
-
196
- it "should throw Errno::EPIPE exceptions with -VV (CHEF-3516)" do
197
- @config[:verbosity] = 2
198
- allow(@out).to receive(:puts).and_raise(Errno::EPIPE)
199
- allow(@err).to receive(:puts).and_raise(Errno::EPIPE)
200
- expect { @ui.send(method, "hi") }.to raise_error(Errno::EPIPE)
201
- end
202
- end
203
-
204
- describe "output" do
205
- it_behaves_like "an output mehthod handling IO exceptions", :output
206
-
207
- it "formats strings appropriately" do
208
- @ui.output("hi")
209
- expect(@out.string).to eq("hi\n")
210
- end
211
-
212
- it "formats hashes appropriately" do
213
- @ui.output({ "hi" => "a", "lo" => "b" })
214
- expect(@out.string).to eq <<~EOM
215
- hi: a
216
- lo: b
217
- EOM
218
- end
219
-
220
- it "formats empty hashes appropriately" do
221
- @ui.output({})
222
- expect(@out.string).to eq("\n")
223
- end
224
-
225
- it "formats arrays appropriately" do
226
- @ui.output(%w{a b})
227
- expect(@out.string).to eq <<~EOM
228
- a
229
- b
230
- EOM
231
- end
232
-
233
- it "formats empty arrays appropriately" do
234
- @ui.output([ ])
235
- expect(@out.string).to eq("\n")
236
- end
237
-
238
- it "formats single-member arrays appropriately" do
239
- @ui.output([ "a" ])
240
- expect(@out.string).to eq("a\n")
241
- end
242
-
243
- it "formats nested single-member arrays appropriately" do
244
- @ui.output([ [ "a" ] ])
245
- expect(@out.string).to eq("a\n")
246
- end
247
-
248
- it "formats nested arrays appropriately" do
249
- @ui.output([ %w{a b}, %w{c d}])
250
- expect(@out.string).to eq <<~EOM
251
- a
252
- b
253
-
254
- c
255
- d
256
- EOM
257
- end
258
-
259
- it "formats nested arrays with single- and empty subarrays appropriately" do
260
- @ui.output([ %w{a b}, [ "c" ], [], %w{d e}])
261
- expect(@out.string).to eq <<~EOM
262
- a
263
- b
264
-
265
- c
266
-
267
-
268
- d
269
- e
270
- EOM
271
- end
272
-
273
- it "formats arrays of hashes with extra lines in between for readability" do
274
- @ui.output([ { "a" => "b", "c" => "d" }, { "x" => "y" }, { "m" => "n", "o" => "p" }])
275
- expect(@out.string).to eq <<~EOM
276
- a: b
277
- c: d
278
-
279
- x: y
280
-
281
- m: n
282
- o: p
283
- EOM
284
- end
285
-
286
- it "formats hashes with empty array members appropriately" do
287
- @ui.output({ "a" => [], "b" => "c" })
288
- expect(@out.string).to eq <<~EOM
289
- a:
290
- b: c
291
- EOM
292
- end
293
-
294
- it "formats hashes with single-member array values appropriately" do
295
- @ui.output({ "a" => [ "foo" ], "b" => "c" })
296
- expect(@out.string).to eq <<~EOM
297
- a: foo
298
- b: c
299
- EOM
300
- end
301
-
302
- it "formats hashes with array members appropriately" do
303
- @ui.output({ "a" => %w{foo bar}, "b" => "c" })
304
- expect(@out.string).to eq <<~EOM
305
- a:
306
- foo
307
- bar
308
- b: c
309
- EOM
310
- end
311
-
312
- it "formats hashes with single-member nested array values appropriately" do
313
- @ui.output({ "a" => [ [ "foo" ] ], "b" => "c" })
314
- expect(@out.string).to eq <<~EOM
315
- a:
316
- foo
317
- b: c
318
- EOM
319
- end
320
-
321
- it "formats hashes with nested array values appropriately" do
322
- @ui.output({ "a" => [ %w{foo bar}, %w{baz bjork} ], "b" => "c" })
323
- # XXX: using a HEREDOC at this point results in a line with required spaces which auto-whitespace removal settings
324
- # on editors will remove and will break this test.
325
- expect(@out.string).to eq("a:\n foo\n bar\n \n baz\n bjork\nb: c\n")
326
- end
327
-
328
- it "formats hashes with hash values appropriately" do
329
- @ui.output({ "a" => { "aa" => "bb", "cc" => "dd" }, "b" => "c" })
330
- expect(@out.string).to eq <<~EOM
331
- a:
332
- aa: bb
333
- cc: dd
334
- b: c
335
- EOM
336
- end
337
-
338
- it "formats hashes with empty hash values appropriately" do
339
- @ui.output({ "a" => {}, "b" => "c" })
340
- expect(@out.string).to eq <<~EOM
341
- a:
342
- b: c
343
- EOM
344
- end
345
- end
346
-
347
- describe "warn" do
348
- it_behaves_like "an output mehthod handling IO exceptions", :warn
349
- end
350
-
351
- describe "error" do
352
- it_behaves_like "an output mehthod handling IO exceptions", :warn
353
- end
354
-
355
- describe "fatal" do
356
- it_behaves_like "an output mehthod handling IO exceptions", :warn
357
- end
358
-
359
- describe "format_for_display" do
360
- it "should return the raw data" do
361
- input = { gi: :go }
362
- expect(@ui.format_for_display(input)).to eq(input)
363
- end
364
-
365
- describe "with --attribute passed" do
366
- it "should return the deeply nested attribute" do
367
- input = { "gi" => { "go" => "ge" }, "id" => "sample-data-bag-item" }
368
- @ui.config[:attribute] = "gi.go"
369
- expect(@ui.format_for_display(input)).to eq({ "sample-data-bag-item" => { "gi.go" => "ge" } })
370
- end
371
-
372
- it "should return multiple attributes" do
373
- input = { "gi" => "go", "hi" => "ho", "id" => "sample-data-bag-item" }
374
- @ui.config[:attribute] = %w{gi hi}
375
- expect(@ui.format_for_display(input)).to eq({ "sample-data-bag-item" => { "gi" => "go", "hi" => "ho" } })
376
- end
377
-
378
- it "should handle attributes named the same as methods" do
379
- input = { "keys" => "values", "hi" => "ho", "id" => "sample-data-bag-item" }
380
- @ui.config[:attribute] = "keys"
381
- expect(@ui.format_for_display(input)).to eq({ "sample-data-bag-item" => { "keys" => "values" } })
382
- end
383
-
384
- it "should handle nested attributes named the same as methods" do
385
- input = { "keys" => { "keys" => "values" }, "hi" => "ho", "id" => "sample-data-bag-item" }
386
- @ui.config[:attribute] = "keys.keys"
387
- expect(@ui.format_for_display(input)).to eq({ "sample-data-bag-item" => { "keys.keys" => "values" } })
388
- end
389
-
390
- it "should return the name attribute" do
391
- input = Chef::Node.new
392
- input.name("chef.localdomain")
393
- @ui.config[:attribute] = "name"
394
- expect(@ui.format_for_display(input)).to eq( { "chef.localdomain" => { "name" => "chef.localdomain" } })
395
- end
396
-
397
- it "should return a 'class' attribute and not the node.class" do
398
- input = Chef::Node.new
399
- input.default["class"] = "classy!"
400
- @ui.config[:attribute] = "class"
401
- expect(@ui.format_for_display(input)).to eq( { nil => { "class" => "classy!" } } )
402
- end
403
-
404
- it "should return the chef_environment attribute" do
405
- input = Chef::Node.new
406
- input.chef_environment = "production-partner-load-integration-preview-testing"
407
- @ui.config[:attribute] = "chef_environment"
408
- expect(@ui.format_for_display(input)).to eq( { nil => { "chef_environment" => "production-partner-load-integration-preview-testing" } } )
409
- end
410
-
411
- it "works with arrays" do
412
- input = Chef::Node.new
413
- input.default["array"] = %w{zero one two}
414
- @ui.config[:attribute] = "array.1"
415
- expect(@ui.format_for_display(input)).to eq( { nil => { "array.1" => "one" } } )
416
- end
417
-
418
- it "returns nil when given an attribute path that isn't a name or attribute" do
419
- input = { "keys" => { "keys" => "values" }, "hi" => "ho", "id" => "sample-data-bag-item" }
420
- non_existing_path = "nope.nada.nothingtoseehere"
421
- @ui.config[:attribute] = non_existing_path
422
- expect(@ui.format_for_display(input)).to eq({ "sample-data-bag-item" => { non_existing_path => nil } })
423
- end
424
-
425
- describe "when --field-separator is passed" do
426
- it "honors that separator" do
427
- input = { "keys" => { "with spaces" => { "open" => { "doors" => { "with many.dots" => "when asked" } } } } }
428
- @ui.config[:field_separator] = ";"
429
- @ui.config[:attribute] = "keys;with spaces;open;doors;with many.dots"
430
- expect(@ui.format_for_display(input)).to eq({ nil => { "keys;with spaces;open;doors;with many.dots" => "when asked" } })
431
- end
432
- end
433
- end
434
-
435
- describe "with --run-list passed" do
436
- it "should return the run list" do
437
- input = Chef::Node.new
438
- input.name("sample-node")
439
- input.run_list("role[monkey]", "role[churchmouse]")
440
- @ui.config[:run_list] = true
441
- response = @ui.format_for_display(input)
442
- expect(response["sample-node"]["run_list"][0]).to eq("role[monkey]")
443
- expect(response["sample-node"]["run_list"][1]).to eq("role[churchmouse]")
444
- end
445
- end
446
- end
447
-
448
- describe "format_cookbook_list_for_display" do
449
- before(:each) do
450
- @item = {
451
- "cookbook_name" => {
452
- "url" => "http://url/cookbooks/cookbook",
453
- "versions" => [
454
- { "version" => "3.0.0", "url" => "http://url/cookbooks/3.0.0" },
455
- { "version" => "2.0.0", "url" => "http://url/cookbooks/2.0.0" },
456
- { "version" => "1.0.0", "url" => "http://url/cookbooks/1.0.0" },
457
- ],
458
- },
459
- }
460
- end
461
-
462
- it "should return an hash of the cookbooks with versions" do
463
- expected_response = { "cookbook_name" => "3.0.0 2.0.0 1.0.0" }
464
- response = @ui.format_cookbook_list_for_display(@item)
465
- expect(response).to eq(expected_response)
466
- end
467
-
468
- describe "with --with-uri" do
469
- it "should return the URIs" do
470
- response = {
471
- "cookbook_name" => {
472
- "1.0.0" => "http://url/cookbooks/1.0.0",
473
- "2.0.0" => "http://url/cookbooks/2.0.0",
474
- "3.0.0" => "http://url/cookbooks/3.0.0" },
475
- }
476
- @ui.config[:with_uri] = true
477
- expect(@ui.format_cookbook_list_for_display(@item)).to eq(response)
478
- end
479
- end
480
-
481
- context "when running on Windows" do
482
- before(:each) do
483
- stdout = double("StringIO", tty?: true)
484
- allow(@ui).to receive(:stdout).and_return(stdout)
485
- allow(ChefUtils).to receive(:windows?) { true }
486
- Chef::Config.reset
487
- end
488
-
489
- after(:each) do
490
- Chef::Config.reset
491
- end
492
-
493
- it "should have color set to true if knife config has color explicitly set to true" do
494
- Chef::Config[:color] = true
495
- @ui.config[:color] = true
496
- expect(@ui.color?).to eql(true)
497
- end
498
-
499
- it "should have color set to false if knife config has color explicitly set to false" do
500
- Chef::Config[:color] = false
501
- expect(@ui.color?).to eql(false)
502
- end
503
-
504
- it "should not have color set to false by default" do
505
- expect(@ui.color?).to eql(false)
506
- end
507
- end
508
- end
509
-
510
- describe "color" do
511
- context "when ui.color? => true" do
512
- it "returns colored output" do
513
- skip "doesn't work on systems that don't correctly have terminals setup for color"
514
- expect(@ui).to receive(:color?).and_return(true)
515
- expect(@ui.color("a_bus_is", :yellow)).to eql("\e[33ma_bus_is\e[0m")
516
- end
517
- end
518
-
519
- context "when ui.color? => false" do
520
- it "returns plain output" do
521
- expect(@ui).to receive(:color?).and_return(false)
522
- expect(@ui.color("a_bus_is", :yellow)).to eql("a_bus_is")
523
- end
524
- end
525
- end
526
-
527
- describe "confirm" do
528
- let(:stdout) { StringIO.new }
529
- let(:output) { stdout.string }
530
-
531
- let(:question) { "monkeys rule" }
532
- let(:answer) { "y" }
533
-
534
- let(:default_choice) { nil }
535
- let(:append_instructions) { true }
536
-
537
- def run_confirm
538
- allow(@ui).to receive(:stdout).and_return(stdout)
539
- allow(@ui.stdin).to receive(:readline).and_return(answer)
540
- @ui.confirm(question, append_instructions, default_choice)
541
- end
542
-
543
- def run_confirm_without_exit
544
- allow(@ui).to receive(:stdout).and_return(stdout)
545
- allow(@ui.stdin).to receive(:readline).and_return(answer)
546
- @ui.confirm_without_exit(question, append_instructions, default_choice)
547
- end
548
-
549
- shared_examples_for "confirm with positive answer" do
550
- it "confirm should return true" do
551
- expect(run_confirm).to be_truthy
552
- end
553
-
554
- it "confirm_without_exit should return true" do
555
- expect(run_confirm_without_exit).to be_truthy
556
- end
557
- end
558
-
559
- shared_examples_for "confirm with negative answer" do
560
- it "confirm should exit 3" do
561
- expect do
562
- run_confirm
563
- end.to raise_error(SystemExit) { |e| expect(e.status).to eq(3) }
564
- end
565
-
566
- it "confirm_without_exit should return false" do
567
- expect(run_confirm_without_exit).to be_falsey
568
- end
569
- end
570
-
571
- describe "with default choice set to true" do
572
- let(:default_choice) { true }
573
-
574
- it "should show 'Y/n' in the instructions" do
575
- run_confirm
576
- expect(output).to include("Y/n")
577
- end
578
-
579
- describe "with empty answer" do
580
- let(:answer) { "" }
581
-
582
- it_behaves_like "confirm with positive answer"
583
- end
584
-
585
- describe "with answer N " do
586
- let(:answer) { "N" }
587
-
588
- it_behaves_like "confirm with negative answer"
589
- end
590
- end
591
-
592
- describe "with default choice set to false" do
593
- let(:default_choice) { false }
594
-
595
- it "should show 'y/N' in the instructions" do
596
- run_confirm
597
- expect(output).to include("y/N")
598
- end
599
-
600
- describe "with empty answer" do
601
- let(:answer) { "" }
602
-
603
- it_behaves_like "confirm with negative answer"
604
- end
605
-
606
- describe "with answer N " do
607
- let(:answer) { "Y" }
608
-
609
- it_behaves_like "confirm with positive answer"
610
- end
611
- end
612
-
613
- %w{Y y}.each do |answer|
614
- describe "with answer #{answer}" do
615
- let(:answer) { answer }
616
-
617
- it_behaves_like "confirm with positive answer"
618
- end
619
- end
620
-
621
- %w{N n}.each do |answer|
622
- describe "with answer #{answer}" do
623
- let(:answer) { answer }
624
-
625
- it_behaves_like "confirm with negative answer"
626
- end
627
- end
628
-
629
- describe "with --y or --yes passed" do
630
- it "should return true" do
631
- @ui.config[:yes] = true
632
- expect(run_confirm).to be_truthy
633
- expect(output).to eq("")
634
- end
635
- end
636
- end
637
-
638
- describe "when asking for free-form user input" do
639
- it "asks a question and returns the answer provided by the user" do
640
- out = StringIO.new
641
- allow(@ui).to receive(:stdout).and_return(out)
642
- allow(@ui).to receive(:stdin).and_return(StringIO.new("http://mychefserver.example.com\n"))
643
- expect(@ui.ask_question("your chef server URL?")).to eq("http://mychefserver.example.com")
644
- expect(out.string).to eq("your chef server URL?")
645
- end
646
-
647
- it "suggests a default setting and returns the default when the user's response only contains whitespace" do
648
- out = StringIO.new
649
- allow(@ui).to receive(:stdout).and_return(out)
650
- allow(@ui).to receive(:stdin).and_return(StringIO.new(" \n"))
651
- expect(@ui.ask_question("your chef server URL? ", default: "http://localhost:4000")).to eq("http://localhost:4000")
652
- expect(out.string).to eq("your chef server URL? [http://localhost:4000] ")
653
- end
654
- end
655
-
656
- end
1
+ #
2
+ # Author:: Adam Jacob (<adam@chef.io>)
3
+ # Author:: Tim Hinderliter (<tim@chef.io>)
4
+ # Author:: Daniel DeLeo (<dan@chef.io>)
5
+ # Author:: John Keiser (<jkeiser@chef.io>)
6
+ # Copyright:: Copyright (c) Chef Software Inc.
7
+ # License:: Apache License, Version 2.0
8
+ #
9
+ # Licensed under the Apache License, Version 2.0 (the "License");
10
+ # you may not use this file except in compliance with the License.
11
+ # You may obtain a copy of the License at
12
+ #
13
+ # http://www.apache.org/licenses/LICENSE-2.0
14
+ #
15
+ # Unless required by applicable law or agreed to in writing, software
16
+ # distributed under the License is distributed on an "AS IS" BASIS,
17
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
18
+ # See the License for the specific language governing permissions and
19
+ # limitations under the License.
20
+ #
21
+
22
+ require "knife_spec_helper"
23
+
24
+ describe Chef::Knife::UI do
25
+ before do
26
+ @out, @err, @in = StringIO.new, StringIO.new, StringIO.new
27
+ @config = {
28
+ verbosity: 0,
29
+ yes: nil,
30
+ format: "summary",
31
+ field_separator: ".",
32
+ }
33
+ @ui = Chef::Knife::UI.new(@out, @err, @in, @config)
34
+ end
35
+
36
+ class TestObject < OpenStruct
37
+ def self.from_hash(hsh)
38
+ new(hsh)
39
+ end
40
+ end
41
+
42
+ describe "edit" do
43
+ ruby_for_json = { "foo" => "bar" }
44
+ ruby_from_json = TestObject.from_hash(ruby_for_json)
45
+ json_from_ruby = "{\n \"foo\": \"bar\"\n}"
46
+ json_from_editor = "{\n \"bar\": \"foo\"\n}"
47
+ ruby_from_editor = TestObject.from_hash({ "bar" => "foo" })
48
+ my_editor = "veeeye"
49
+ temp_path = "/tmp/bar/baz"
50
+
51
+ let(:subject) { @ui.edit_data(ruby_for_json, parse_output, object_class: klass) }
52
+ let(:parse_output) { false }
53
+ let(:klass) { nil }
54
+
55
+ context "when editing is disabled" do
56
+ before do
57
+ @ui.config[:disable_editing] = true
58
+ stub_const("Tempfile", double) # Tempfiles should never be invoked
59
+ end
60
+ context "when parse_output is false" do
61
+ it "returns pretty json string" do
62
+ expect(subject).to eql(json_from_ruby)
63
+ end
64
+ end
65
+ context "when parse_output is true" do
66
+ let(:parse_output) { true }
67
+ let(:klass) { TestObject }
68
+ it "returns a ruby object" do
69
+ expect(subject).to eql(ruby_from_json)
70
+ end
71
+ context "but no object class is provided" do
72
+ let(:klass) { nil }
73
+ it "raises an error" do
74
+ expect { subject }.to raise_error ArgumentError,
75
+ /Please pass in the object class to hydrate or use #edit_hash/
76
+ end
77
+ end
78
+ end
79
+ end
80
+
81
+ context "when editing is enabled" do
82
+ before do
83
+ @ui.config[:disable_editing] = false
84
+ @ui.config[:editor] = my_editor
85
+ @mock = double("Tempfile")
86
+ expect(@mock).to receive(:sync=).with(true)
87
+ expect(@mock).to receive(:puts).with(json_from_ruby)
88
+ expect(@mock).to receive(:close)
89
+ expect(@mock).to receive(:path).at_least(:once).and_return(temp_path)
90
+ expect(Tempfile).to receive(:open).with([ "knife-edit-", ".json" ]).and_yield(@mock)
91
+ end
92
+ context "and the editor works" do
93
+ before do
94
+ expect(@ui).to receive(:system).with("#{my_editor} #{temp_path}").and_return(true)
95
+ expect(IO).to receive(:read).with(temp_path).and_return(json_from_editor)
96
+ end
97
+
98
+ context "when parse_output is false" do
99
+ it "returns an edited pretty json string" do
100
+ expect(subject).to eql(json_from_editor)
101
+ end
102
+ end
103
+ context "when parse_output is true" do
104
+ let(:parse_output) { true }
105
+ let(:klass) { TestObject }
106
+ it "returns an edited ruby object" do
107
+ expect(subject).to eql(ruby_from_editor)
108
+ end
109
+ end
110
+ end
111
+ context "when running the editor fails with nil" do
112
+ before do
113
+ expect(@ui).to receive(:system).with("#{my_editor} #{temp_path}").and_return(nil)
114
+ expect(IO).not_to receive(:read)
115
+ end
116
+ it "throws an exception" do
117
+ expect { subject }.to raise_error(RuntimeError)
118
+ end
119
+ end
120
+ context "when running the editor fails with false" do
121
+ before do
122
+ expect(@ui).to receive(:system).with("#{my_editor} #{temp_path}").and_return(false)
123
+ expect(IO).not_to receive(:read)
124
+ end
125
+ it "throws an exception" do
126
+ expect { subject }.to raise_error(RuntimeError)
127
+ end
128
+ end
129
+ end
130
+ context "when editing and not stubbing Tempfile (semi-functional test)" do
131
+ before do
132
+ @ui.config[:disable_editing] = false
133
+ @ui.config[:editor] = my_editor
134
+ @tempfile = Tempfile.new([ "knife-edit-", ".json" ])
135
+ expect(Tempfile).to receive(:open).with([ "knife-edit-", ".json" ]).and_yield(@tempfile)
136
+ end
137
+
138
+ context "and the editor works" do
139
+ before do
140
+ expect(@ui).to receive(:system).with("#{my_editor} #{@tempfile.path}").and_return(true)
141
+ expect(IO).to receive(:read).with(@tempfile.path).and_return(json_from_editor)
142
+ end
143
+
144
+ context "when parse_output is false" do
145
+ it "returns an edited pretty json string" do
146
+ expect(subject).to eql(json_from_editor)
147
+ end
148
+ it "the tempfile should have mode 0600", :unix_only do
149
+ # XXX: this looks odd because we're really testing Tempfile.new here
150
+ expect(File.stat(@tempfile.path).mode & 0777).to eql(0600)
151
+ expect(subject).to eql(json_from_editor)
152
+ end
153
+ end
154
+
155
+ context "when parse_output is true" do
156
+ let(:parse_output) { true }
157
+ let(:klass) { TestObject }
158
+ it "returns an edited ruby object" do
159
+ expect(subject).to eql(ruby_from_editor)
160
+ end
161
+ it "the tempfile should have mode 0600", :unix_only do
162
+ # XXX: this looks odd because we're really testing Tempfile.new here
163
+ expect(File.stat(@tempfile.path).mode & 0777).to eql(0600)
164
+ expect(subject).to eql(ruby_from_editor)
165
+ end
166
+ end
167
+ end
168
+ end
169
+ end
170
+
171
+ describe "format_list_for_display" do
172
+ it "should print the full hash if --with-uri is true" do
173
+ @ui.config[:with_uri] = true
174
+ expect(@ui.format_list_for_display({ marcy: :playground })).to eq({ marcy: :playground })
175
+ end
176
+
177
+ it "should print only the keys if --with-uri is false" do
178
+ @ui.config[:with_uri] = false
179
+ expect(@ui.format_list_for_display({ marcy: :playground })).to eq([ :marcy ])
180
+ end
181
+ end
182
+
183
+ shared_examples "an output mehthod handling IO exceptions" do |method|
184
+ it "should throw Errno::EIO exceptions" do
185
+ allow(@out).to receive(:puts).and_raise(Errno::EIO)
186
+ allow(@err).to receive(:puts).and_raise(Errno::EIO)
187
+ expect { @ui.send(method, "hi") }.to raise_error(Errno::EIO)
188
+ end
189
+
190
+ it "should ignore Errno::EPIPE exceptions (CHEF-3516)" do
191
+ allow(@out).to receive(:puts).and_raise(Errno::EPIPE)
192
+ allow(@err).to receive(:puts).and_raise(Errno::EPIPE)
193
+ expect { @ui.send(method, "hi") }.to raise_error(SystemExit)
194
+ end
195
+
196
+ it "should throw Errno::EPIPE exceptions with -VV (CHEF-3516)" do
197
+ @config[:verbosity] = 2
198
+ allow(@out).to receive(:puts).and_raise(Errno::EPIPE)
199
+ allow(@err).to receive(:puts).and_raise(Errno::EPIPE)
200
+ expect { @ui.send(method, "hi") }.to raise_error(Errno::EPIPE)
201
+ end
202
+ end
203
+
204
+ describe "output" do
205
+ it_behaves_like "an output mehthod handling IO exceptions", :output
206
+
207
+ it "formats strings appropriately" do
208
+ @ui.output("hi")
209
+ expect(@out.string).to eq("hi\n")
210
+ end
211
+
212
+ it "formats hashes appropriately" do
213
+ @ui.output({ "hi" => "a", "lo" => "b" })
214
+ expect(@out.string).to eq <<~EOM
215
+ hi: a
216
+ lo: b
217
+ EOM
218
+ end
219
+
220
+ it "formats empty hashes appropriately" do
221
+ @ui.output({})
222
+ expect(@out.string).to eq("\n")
223
+ end
224
+
225
+ it "formats arrays appropriately" do
226
+ @ui.output(%w{a b})
227
+ expect(@out.string).to eq <<~EOM
228
+ a
229
+ b
230
+ EOM
231
+ end
232
+
233
+ it "formats empty arrays appropriately" do
234
+ @ui.output([ ])
235
+ expect(@out.string).to eq("\n")
236
+ end
237
+
238
+ it "formats single-member arrays appropriately" do
239
+ @ui.output([ "a" ])
240
+ expect(@out.string).to eq("a\n")
241
+ end
242
+
243
+ it "formats nested single-member arrays appropriately" do
244
+ @ui.output([ [ "a" ] ])
245
+ expect(@out.string).to eq("a\n")
246
+ end
247
+
248
+ it "formats nested arrays appropriately" do
249
+ @ui.output([ %w{a b}, %w{c d}])
250
+ expect(@out.string).to eq <<~EOM
251
+ a
252
+ b
253
+
254
+ c
255
+ d
256
+ EOM
257
+ end
258
+
259
+ it "formats nested arrays with single- and empty subarrays appropriately" do
260
+ @ui.output([ %w{a b}, [ "c" ], [], %w{d e}])
261
+ expect(@out.string).to eq <<~EOM
262
+ a
263
+ b
264
+
265
+ c
266
+
267
+
268
+ d
269
+ e
270
+ EOM
271
+ end
272
+
273
+ it "formats arrays of hashes with extra lines in between for readability" do
274
+ @ui.output([ { "a" => "b", "c" => "d" }, { "x" => "y" }, { "m" => "n", "o" => "p" }])
275
+ expect(@out.string).to eq <<~EOM
276
+ a: b
277
+ c: d
278
+
279
+ x: y
280
+
281
+ m: n
282
+ o: p
283
+ EOM
284
+ end
285
+
286
+ it "formats hashes with empty array members appropriately" do
287
+ @ui.output({ "a" => [], "b" => "c" })
288
+ expect(@out.string).to eq <<~EOM
289
+ a:
290
+ b: c
291
+ EOM
292
+ end
293
+
294
+ it "formats hashes with single-member array values appropriately" do
295
+ @ui.output({ "a" => [ "foo" ], "b" => "c" })
296
+ expect(@out.string).to eq <<~EOM
297
+ a: foo
298
+ b: c
299
+ EOM
300
+ end
301
+
302
+ it "formats hashes with array members appropriately" do
303
+ @ui.output({ "a" => %w{foo bar}, "b" => "c" })
304
+ expect(@out.string).to eq <<~EOM
305
+ a:
306
+ foo
307
+ bar
308
+ b: c
309
+ EOM
310
+ end
311
+
312
+ it "formats hashes with single-member nested array values appropriately" do
313
+ @ui.output({ "a" => [ [ "foo" ] ], "b" => "c" })
314
+ expect(@out.string).to eq <<~EOM
315
+ a:
316
+ foo
317
+ b: c
318
+ EOM
319
+ end
320
+
321
+ it "formats hashes with nested array values appropriately" do
322
+ @ui.output({ "a" => [ %w{foo bar}, %w{baz bjork} ], "b" => "c" })
323
+ # XXX: using a HEREDOC at this point results in a line with required spaces which auto-whitespace removal settings
324
+ # on editors will remove and will break this test.
325
+ expect(@out.string).to eq("a:\n foo\n bar\n \n baz\n bjork\nb: c\n")
326
+ end
327
+
328
+ it "formats hashes with hash values appropriately" do
329
+ @ui.output({ "a" => { "aa" => "bb", "cc" => "dd" }, "b" => "c" })
330
+ expect(@out.string).to eq <<~EOM
331
+ a:
332
+ aa: bb
333
+ cc: dd
334
+ b: c
335
+ EOM
336
+ end
337
+
338
+ it "formats hashes with empty hash values appropriately" do
339
+ @ui.output({ "a" => {}, "b" => "c" })
340
+ expect(@out.string).to eq <<~EOM
341
+ a:
342
+ b: c
343
+ EOM
344
+ end
345
+ end
346
+
347
+ describe "warn" do
348
+ it_behaves_like "an output mehthod handling IO exceptions", :warn
349
+ end
350
+
351
+ describe "error" do
352
+ it_behaves_like "an output mehthod handling IO exceptions", :warn
353
+ end
354
+
355
+ describe "fatal" do
356
+ it_behaves_like "an output mehthod handling IO exceptions", :warn
357
+ end
358
+
359
+ describe "format_for_display" do
360
+ it "should return the raw data" do
361
+ input = { gi: :go }
362
+ expect(@ui.format_for_display(input)).to eq(input)
363
+ end
364
+
365
+ describe "with --attribute passed" do
366
+ it "should return the deeply nested attribute" do
367
+ input = { "gi" => { "go" => "ge" }, "id" => "sample-data-bag-item" }
368
+ @ui.config[:attribute] = "gi.go"
369
+ expect(@ui.format_for_display(input)).to eq({ "sample-data-bag-item" => { "gi.go" => "ge" } })
370
+ end
371
+
372
+ it "should return multiple attributes" do
373
+ input = { "gi" => "go", "hi" => "ho", "id" => "sample-data-bag-item" }
374
+ @ui.config[:attribute] = %w{gi hi}
375
+ expect(@ui.format_for_display(input)).to eq({ "sample-data-bag-item" => { "gi" => "go", "hi" => "ho" } })
376
+ end
377
+
378
+ it "should handle attributes named the same as methods" do
379
+ input = { "keys" => "values", "hi" => "ho", "id" => "sample-data-bag-item" }
380
+ @ui.config[:attribute] = "keys"
381
+ expect(@ui.format_for_display(input)).to eq({ "sample-data-bag-item" => { "keys" => "values" } })
382
+ end
383
+
384
+ it "should handle nested attributes named the same as methods" do
385
+ input = { "keys" => { "keys" => "values" }, "hi" => "ho", "id" => "sample-data-bag-item" }
386
+ @ui.config[:attribute] = "keys.keys"
387
+ expect(@ui.format_for_display(input)).to eq({ "sample-data-bag-item" => { "keys.keys" => "values" } })
388
+ end
389
+
390
+ it "should return the name attribute" do
391
+ input = Chef::Node.new
392
+ input.name("chef.localdomain")
393
+ @ui.config[:attribute] = "name"
394
+ expect(@ui.format_for_display(input)).to eq( { "chef.localdomain" => { "name" => "chef.localdomain" } })
395
+ end
396
+
397
+ it "should return a 'class' attribute and not the node.class" do
398
+ input = Chef::Node.new
399
+ input.default["class"] = "classy!"
400
+ @ui.config[:attribute] = "class"
401
+ expect(@ui.format_for_display(input)).to eq( { nil => { "class" => "classy!" } } )
402
+ end
403
+
404
+ it "should return the chef_environment attribute" do
405
+ input = Chef::Node.new
406
+ input.chef_environment = "production-partner-load-integration-preview-testing"
407
+ @ui.config[:attribute] = "chef_environment"
408
+ expect(@ui.format_for_display(input)).to eq( { nil => { "chef_environment" => "production-partner-load-integration-preview-testing" } } )
409
+ end
410
+
411
+ it "works with arrays" do
412
+ input = Chef::Node.new
413
+ input.default["array"] = %w{zero one two}
414
+ @ui.config[:attribute] = "array.1"
415
+ expect(@ui.format_for_display(input)).to eq( { nil => { "array.1" => "one" } } )
416
+ end
417
+
418
+ it "returns nil when given an attribute path that isn't a name or attribute" do
419
+ input = { "keys" => { "keys" => "values" }, "hi" => "ho", "id" => "sample-data-bag-item" }
420
+ non_existing_path = "nope.nada.nothingtoseehere"
421
+ @ui.config[:attribute] = non_existing_path
422
+ expect(@ui.format_for_display(input)).to eq({ "sample-data-bag-item" => { non_existing_path => nil } })
423
+ end
424
+
425
+ describe "when --field-separator is passed" do
426
+ it "honors that separator" do
427
+ input = { "keys" => { "with spaces" => { "open" => { "doors" => { "with many.dots" => "when asked" } } } } }
428
+ @ui.config[:field_separator] = ";"
429
+ @ui.config[:attribute] = "keys;with spaces;open;doors;with many.dots"
430
+ expect(@ui.format_for_display(input)).to eq({ nil => { "keys;with spaces;open;doors;with many.dots" => "when asked" } })
431
+ end
432
+ end
433
+ end
434
+
435
+ describe "with --run-list passed" do
436
+ it "should return the run list" do
437
+ input = Chef::Node.new
438
+ input.name("sample-node")
439
+ input.run_list("role[monkey]", "role[churchmouse]")
440
+ @ui.config[:run_list] = true
441
+ response = @ui.format_for_display(input)
442
+ expect(response["sample-node"]["run_list"][0]).to eq("role[monkey]")
443
+ expect(response["sample-node"]["run_list"][1]).to eq("role[churchmouse]")
444
+ end
445
+ end
446
+ end
447
+
448
+ describe "format_cookbook_list_for_display" do
449
+ before(:each) do
450
+ @item = {
451
+ "cookbook_name" => {
452
+ "url" => "http://url/cookbooks/cookbook",
453
+ "versions" => [
454
+ { "version" => "3.0.0", "url" => "http://url/cookbooks/3.0.0" },
455
+ { "version" => "2.0.0", "url" => "http://url/cookbooks/2.0.0" },
456
+ { "version" => "1.0.0", "url" => "http://url/cookbooks/1.0.0" },
457
+ ],
458
+ },
459
+ }
460
+ end
461
+
462
+ it "should return an hash of the cookbooks with versions" do
463
+ expected_response = { "cookbook_name" => "3.0.0 2.0.0 1.0.0" }
464
+ response = @ui.format_cookbook_list_for_display(@item)
465
+ expect(response).to eq(expected_response)
466
+ end
467
+
468
+ describe "with --with-uri" do
469
+ it "should return the URIs" do
470
+ response = {
471
+ "cookbook_name" => {
472
+ "1.0.0" => "http://url/cookbooks/1.0.0",
473
+ "2.0.0" => "http://url/cookbooks/2.0.0",
474
+ "3.0.0" => "http://url/cookbooks/3.0.0" },
475
+ }
476
+ @ui.config[:with_uri] = true
477
+ expect(@ui.format_cookbook_list_for_display(@item)).to eq(response)
478
+ end
479
+ end
480
+
481
+ context "when running on Windows" do
482
+ before(:each) do
483
+ stdout = double("StringIO", tty?: true)
484
+ allow(@ui).to receive(:stdout).and_return(stdout)
485
+ allow(ChefUtils).to receive(:windows?) { true }
486
+ Chef::Config.reset
487
+ end
488
+
489
+ after(:each) do
490
+ Chef::Config.reset
491
+ end
492
+
493
+ it "should have color set to true if knife config has color explicitly set to true" do
494
+ Chef::Config[:color] = true
495
+ @ui.config[:color] = true
496
+ expect(@ui.color?).to eql(true)
497
+ end
498
+
499
+ it "should have color set to false if knife config has color explicitly set to false" do
500
+ Chef::Config[:color] = false
501
+ expect(@ui.color?).to eql(false)
502
+ end
503
+
504
+ it "should not have color set to false by default" do
505
+ expect(@ui.color?).to eql(false)
506
+ end
507
+ end
508
+ end
509
+
510
+ describe "color" do
511
+ context "when ui.color? => true" do
512
+ it "returns colored output" do
513
+ skip "doesn't work on systems that don't correctly have terminals setup for color"
514
+ expect(@ui).to receive(:color?).and_return(true)
515
+ expect(@ui.color("a_bus_is", :yellow)).to eql("\e[33ma_bus_is\e[0m")
516
+ end
517
+ end
518
+
519
+ context "when ui.color? => false" do
520
+ it "returns plain output" do
521
+ expect(@ui).to receive(:color?).and_return(false)
522
+ expect(@ui.color("a_bus_is", :yellow)).to eql("a_bus_is")
523
+ end
524
+ end
525
+ end
526
+
527
+ describe "confirm" do
528
+ let(:stdout) { StringIO.new }
529
+ let(:output) { stdout.string }
530
+
531
+ let(:question) { "monkeys rule" }
532
+ let(:answer) { "y" }
533
+
534
+ let(:default_choice) { nil }
535
+ let(:append_instructions) { true }
536
+
537
+ def run_confirm
538
+ allow(@ui).to receive(:stdout).and_return(stdout)
539
+ allow(@ui.stdin).to receive(:readline).and_return(answer)
540
+ @ui.confirm(question, append_instructions, default_choice)
541
+ end
542
+
543
+ def run_confirm_without_exit
544
+ allow(@ui).to receive(:stdout).and_return(stdout)
545
+ allow(@ui.stdin).to receive(:readline).and_return(answer)
546
+ @ui.confirm_without_exit(question, append_instructions, default_choice)
547
+ end
548
+
549
+ shared_examples_for "confirm with positive answer" do
550
+ it "confirm should return true" do
551
+ expect(run_confirm).to be_truthy
552
+ end
553
+
554
+ it "confirm_without_exit should return true" do
555
+ expect(run_confirm_without_exit).to be_truthy
556
+ end
557
+ end
558
+
559
+ shared_examples_for "confirm with negative answer" do
560
+ it "confirm should exit 3" do
561
+ expect do
562
+ run_confirm
563
+ end.to raise_error(SystemExit) { |e| expect(e.status).to eq(3) }
564
+ end
565
+
566
+ it "confirm_without_exit should return false" do
567
+ expect(run_confirm_without_exit).to be_falsey
568
+ end
569
+ end
570
+
571
+ describe "with default choice set to true" do
572
+ let(:default_choice) { true }
573
+
574
+ it "should show 'Y/n' in the instructions" do
575
+ run_confirm
576
+ expect(output).to include("Y/n")
577
+ end
578
+
579
+ describe "with empty answer" do
580
+ let(:answer) { "" }
581
+
582
+ it_behaves_like "confirm with positive answer"
583
+ end
584
+
585
+ describe "with answer N " do
586
+ let(:answer) { "N" }
587
+
588
+ it_behaves_like "confirm with negative answer"
589
+ end
590
+ end
591
+
592
+ describe "with default choice set to false" do
593
+ let(:default_choice) { false }
594
+
595
+ it "should show 'y/N' in the instructions" do
596
+ run_confirm
597
+ expect(output).to include("y/N")
598
+ end
599
+
600
+ describe "with empty answer" do
601
+ let(:answer) { "" }
602
+
603
+ it_behaves_like "confirm with negative answer"
604
+ end
605
+
606
+ describe "with answer N " do
607
+ let(:answer) { "Y" }
608
+
609
+ it_behaves_like "confirm with positive answer"
610
+ end
611
+ end
612
+
613
+ %w{Y y}.each do |answer|
614
+ describe "with answer #{answer}" do
615
+ let(:answer) { answer }
616
+
617
+ it_behaves_like "confirm with positive answer"
618
+ end
619
+ end
620
+
621
+ %w{N n}.each do |answer|
622
+ describe "with answer #{answer}" do
623
+ let(:answer) { answer }
624
+
625
+ it_behaves_like "confirm with negative answer"
626
+ end
627
+ end
628
+
629
+ describe "with --y or --yes passed" do
630
+ it "should return true" do
631
+ @ui.config[:yes] = true
632
+ expect(run_confirm).to be_truthy
633
+ expect(output).to eq("")
634
+ end
635
+ end
636
+ end
637
+
638
+ describe "when asking for free-form user input" do
639
+ it "asks a question and returns the answer provided by the user" do
640
+ out = StringIO.new
641
+ allow(@ui).to receive(:stdout).and_return(out)
642
+ allow(@ui).to receive(:stdin).and_return(StringIO.new("http://mychefserver.example.com\n"))
643
+ expect(@ui.ask_question("your chef server URL?")).to eq("http://mychefserver.example.com")
644
+ expect(out.string).to eq("your chef server URL?")
645
+ end
646
+
647
+ it "suggests a default setting and returns the default when the user's response only contains whitespace" do
648
+ out = StringIO.new
649
+ allow(@ui).to receive(:stdout).and_return(out)
650
+ allow(@ui).to receive(:stdin).and_return(StringIO.new(" \n"))
651
+ expect(@ui.ask_question("your chef server URL? ", default: "http://localhost:4000")).to eq("http://localhost:4000")
652
+ expect(out.string).to eq("your chef server URL? [http://localhost:4000] ")
653
+ end
654
+ end
655
+
656
+ end