knife 18.2.7 → 18.3.0

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 (667) hide show
  1. checksums.yaml +4 -4
  2. data/Gemfile +26 -26
  3. data/LICENSE +201 -201
  4. data/Rakefile +33 -33
  5. data/bin/knife +24 -24
  6. data/knife.gemspec +62 -62
  7. data/lib/chef/application/knife.rb +234 -234
  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 +242 -242
  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 +1193 -1193
  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 +264 -264
  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 -232
  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 +338 -338
  65. data/lib/chef/knife/core/windows_bootstrap_context.rb +405 -405
  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/list.rb +177 -177
  101. data/lib/chef/knife/node_bulk_delete.rb +75 -75
  102. data/lib/chef/knife/node_create.rb +47 -47
  103. data/lib/chef/knife/node_delete.rb +46 -46
  104. data/lib/chef/knife/node_edit.rb +70 -70
  105. data/lib/chef/knife/node_environment_set.rb +53 -53
  106. data/lib/chef/knife/node_from_file.rb +51 -51
  107. data/lib/chef/knife/node_list.rb +44 -44
  108. data/lib/chef/knife/node_policy_set.rb +79 -79
  109. data/lib/chef/knife/node_run_list_add.rb +104 -104
  110. data/lib/chef/knife/node_run_list_remove.rb +67 -67
  111. data/lib/chef/knife/node_run_list_set.rb +66 -66
  112. data/lib/chef/knife/node_show.rb +63 -63
  113. data/lib/chef/knife/null.rb +12 -12
  114. data/lib/chef/knife/org_create.rb +70 -70
  115. data/lib/chef/knife/org_delete.rb +32 -32
  116. data/lib/chef/knife/org_edit.rb +48 -48
  117. data/lib/chef/knife/org_list.rb +44 -44
  118. data/lib/chef/knife/org_show.rb +31 -31
  119. data/lib/chef/knife/org_user_add.rb +62 -62
  120. data/lib/chef/knife/org_user_remove.rb +103 -103
  121. data/lib/chef/knife/raw.rb +123 -123
  122. data/lib/chef/knife/recipe_list.rb +32 -32
  123. data/lib/chef/knife/rehash.rb +50 -50
  124. data/lib/chef/knife/role_bulk_delete.rb +66 -66
  125. data/lib/chef/knife/role_create.rb +53 -53
  126. data/lib/chef/knife/role_delete.rb +46 -46
  127. data/lib/chef/knife/role_edit.rb +45 -45
  128. data/lib/chef/knife/role_env_run_list_add.rb +87 -87
  129. data/lib/chef/knife/role_env_run_list_clear.rb +55 -55
  130. data/lib/chef/knife/role_env_run_list_remove.rb +57 -57
  131. data/lib/chef/knife/role_env_run_list_replace.rb +60 -60
  132. data/lib/chef/knife/role_env_run_list_set.rb +70 -70
  133. data/lib/chef/knife/role_from_file.rb +51 -51
  134. data/lib/chef/knife/role_list.rb +42 -42
  135. data/lib/chef/knife/role_run_list_add.rb +87 -87
  136. data/lib/chef/knife/role_run_list_clear.rb +55 -55
  137. data/lib/chef/knife/role_run_list_remove.rb +56 -56
  138. data/lib/chef/knife/role_run_list_replace.rb +60 -60
  139. data/lib/chef/knife/role_run_list_set.rb +69 -69
  140. data/lib/chef/knife/role_show.rb +48 -48
  141. data/lib/chef/knife/search.rb +194 -194
  142. data/lib/chef/knife/serve.rb +65 -65
  143. data/lib/chef/knife/show.rb +72 -72
  144. data/lib/chef/knife/ssh.rb +657 -657
  145. data/lib/chef/knife/ssl_check.rb +284 -284
  146. data/lib/chef/knife/ssl_fetch.rb +162 -162
  147. data/lib/chef/knife/status.rb +95 -95
  148. data/lib/chef/knife/supermarket_download.rb +119 -119
  149. data/lib/chef/knife/supermarket_install.rb +192 -192
  150. data/lib/chef/knife/supermarket_list.rb +76 -76
  151. data/lib/chef/knife/supermarket_search.rb +53 -53
  152. data/lib/chef/knife/supermarket_share.rb +166 -166
  153. data/lib/chef/knife/supermarket_show.rb +66 -66
  154. data/lib/chef/knife/supermarket_unshare.rb +62 -62
  155. data/lib/chef/knife/tag_create.rb +52 -52
  156. data/lib/chef/knife/tag_delete.rb +60 -60
  157. data/lib/chef/knife/tag_list.rb +47 -47
  158. data/lib/chef/knife/upload.rb +87 -87
  159. data/lib/chef/knife/user_create.rb +180 -180
  160. data/lib/chef/knife/user_delete.rb +151 -151
  161. data/lib/chef/knife/user_dissociate.rb +42 -42
  162. data/lib/chef/knife/user_edit.rb +94 -94
  163. data/lib/chef/knife/user_invite_add.rb +43 -43
  164. data/lib/chef/knife/user_invite_list.rb +34 -34
  165. data/lib/chef/knife/user_invite_rescind.rb +63 -63
  166. data/lib/chef/knife/user_key_create.rb +73 -73
  167. data/lib/chef/knife/user_key_delete.rb +80 -80
  168. data/lib/chef/knife/user_key_edit.rb +83 -83
  169. data/lib/chef/knife/user_key_list.rb +73 -73
  170. data/lib/chef/knife/user_key_show.rb +80 -80
  171. data/lib/chef/knife/user_list.rb +43 -43
  172. data/lib/chef/knife/user_password.rb +70 -70
  173. data/lib/chef/knife/user_reregister.rb +59 -59
  174. data/lib/chef/knife/user_show.rb +52 -52
  175. data/lib/chef/knife/version.rb +24 -24
  176. data/lib/chef/knife/xargs.rb +282 -282
  177. data/lib/chef/knife/yaml_convert.rb +91 -91
  178. data/lib/chef/knife.rb +673 -673
  179. data/spec/data/apt/chef-integration-test-1.0/debian/changelog +5 -5
  180. data/spec/data/apt/chef-integration-test-1.0/debian/compat +1 -1
  181. data/spec/data/apt/chef-integration-test-1.0/debian/control +13 -13
  182. data/spec/data/apt/chef-integration-test-1.0/debian/copyright +34 -34
  183. data/spec/data/apt/chef-integration-test-1.0/debian/files +1 -1
  184. data/spec/data/apt/chef-integration-test-1.0/debian/rules +13 -13
  185. data/spec/data/apt/chef-integration-test-1.0/debian/source/format +1 -1
  186. data/spec/data/apt/chef-integration-test-1.1/debian/changelog +11 -11
  187. data/spec/data/apt/chef-integration-test-1.1/debian/compat +1 -1
  188. data/spec/data/apt/chef-integration-test-1.1/debian/control +13 -13
  189. data/spec/data/apt/chef-integration-test-1.1/debian/copyright +34 -34
  190. data/spec/data/apt/chef-integration-test-1.1/debian/files +1 -1
  191. data/spec/data/apt/chef-integration-test-1.1/debian/rules +13 -13
  192. data/spec/data/apt/chef-integration-test-1.1/debian/source/format +1 -1
  193. data/spec/data/apt/chef-integration-test2-1.0/debian/changelog +5 -5
  194. data/spec/data/apt/chef-integration-test2-1.0/debian/chef-integration-test2/DEBIAN/conffiles +1 -1
  195. data/spec/data/apt/chef-integration-test2-1.0/debian/chef-integration-test2/DEBIAN/control +10 -10
  196. data/spec/data/apt/chef-integration-test2-1.0/debian/chef-integration-test2/DEBIAN/md5sums +1 -1
  197. data/spec/data/apt/chef-integration-test2-1.0/debian/chef-integration-test2.debhelper.log +45 -45
  198. data/spec/data/apt/chef-integration-test2-1.0/debian/chef-integration-test2.substvars +1 -1
  199. data/spec/data/apt/chef-integration-test2-1.0/debian/compat +1 -1
  200. data/spec/data/apt/chef-integration-test2-1.0/debian/conffiles +1 -1
  201. data/spec/data/apt/chef-integration-test2-1.0/debian/control +13 -13
  202. data/spec/data/apt/chef-integration-test2-1.0/debian/copyright +34 -34
  203. data/spec/data/apt/chef-integration-test2-1.0/debian/files +1 -1
  204. data/spec/data/apt/chef-integration-test2-1.0/debian/rules +13 -13
  205. data/spec/data/apt/chef-integration-test2-1.0/debian/source/format +1 -1
  206. data/spec/data/apt/chef-integration-test2_1.0-1.dsc +18 -18
  207. data/spec/data/apt/chef-integration-test2_1.0-1_amd64.build +91 -91
  208. data/spec/data/apt/chef-integration-test2_1.0-1_amd64.changes +31 -31
  209. data/spec/data/apt/chef-integration-test_1.0-1_amd64.changes +22 -22
  210. data/spec/data/apt/chef-integration-test_1.1-1_amd64.changes +22 -22
  211. data/spec/data/apt/var/www/apt/conf/distributions +7 -7
  212. data/spec/data/apt/var/www/apt/conf/incoming +4 -4
  213. data/spec/data/apt/var/www/apt/conf/pulls +3 -3
  214. data/spec/data/apt/var/www/apt/db/version +4 -4
  215. data/spec/data/apt/var/www/apt/dists/sid/Release +19 -19
  216. data/spec/data/apt/var/www/apt/dists/sid/main/binary-amd64/Packages +16 -16
  217. data/spec/data/apt/var/www/apt/dists/sid/main/binary-amd64/Release +5 -5
  218. data/spec/data/bootstrap/encrypted_data_bag_secret +1 -1
  219. data/spec/data/bootstrap/no_proxy.erb +2 -2
  220. data/spec/data/bootstrap/secret.erb +9 -9
  221. data/spec/data/bootstrap/test-hints.erb +12 -12
  222. data/spec/data/bootstrap/test.erb +1 -1
  223. data/spec/data/cb_version_cookbooks/tatft/README.rdoc +2 -2
  224. data/spec/data/cb_version_cookbooks/tatft/attributes/default.rb +1 -1
  225. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-600hhz-0 +1 -1
  226. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-ahd2gq-0 +1 -1
  227. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-api8ux-0 +1 -1
  228. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-b0r1m1-0 +1 -1
  229. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-bfygsi-0 +1 -1
  230. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-el14l6-0 +1 -1
  231. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-ivrl3y-0 +1 -1
  232. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-kkbs85-0 +1 -1
  233. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-ory1ux-0 +1 -1
  234. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-pgsq76-0 +1 -1
  235. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-ra8uim-0 +1 -1
  236. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-t7k1g-0 +1 -1
  237. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-t8g0sv-0 +1 -1
  238. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-ufy6g3-0 +1 -1
  239. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-x2d6j9-0 +1 -1
  240. data/spec/data/checksum_cache/chef-file--tmp-chef-rendered-template20100929-10863-xi0l6h-0 +1 -1
  241. data/spec/data/client.d_00/00-foo.rb +2 -2
  242. data/spec/data/client.d_00/01-bar.rb +1 -1
  243. data/spec/data/client.d_00/02-strings.rb +2 -2
  244. data/spec/data/client.d_00/bar +1 -1
  245. data/spec/data/client.d_01/foo/bar.rb +1 -1
  246. data/spec/data/client.d_02/foo.rb/foo.txt +1 -1
  247. data/spec/data/config.rb +6 -6
  248. data/spec/data/cookbooks/angrybash/metadata.rb +2 -2
  249. data/spec/data/cookbooks/angrybash/recipes/default.rb +8 -8
  250. data/spec/data/cookbooks/apache2/files/default/apache2_module_conf_generate.pl +2 -2
  251. data/spec/data/cookbooks/apache2/metadata.json +33 -33
  252. data/spec/data/cookbooks/apache2/metadata.rb +2 -2
  253. data/spec/data/cookbooks/apache2/recipes/default.rb +2 -2
  254. data/spec/data/cookbooks/borken/metadata.rb +2 -2
  255. data/spec/data/cookbooks/borken/recipes/default.rb +1 -1
  256. data/spec/data/cookbooks/borken/templates/default/borken.erb +1 -1
  257. data/spec/data/cookbooks/chefignore +8 -8
  258. data/spec/data/cookbooks/ignorken/files/default/not_me.rb +2 -2
  259. data/spec/data/cookbooks/ignorken/metadata.rb +2 -2
  260. data/spec/data/cookbooks/ignorken/recipes/ignoreme.rb +1 -1
  261. data/spec/data/cookbooks/ignorken/templates/ubuntu-12.10/not_me.rb +2 -2
  262. data/spec/data/cookbooks/irssi/files/default/irssi.response +2 -2
  263. data/spec/data/cookbooks/java/files/default/java.response +1 -1
  264. data/spec/data/cookbooks/java/metadata.json +33 -33
  265. data/spec/data/cookbooks/java/metadata.rb +2 -2
  266. data/spec/data/cookbooks/name-mismatch-versionnumber/README.md +4 -4
  267. data/spec/data/cookbooks/name-mismatch-versionnumber/metadata.rb +8 -8
  268. data/spec/data/cookbooks/name-mismatch-versionnumber/recipes/default.rb +8 -8
  269. data/spec/data/cookbooks/openldap/attributes/default.rb +16 -16
  270. data/spec/data/cookbooks/openldap/attributes/smokey.rb +1 -1
  271. data/spec/data/cookbooks/openldap/definitions/client.rb +5 -5
  272. data/spec/data/cookbooks/openldap/definitions/server.rb +5 -5
  273. data/spec/data/cookbooks/openldap/files/default/.dotfile +1 -1
  274. data/spec/data/cookbooks/openldap/files/default/remotedir/.a_dotdir/.a_dotfile_in_a_dotdir +1 -1
  275. data/spec/data/cookbooks/openldap/files/default/remotedir/not_a_template.erb +2 -2
  276. data/spec/data/cookbooks/openldap/files/default/remotedir/remote_dir_file1.txt +2 -2
  277. data/spec/data/cookbooks/openldap/files/default/remotedir/remote_dir_file2.txt +2 -2
  278. data/spec/data/cookbooks/openldap/files/default/remotedir/remotesubdir/.a_dotfile +1 -1
  279. data/spec/data/cookbooks/openldap/files/default/remotedir/remotesubdir/remote_subdir_file1.txt +2 -2
  280. data/spec/data/cookbooks/openldap/files/default/remotedir/remotesubdir/remote_subdir_file2.txt +2 -2
  281. data/spec/data/cookbooks/openldap/files/default/remotedir/subdir_with_no_file_just_a_subsubdir/the_subsubdir/some_file.txt +3 -3
  282. data/spec/data/cookbooks/openldap/libraries/openldap/version.rb +3 -3
  283. data/spec/data/cookbooks/openldap/libraries/openldap.rb +4 -4
  284. data/spec/data/cookbooks/openldap/metadata.rb +8 -8
  285. data/spec/data/cookbooks/openldap/recipes/default.rb +4 -4
  286. data/spec/data/cookbooks/openldap/recipes/gigantor.rb +3 -3
  287. data/spec/data/cookbooks/openldap/recipes/one.rb +15 -15
  288. data/spec/data/cookbooks/openldap/recipes/return.rb +2 -2
  289. data/spec/data/cookbooks/openldap/templates/default/helper_test.erb +1 -1
  290. data/spec/data/cookbooks/openldap/templates/default/helpers.erb +14 -14
  291. data/spec/data/cookbooks/openldap/templates/default/helpers_via_partial_test.erb +1 -1
  292. data/spec/data/cookbooks/openldap/templates/default/nested_openldap_partials.erb +1 -1
  293. data/spec/data/cookbooks/openldap/templates/default/nested_partial.erb +1 -1
  294. data/spec/data/cookbooks/openldap/templates/default/no_windows_line_endings.erb +4 -4
  295. data/spec/data/cookbooks/openldap/templates/default/openldap_nested_variable_stuff.erb +1 -1
  296. data/spec/data/cookbooks/openldap/templates/default/openldap_variable_stuff.conf.erb +1 -1
  297. data/spec/data/cookbooks/openldap/templates/default/test.erb +1 -1
  298. data/spec/data/cookbooks/preseed/files/default/preseed-file.seed +1 -1
  299. data/spec/data/cookbooks/preseed/files/default/preseed-template.seed +4 -4
  300. data/spec/data/cookbooks/preseed/metadata.rb +2 -2
  301. data/spec/data/cookbooks/preseed/templates/default/preseed-template-variables.seed +1 -1
  302. data/spec/data/cookbooks/preseed/templates/default/preseed-template.seed +1 -1
  303. data/spec/data/cookbooks/starter/chefignore +8 -8
  304. data/spec/data/cookbooks/starter/metadata.rb +2 -2
  305. data/spec/data/cookbooks/starter/recipes/default.rb +4 -4
  306. data/spec/data/cookbooks/supports-platform-constraints/metadata.rb +5 -5
  307. data/spec/data/cookbooks/wget/files/default/wget.response +2 -2
  308. data/spec/data/definitions/test.rb +4 -4
  309. data/spec/data/environment-config.rb +4 -4
  310. data/spec/data/file-providers-method-snapshot-chef-11-4.json +127 -127
  311. data/spec/data/fileedit/hosts +4 -4
  312. data/spec/data/incomplete-metadata-chef-repo/incomplete-metadata/README.md +4 -4
  313. data/spec/data/incomplete-metadata-chef-repo/incomplete-metadata/metadata.rb +13 -13
  314. data/spec/data/incomplete-metadata-chef-repo/incomplete-metadata/recipes/default.rb +8 -8
  315. data/spec/data/invalid-metadata-chef-repo/invalid-metadata/README.md +4 -4
  316. data/spec/data/invalid-metadata-chef-repo/invalid-metadata/metadata.rb +9 -9
  317. data/spec/data/invalid-metadata-chef-repo/invalid-metadata/recipes/default.rb +8 -8
  318. data/spec/data/kitchen/chefignore +6 -6
  319. data/spec/data/kitchen/openldap/attributes/default.rb +3 -3
  320. data/spec/data/kitchen/openldap/attributes/robinson.rb +2 -2
  321. data/spec/data/kitchen/openldap/definitions/client.rb +3 -3
  322. data/spec/data/kitchen/openldap/definitions/drewbarrymore.rb +2 -2
  323. data/spec/data/kitchen/openldap/recipes/gigantor.rb +2 -2
  324. data/spec/data/kitchen/openldap/recipes/ignoreme.rb +2 -2
  325. data/spec/data/kitchen/openldap/recipes/woot.rb +3 -3
  326. data/spec/data/knife_subcommand/test_explicit_category.rb +6 -6
  327. data/spec/data/knife_subcommand/test_name_mapping.rb +4 -4
  328. data/spec/data/knife_subcommand/test_yourself.rb +21 -21
  329. data/spec/data/lwrp/providers/buck_passer.rb +28 -28
  330. data/spec/data/lwrp/providers/buck_passer_2.rb +26 -26
  331. data/spec/data/lwrp/providers/embedded_resource_accesses_providers_scope.rb +28 -28
  332. data/spec/data/lwrp/providers/inline_compiler.rb +24 -24
  333. data/spec/data/lwrp/providers/monkey_name_printer.rb +5 -5
  334. data/spec/data/lwrp/providers/paint_drying_watcher.rb +7 -7
  335. data/spec/data/lwrp/providers/thumb_twiddler.rb +7 -7
  336. data/spec/data/lwrp/resources/bar.rb +4 -4
  337. data/spec/data/lwrp/resources/buck_passer.rb +6 -6
  338. data/spec/data/lwrp/resources/buck_passer_2.rb +4 -4
  339. data/spec/data/lwrp/resources/embedded_resource_accesses_providers_scope.rb +4 -4
  340. data/spec/data/lwrp/resources/foo.rb +6 -6
  341. data/spec/data/lwrp/resources/inline_compiler.rb +4 -4
  342. data/spec/data/lwrp/resources/monkey_name_printer.rb +6 -6
  343. data/spec/data/lwrp/resources/paint_drying_watcher.rb +4 -4
  344. data/spec/data/lwrp/resources/thumb_twiddler.rb +4 -4
  345. data/spec/data/lwrp/resources_with_default_attributes/nodeattr.rb +3 -3
  346. data/spec/data/lwrp_const_scoping/resources/conflict.rb +1 -1
  347. data/spec/data/lwrp_override/providers/buck_passer.rb +5 -5
  348. data/spec/data/lwrp_override/resources/foo.rb +11 -11
  349. data/spec/data/mac_users/10.9.plist.xml +560 -560
  350. data/spec/data/mac_users/10.9.shadow.xml +21 -21
  351. data/spec/data/metadata/quick_start/metadata.rb +14 -14
  352. data/spec/data/mixin/invalid_data.rb +3 -3
  353. data/spec/data/mixin/real_data.rb +2 -2
  354. data/spec/data/nested.json +2 -2
  355. data/spec/data/nodes/default.rb +15 -15
  356. data/spec/data/nodes/test.example.com.rb +17 -17
  357. data/spec/data/nodes/test.rb +15 -15
  358. data/spec/data/null_config.rb +1 -1
  359. data/spec/data/object_loader/environments/test.json +7 -7
  360. data/spec/data/object_loader/environments/test.rb +2 -2
  361. data/spec/data/object_loader/environments/test_json_class.json +8 -8
  362. data/spec/data/object_loader/nodes/test.json +7 -7
  363. data/spec/data/object_loader/nodes/test.rb +2 -2
  364. data/spec/data/object_loader/nodes/test_json_class.json +8 -8
  365. data/spec/data/object_loader/roles/test.json +7 -7
  366. data/spec/data/object_loader/roles/test.rb +2 -2
  367. data/spec/data/object_loader/roles/test_json_class.json +8 -8
  368. data/spec/data/partial_one.erb +1 -1
  369. data/spec/data/prefer_metadata_json/metadata.json +51 -51
  370. data/spec/data/prefer_metadata_json/metadata.rb +6 -6
  371. data/spec/data/recipes/test.rb +7 -7
  372. data/spec/data/root_alias_cookbooks/dup_attr/attributes/default.rb +1 -1
  373. data/spec/data/root_alias_cookbooks/dup_attr/attributes.rb +1 -1
  374. data/spec/data/root_alias_cookbooks/dup_attr/metadata.rb +2 -2
  375. data/spec/data/root_alias_cookbooks/dup_attr/recipe.rb +3 -3
  376. data/spec/data/root_alias_cookbooks/dup_recipe/attributes.rb +1 -1
  377. data/spec/data/root_alias_cookbooks/dup_recipe/metadata.rb +2 -2
  378. data/spec/data/root_alias_cookbooks/dup_recipe/recipe.rb +3 -3
  379. data/spec/data/root_alias_cookbooks/dup_recipe/recipes/default.rb +3 -3
  380. data/spec/data/root_alias_cookbooks/simple/attributes.rb +1 -1
  381. data/spec/data/root_alias_cookbooks/simple/metadata.rb +2 -2
  382. data/spec/data/root_alias_cookbooks/simple/recipe.rb +3 -3
  383. data/spec/data/rubygems.org/sexp_processor-info +49 -49
  384. data/spec/data/run_context/cookbooks/circular-dep1/attributes/default.rb +2 -2
  385. data/spec/data/run_context/cookbooks/circular-dep1/definitions/circular_dep1_res.rb +1 -1
  386. data/spec/data/run_context/cookbooks/circular-dep1/libraries/lib.rb +2 -2
  387. data/spec/data/run_context/cookbooks/circular-dep1/metadata.rb +2 -2
  388. data/spec/data/run_context/cookbooks/circular-dep1/providers/provider.rb +1 -1
  389. data/spec/data/run_context/cookbooks/circular-dep1/resources/resource.rb +2 -2
  390. data/spec/data/run_context/cookbooks/circular-dep2/attributes/default.rb +2 -2
  391. data/spec/data/run_context/cookbooks/circular-dep2/definitions/circular_dep2_res.rb +1 -1
  392. data/spec/data/run_context/cookbooks/circular-dep2/libraries/lib.rb +2 -2
  393. data/spec/data/run_context/cookbooks/circular-dep2/metadata.rb +2 -2
  394. data/spec/data/run_context/cookbooks/circular-dep2/providers/provider.rb +1 -1
  395. data/spec/data/run_context/cookbooks/circular-dep2/resources/resource.rb +2 -2
  396. data/spec/data/run_context/cookbooks/dependency1/attributes/aa_first.rb +2 -2
  397. data/spec/data/run_context/cookbooks/dependency1/attributes/default.rb +2 -2
  398. data/spec/data/run_context/cookbooks/dependency1/attributes/unparsed_file +1 -1
  399. data/spec/data/run_context/cookbooks/dependency1/attributes/zz_last.rb +2 -2
  400. data/spec/data/run_context/cookbooks/dependency1/definitions/dependency1_res.rb +1 -1
  401. data/spec/data/run_context/cookbooks/dependency1/definitions/unparsed_file +1 -1
  402. data/spec/data/run_context/cookbooks/dependency1/libraries/lib.rb +2 -2
  403. data/spec/data/run_context/cookbooks/dependency1/libraries/unparsed_file +1 -1
  404. data/spec/data/run_context/cookbooks/dependency1/providers/provider.rb +1 -1
  405. data/spec/data/run_context/cookbooks/dependency1/providers/unparsed_file +1 -1
  406. data/spec/data/run_context/cookbooks/dependency1/recipes/unparsed_file +1 -1
  407. data/spec/data/run_context/cookbooks/dependency1/resources/resource.rb +2 -2
  408. data/spec/data/run_context/cookbooks/dependency1/resources/unparsed_file +1 -1
  409. data/spec/data/run_context/cookbooks/dependency2/attributes/default.rb +2 -2
  410. data/spec/data/run_context/cookbooks/dependency2/definitions/dependency2_res.rb +1 -1
  411. data/spec/data/run_context/cookbooks/dependency2/libraries/lib.rb +2 -2
  412. data/spec/data/run_context/cookbooks/dependency2/providers/provider.rb +1 -1
  413. data/spec/data/run_context/cookbooks/dependency2/resources/resource.rb +2 -2
  414. data/spec/data/run_context/cookbooks/include/recipes/default.rb +24 -24
  415. data/spec/data/run_context/cookbooks/include/recipes/includee.rb +3 -3
  416. data/spec/data/run_context/cookbooks/no-default-attr/attributes/server.rb +2 -2
  417. data/spec/data/run_context/cookbooks/no-default-attr/definitions/no_default-attr_res.rb +1 -1
  418. data/spec/data/run_context/cookbooks/no-default-attr/providers/provider.rb +1 -1
  419. data/spec/data/run_context/cookbooks/no-default-attr/resources/resource.rb +2 -2
  420. data/spec/data/run_context/cookbooks/test/attributes/george.rb +1 -1
  421. data/spec/data/run_context/cookbooks/test/definitions/new_animals.rb +9 -9
  422. data/spec/data/run_context/cookbooks/test/definitions/new_cat.rb +5 -5
  423. data/spec/data/run_context/cookbooks/test/definitions/test_res.rb +1 -1
  424. data/spec/data/run_context/cookbooks/test/providers/provider.rb +1 -1
  425. data/spec/data/run_context/cookbooks/test/recipes/default.rb +5 -5
  426. data/spec/data/run_context/cookbooks/test/recipes/one.rb +7 -7
  427. data/spec/data/run_context/cookbooks/test/recipes/two.rb +7 -7
  428. data/spec/data/run_context/cookbooks/test/resources/resource.rb +3 -3
  429. data/spec/data/run_context/cookbooks/test-with-circular-deps/attributes/default.rb +2 -2
  430. data/spec/data/run_context/cookbooks/test-with-circular-deps/definitions/test_with-circular-deps_res.rb +1 -1
  431. data/spec/data/run_context/cookbooks/test-with-circular-deps/libraries/lib.rb +2 -2
  432. data/spec/data/run_context/cookbooks/test-with-circular-deps/metadata.rb +2 -2
  433. data/spec/data/run_context/cookbooks/test-with-circular-deps/providers/provider.rb +1 -1
  434. data/spec/data/run_context/cookbooks/test-with-circular-deps/resources/resource.rb +3 -3
  435. data/spec/data/run_context/cookbooks/test-with-deps/attributes/default.rb +2 -2
  436. data/spec/data/run_context/cookbooks/test-with-deps/definitions/test_with-deps_res.rb +1 -1
  437. data/spec/data/run_context/cookbooks/test-with-deps/libraries/lib.rb +1 -1
  438. data/spec/data/run_context/cookbooks/test-with-deps/metadata.rb +3 -3
  439. data/spec/data/run_context/cookbooks/test-with-deps/providers/provider.rb +1 -1
  440. data/spec/data/run_context/cookbooks/test-with-deps/resources/resource.rb +2 -2
  441. data/spec/data/run_context/nodes/run_context.rb +5 -5
  442. data/spec/data/sample_msu1.xml +10 -10
  443. data/spec/data/sample_msu2.xml +14 -14
  444. data/spec/data/sample_msu3.xml +16 -16
  445. data/spec/data/search_queries_to_transform.txt +98 -98
  446. data/spec/data/shef-config.rb +11 -11
  447. data/spec/data/snap_package/async_result_success.json +6 -6
  448. data/spec/data/snap_package/change_id_result.json +175 -175
  449. data/spec/data/snap_package/find_result_failure.json +10 -10
  450. data/spec/data/snap_package/find_result_success.json +70 -70
  451. data/spec/data/snap_package/get_by_name_result_failure.json +10 -10
  452. data/spec/data/snap_package/get_by_name_result_success.json +38 -38
  453. data/spec/data/snap_package/get_conf_success.json +10 -10
  454. data/spec/data/snap_package/result_failure.json +9 -9
  455. data/spec/data/ssl/5e707473.0 +18 -18
  456. data/spec/data/ssl/chef-rspec.cert +27 -27
  457. data/spec/data/ssl/chef-rspec.key +27 -27
  458. data/spec/data/ssl/key.pem +15 -15
  459. data/spec/data/ssl/private_key.pem +27 -27
  460. data/spec/data/ssl/private_key_with_whitespace.pem +32 -32
  461. data/spec/data/standalone_cookbook/chefignore +9 -9
  462. data/spec/data/standalone_cookbook/recipes/default.rb +2 -2
  463. data/spec/data/templates/failed.erb +5 -5
  464. data/spec/data/trusted_certs/example.crt +22 -22
  465. data/spec/data/trusted_certs/example_no_cn.crt +36 -36
  466. data/spec/data/trusted_certs/intermediate.pem +27 -27
  467. data/spec/data/trusted_certs/opscode.pem +57 -57
  468. data/spec/data/trusted_certs/root.pem +22 -22
  469. data/spec/data/windows_certificates/base64_test.cer +20 -20
  470. data/spec/data/windows_certificates/othertest.cer +20 -20
  471. data/spec/data/windows_certificates/test.cer +20 -20
  472. data/spec/data/windows_certificates/test.pem +20 -20
  473. data/spec/functional/configure_spec.rb +33 -33
  474. data/spec/functional/cookbook_delete_spec.rb +156 -156
  475. data/spec/functional/exec_spec.rb +55 -55
  476. data/spec/functional/rehash_spec.rb +39 -39
  477. data/spec/functional/smoke_test.rb +42 -42
  478. data/spec/functional/ssh_spec.rb +352 -352
  479. data/spec/functional/version_spec.rb +26 -26
  480. data/spec/integration/chef_fs_data_store_spec.rb +557 -557
  481. data/spec/integration/chef_repo_path_spec.rb +962 -962
  482. data/spec/integration/chef_repository_file_system_spec.rb +200 -200
  483. data/spec/integration/chefignore_spec.rb +301 -301
  484. data/spec/integration/client_bulk_delete_spec.rb +131 -131
  485. data/spec/integration/client_create_spec.rb +70 -70
  486. data/spec/integration/client_delete_spec.rb +64 -64
  487. data/spec/integration/client_key_create_spec.rb +66 -66
  488. data/spec/integration/client_key_delete_spec.rb +43 -43
  489. data/spec/integration/client_key_list_spec.rb +61 -61
  490. data/spec/integration/client_key_show_spec.rb +45 -45
  491. data/spec/integration/client_list_spec.rb +49 -49
  492. data/spec/integration/client_show_spec.rb +37 -37
  493. data/spec/integration/commands_spec.rb +55 -55
  494. data/spec/integration/common_options_spec.rb +174 -174
  495. data/spec/integration/config_list_spec.rb +220 -220
  496. data/spec/integration/config_show_spec.rb +192 -192
  497. data/spec/integration/config_use_spec.rb +198 -198
  498. data/spec/integration/cookbook_api_ipv6_spec.rb +113 -113
  499. data/spec/integration/cookbook_bulk_delete_spec.rb +65 -65
  500. data/spec/integration/cookbook_download_spec.rb +72 -72
  501. data/spec/integration/cookbook_list_spec.rb +55 -55
  502. data/spec/integration/cookbook_show_spec.rb +149 -149
  503. data/spec/integration/cookbook_upload_spec.rb +128 -128
  504. data/spec/integration/data_bag_create_spec.rb +125 -125
  505. data/spec/integration/data_bag_delete_spec.rb +59 -59
  506. data/spec/integration/data_bag_edit_spec.rb +105 -105
  507. data/spec/integration/data_bag_from_file_spec.rb +116 -116
  508. data/spec/integration/data_bag_list_spec.rb +44 -44
  509. data/spec/integration/data_bag_show_spec.rb +95 -95
  510. data/spec/integration/delete_spec.rb +1018 -1018
  511. data/spec/integration/deps_spec.rb +703 -703
  512. data/spec/integration/diff_spec.rb +605 -605
  513. data/spec/integration/download_spec.rb +1336 -1336
  514. data/spec/integration/environment_compare_spec.rb +75 -75
  515. data/spec/integration/environment_create_spec.rb +41 -41
  516. data/spec/integration/environment_delete_spec.rb +37 -37
  517. data/spec/integration/environment_from_file_spec.rb +116 -116
  518. data/spec/integration/environment_list_spec.rb +42 -42
  519. data/spec/integration/environment_show_spec.rb +77 -77
  520. data/spec/integration/list_spec.rb +1060 -1060
  521. data/spec/integration/node_bulk_delete_spec.rb +52 -52
  522. data/spec/integration/node_create_spec.rb +47 -47
  523. data/spec/integration/node_delete_spec.rb +48 -48
  524. data/spec/integration/node_environment_set_spec.rb +46 -46
  525. data/spec/integration/node_from_file_spec.rb +59 -59
  526. data/spec/integration/node_list_spec.rb +45 -45
  527. data/spec/integration/node_run_list_add_spec.rb +54 -54
  528. data/spec/integration/node_run_list_remove_spec.rb +36 -36
  529. data/spec/integration/node_run_list_set_spec.rb +41 -41
  530. data/spec/integration/node_show_spec.rb +36 -36
  531. data/spec/integration/raw_spec.rb +297 -297
  532. data/spec/integration/redirection_spec.rb +64 -64
  533. data/spec/integration/role_bulk_delete_spec.rb +52 -52
  534. data/spec/integration/role_create_spec.rb +41 -41
  535. data/spec/integration/role_delete_spec.rb +48 -48
  536. data/spec/integration/role_from_file_spec.rb +96 -96
  537. data/spec/integration/role_list_spec.rb +45 -45
  538. data/spec/integration/role_show_spec.rb +51 -51
  539. data/spec/integration/search_node_spec.rb +40 -40
  540. data/spec/integration/serve_spec.rb +92 -92
  541. data/spec/integration/show_spec.rb +197 -197
  542. data/spec/integration/upload_spec.rb +1616 -1616
  543. data/spec/knife_spec_helper.rb +241 -241
  544. data/spec/support/chef_helpers.rb +79 -79
  545. data/spec/support/key_helpers.rb +102 -102
  546. data/spec/support/platform_helpers.rb +255 -255
  547. data/spec/support/platforms/prof/gc.rb +51 -51
  548. data/spec/support/platforms/prof/win32.rb +45 -45
  549. data/spec/support/platforms/win32/spec_service.rb +57 -57
  550. data/spec/support/recipe_dsl_helper.rb +83 -83
  551. data/spec/support/shared/context/config.rb +18 -18
  552. data/spec/support/shared/functional/knife.rb +37 -37
  553. data/spec/support/shared/integration/integration_helper.rb +122 -122
  554. data/spec/support/shared/integration/knife_support.rb +192 -192
  555. data/spec/support/shared/matchers/exit_with_code.rb +32 -32
  556. data/spec/support/shared/matchers/match_environment_variable.rb +17 -17
  557. data/spec/support/shared/unit/knife_shared.rb +39 -39
  558. data/spec/support/shared/unit/mock_shellout.rb +49 -49
  559. data/spec/tiny_server.rb +190 -190
  560. data/spec/unit/application/knife_spec.rb +241 -241
  561. data/spec/unit/knife/bootstrap/chef_vault_handler_spec.rb +152 -152
  562. data/spec/unit/knife/bootstrap/client_builder_spec.rb +207 -207
  563. data/spec/unit/knife/bootstrap/train_connector_spec.rb +244 -244
  564. data/spec/unit/knife/bootstrap_spec.rb +2288 -2233
  565. data/spec/unit/knife/client_bulk_delete_spec.rb +166 -166
  566. data/spec/unit/knife/client_create_spec.rb +232 -232
  567. data/spec/unit/knife/client_delete_spec.rb +99 -99
  568. data/spec/unit/knife/client_edit_spec.rb +53 -53
  569. data/spec/unit/knife/client_list_spec.rb +34 -34
  570. data/spec/unit/knife/client_reregister_spec.rb +62 -62
  571. data/spec/unit/knife/client_show_spec.rb +52 -52
  572. data/spec/unit/knife/configure_client_spec.rb +81 -81
  573. data/spec/unit/knife/configure_spec.rb +190 -190
  574. data/spec/unit/knife/cookbook_bulk_delete_spec.rb +87 -87
  575. data/spec/unit/knife/cookbook_delete_spec.rb +239 -239
  576. data/spec/unit/knife/cookbook_download_spec.rb +255 -255
  577. data/spec/unit/knife/cookbook_list_spec.rb +88 -88
  578. data/spec/unit/knife/cookbook_metadata_from_file_spec.rb +72 -72
  579. data/spec/unit/knife/cookbook_metadata_spec.rb +182 -182
  580. data/spec/unit/knife/cookbook_show_spec.rb +253 -253
  581. data/spec/unit/knife/cookbook_upload_spec.rb +426 -426
  582. data/spec/unit/knife/core/bootstrap_context_spec.rb +287 -287
  583. data/spec/unit/knife/core/cookbook_scm_repo_spec.rb +187 -187
  584. data/spec/unit/knife/core/cookbook_site_streaming_uploader_spec.rb +198 -198
  585. data/spec/unit/knife/core/gem_glob_loader_spec.rb +242 -242
  586. data/spec/unit/knife/core/hashed_command_loader_spec.rb +112 -112
  587. data/spec/unit/knife/core/node_editor_spec.rb +211 -211
  588. data/spec/unit/knife/core/object_loader_spec.rb +81 -81
  589. data/spec/unit/knife/core/status_presenter_spec.rb +54 -54
  590. data/spec/unit/knife/core/subcommand_loader_spec.rb +64 -64
  591. data/spec/unit/knife/core/ui_spec.rb +656 -656
  592. data/spec/unit/knife/core/windows_bootstrap_context_spec.rb +238 -238
  593. data/spec/unit/knife/data_bag_create_spec.rb +175 -175
  594. data/spec/unit/knife/data_bag_edit_spec.rb +126 -126
  595. data/spec/unit/knife/data_bag_from_file_spec.rb +174 -174
  596. data/spec/unit/knife/data_bag_secret_options_spec.rb +173 -173
  597. data/spec/unit/knife/data_bag_show_spec.rb +139 -139
  598. data/spec/unit/knife/environment_compare_spec.rb +112 -112
  599. data/spec/unit/knife/environment_create_spec.rb +91 -91
  600. data/spec/unit/knife/environment_delete_spec.rb +71 -71
  601. data/spec/unit/knife/environment_edit_spec.rb +79 -79
  602. data/spec/unit/knife/environment_from_file_spec.rb +90 -90
  603. data/spec/unit/knife/environment_list_spec.rb +54 -54
  604. data/spec/unit/knife/environment_show_spec.rb +52 -52
  605. data/spec/unit/knife/key_create_spec.rb +223 -223
  606. data/spec/unit/knife/key_delete_spec.rb +133 -133
  607. data/spec/unit/knife/key_edit_spec.rb +264 -264
  608. data/spec/unit/knife/key_helper.rb +74 -74
  609. data/spec/unit/knife/key_list_spec.rb +216 -216
  610. data/spec/unit/knife/key_show_spec.rb +126 -126
  611. data/spec/unit/knife/node_bulk_delete_spec.rb +94 -94
  612. data/spec/unit/knife/node_delete_spec.rb +77 -77
  613. data/spec/unit/knife/node_edit_spec.rb +116 -116
  614. data/spec/unit/knife/node_environment_set_spec.rb +61 -61
  615. data/spec/unit/knife/node_from_file_spec.rb +59 -59
  616. data/spec/unit/knife/node_list_spec.rb +62 -62
  617. data/spec/unit/knife/node_policy_set_spec.rb +122 -122
  618. data/spec/unit/knife/node_run_list_add_spec.rb +145 -145
  619. data/spec/unit/knife/node_run_list_remove_spec.rb +106 -106
  620. data/spec/unit/knife/node_run_list_set_spec.rb +115 -115
  621. data/spec/unit/knife/node_show_spec.rb +65 -65
  622. data/spec/unit/knife/org_create_spec.rb +76 -76
  623. data/spec/unit/knife/org_delete_spec.rb +41 -41
  624. data/spec/unit/knife/org_edit_spec.rb +49 -49
  625. data/spec/unit/knife/org_list_spec.rb +58 -58
  626. data/spec/unit/knife/org_show_spec.rb +45 -45
  627. data/spec/unit/knife/org_user_add_spec.rb +39 -39
  628. data/spec/unit/knife/raw_spec.rb +43 -43
  629. data/spec/unit/knife/role_bulk_delete_spec.rb +80 -80
  630. data/spec/unit/knife/role_create_spec.rb +80 -80
  631. data/spec/unit/knife/role_delete_spec.rb +67 -67
  632. data/spec/unit/knife/role_edit_spec.rb +77 -77
  633. data/spec/unit/knife/role_env_run_list_add_spec.rb +217 -217
  634. data/spec/unit/knife/role_env_run_list_clear_spec.rb +94 -94
  635. data/spec/unit/knife/role_env_run_list_remove_spec.rb +102 -102
  636. data/spec/unit/knife/role_env_run_list_replace_spec.rb +105 -105
  637. data/spec/unit/knife/role_env_run_list_set_spec.rb +99 -99
  638. data/spec/unit/knife/role_from_file_spec.rb +69 -69
  639. data/spec/unit/knife/role_list_spec.rb +54 -54
  640. data/spec/unit/knife/role_run_list_add_spec.rb +179 -179
  641. data/spec/unit/knife/role_run_list_clear_spec.rb +84 -84
  642. data/spec/unit/knife/role_run_list_remove_spec.rb +92 -92
  643. data/spec/unit/knife/role_run_list_replace_spec.rb +98 -98
  644. data/spec/unit/knife/role_run_list_set_spec.rb +89 -89
  645. data/spec/unit/knife/role_show_spec.rb +59 -59
  646. data/spec/unit/knife/ssh_spec.rb +403 -403
  647. data/spec/unit/knife/ssl_check_spec.rb +256 -256
  648. data/spec/unit/knife/ssl_fetch_spec.rb +222 -222
  649. data/spec/unit/knife/status_spec.rb +112 -112
  650. data/spec/unit/knife/supermarket_download_spec.rb +152 -152
  651. data/spec/unit/knife/supermarket_install_spec.rb +203 -203
  652. data/spec/unit/knife/supermarket_list_spec.rb +70 -70
  653. data/spec/unit/knife/supermarket_search_spec.rb +85 -85
  654. data/spec/unit/knife/supermarket_share_spec.rb +208 -208
  655. data/spec/unit/knife/supermarket_unshare_spec.rb +78 -78
  656. data/spec/unit/knife/tag_create_spec.rb +23 -23
  657. data/spec/unit/knife/tag_delete_spec.rb +25 -25
  658. data/spec/unit/knife/tag_list_spec.rb +23 -23
  659. data/spec/unit/knife/user_create_spec.rb +282 -282
  660. data/spec/unit/knife/user_delete_spec.rb +171 -171
  661. data/spec/unit/knife/user_edit_spec.rb +54 -54
  662. data/spec/unit/knife/user_list_spec.rb +73 -73
  663. data/spec/unit/knife/user_password_spec.rb +64 -64
  664. data/spec/unit/knife/user_reregister_spec.rb +56 -56
  665. data/spec/unit/knife/user_show_spec.rb +91 -91
  666. data/spec/unit/knife_spec.rb +634 -634
  667. metadata +5 -5
@@ -1,657 +1,657 @@
1
- #
2
- # Author:: Adam Jacob (<adam@chef.io>)
3
- # Copyright:: Copyright (c) Chef Software Inc.
4
- # License:: Apache License, Version 2.0
5
- #
6
- # Licensed under the Apache License, Version 2.0 (the "License");
7
- # you may not use this file except in compliance with the License.
8
- # You may obtain a copy of the License at
9
- #
10
- # http://www.apache.org/licenses/LICENSE-2.0
11
- #
12
- # Unless required by applicable law or agreed to in writing, software
13
- # distributed under the License is distributed on an "AS IS" BASIS,
14
- # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15
- # See the License for the specific language governing permissions and
16
- # limitations under the License.
17
- #
18
-
19
- require_relative "../knife"
20
-
21
- class Chef
22
- class Knife
23
- class Ssh < Knife
24
-
25
- deps do
26
- require "chef/mixin/shell_out" unless defined?(Chef::Mixin::ShellOut)
27
- require "net/ssh" unless defined?(Net::SSH)
28
- require "net/ssh/multi"
29
- require "readline"
30
- require "chef/exceptions" unless defined?(Chef::Exceptions)
31
- require "chef/search/query" unless defined?(Chef::Search::Query)
32
- require "chef-config/path_helper" unless defined?(ChefConfig::PathHelper)
33
-
34
- include Chef::Mixin::ShellOut
35
- end
36
-
37
- attr_writer :password
38
-
39
- banner "knife ssh QUERY COMMAND (options)"
40
-
41
- option :concurrency,
42
- short: "-C NUM",
43
- long: "--concurrency NUM",
44
- description: "The number of concurrent connections.",
45
- default: nil,
46
- proc: lambda { |o| o.to_i }
47
-
48
- option :ssh_attribute,
49
- short: "-a ATTR",
50
- long: "--attribute ATTR",
51
- description: "The attribute to use for opening the connection - default depends on the context."
52
-
53
- option :manual,
54
- short: "-m",
55
- long: "--manual-list",
56
- boolean: true,
57
- description: "QUERY is a space separated list of servers.",
58
- default: false
59
-
60
- option :prefix_attribute,
61
- long: "--prefix-attribute ATTR",
62
- description: "The attribute to use for prefixing the output - default depends on the context."
63
-
64
- option :ssh_user,
65
- short: "-x USERNAME",
66
- long: "--ssh-user USERNAME",
67
- description: "The ssh username."
68
-
69
- option :ssh_password,
70
- short: "-P [PASSWORD]",
71
- long: "--ssh-password [PASSWORD]",
72
- description: "The ssh password - will prompt if flag is specified but no password is given.",
73
- # default to a value that can not be a password (boolean)
74
- # so we can effectively test if this parameter was specified
75
- # without a value
76
- default: false
77
-
78
- option :ssh_port,
79
- short: "-p PORT",
80
- long: "--ssh-port PORT",
81
- description: "The ssh port.",
82
- proc: Proc.new { |key| key.strip }
83
-
84
- option :ssh_timeout,
85
- short: "-t SECONDS",
86
- long: "--ssh-timeout SECONDS",
87
- description: "The ssh connection timeout.",
88
- proc: Proc.new { |key| key.strip.to_i },
89
- default: 120
90
-
91
- option :ssh_gateway,
92
- short: "-G GATEWAY",
93
- long: "--ssh-gateway GATEWAY",
94
- description: "The ssh gateway.",
95
- proc: Proc.new { |key| key.strip }
96
-
97
- option :ssh_gateway_identity,
98
- long: "--ssh-gateway-identity SSH_GATEWAY_IDENTITY",
99
- description: "The SSH identity file used for gateway authentication."
100
-
101
- option :forward_agent,
102
- short: "-A",
103
- long: "--forward-agent",
104
- description: "Enable SSH agent forwarding.",
105
- boolean: true
106
-
107
- option :ssh_identity_file,
108
- short: "-i IDENTITY_FILE",
109
- long: "--ssh-identity-file IDENTITY_FILE",
110
- description: "The SSH identity file used for authentication."
111
-
112
- option :host_key_verify,
113
- long: "--[no-]host-key-verify",
114
- description: "Verify host key, enabled by default.",
115
- boolean: true,
116
- default: true
117
-
118
- option :on_error,
119
- short: "-e",
120
- long: "--exit-on-error",
121
- description: "Immediately exit if an error is encountered.",
122
- boolean: true,
123
- default: false
124
-
125
- option :duplicated_fqdns,
126
- long: "--duplicated-fqdns",
127
- description: "Behavior if FQDNs are duplicated, ignored by default.",
128
- proc: Proc.new { |key| key.strip.to_sym },
129
- default: :ignore
130
-
131
- option :tmux_split,
132
- long: "--tmux-split",
133
- description: "Split tmux window.",
134
- boolean: true,
135
- default: false
136
-
137
- option :pty,
138
- long: "--[no-]pty",
139
- description: "Request a PTY, enabled by default.",
140
- boolean: true,
141
- default: true
142
-
143
- option :require_pty,
144
- long: "--[no-]require-pty",
145
- description: "Raise exception if a PTY cannot be acquired, disabled by default.",
146
- boolean: true,
147
- default: false
148
-
149
- def session
150
- ssh_error_handler = Proc.new do |server|
151
- if config[:on_error]
152
- # Net::SSH::Multi magic to force exception to be re-raised.
153
- throw :go, :raise
154
- else
155
- ui.warn "Failed to connect to #{server.host} -- #{$!.class.name}: #{$!.message}"
156
- $!.backtrace.each { |l| Chef::Log.debug(l) }
157
- end
158
- end
159
-
160
- @session ||= Net::SSH::Multi.start(concurrent_connections: config[:concurrency], on_error: ssh_error_handler)
161
- end
162
-
163
- def configure_gateway
164
- if config[:ssh_gateway]
165
- gw_host, gw_user = config[:ssh_gateway].split("@").reverse
166
- gw_host, gw_port = gw_host.split(":")
167
- gw_opts = session_options(gw_host, gw_port, gw_user, gateway: true)
168
- user = gw_opts.delete(:user)
169
-
170
- begin
171
- # Try to connect with a key.
172
- session.via(gw_host, user, gw_opts)
173
- rescue Net::SSH::AuthenticationFailed
174
- prompt = "Enter the password for #{user}@#{gw_host}: "
175
- gw_opts[:password] = prompt_for_password(prompt)
176
- # Try again with a password.
177
- session.via(gw_host, user, gw_opts)
178
- end
179
- end
180
- end
181
-
182
- def configure_session
183
- list = config[:manual] ? @name_args[0].split(" ") : search_nodes
184
- if list.length == 0
185
- if @search_count == 0
186
- ui.fatal("No nodes returned from search")
187
- else
188
- ui.fatal("#{@search_count} #{@search_count > 1 ? "nodes" : "node"} found, " +
189
- "but does not have the required attribute to establish the connection. " +
190
- "Try setting another attribute to open the connection using --attribute.")
191
- end
192
- exit 10
193
- end
194
- if %i{warn fatal}.include?(config[:duplicated_fqdns])
195
- fqdns = list.map { |v| v[0] }
196
- if fqdns.count != fqdns.uniq.count
197
- duplicated_fqdns = fqdns.uniq
198
- ui.send(config[:duplicated_fqdns],
199
- "SSH #{duplicated_fqdns.count > 1 ? "nodes are" : "node is"} " +
200
- "duplicated: #{duplicated_fqdns.join(",")}")
201
- exit 10 if config[:duplicated_fqdns] == :fatal
202
- end
203
- end
204
- session_from_list(list)
205
- end
206
-
207
- def get_prefix_attribute(item)
208
- # Order of precedence for prefix
209
- # 1) config value (cli or knife config)
210
- # 2) nil
211
- msg = "Using node attribute '%s' as the prefix: %s"
212
- if item["prefix"]
213
- Chef::Log.debug(sprintf(msg, config[:prefix_attribute], item["prefix"]))
214
- item["prefix"]
215
- else
216
- nil
217
- end
218
- end
219
-
220
- def get_ssh_attribute(item)
221
- # Order of precedence for ssh target
222
- # 1) config value (cli or knife config)
223
- # 2) cloud attribute
224
- # 3) fqdn
225
- msg = "Using node attribute '%s' as the ssh target: %s"
226
- if item["target"]
227
- Chef::Log.debug(sprintf(msg, config[:ssh_attribute], item["target"]))
228
- item["target"]
229
- elsif !item.dig("cloud", "public_hostname").to_s.empty?
230
- Chef::Log.debug(sprintf(msg, "cloud.public_hostname", item["cloud"]["public_hostname"]))
231
- item["cloud"]["public_hostname"]
232
- else
233
- Chef::Log.debug(sprintf(msg, "fqdn", item["fqdn"]))
234
- item["fqdn"]
235
- end
236
- end
237
-
238
- def search_nodes
239
- list = []
240
- query = Chef::Search::Query.new
241
- required_attributes = { fqdn: ["fqdn"], cloud: ["cloud"] }
242
-
243
- separator = ui.presenter.attribute_field_separator
244
-
245
- if config[:prefix_attribute]
246
- required_attributes[:prefix] = config[:prefix_attribute].split(separator)
247
- end
248
-
249
- if config[:ssh_attribute]
250
- required_attributes[:target] = config[:ssh_attribute].split(separator)
251
- end
252
-
253
- @search_count = 0
254
- query.search(:node, @name_args[0], filter_result: required_attributes, fuzz: true) do |item|
255
- @search_count += 1
256
- # we should skip the loop to next iteration if the item
257
- # returned by the search is nil
258
- next if item.nil?
259
-
260
- # next if we couldn't find the specified attribute in the
261
- # returned node object
262
- host = get_ssh_attribute(item)
263
- next if host.nil?
264
-
265
- prefix = get_prefix_attribute(item)
266
- ssh_port = item.dig("cloud", "public_ssh_port")
267
- srv = [host, ssh_port, prefix]
268
- list.push(srv)
269
- end
270
-
271
- list
272
- end
273
-
274
- # Net::SSH session options hash for global options. These should be
275
- # options that will apply to the gateway connection in addition to the
276
- # main one.
277
- #
278
- # @since 12.5.0
279
- # @param host [String] Hostname for this session.
280
- # @param port [String] SSH port for this session.
281
- # @param user [String] Optional username for this session.
282
- # @param gateway [Boolean] Flag: host or gateway key
283
- # @return [Hash<Symbol, Object>]
284
- def session_options(host, port, user = nil, gateway: false)
285
- ssh_config = Net::SSH.configuration_for(host, true)
286
- {}.tap do |opts|
287
- opts[:user] = user || config[:ssh_user] || ssh_config[:user]
288
- if !gateway && config[:ssh_identity_file]
289
- opts[:keys] = File.expand_path(config[:ssh_identity_file])
290
- opts[:keys_only] = true
291
- elsif gateway && config[:ssh_gateway_identity]
292
- opts[:keys] = File.expand_path(config[:ssh_gateway_identity])
293
- opts[:keys_only] = true
294
- elsif config[:ssh_password]
295
- opts[:password] = config[:ssh_password]
296
- end
297
- # Don't set the keys to nil if we don't have them.
298
- forward_agent = config[:forward_agent] || ssh_config[:forward_agent]
299
- opts[:forward_agent] = forward_agent unless forward_agent.nil?
300
- port ||= ssh_config[:port]
301
- opts[:port] = port unless port.nil?
302
- opts[:logger] = Chef::Log.with_child(subsystem: "net/ssh") if Chef::Log.level == :trace
303
- unless config[:host_key_verify]
304
- opts[:verify_host_key] = :never
305
- opts[:user_known_hosts_file] = "/dev/null"
306
- end
307
- if ssh_config[:keepalive]
308
- opts[:keepalive] = true
309
- opts[:keepalive_interval] = ssh_config[:keepalive_interval]
310
- end
311
- # maintain support for legacy key types / ciphers / key exchange algorithms.
312
- # most importantly this adds back support for DSS host keys
313
- # See https://github.com/net-ssh/net-ssh/pull/709
314
- opts[:append_all_supported_algorithms] = true
315
- end
316
- end
317
-
318
- def session_from_list(list)
319
- list.each do |item|
320
- host, ssh_port, prefix = item
321
- prefix = host unless prefix
322
- Chef::Log.debug("Adding #{host}")
323
- session_opts = session_options(host, ssh_port, gateway: false)
324
- # Handle port overrides for the main connection.
325
- session_opts[:port] = config[:ssh_port] if config[:ssh_port]
326
- # Handle connection timeout
327
- session_opts[:timeout] = config[:ssh_timeout] if config[:ssh_timeout]
328
- # Handle session prefix
329
- session_opts[:properties] = { prefix: prefix }
330
- # Create the hostspec.
331
- hostspec = session_opts[:user] ? "#{session_opts.delete(:user)}@#{host}" : host
332
- # Connect a new session on the multi.
333
- session.use(hostspec, session_opts)
334
-
335
- @longest = prefix.length if prefix.length > @longest
336
- end
337
-
338
- session
339
- end
340
-
341
- def fixup_sudo(command)
342
- command.sub(/^sudo/, "sudo -p 'knife sudo password: '")
343
- end
344
-
345
- def print_data(host, data)
346
- @buffers ||= {}
347
- if leftover = @buffers[host]
348
- @buffers[host] = nil
349
- print_data(host, leftover + data)
350
- else
351
- if newline_index = data.index("\n")
352
- line = data.slice!(0...newline_index)
353
- data.slice!(0)
354
- print_line(host, line)
355
- print_data(host, data)
356
- else
357
- @buffers[host] = data
358
- end
359
- end
360
- end
361
-
362
- def print_line(host, data)
363
- padding = @longest - host.length
364
- str = ui.color(host, :cyan) + (" " * (padding + 1)) + data
365
- ui.msg(str)
366
- end
367
-
368
- # @param command [String] the command to run
369
- # @param session_list [???] list of sessions, one per node
370
- #
371
- def ssh_command(command, session_list = session)
372
- stderr = ""
373
- exit_status = 0
374
- command = fixup_sudo(command)
375
- command.force_encoding("binary") if command.respond_to?(:force_encoding)
376
- session_list.open_channel do |chan|
377
- if config[:on_error] && exit_status != 0
378
- chan.close
379
- else
380
- if config[:pty]
381
- chan.request_pty do |ch, success|
382
- unless success
383
- ui.warn("Failed to obtain a PTY from #{ch.connection.host}")
384
- raise ArgumentError, "Request for PTY failed" if config[:require_pty]
385
- end
386
- end
387
- end
388
- chan.exec command do |ch, success|
389
- raise ArgumentError, "Cannot execute #{command}" unless success
390
-
391
- ch.on_data do |ichannel, data|
392
- print_data(ichannel.connection[:prefix], data)
393
- if /^knife sudo password: /.match?(data)
394
- print_data(ichannel.connection[:prefix], "\n")
395
- ichannel.send_data("#{get_password}\n")
396
- end
397
- end
398
- ch.on_extended_data do |_, _type, data|
399
- raise ArgumentError, "No PTY present. If a PTY is required use --require-pty" if data.eql?("sudo: no tty present and no askpass program specified\n")
400
-
401
- stderr += data
402
- end
403
- ch.on_request "exit-status" do |ichannel, data|
404
- exit_status = [exit_status, data.read_long].max
405
- end
406
- end
407
- end
408
- end
409
- session.loop
410
- exit_status
411
- ensure
412
- session_list.close
413
- end
414
-
415
- def get_password
416
- @password ||= prompt_for_password
417
- end
418
-
419
- def prompt_for_password(prompt = "Enter your password: ")
420
- ui.ask(prompt, echo: false)
421
- end
422
-
423
- # Present the prompt and read a single line from the console. It also
424
- # detects ^D and returns "exit" in that case. Adds the input to the
425
- # history, unless the input is empty. Loops repeatedly until a non-empty
426
- # line is input.
427
- def read_line
428
- loop do
429
- command = reader.readline("#{ui.color("knife-ssh>", :bold)} ", true)
430
-
431
- if command.nil?
432
- command = "exit"
433
- puts(command)
434
- else
435
- command.strip!
436
- end
437
-
438
- unless command.empty?
439
- return command
440
- end
441
- end
442
- end
443
-
444
- def reader
445
- Readline
446
- end
447
-
448
- def interactive
449
- puts "Connected to #{ui.list(session.servers_for.collect { |s| ui.color(s.host, :cyan) }, :inline, " and ")}"
450
- puts
451
- puts "To run a command on a list of servers, do:"
452
- puts " on SERVER1 SERVER2 SERVER3; COMMAND"
453
- puts " Example: on latte foamy; echo foobar"
454
- puts
455
- puts "To exit interactive mode, use 'quit!'"
456
- puts
457
- loop do
458
- command = read_line
459
- case command
460
- when "quit!"
461
- puts "Bye!"
462
- break
463
- when /^on (.+?); (.+)$/
464
- raw_list = $1.split(" ")
465
- server_list = []
466
- session.servers.each do |session_server|
467
- server_list << session_server if raw_list.include?(session_server.host)
468
- end
469
- command = $2
470
- ssh_command(command, session.on(*server_list))
471
- else
472
- ssh_command(command)
473
- end
474
- end
475
- end
476
-
477
- def screen
478
- tf = Tempfile.new("knife-ssh-screen")
479
- ChefConfig::PathHelper.home(".screenrc") do |screenrc_path|
480
- if File.exist? screenrc_path
481
- tf.puts("source #{screenrc_path}")
482
- end
483
- end
484
- tf.puts("caption always '%-Lw%{= BW}%50>%n%f* %t%{-}%+Lw%<'")
485
- tf.puts("hardstatus alwayslastline 'knife ssh #{@name_args[0]}'")
486
- window = 0
487
- session.servers_for.each do |server|
488
- tf.print("screen -t \"#{server.host}\" #{window} ssh ")
489
- tf.print("-i #{config[:ssh_identity_file]} ") if config[:ssh_identity_file]
490
- server.user ? tf.puts("#{server.user}@#{server.host}") : tf.puts(server.host)
491
- window += 1
492
- end
493
- tf.close
494
- exec("screen -c #{tf.path}")
495
- end
496
-
497
- def tmux
498
- ssh_dest = lambda do |server|
499
- identity = "-i #{config[:ssh_identity_file]} " if config[:ssh_identity_file]
500
- prefix = server.user ? "#{server.user}@" : ""
501
- "'ssh #{identity}#{prefix}#{server.host}'"
502
- end
503
-
504
- new_window_cmds = lambda do
505
- if session.servers_for.size > 1
506
- [""] + session.servers_for[1..].map do |server|
507
- if config[:tmux_split]
508
- "split-window #{ssh_dest.call(server)}; tmux select-layout tiled"
509
- else
510
- "new-window -a -n '#{server.host}' #{ssh_dest.call(server)}"
511
- end
512
- end
513
- else
514
- []
515
- end.join(" \\; ")
516
- end
517
-
518
- tmux_name = "'knife ssh #{@name_args[0].tr(":.", "=-")}'"
519
- begin
520
- server = session.servers_for.first
521
- cmd = ["tmux new-session -d -s #{tmux_name}",
522
- "-n '#{server.host}'", ssh_dest.call(server),
523
- new_window_cmds.call].join(" ")
524
- shell_out!(cmd)
525
- exec("tmux attach-session -t #{tmux_name}")
526
- rescue Chef::Exceptions::Exec
527
- end
528
- end
529
-
530
- def macterm
531
- begin
532
- require "appscript" unless defined?(Appscript)
533
- rescue LoadError
534
- STDERR.puts "You need the rb-appscript gem to use knife ssh macterm. `(sudo) gem install rb-appscript` to install"
535
- raise
536
- end
537
-
538
- Appscript.app("/Applications/Utilities/Terminal.app").windows.first.activate
539
- Appscript.app("System Events").application_processes["Terminal.app"].keystroke("n", using: :command_down)
540
- term = Appscript.app("Terminal")
541
- window = term.windows.first.get
542
-
543
- (session.servers_for.size - 1).times do |i|
544
- window.activate
545
- Appscript.app("System Events").application_processes["Terminal.app"].keystroke("t", using: :command_down)
546
- end
547
-
548
- session.servers_for.each_with_index do |server, tab_number|
549
- cmd = "unset PROMPT_COMMAND; echo -e \"\\033]0;#{server.host}\\007\"; ssh #{server.user ? "#{server.user}@#{server.host}" : server.host}"
550
- Appscript.app("Terminal").do_script(cmd, in: window.tabs[tab_number + 1].get)
551
- end
552
- end
553
-
554
- def cssh
555
- cssh_cmd = nil
556
- %w{csshX cssh}.each do |cmd|
557
-
558
- # Unix and Mac only
559
- cssh_cmd = shell_out!("which #{cmd}").stdout.strip
560
- break
561
- rescue Mixlib::ShellOut::ShellCommandFailed
562
-
563
- end
564
- raise Chef::Exceptions::Exec, "no command found for cssh" unless cssh_cmd
565
-
566
- # pass in the consolidated identity file option to cssh(X)
567
- if config[:ssh_identity_file]
568
- cssh_cmd << " --ssh_args '-i #{File.expand_path(config[:ssh_identity_file])}'"
569
- end
570
-
571
- session.servers_for.each do |server|
572
- cssh_cmd << " #{server.user ? "#{server.user}@#{server.host}" : server.host}"
573
- end
574
- Chef::Log.debug("Starting cssh session with command: #{cssh_cmd}")
575
- exec(cssh_cmd)
576
- end
577
-
578
- def get_stripped_unfrozen_value(value)
579
- return nil unless value
580
-
581
- value.strip
582
- end
583
-
584
- def configure_user
585
- config[:ssh_user] = get_stripped_unfrozen_value(config[:ssh_user] ||
586
- Chef::Config[:knife][:ssh_user])
587
- end
588
-
589
- def configure_password
590
- if config.key?(:ssh_password) && config[:ssh_password].nil?
591
- # if we have an actual nil that means someone called "--ssh-password" with no value, so we prompt for a password
592
- config[:ssh_password] = get_password
593
- else
594
- # the false default of ssh_password results in a nil here
595
- config[:ssh_password] = get_stripped_unfrozen_value(config[:ssh_password])
596
- end
597
- end
598
-
599
- def configure_ssh_identity_file
600
- config[:ssh_identity_file] = get_stripped_unfrozen_value(config[:ssh_identity_file])
601
- end
602
-
603
- def configure_ssh_gateway_identity
604
- config[:ssh_gateway_identity] = get_stripped_unfrozen_value(config[:ssh_gateway_identity])
605
- end
606
-
607
- def run
608
- @longest = 0
609
-
610
- if @name_args.length < 1
611
- show_usage
612
- ui.fatal("You must specify the SEARCH QUERY.")
613
- exit(1)
614
- end
615
-
616
- configure_user
617
- configure_password
618
- @password = config[:ssh_password] if config[:ssh_password]
619
-
620
- # If a password was not given, check for SSH identity file.
621
- unless @password
622
- configure_ssh_identity_file
623
- configure_ssh_gateway_identity
624
- end
625
-
626
- configure_gateway
627
- configure_session
628
-
629
- exit_status =
630
- case @name_args[1]
631
- when "interactive"
632
- interactive
633
- when "screen"
634
- screen
635
- when "tmux"
636
- tmux
637
- when "macterm"
638
- macterm
639
- when "cssh"
640
- cssh
641
- else
642
- ssh_command(@name_args[1..].join(" "))
643
- end
644
-
645
- session.close
646
- if exit_status && exit_status != 0
647
- exit exit_status
648
- else
649
- exit_status
650
- end
651
- end
652
-
653
- private :search_nodes
654
-
655
- end
656
- end
657
- end
1
+ #
2
+ # Author:: Adam Jacob (<adam@chef.io>)
3
+ # Copyright:: Copyright (c) Chef Software Inc.
4
+ # License:: Apache License, Version 2.0
5
+ #
6
+ # Licensed under the Apache License, Version 2.0 (the "License");
7
+ # you may not use this file except in compliance with the License.
8
+ # You may obtain a copy of the License at
9
+ #
10
+ # http://www.apache.org/licenses/LICENSE-2.0
11
+ #
12
+ # Unless required by applicable law or agreed to in writing, software
13
+ # distributed under the License is distributed on an "AS IS" BASIS,
14
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15
+ # See the License for the specific language governing permissions and
16
+ # limitations under the License.
17
+ #
18
+
19
+ require_relative "../knife"
20
+
21
+ class Chef
22
+ class Knife
23
+ class Ssh < Knife
24
+
25
+ deps do
26
+ require "chef/mixin/shell_out" unless defined?(Chef::Mixin::ShellOut)
27
+ require "net/ssh" unless defined?(Net::SSH)
28
+ require "net/ssh/multi"
29
+ require "readline"
30
+ require "chef/exceptions" unless defined?(Chef::Exceptions)
31
+ require "chef/search/query" unless defined?(Chef::Search::Query)
32
+ require "chef-config/path_helper" unless defined?(ChefConfig::PathHelper)
33
+
34
+ include Chef::Mixin::ShellOut
35
+ end
36
+
37
+ attr_writer :password
38
+
39
+ banner "knife ssh QUERY COMMAND (options)"
40
+
41
+ option :concurrency,
42
+ short: "-C NUM",
43
+ long: "--concurrency NUM",
44
+ description: "The number of concurrent connections.",
45
+ default: nil,
46
+ proc: lambda { |o| o.to_i }
47
+
48
+ option :ssh_attribute,
49
+ short: "-a ATTR",
50
+ long: "--attribute ATTR",
51
+ description: "The attribute to use for opening the connection - default depends on the context."
52
+
53
+ option :manual,
54
+ short: "-m",
55
+ long: "--manual-list",
56
+ boolean: true,
57
+ description: "QUERY is a space separated list of servers.",
58
+ default: false
59
+
60
+ option :prefix_attribute,
61
+ long: "--prefix-attribute ATTR",
62
+ description: "The attribute to use for prefixing the output - default depends on the context."
63
+
64
+ option :ssh_user,
65
+ short: "-x USERNAME",
66
+ long: "--ssh-user USERNAME",
67
+ description: "The ssh username."
68
+
69
+ option :ssh_password,
70
+ short: "-P [PASSWORD]",
71
+ long: "--ssh-password [PASSWORD]",
72
+ description: "The ssh password - will prompt if flag is specified but no password is given.",
73
+ # default to a value that can not be a password (boolean)
74
+ # so we can effectively test if this parameter was specified
75
+ # without a value
76
+ default: false
77
+
78
+ option :ssh_port,
79
+ short: "-p PORT",
80
+ long: "--ssh-port PORT",
81
+ description: "The ssh port.",
82
+ proc: Proc.new { |key| key.strip }
83
+
84
+ option :ssh_timeout,
85
+ short: "-t SECONDS",
86
+ long: "--ssh-timeout SECONDS",
87
+ description: "The ssh connection timeout.",
88
+ proc: Proc.new { |key| key.strip.to_i },
89
+ default: 120
90
+
91
+ option :ssh_gateway,
92
+ short: "-G GATEWAY",
93
+ long: "--ssh-gateway GATEWAY",
94
+ description: "The ssh gateway.",
95
+ proc: Proc.new { |key| key.strip }
96
+
97
+ option :ssh_gateway_identity,
98
+ long: "--ssh-gateway-identity SSH_GATEWAY_IDENTITY",
99
+ description: "The SSH identity file used for gateway authentication."
100
+
101
+ option :forward_agent,
102
+ short: "-A",
103
+ long: "--forward-agent",
104
+ description: "Enable SSH agent forwarding.",
105
+ boolean: true
106
+
107
+ option :ssh_identity_file,
108
+ short: "-i IDENTITY_FILE",
109
+ long: "--ssh-identity-file IDENTITY_FILE",
110
+ description: "The SSH identity file used for authentication."
111
+
112
+ option :host_key_verify,
113
+ long: "--[no-]host-key-verify",
114
+ description: "Verify host key, enabled by default.",
115
+ boolean: true,
116
+ default: true
117
+
118
+ option :on_error,
119
+ short: "-e",
120
+ long: "--exit-on-error",
121
+ description: "Immediately exit if an error is encountered.",
122
+ boolean: true,
123
+ default: false
124
+
125
+ option :duplicated_fqdns,
126
+ long: "--duplicated-fqdns",
127
+ description: "Behavior if FQDNs are duplicated, ignored by default.",
128
+ proc: Proc.new { |key| key.strip.to_sym },
129
+ default: :ignore
130
+
131
+ option :tmux_split,
132
+ long: "--tmux-split",
133
+ description: "Split tmux window.",
134
+ boolean: true,
135
+ default: false
136
+
137
+ option :pty,
138
+ long: "--[no-]pty",
139
+ description: "Request a PTY, enabled by default.",
140
+ boolean: true,
141
+ default: true
142
+
143
+ option :require_pty,
144
+ long: "--[no-]require-pty",
145
+ description: "Raise exception if a PTY cannot be acquired, disabled by default.",
146
+ boolean: true,
147
+ default: false
148
+
149
+ def session
150
+ ssh_error_handler = Proc.new do |server|
151
+ if config[:on_error]
152
+ # Net::SSH::Multi magic to force exception to be re-raised.
153
+ throw :go, :raise
154
+ else
155
+ ui.warn "Failed to connect to #{server.host} -- #{$!.class.name}: #{$!.message}"
156
+ $!.backtrace.each { |l| Chef::Log.debug(l) }
157
+ end
158
+ end
159
+
160
+ @session ||= Net::SSH::Multi.start(concurrent_connections: config[:concurrency], on_error: ssh_error_handler)
161
+ end
162
+
163
+ def configure_gateway
164
+ if config[:ssh_gateway]
165
+ gw_host, gw_user = config[:ssh_gateway].split("@").reverse
166
+ gw_host, gw_port = gw_host.split(":")
167
+ gw_opts = session_options(gw_host, gw_port, gw_user, gateway: true)
168
+ user = gw_opts.delete(:user)
169
+
170
+ begin
171
+ # Try to connect with a key.
172
+ session.via(gw_host, user, gw_opts)
173
+ rescue Net::SSH::AuthenticationFailed
174
+ prompt = "Enter the password for #{user}@#{gw_host}: "
175
+ gw_opts[:password] = prompt_for_password(prompt)
176
+ # Try again with a password.
177
+ session.via(gw_host, user, gw_opts)
178
+ end
179
+ end
180
+ end
181
+
182
+ def configure_session
183
+ list = config[:manual] ? @name_args[0].split(" ") : search_nodes
184
+ if list.length == 0
185
+ if @search_count == 0
186
+ ui.fatal("No nodes returned from search")
187
+ else
188
+ ui.fatal("#{@search_count} #{@search_count > 1 ? "nodes" : "node"} found, " +
189
+ "but does not have the required attribute to establish the connection. " +
190
+ "Try setting another attribute to open the connection using --attribute.")
191
+ end
192
+ exit 10
193
+ end
194
+ if %i{warn fatal}.include?(config[:duplicated_fqdns])
195
+ fqdns = list.map { |v| v[0] }
196
+ if fqdns.count != fqdns.uniq.count
197
+ duplicated_fqdns = fqdns.uniq
198
+ ui.send(config[:duplicated_fqdns],
199
+ "SSH #{duplicated_fqdns.count > 1 ? "nodes are" : "node is"} " +
200
+ "duplicated: #{duplicated_fqdns.join(",")}")
201
+ exit 10 if config[:duplicated_fqdns] == :fatal
202
+ end
203
+ end
204
+ session_from_list(list)
205
+ end
206
+
207
+ def get_prefix_attribute(item)
208
+ # Order of precedence for prefix
209
+ # 1) config value (cli or knife config)
210
+ # 2) nil
211
+ msg = "Using node attribute '%s' as the prefix: %s"
212
+ if item["prefix"]
213
+ Chef::Log.debug(sprintf(msg, config[:prefix_attribute], item["prefix"]))
214
+ item["prefix"]
215
+ else
216
+ nil
217
+ end
218
+ end
219
+
220
+ def get_ssh_attribute(item)
221
+ # Order of precedence for ssh target
222
+ # 1) config value (cli or knife config)
223
+ # 2) cloud attribute
224
+ # 3) fqdn
225
+ msg = "Using node attribute '%s' as the ssh target: %s"
226
+ if item["target"]
227
+ Chef::Log.debug(sprintf(msg, config[:ssh_attribute], item["target"]))
228
+ item["target"]
229
+ elsif !item.dig("cloud", "public_hostname").to_s.empty?
230
+ Chef::Log.debug(sprintf(msg, "cloud.public_hostname", item["cloud"]["public_hostname"]))
231
+ item["cloud"]["public_hostname"]
232
+ else
233
+ Chef::Log.debug(sprintf(msg, "fqdn", item["fqdn"]))
234
+ item["fqdn"]
235
+ end
236
+ end
237
+
238
+ def search_nodes
239
+ list = []
240
+ query = Chef::Search::Query.new
241
+ required_attributes = { fqdn: ["fqdn"], cloud: ["cloud"] }
242
+
243
+ separator = ui.presenter.attribute_field_separator
244
+
245
+ if config[:prefix_attribute]
246
+ required_attributes[:prefix] = config[:prefix_attribute].split(separator)
247
+ end
248
+
249
+ if config[:ssh_attribute]
250
+ required_attributes[:target] = config[:ssh_attribute].split(separator)
251
+ end
252
+
253
+ @search_count = 0
254
+ query.search(:node, @name_args[0], filter_result: required_attributes, fuzz: true) do |item|
255
+ @search_count += 1
256
+ # we should skip the loop to next iteration if the item
257
+ # returned by the search is nil
258
+ next if item.nil?
259
+
260
+ # next if we couldn't find the specified attribute in the
261
+ # returned node object
262
+ host = get_ssh_attribute(item)
263
+ next if host.nil?
264
+
265
+ prefix = get_prefix_attribute(item)
266
+ ssh_port = item.dig("cloud", "public_ssh_port")
267
+ srv = [host, ssh_port, prefix]
268
+ list.push(srv)
269
+ end
270
+
271
+ list
272
+ end
273
+
274
+ # Net::SSH session options hash for global options. These should be
275
+ # options that will apply to the gateway connection in addition to the
276
+ # main one.
277
+ #
278
+ # @since 12.5.0
279
+ # @param host [String] Hostname for this session.
280
+ # @param port [String] SSH port for this session.
281
+ # @param user [String] Optional username for this session.
282
+ # @param gateway [Boolean] Flag: host or gateway key
283
+ # @return [Hash<Symbol, Object>]
284
+ def session_options(host, port, user = nil, gateway: false)
285
+ ssh_config = Net::SSH.configuration_for(host, true)
286
+ {}.tap do |opts|
287
+ opts[:user] = user || config[:ssh_user] || ssh_config[:user]
288
+ if !gateway && config[:ssh_identity_file]
289
+ opts[:keys] = File.expand_path(config[:ssh_identity_file])
290
+ opts[:keys_only] = true
291
+ elsif gateway && config[:ssh_gateway_identity]
292
+ opts[:keys] = File.expand_path(config[:ssh_gateway_identity])
293
+ opts[:keys_only] = true
294
+ elsif config[:ssh_password]
295
+ opts[:password] = config[:ssh_password]
296
+ end
297
+ # Don't set the keys to nil if we don't have them.
298
+ forward_agent = config[:forward_agent] || ssh_config[:forward_agent]
299
+ opts[:forward_agent] = forward_agent unless forward_agent.nil?
300
+ port ||= ssh_config[:port]
301
+ opts[:port] = port unless port.nil?
302
+ opts[:logger] = Chef::Log.with_child(subsystem: "net/ssh") if Chef::Log.level == :trace
303
+ unless config[:host_key_verify]
304
+ opts[:verify_host_key] = :never
305
+ opts[:user_known_hosts_file] = "/dev/null"
306
+ end
307
+ if ssh_config[:keepalive]
308
+ opts[:keepalive] = true
309
+ opts[:keepalive_interval] = ssh_config[:keepalive_interval]
310
+ end
311
+ # maintain support for legacy key types / ciphers / key exchange algorithms.
312
+ # most importantly this adds back support for DSS host keys
313
+ # See https://github.com/net-ssh/net-ssh/pull/709
314
+ opts[:append_all_supported_algorithms] = true
315
+ end
316
+ end
317
+
318
+ def session_from_list(list)
319
+ list.each do |item|
320
+ host, ssh_port, prefix = item
321
+ prefix = host unless prefix
322
+ Chef::Log.debug("Adding #{host}")
323
+ session_opts = session_options(host, ssh_port, gateway: false)
324
+ # Handle port overrides for the main connection.
325
+ session_opts[:port] = config[:ssh_port] if config[:ssh_port]
326
+ # Handle connection timeout
327
+ session_opts[:timeout] = config[:ssh_timeout] if config[:ssh_timeout]
328
+ # Handle session prefix
329
+ session_opts[:properties] = { prefix: prefix }
330
+ # Create the hostspec.
331
+ hostspec = session_opts[:user] ? "#{session_opts.delete(:user)}@#{host}" : host
332
+ # Connect a new session on the multi.
333
+ session.use(hostspec, session_opts)
334
+
335
+ @longest = prefix.length if prefix.length > @longest
336
+ end
337
+
338
+ session
339
+ end
340
+
341
+ def fixup_sudo(command)
342
+ command.sub(/^sudo/, "sudo -p 'knife sudo password: '")
343
+ end
344
+
345
+ def print_data(host, data)
346
+ @buffers ||= {}
347
+ if leftover = @buffers[host]
348
+ @buffers[host] = nil
349
+ print_data(host, leftover + data)
350
+ else
351
+ if newline_index = data.index("\n")
352
+ line = data.slice!(0...newline_index)
353
+ data.slice!(0)
354
+ print_line(host, line)
355
+ print_data(host, data)
356
+ else
357
+ @buffers[host] = data
358
+ end
359
+ end
360
+ end
361
+
362
+ def print_line(host, data)
363
+ padding = @longest - host.length
364
+ str = ui.color(host, :cyan) + (" " * (padding + 1)) + data
365
+ ui.msg(str)
366
+ end
367
+
368
+ # @param command [String] the command to run
369
+ # @param session_list [???] list of sessions, one per node
370
+ #
371
+ def ssh_command(command, session_list = session)
372
+ stderr = ""
373
+ exit_status = 0
374
+ command = fixup_sudo(command)
375
+ command.force_encoding("binary") if command.respond_to?(:force_encoding)
376
+ session_list.open_channel do |chan|
377
+ if config[:on_error] && exit_status != 0
378
+ chan.close
379
+ else
380
+ if config[:pty]
381
+ chan.request_pty do |ch, success|
382
+ unless success
383
+ ui.warn("Failed to obtain a PTY from #{ch.connection.host}")
384
+ raise ArgumentError, "Request for PTY failed" if config[:require_pty]
385
+ end
386
+ end
387
+ end
388
+ chan.exec command do |ch, success|
389
+ raise ArgumentError, "Cannot execute #{command}" unless success
390
+
391
+ ch.on_data do |ichannel, data|
392
+ print_data(ichannel.connection[:prefix], data)
393
+ if /^knife sudo password: /.match?(data)
394
+ print_data(ichannel.connection[:prefix], "\n")
395
+ ichannel.send_data("#{get_password}\n")
396
+ end
397
+ end
398
+ ch.on_extended_data do |_, _type, data|
399
+ raise ArgumentError, "No PTY present. If a PTY is required use --require-pty" if data.eql?("sudo: no tty present and no askpass program specified\n")
400
+
401
+ stderr += data
402
+ end
403
+ ch.on_request "exit-status" do |ichannel, data|
404
+ exit_status = [exit_status, data.read_long].max
405
+ end
406
+ end
407
+ end
408
+ end
409
+ session.loop
410
+ exit_status
411
+ ensure
412
+ session_list.close
413
+ end
414
+
415
+ def get_password
416
+ @password ||= prompt_for_password
417
+ end
418
+
419
+ def prompt_for_password(prompt = "Enter your password: ")
420
+ ui.ask(prompt, echo: false)
421
+ end
422
+
423
+ # Present the prompt and read a single line from the console. It also
424
+ # detects ^D and returns "exit" in that case. Adds the input to the
425
+ # history, unless the input is empty. Loops repeatedly until a non-empty
426
+ # line is input.
427
+ def read_line
428
+ loop do
429
+ command = reader.readline("#{ui.color("knife-ssh>", :bold)} ", true)
430
+
431
+ if command.nil?
432
+ command = "exit"
433
+ puts(command)
434
+ else
435
+ command.strip!
436
+ end
437
+
438
+ unless command.empty?
439
+ return command
440
+ end
441
+ end
442
+ end
443
+
444
+ def reader
445
+ Readline
446
+ end
447
+
448
+ def interactive
449
+ puts "Connected to #{ui.list(session.servers_for.collect { |s| ui.color(s.host, :cyan) }, :inline, " and ")}"
450
+ puts
451
+ puts "To run a command on a list of servers, do:"
452
+ puts " on SERVER1 SERVER2 SERVER3; COMMAND"
453
+ puts " Example: on latte foamy; echo foobar"
454
+ puts
455
+ puts "To exit interactive mode, use 'quit!'"
456
+ puts
457
+ loop do
458
+ command = read_line
459
+ case command
460
+ when "quit!"
461
+ puts "Bye!"
462
+ break
463
+ when /^on (.+?); (.+)$/
464
+ raw_list = $1.split(" ")
465
+ server_list = []
466
+ session.servers.each do |session_server|
467
+ server_list << session_server if raw_list.include?(session_server.host)
468
+ end
469
+ command = $2
470
+ ssh_command(command, session.on(*server_list))
471
+ else
472
+ ssh_command(command)
473
+ end
474
+ end
475
+ end
476
+
477
+ def screen
478
+ tf = Tempfile.new("knife-ssh-screen")
479
+ ChefConfig::PathHelper.home(".screenrc") do |screenrc_path|
480
+ if File.exist? screenrc_path
481
+ tf.puts("source #{screenrc_path}")
482
+ end
483
+ end
484
+ tf.puts("caption always '%-Lw%{= BW}%50>%n%f* %t%{-}%+Lw%<'")
485
+ tf.puts("hardstatus alwayslastline 'knife ssh #{@name_args[0]}'")
486
+ window = 0
487
+ session.servers_for.each do |server|
488
+ tf.print("screen -t \"#{server.host}\" #{window} ssh ")
489
+ tf.print("-i #{config[:ssh_identity_file]} ") if config[:ssh_identity_file]
490
+ server.user ? tf.puts("#{server.user}@#{server.host}") : tf.puts(server.host)
491
+ window += 1
492
+ end
493
+ tf.close
494
+ exec("screen -c #{tf.path}")
495
+ end
496
+
497
+ def tmux
498
+ ssh_dest = lambda do |server|
499
+ identity = "-i #{config[:ssh_identity_file]} " if config[:ssh_identity_file]
500
+ prefix = server.user ? "#{server.user}@" : ""
501
+ "'ssh #{identity}#{prefix}#{server.host}'"
502
+ end
503
+
504
+ new_window_cmds = lambda do
505
+ if session.servers_for.size > 1
506
+ [""] + session.servers_for[1..].map do |server|
507
+ if config[:tmux_split]
508
+ "split-window #{ssh_dest.call(server)}; tmux select-layout tiled"
509
+ else
510
+ "new-window -a -n '#{server.host}' #{ssh_dest.call(server)}"
511
+ end
512
+ end
513
+ else
514
+ []
515
+ end.join(" \\; ")
516
+ end
517
+
518
+ tmux_name = "'knife ssh #{@name_args[0].tr(":.", "=-")}'"
519
+ begin
520
+ server = session.servers_for.first
521
+ cmd = ["tmux new-session -d -s #{tmux_name}",
522
+ "-n '#{server.host}'", ssh_dest.call(server),
523
+ new_window_cmds.call].join(" ")
524
+ shell_out!(cmd)
525
+ exec("tmux attach-session -t #{tmux_name}")
526
+ rescue Chef::Exceptions::Exec
527
+ end
528
+ end
529
+
530
+ def macterm
531
+ begin
532
+ require "appscript" unless defined?(Appscript)
533
+ rescue LoadError
534
+ STDERR.puts "You need the rb-appscript gem to use knife ssh macterm. `(sudo) gem install rb-appscript` to install"
535
+ raise
536
+ end
537
+
538
+ Appscript.app("/Applications/Utilities/Terminal.app").windows.first.activate
539
+ Appscript.app("System Events").application_processes["Terminal.app"].keystroke("n", using: :command_down)
540
+ term = Appscript.app("Terminal")
541
+ window = term.windows.first.get
542
+
543
+ (session.servers_for.size - 1).times do |i|
544
+ window.activate
545
+ Appscript.app("System Events").application_processes["Terminal.app"].keystroke("t", using: :command_down)
546
+ end
547
+
548
+ session.servers_for.each_with_index do |server, tab_number|
549
+ cmd = "unset PROMPT_COMMAND; echo -e \"\\033]0;#{server.host}\\007\"; ssh #{server.user ? "#{server.user}@#{server.host}" : server.host}"
550
+ Appscript.app("Terminal").do_script(cmd, in: window.tabs[tab_number + 1].get)
551
+ end
552
+ end
553
+
554
+ def cssh
555
+ cssh_cmd = nil
556
+ %w{csshX cssh}.each do |cmd|
557
+
558
+ # Unix and Mac only
559
+ cssh_cmd = shell_out!("which #{cmd}").stdout.strip
560
+ break
561
+ rescue Mixlib::ShellOut::ShellCommandFailed
562
+
563
+ end
564
+ raise Chef::Exceptions::Exec, "no command found for cssh" unless cssh_cmd
565
+
566
+ # pass in the consolidated identity file option to cssh(X)
567
+ if config[:ssh_identity_file]
568
+ cssh_cmd << " --ssh_args '-i #{File.expand_path(config[:ssh_identity_file])}'"
569
+ end
570
+
571
+ session.servers_for.each do |server|
572
+ cssh_cmd << " #{server.user ? "#{server.user}@#{server.host}" : server.host}"
573
+ end
574
+ Chef::Log.debug("Starting cssh session with command: #{cssh_cmd}")
575
+ exec(cssh_cmd)
576
+ end
577
+
578
+ def get_stripped_unfrozen_value(value)
579
+ return nil unless value
580
+
581
+ value.strip
582
+ end
583
+
584
+ def configure_user
585
+ config[:ssh_user] = get_stripped_unfrozen_value(config[:ssh_user] ||
586
+ Chef::Config[:knife][:ssh_user])
587
+ end
588
+
589
+ def configure_password
590
+ if config.key?(:ssh_password) && config[:ssh_password].nil?
591
+ # if we have an actual nil that means someone called "--ssh-password" with no value, so we prompt for a password
592
+ config[:ssh_password] = get_password
593
+ else
594
+ # the false default of ssh_password results in a nil here
595
+ config[:ssh_password] = get_stripped_unfrozen_value(config[:ssh_password])
596
+ end
597
+ end
598
+
599
+ def configure_ssh_identity_file
600
+ config[:ssh_identity_file] = get_stripped_unfrozen_value(config[:ssh_identity_file])
601
+ end
602
+
603
+ def configure_ssh_gateway_identity
604
+ config[:ssh_gateway_identity] = get_stripped_unfrozen_value(config[:ssh_gateway_identity])
605
+ end
606
+
607
+ def run
608
+ @longest = 0
609
+
610
+ if @name_args.length < 1
611
+ show_usage
612
+ ui.fatal("You must specify the SEARCH QUERY.")
613
+ exit(1)
614
+ end
615
+
616
+ configure_user
617
+ configure_password
618
+ @password = config[:ssh_password] if config[:ssh_password]
619
+
620
+ # If a password was not given, check for SSH identity file.
621
+ unless @password
622
+ configure_ssh_identity_file
623
+ configure_ssh_gateway_identity
624
+ end
625
+
626
+ configure_gateway
627
+ configure_session
628
+
629
+ exit_status =
630
+ case @name_args[1]
631
+ when "interactive"
632
+ interactive
633
+ when "screen"
634
+ screen
635
+ when "tmux"
636
+ tmux
637
+ when "macterm"
638
+ macterm
639
+ when "cssh"
640
+ cssh
641
+ else
642
+ ssh_command(@name_args[1..].join(" "))
643
+ end
644
+
645
+ session.close
646
+ if exit_status && exit_status != 0
647
+ exit exit_status
648
+ else
649
+ exit_status
650
+ end
651
+ end
652
+
653
+ private :search_nodes
654
+
655
+ end
656
+ end
657
+ end