knife 18.0.185 → 18.1.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (667) hide show
  1. checksums.yaml +4 -4
  2. data/Gemfile +29 -29
  3. data/LICENSE +201 -201
  4. data/Rakefile +33 -33
  5. data/bin/knife +24 -24
  6. data/knife.gemspec +60 -60
  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 +232 -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 -84
  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 -86
  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 +2233 -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 +3 -3
@@ -1,1616 +1,1616 @@
1
- #
2
- # Author:: John Keiser (<jkeiser@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
- require "knife_spec_helper"
19
- require "support/shared/integration/integration_helper"
20
- require "chef/knife/upload"
21
- require "chef/knife/diff"
22
- require "chef/knife/raw"
23
- require "chef/json_compat"
24
-
25
- describe "knife upload", :workstation do
26
- include IntegrationSupport
27
- include KnifeSupport
28
-
29
- context "without versioned cookbooks" do
30
-
31
- when_the_chef_server "has one of each thing" do
32
-
33
- before do
34
- client "x", {}
35
- cookbook "x", "1.0.0"
36
- data_bag "x", { "y" => {} }
37
- environment "x", {}
38
- node "x", {}
39
- role "x", {}
40
- user "x", {}
41
- end
42
-
43
- when_the_repository "has only top-level directories" do
44
- before do
45
- directory "clients"
46
- directory "cookbooks"
47
- directory "data_bags"
48
- directory "environments"
49
- directory "nodes"
50
- directory "roles"
51
- directory "users"
52
- end
53
-
54
- it "knife upload does nothing" do
55
- knife("upload /").should_succeed ""
56
- knife("diff --name-status /").should_succeed <<~EOM
57
- D\t/clients/chef-validator.json
58
- D\t/clients/chef-webui.json
59
- D\t/clients/x.json
60
- D\t/cookbooks/x
61
- D\t/data_bags/x
62
- D\t/environments/_default.json
63
- D\t/environments/x.json
64
- D\t/nodes/x.json
65
- D\t/roles/x.json
66
- D\t/users/admin.json
67
- D\t/users/x.json
68
- EOM
69
- end
70
-
71
- it "knife upload --purge deletes everything" do
72
- knife("upload --purge /").should_succeed(<<~EOM, stderr: "WARNING: /environments/_default.json cannot be deleted (default environment cannot be modified).\n")
73
- Deleted extra entry /clients/chef-validator.json (purge is on)
74
- Deleted extra entry /clients/chef-webui.json (purge is on)
75
- Deleted extra entry /clients/x.json (purge is on)
76
- Deleted extra entry /cookbooks/x (purge is on)
77
- Deleted extra entry /data_bags/x (purge is on)
78
- Deleted extra entry /environments/x.json (purge is on)
79
- Deleted extra entry /nodes/x.json (purge is on)
80
- Deleted extra entry /roles/x.json (purge is on)
81
- Deleted extra entry /users/admin.json (purge is on)
82
- Deleted extra entry /users/x.json (purge is on)
83
- EOM
84
- knife("diff --name-status /").should_succeed <<~EOM
85
- D\t/environments/_default.json
86
- EOM
87
- end
88
- end
89
-
90
- when_the_repository "has an identical copy of each thing" do
91
-
92
- before do
93
- file "clients/chef-validator.json", { "validator" => true, "public_key" => ChefZero::PUBLIC_KEY }
94
- file "clients/chef-webui.json", { "admin" => true, "public_key" => ChefZero::PUBLIC_KEY }
95
- file "clients/x.json", { "public_key" => ChefZero::PUBLIC_KEY }
96
- file "cookbooks/x/metadata.rb", cb_metadata("x", "1.0.0")
97
- file "data_bags/x/y.json", {}
98
- file "environments/_default.json", { "description" => "The default Chef environment" }
99
- file "environments/x.json", {}
100
- file "nodes/x.json", { "normal" => { "tags" => [] } }
101
- file "roles/x.json", {}
102
- file "users/admin.json", { "admin" => true, "public_key" => ChefZero::PUBLIC_KEY }
103
- file "users/x.json", { "public_key" => ChefZero::PUBLIC_KEY }
104
- end
105
-
106
- it "knife upload makes no changes" do
107
- knife("upload /cookbooks/x").should_succeed ""
108
- knife("diff --name-status /").should_succeed ""
109
- end
110
-
111
- it "knife upload --purge makes no changes" do
112
- knife("upload --purge /").should_succeed ""
113
- knife("diff --name-status /").should_succeed ""
114
- end
115
-
116
- context "except the role file" do
117
- before do
118
- file "roles/x.json", { "description" => "blarghle" }
119
- end
120
-
121
- it "knife upload changes the role" do
122
- knife("upload /").should_succeed "Updated /roles/x.json\n"
123
- knife("diff --name-status /").should_succeed ""
124
- end
125
- it "knife upload --no-diff does not change the role" do
126
- knife("upload --no-diff /").should_succeed ""
127
- knife("diff --name-status /").should_succeed "M\t/roles/x.json\n"
128
- end
129
- end
130
-
131
- context "except the role file is textually different, but not ACTUALLY different" do
132
- before do
133
- file "roles/x.json", <<~EOM
134
- {
135
- "chef_type": "role",
136
- "default_attributes": {
137
- },
138
- "env_run_lists": {
139
- },
140
- "json_class": "Chef::Role",
141
- "name": "x",
142
- "description": "",
143
- "override_attributes": {
144
- },
145
- "run_list": [
146
-
147
- ]
148
- }
149
- EOM
150
- end
151
-
152
- it "knife upload / does not change anything" do
153
- knife("upload /").should_succeed ""
154
- knife("diff --name-status /").should_succeed ""
155
- end
156
- end
157
-
158
- context "the role is in ruby" do
159
- before do
160
- file "roles/x.rb", <<~EOM
161
- name "x"
162
- description "blargle"
163
- EOM
164
- end
165
-
166
- it "knife upload changes the role" do
167
- knife("upload /").should_succeed "Updated /roles/x.json\n"
168
- knife("diff --name-status /").should_succeed ""
169
- end
170
-
171
- it "knife upload --no-diff does not change the role" do
172
- knife("upload --no-diff /").should_succeed ""
173
- knife("diff --name-status /").should_succeed "M\t/roles/x.rb\n"
174
- end
175
- end
176
-
177
- context "when cookbook metadata has a self-dependency" do
178
- before do
179
- file "cookbooks/x/metadata.rb", "name 'x'; version '1.0.0'; depends 'x'"
180
- end
181
-
182
- it "fails with RuntimeError" do
183
- expect { knife("upload /cookbooks") }.to raise_error RuntimeError, /Cookbook depends on itself/
184
- end
185
- end
186
-
187
- context "as well as one extra copy of each thing" do
188
- before do
189
- file "clients/y.json", { "public_key" => ChefZero::PUBLIC_KEY }
190
- file "cookbooks/x/blah.rb", ""
191
- file "cookbooks/y/metadata.rb", cb_metadata("y", "1.0.0")
192
- file "data_bags/x/z.json", {}
193
- file "data_bags/y/zz.json", {}
194
- file "environments/y.json", {}
195
- file "nodes/y.json", {}
196
- file "roles/y.json", {}
197
- file "users/y.json", { "public_key" => ChefZero::PUBLIC_KEY }
198
- end
199
-
200
- it "knife upload adds the new files" do
201
- knife("upload /").should_succeed <<~EOM
202
- Created /clients/y.json
203
- Updated /cookbooks/x
204
- Created /cookbooks/y
205
- Created /data_bags/x/z.json
206
- Created /data_bags/y
207
- Created /data_bags/y/zz.json
208
- Created /environments/y.json
209
- Created /nodes/y.json
210
- Created /roles/y.json
211
- Created /users/y.json
212
- EOM
213
- knife("diff --name-status /").should_succeed <<~EOM
214
- D\t/cookbooks/x/metadata.json
215
- D\t/cookbooks/y/metadata.json
216
- EOM
217
- end
218
-
219
- it "knife upload --no-diff adds the new files" do
220
- knife("upload --no-diff /").should_succeed <<~EOM
221
- Created /clients/y.json
222
- Updated /cookbooks/x
223
- Created /cookbooks/y
224
- Created /data_bags/x/z.json
225
- Created /data_bags/y
226
- Created /data_bags/y/zz.json
227
- Created /environments/y.json
228
- Created /nodes/y.json
229
- Created /roles/y.json
230
- Created /users/y.json
231
- EOM
232
- knife("diff --name-status /").should_succeed <<~EOM
233
- D\t/cookbooks/x/metadata.json
234
- D\t/cookbooks/y/metadata.json
235
- EOM
236
- end
237
- end
238
- end
239
-
240
- when_the_repository "is empty" do
241
- it "knife upload does nothing" do
242
- knife("upload /").should_succeed ""
243
- knife("diff --name-status /").should_succeed <<~EOM
244
- D\t/clients
245
- D\t/cookbooks
246
- D\t/data_bags
247
- D\t/environments
248
- D\t/nodes
249
- D\t/roles
250
- D\t/users
251
- EOM
252
- end
253
-
254
- it "knife upload --purge deletes nothing" do
255
- knife("upload --purge /").should_fail <<~EOM
256
- ERROR: /clients cannot be deleted.
257
- ERROR: /cookbooks cannot be deleted.
258
- ERROR: /data_bags cannot be deleted.
259
- ERROR: /environments cannot be deleted.
260
- ERROR: /nodes cannot be deleted.
261
- ERROR: /roles cannot be deleted.
262
- ERROR: /users cannot be deleted.
263
- EOM
264
- knife("diff --name-status /").should_succeed <<~EOM
265
- D\t/clients
266
- D\t/cookbooks
267
- D\t/data_bags
268
- D\t/environments
269
- D\t/nodes
270
- D\t/roles
271
- D\t/users
272
- EOM
273
- end
274
-
275
- context "when current directory is top level" do
276
- before do
277
- cwd "."
278
- end
279
-
280
- it "knife upload with no parameters reports an error" do
281
- knife("upload").should_fail "FATAL: You must specify at least one argument. If you want to upload everything in this directory, run \"knife upload .\"\n", stdout: /USAGE/
282
- end
283
- end
284
- end
285
- end
286
-
287
- when_the_chef_server "is empty" do
288
- when_the_repository "has a data bag item" do
289
-
290
- before do
291
- file "data_bags/x/y.json", { "foo" => "bar" }
292
- end
293
-
294
- it "knife upload of the data bag uploads only the values in the data bag item and no other" do
295
- knife("upload /data_bags/x/y.json").should_succeed <<~EOM
296
- Created /data_bags/x
297
- Created /data_bags/x/y.json
298
- EOM
299
- knife("diff --name-status /data_bags").should_succeed <<~EOM
300
- EOM
301
- expect(Chef::JSONCompat.parse(knife("raw /data/x/y").stdout, create_additions: false).keys.sort).to eq(%w{foo id})
302
- end
303
-
304
- it "knife upload /data_bags/x /data_bags/x/y.json uploads x once" do
305
- knife("upload /data_bags/x /data_bags/x/y.json").should_succeed <<~EOM
306
- Created /data_bags/x
307
- Created /data_bags/x/y.json
308
- EOM
309
- end
310
- end
311
-
312
- when_the_repository "has a data bag item with keys chef_type and data_bag" do
313
-
314
- before do
315
- file "data_bags/x/y.json", { "chef_type" => "aaa", "data_bag" => "bbb" }
316
- end
317
-
318
- it "upload preserves chef_type and data_bag" do
319
- knife("upload /data_bags/x/y.json").should_succeed <<~EOM
320
- Created /data_bags/x
321
- Created /data_bags/x/y.json
322
- EOM
323
- knife("diff --name-status /data_bags").should_succeed ""
324
- result = Chef::JSONCompat.parse(knife("raw /data/x/y").stdout, create_additions: false)
325
- expect(result.keys.sort).to eq(%w{chef_type data_bag id})
326
- expect(result["chef_type"]).to eq("aaa")
327
- expect(result["data_bag"]).to eq("bbb")
328
- end
329
- end
330
-
331
- # Test upload of an item when the other end doesn't even have the container
332
- when_the_repository "has two data bag items" do
333
- before do
334
- file "data_bags/x/y.json", {}
335
- file "data_bags/x/z.json", {}
336
- end
337
- it "knife upload of one data bag item itself succeeds" do
338
- knife("upload /data_bags/x/y.json").should_succeed <<~EOM
339
- Created /data_bags/x
340
- Created /data_bags/x/y.json
341
- EOM
342
- knife("diff --name-status /data_bags").should_succeed <<~EOM
343
- A\t/data_bags/x/z.json
344
- EOM
345
- end
346
- end
347
- end
348
-
349
- when_the_chef_server "has three data bag items" do
350
-
351
- before do
352
- data_bag "x", { "deleted" => {}, "modified" => {}, "unmodified" => {} }
353
- end
354
-
355
- when_the_repository "has a modified, unmodified, added and deleted data bag item" do
356
- before do
357
- file "data_bags/x/added.json", {}
358
- file "data_bags/x/modified.json", { "foo" => "bar" }
359
- file "data_bags/x/unmodified.json", {}
360
- end
361
-
362
- it "knife upload of the modified file succeeds" do
363
- knife("upload /data_bags/x/modified.json").should_succeed <<~EOM
364
- Updated /data_bags/x/modified.json
365
- EOM
366
- knife("diff --name-status /data_bags").should_succeed <<~EOM
367
- D\t/data_bags/x/deleted.json
368
- A\t/data_bags/x/added.json
369
- EOM
370
- end
371
- it "knife upload of the unmodified file does nothing" do
372
- knife("upload /data_bags/x/unmodified.json").should_succeed ""
373
- knife("diff --name-status /data_bags").should_succeed <<~EOM
374
- D\t/data_bags/x/deleted.json
375
- M\t/data_bags/x/modified.json
376
- A\t/data_bags/x/added.json
377
- EOM
378
- end
379
- it "knife upload of the added file succeeds" do
380
- knife("upload /data_bags/x/added.json").should_succeed <<~EOM
381
- Created /data_bags/x/added.json
382
- EOM
383
- knife("diff --name-status /data_bags").should_succeed <<~EOM
384
- D\t/data_bags/x/deleted.json
385
- M\t/data_bags/x/modified.json
386
- EOM
387
- end
388
- it "knife upload of the deleted file does nothing" do
389
- knife("upload /data_bags/x/deleted.json").should_succeed ""
390
- knife("diff --name-status /data_bags").should_succeed <<~EOM
391
- D\t/data_bags/x/deleted.json
392
- M\t/data_bags/x/modified.json
393
- A\t/data_bags/x/added.json
394
- EOM
395
- end
396
- it "knife upload --purge of the deleted file deletes it" do
397
- knife("upload --purge /data_bags/x/deleted.json").should_succeed <<~EOM
398
- Deleted extra entry /data_bags/x/deleted.json (purge is on)
399
- EOM
400
- knife("diff --name-status /data_bags").should_succeed <<~EOM
401
- M\t/data_bags/x/modified.json
402
- A\t/data_bags/x/added.json
403
- EOM
404
- end
405
- it "knife upload of the entire data bag uploads everything" do
406
- knife("upload /data_bags/x").should_succeed <<~EOM
407
- Created /data_bags/x/added.json
408
- Updated /data_bags/x/modified.json
409
- EOM
410
- knife("diff --name-status /data_bags").should_succeed <<~EOM
411
- D\t/data_bags/x/deleted.json
412
- EOM
413
- end
414
- it "knife upload --purge of the entire data bag uploads everything" do
415
- knife("upload --purge /data_bags/x").should_succeed <<~EOM
416
- Created /data_bags/x/added.json
417
- Updated /data_bags/x/modified.json
418
- Deleted extra entry /data_bags/x/deleted.json (purge is on)
419
- EOM
420
- knife("diff --name-status /data_bags").should_succeed ""
421
- end
422
- context "when cwd is the /data_bags directory" do
423
-
424
- before do
425
- cwd "data_bags"
426
- end
427
-
428
- it "knife upload fails" do
429
- knife("upload").should_fail "FATAL: You must specify at least one argument. If you want to upload everything in this directory, run \"knife upload .\"\n", stdout: /USAGE/
430
- end
431
-
432
- it "knife upload --purge . uploads everything" do
433
- knife("upload --purge .").should_succeed <<~EOM
434
- Created x/added.json
435
- Updated x/modified.json
436
- Deleted extra entry x/deleted.json (purge is on)
437
- EOM
438
- knife("diff --name-status /data_bags").should_succeed ""
439
- end
440
- it "knife upload --purge * uploads everything" do
441
- knife("upload --purge *").should_succeed <<~EOM
442
- Created x/added.json
443
- Updated x/modified.json
444
- Deleted extra entry x/deleted.json (purge is on)
445
- EOM
446
- knife("diff --name-status /data_bags").should_succeed ""
447
- end
448
- end
449
- end
450
- end
451
-
452
- # Cookbook upload is a funny thing ... direct cookbook upload works, but
453
- # upload of a file is designed not to work at present. Make sure that is the
454
- # case.
455
- when_the_chef_server "has a cookbook" do
456
- before do
457
- cookbook "x", "1.0.0", { "z.rb" => "" }
458
- end
459
-
460
- when_the_repository "does not have metadata file" do
461
- before do
462
- file "cookbooks/x/y.rb", "hi"
463
- end
464
-
465
- it "raises MetadataNotFound exception" do
466
- expect { knife("upload /cookbooks/x") }.to raise_error(Chef::Exceptions::MetadataNotFound)
467
- end
468
- end
469
-
470
- when_the_repository "does not have valid metadata" do
471
- before do
472
- file "cookbooks/x/metadata.rb", cb_metadata(nil, "1.0.0")
473
- end
474
-
475
- it "raises exception for invalid metadata" do
476
- expect { knife("upload /cookbooks/x") }.to raise_error(Chef::Exceptions::MetadataNotValid)
477
- end
478
- end
479
-
480
- when_the_repository "has a modified, extra and missing file for the cookbook" do
481
- before do
482
- file "cookbooks/x/metadata.rb", cb_metadata("x", "1.0.0", "#modified")
483
- file "cookbooks/x/y.rb", "hi"
484
- end
485
-
486
- it "knife upload of any individual file fails" do
487
- knife("upload /cookbooks/x/metadata.rb").should_fail "ERROR: /cookbooks/x/metadata.rb cannot be updated.\n"
488
- knife("upload /cookbooks/x/y.rb").should_fail "ERROR: /cookbooks/x cannot have a child created under it.\n"
489
- knife("upload --purge /cookbooks/x/z.rb").should_fail "ERROR: /cookbooks/x/z.rb cannot be deleted.\n"
490
- end
491
-
492
- # TODO this is a bit of an inconsistency: if we didn't specify --purge,
493
- # technically we shouldn't have deleted missing files. But ... cookbooks
494
- # are a special case.
495
- it "knife upload of the cookbook itself succeeds" do
496
- knife("upload /cookbooks/x").should_succeed <<~EOM
497
- Updated /cookbooks/x
498
- EOM
499
- knife("diff --name-status /cookbooks").should_succeed <<~EOM
500
- D\t/cookbooks/x/metadata.json
501
- EOM
502
- end
503
-
504
- it "knife upload --purge of the cookbook itself succeeds" do
505
- knife("upload /cookbooks/x").should_succeed <<~EOM
506
- Updated /cookbooks/x
507
- EOM
508
- knife("diff --name-status /cookbooks").should_succeed <<~EOM
509
- D\t/cookbooks/x/metadata.json
510
- EOM
511
- end
512
- end
513
- when_the_repository "has a missing file for the cookbook" do
514
-
515
- before do
516
- file "cookbooks/x/metadata.rb", cb_metadata("x", "1.0.0")
517
- end
518
-
519
- it "knife upload of the cookbook succeeds" do
520
- knife("upload /cookbooks/x").should_succeed <<~EOM
521
- Updated /cookbooks/x
522
- EOM
523
- knife("diff --name-status /cookbooks").should_succeed <<~EOM
524
- D\t/cookbooks/x/metadata.json
525
- EOM
526
- end
527
- end
528
- when_the_repository "has an extra file for the cookbook" do
529
-
530
- before do
531
- file "cookbooks/x/metadata.rb", cb_metadata("x", "1.0.0")
532
- file "cookbooks/x/z.rb", ""
533
- file "cookbooks/x/blah.rb", ""
534
- end
535
-
536
- it "knife upload of the cookbook succeeds" do
537
- knife("upload /cookbooks/x").should_succeed <<~EOM
538
- Updated /cookbooks/x
539
- EOM
540
- knife("diff --name-status /cookbooks").should_succeed <<~EOM
541
- D\t/cookbooks/x/metadata.json
542
- EOM
543
- end
544
- end
545
-
546
- when_the_repository "has a different file in the cookbook" do
547
- before do
548
- file "cookbooks/x/metadata.rb", cb_metadata("x", "1.0.0")
549
- end
550
-
551
- it "knife upload --freeze freezes the cookbook" do
552
- knife("upload --freeze /cookbooks/x").should_succeed <<~EOM
553
- Updated /cookbooks/x
554
- EOM
555
- # Modify a file and attempt to upload
556
- file "cookbooks/x/metadata.rb", 'name "x"; version "1.0.0"#different'
557
- knife("upload /cookbooks/x").should_fail "ERROR: /cookbooks failed to write: Cookbook x is frozen\n"
558
- end
559
- end
560
- end
561
-
562
- when_the_chef_server "has a frozen cookbook" do
563
- before do
564
- cookbook "frozencook", "1.0.0", {}, frozen: true
565
- end
566
-
567
- when_the_repository "has an update to said cookbook" do
568
-
569
- before do
570
- file "cookbooks/frozencook/metadata.rb", cb_metadata("frozencook", "1.0.0", "# This is different")
571
- end
572
-
573
- it "knife upload fails to upload the frozen cookbook" do
574
- knife("upload /cookbooks/frozencook").should_fail "ERROR: /cookbooks failed to write: Cookbook frozencook is frozen\n"
575
- end
576
- it "knife upload --force uploads the frozen cookbook" do
577
- knife("upload --force /cookbooks/frozencook").should_succeed <<~EOM
578
- Updated /cookbooks/frozencook
579
- EOM
580
- end
581
- end
582
- end
583
-
584
- when_the_repository "has a cookbook" do
585
- before do
586
- file "cookbooks/x/metadata.rb", cb_metadata("x", "1.0.0")
587
- file "cookbooks/x/metadata.json", { name: "x", version: "1.0.0" }
588
- file "cookbooks/x/onlyin1.0.0.rb", "old_text"
589
- end
590
-
591
- when_the_chef_server "has a later version for the cookbook" do
592
- before do
593
- cookbook "x", "1.0.0", { "onlyin1.0.0.rb" => "" }
594
- cookbook "x", "1.0.1", { "onlyin1.0.1.rb" => "hi" }
595
- end
596
-
597
- it "knife upload /cookbooks/x uploads the local version" do
598
- knife("diff --name-status /cookbooks").should_succeed <<~EOM
599
- M\t/cookbooks/x/metadata.rb
600
- D\t/cookbooks/x/onlyin1.0.1.rb
601
- A\t/cookbooks/x/metadata.json
602
- A\t/cookbooks/x/onlyin1.0.0.rb
603
- EOM
604
- knife("upload --purge /cookbooks/x").should_succeed <<~EOM
605
- Updated /cookbooks/x
606
- EOM
607
- knife("diff --name-status /cookbooks").should_succeed <<~EOM
608
- M\t/cookbooks/x/metadata.rb
609
- D\t/cookbooks/x/onlyin1.0.1.rb
610
- A\t/cookbooks/x/metadata.json
611
- A\t/cookbooks/x/onlyin1.0.0.rb
612
- EOM
613
- end
614
- end
615
- end
616
-
617
- when_the_repository "has a cookbook" do
618
- before do
619
- file "cookbooks/x/metadata.rb", cb_metadata("x", "1.0.0")
620
- file "cookbooks/x/onlyin1.0.0.rb", "old_text"
621
- end
622
-
623
- when_the_chef_server "has a later version for the cookbook" do
624
- before do
625
- cookbook "x", "1.0.0", { "onlyin1.0.0.rb" => "" }
626
- cookbook "x", "1.0.1", { "onlyin1.0.1.rb" => "hi" }
627
- end
628
-
629
- it "knife upload /cookbooks/x uploads the local version and generates metadata.json from metadata.rb and uploads it." do
630
- knife("diff --name-status /cookbooks").should_succeed <<~EOM
631
- M\t/cookbooks/x/metadata.rb
632
- D\t/cookbooks/x/onlyin1.0.1.rb
633
- A\t/cookbooks/x/onlyin1.0.0.rb
634
- EOM
635
- knife("upload --purge /cookbooks/x").should_succeed <<~EOM
636
- Updated /cookbooks/x
637
- EOM
638
- knife("diff --name-status /cookbooks").should_succeed <<~EOM
639
- M\t/cookbooks/x/metadata.rb
640
- D\t/cookbooks/x/onlyin1.0.1.rb
641
- A\t/cookbooks/x/onlyin1.0.0.rb
642
- EOM
643
- end
644
- end
645
-
646
- when_the_chef_server "has an earlier version for the cookbook" do
647
- before do
648
- cookbook "x", "1.0.0", { "onlyin1.0.0.rb" => "" }
649
- cookbook "x", "0.9.9", { "onlyin0.9.9.rb" => "hi" }
650
- end
651
-
652
- it "knife upload /cookbooks/x uploads the local version generates metadata.json and uploads it." do
653
- knife("upload --purge /cookbooks/x").should_succeed <<~EOM
654
- Updated /cookbooks/x
655
- EOM
656
- knife("diff --name-status /cookbooks").should_succeed <<~EOM
657
- D\t/cookbooks/x/metadata.json
658
- EOM
659
- end
660
- end
661
-
662
- when_the_chef_server "has a later version for the cookbook, and no current version" do
663
- before do
664
- cookbook "x", "1.0.1", { "onlyin1.0.1.rb" => "hi" }
665
- end
666
-
667
- it "knife upload /cookbooks/x uploads the local version and generates metadata.json before upload and uploads it." do
668
- knife("diff --name-status /cookbooks").should_succeed <<~EOM
669
- M\t/cookbooks/x/metadata.rb
670
- D\t/cookbooks/x/onlyin1.0.1.rb
671
- A\t/cookbooks/x/onlyin1.0.0.rb
672
- EOM
673
- knife("upload --purge /cookbooks/x").should_succeed <<~EOM
674
- Updated /cookbooks/x
675
- EOM
676
- knife("diff --name-status /cookbooks").should_succeed <<~EOM
677
- M\t/cookbooks/x/metadata.rb
678
- D\t/cookbooks/x/onlyin1.0.1.rb
679
- A\t/cookbooks/x/onlyin1.0.0.rb
680
- EOM
681
- end
682
- end
683
-
684
- when_the_chef_server "has an earlier version for the cookbook, and no current version" do
685
- before do
686
- cookbook "x", "0.9.9", { "onlyin0.9.9.rb" => "hi" }
687
- end
688
-
689
- it "knife upload /cookbooks/x uploads the new version" do
690
- knife("upload --purge /cookbooks/x").should_succeed <<~EOM
691
- Updated /cookbooks/x
692
- EOM
693
- knife("diff --name-status /cookbooks").should_succeed <<~EOM
694
- D\t/cookbooks/x/metadata.json
695
- EOM
696
- end
697
- end
698
- end
699
-
700
- when_the_chef_server "has an environment" do
701
- before do
702
- environment "x", {}
703
- end
704
-
705
- when_the_repository "has an environment with bad JSON" do
706
- before do
707
- file "environments/x.json", "{"
708
- end
709
-
710
- it "knife upload tries and fails" do
711
- error1 = <<~EOH
712
- WARN: Parse error reading #{path_to("environments/x.json")} as JSON: parse error: premature EOF
713
- {
714
- (right here) ------^
715
-
716
- ERROR: /environments/x.json failed to write: Parse error reading JSON: parse error: premature EOF
717
- {
718
- (right here) ------^
719
- EOH
720
-
721
- warn = <<~EOH
722
- WARN: Parse error reading #{path_to("environments/x.json")} as JSON: parse error: premature EOF
723
- {
724
- (right here) ------^
725
-
726
- EOH
727
- knife("upload /environments/x.json").should_fail(error1)
728
- knife("diff --name-status /environments/x.json").should_succeed("M\t/environments/x.json\n", stderr: warn)
729
- end
730
- end
731
-
732
- when_the_repository "has the same environment with the wrong name in the file" do
733
- before do
734
- file "environments/x.json", { "name" => "y" }
735
- end
736
- it "knife upload fails" do
737
- knife("upload /environments/x.json").should_fail "ERROR: /environments/x.json failed to write: Name must be 'x' (is 'y')\n"
738
- knife("diff --name-status /environments/x.json").should_succeed "M\t/environments/x.json\n"
739
- end
740
- end
741
-
742
- when_the_repository "has the same environment with no name in the file" do
743
- before do
744
- file "environments/x.json", { "description" => "hi" }
745
- end
746
- it "knife upload succeeds" do
747
- knife("upload /environments/x.json").should_succeed "Updated /environments/x.json\n"
748
- knife("diff --name-status /environments/x.json").should_succeed ""
749
- end
750
- end
751
- end
752
-
753
- when_the_chef_server "is empty" do
754
-
755
- when_the_repository "has an environment with the wrong name in the file" do
756
- before do
757
- file "environments/x.json", { "name" => "y" }
758
- end
759
- it "knife upload fails" do
760
- knife("upload /environments/x.json").should_fail "ERROR: /environments failed to create_child: Error creating 'x.json': Name must be 'x' (is 'y')\n"
761
- knife("diff --name-status /environments/x.json").should_succeed "A\t/environments/x.json\n"
762
- end
763
- end
764
-
765
- when_the_repository "has an environment with no name in the file" do
766
-
767
- before do
768
- file "environments/x.json", { "description" => "hi" }
769
- end
770
- it "knife upload succeeds" do
771
- knife("upload /environments/x.json").should_succeed "Created /environments/x.json\n"
772
- knife("diff --name-status /environments/x.json").should_succeed ""
773
- end
774
- end
775
-
776
- when_the_repository "has a data bag with no id in the file" do
777
- before do
778
- file "data_bags/bag/x.json", { "foo" => "bar" }
779
- end
780
- it "knife upload succeeds" do
781
- knife("upload /data_bags/bag/x.json").should_succeed "Created /data_bags/bag\nCreated /data_bags/bag/x.json\n"
782
- knife("diff --name-status /data_bags/bag/x.json").should_succeed ""
783
- end
784
- end
785
- end
786
- when_the_chef_server "is empty" do
787
- when_the_repository "has a cookbook with an invalid chef_version constraint in it" do
788
- before do
789
- file "cookbooks/x/metadata.rb", cb_metadata("x", "1.0.0", "\nchef_version '~> 999.0'")
790
- end
791
- it "knife upload succeeds" do
792
- knife("upload /cookbooks/x").should_succeed <<~EOM
793
- Created /cookbooks/x
794
- EOM
795
- knife("diff --name-status /cookbooks").should_succeed <<~EOM
796
- D\t/cookbooks/x/metadata.json
797
- EOM
798
- end
799
- end
800
- end
801
- end # without versioned cookbooks
802
-
803
- context "with versioned cookbooks" do
804
- before { Chef::Config[:versioned_cookbooks] = true }
805
-
806
- when_the_chef_server "has one of each thing" do
807
-
808
- before do
809
- client "x", {}
810
- cookbook "x", "1.0.0"
811
- data_bag "x", { "y" => {} }
812
- environment "x", {}
813
- node "x", {}
814
- role "x", {}
815
- user "x", {}
816
- end
817
-
818
- when_the_repository "has only top-level directories" do
819
- before do
820
- directory "clients"
821
- directory "cookbooks"
822
- directory "data_bags"
823
- directory "environments"
824
- directory "nodes"
825
- directory "roles"
826
- directory "users"
827
- end
828
-
829
- it "knife upload does nothing" do
830
- knife("upload /").should_succeed ""
831
- knife("diff --name-status /").should_succeed <<~EOM
832
- D\t/clients/chef-validator.json
833
- D\t/clients/chef-webui.json
834
- D\t/clients/x.json
835
- D\t/cookbooks/x-1.0.0
836
- D\t/data_bags/x
837
- D\t/environments/_default.json
838
- D\t/environments/x.json
839
- D\t/nodes/x.json
840
- D\t/roles/x.json
841
- D\t/users/admin.json
842
- D\t/users/x.json
843
- EOM
844
- end
845
-
846
- it "knife upload --purge deletes everything" do
847
- knife("upload --purge /").should_succeed(<<~EOM, stderr: "WARNING: /environments/_default.json cannot be deleted (default environment cannot be modified).\n")
848
- Deleted extra entry /clients/chef-validator.json (purge is on)
849
- Deleted extra entry /clients/chef-webui.json (purge is on)
850
- Deleted extra entry /clients/x.json (purge is on)
851
- Deleted extra entry /cookbooks/x-1.0.0 (purge is on)
852
- Deleted extra entry /data_bags/x (purge is on)
853
- Deleted extra entry /environments/x.json (purge is on)
854
- Deleted extra entry /nodes/x.json (purge is on)
855
- Deleted extra entry /roles/x.json (purge is on)
856
- Deleted extra entry /users/admin.json (purge is on)
857
- Deleted extra entry /users/x.json (purge is on)
858
- EOM
859
- knife("diff --name-status /").should_succeed <<~EOM
860
- D\t/environments/_default.json
861
- EOM
862
- end
863
- end
864
-
865
- when_the_repository "has an identical copy of each thing" do
866
- before do
867
- file "clients/chef-validator.json", { "validator" => true, "public_key" => ChefZero::PUBLIC_KEY }
868
- file "clients/chef-webui.json", { "admin" => true, "public_key" => ChefZero::PUBLIC_KEY }
869
- file "clients/x.json", { "public_key" => ChefZero::PUBLIC_KEY }
870
- file "cookbooks/x-1.0.0/metadata.rb", cb_metadata("x", "1.0.0")
871
- file "data_bags/x/y.json", {}
872
- file "environments/_default.json", { "description" => "The default Chef environment" }
873
- file "environments/x.json", {}
874
- file "nodes/x.json", { "normal" => { "tags" => [] } }
875
- file "roles/x.json", {}
876
- file "users/admin.json", { "admin" => true, "public_key" => ChefZero::PUBLIC_KEY }
877
- file "users/x.json", { "public_key" => ChefZero::PUBLIC_KEY }
878
- end
879
-
880
- it "knife upload makes no changes" do
881
- knife("upload /cookbooks/x-1.0.0").should_succeed ""
882
- knife("diff --name-status /").should_succeed ""
883
- end
884
-
885
- it "knife upload --purge makes no changes" do
886
- knife("upload --purge /").should_succeed ""
887
- knife("diff --name-status /").should_succeed ""
888
- end
889
-
890
- context "except the role file" do
891
- before do
892
- file "roles/x.json", { "description" => "blarghle" }
893
- end
894
-
895
- it "knife upload changes the role" do
896
- knife("upload /").should_succeed "Updated /roles/x.json\n"
897
- knife("diff --name-status /").should_succeed ""
898
- end
899
- end
900
-
901
- context "except the role file is textually different, but not ACTUALLY different" do
902
-
903
- before do
904
- file "roles/x.json", <<~EOM
905
- {
906
- "chef_type": "role",
907
- "default_attributes": {
908
- },
909
- "env_run_lists": {
910
- },
911
- "json_class": "Chef::Role",
912
- "name": "x",
913
- "description": "",
914
- "override_attributes": {
915
- },
916
- "run_list": [
917
-
918
- ]
919
- }
920
- EOM
921
- end
922
-
923
- it "knife upload / does not change anything" do
924
- knife("upload /").should_succeed ""
925
- knife("diff --name-status /").should_succeed ""
926
- end
927
- end
928
-
929
- context "as well as one extra copy of each thing" do
930
- before do
931
- file "clients/y.json", { "public_key" => ChefZero::PUBLIC_KEY }
932
- file "cookbooks/x-1.0.0/blah.rb", ""
933
- file "cookbooks/x-2.0.0/metadata.rb", cb_metadata("x", "2.0.0")
934
- file "cookbooks/y-1.0.0/metadata.rb", cb_metadata("y", "1.0.0")
935
- file "data_bags/x/z.json", {}
936
- file "data_bags/y/zz.json", {}
937
- file "environments/y.json", {}
938
- file "nodes/y.json", {}
939
- file "roles/y.json", {}
940
- file "users/y.json", { "public_key" => ChefZero::PUBLIC_KEY }
941
- end
942
-
943
- it "knife upload adds the new files" do
944
- knife("upload /").should_succeed <<~EOM
945
- Created /clients/y.json
946
- Updated /cookbooks/x-1.0.0
947
- Created /cookbooks/x-2.0.0
948
- Created /cookbooks/y-1.0.0
949
- Created /data_bags/x/z.json
950
- Created /data_bags/y
951
- Created /data_bags/y/zz.json
952
- Created /environments/y.json
953
- Created /nodes/y.json
954
- Created /roles/y.json
955
- Created /users/y.json
956
- EOM
957
- knife("diff --name-status /").should_succeed ""
958
- end
959
- end
960
- end
961
-
962
- when_the_repository "is empty" do
963
- it "knife upload does nothing" do
964
- knife("upload /").should_succeed ""
965
- knife("diff --name-status /").should_succeed <<~EOM
966
- D\t/clients
967
- D\t/cookbooks
968
- D\t/data_bags
969
- D\t/environments
970
- D\t/nodes
971
- D\t/roles
972
- D\t/users
973
- EOM
974
- end
975
-
976
- it "knife upload --purge deletes nothing" do
977
- knife("upload --purge /").should_fail <<~EOM
978
- ERROR: /clients cannot be deleted.
979
- ERROR: /cookbooks cannot be deleted.
980
- ERROR: /data_bags cannot be deleted.
981
- ERROR: /environments cannot be deleted.
982
- ERROR: /nodes cannot be deleted.
983
- ERROR: /roles cannot be deleted.
984
- ERROR: /users cannot be deleted.
985
- EOM
986
- knife("diff --name-status /").should_succeed <<~EOM
987
- D\t/clients
988
- D\t/cookbooks
989
- D\t/data_bags
990
- D\t/environments
991
- D\t/nodes
992
- D\t/roles
993
- D\t/users
994
- EOM
995
- end
996
-
997
- context "when current directory is top level" do
998
- before do
999
- cwd "."
1000
- end
1001
- it "knife upload with no parameters reports an error" do
1002
- knife("upload").should_fail "FATAL: You must specify at least one argument. If you want to upload everything in this directory, run \"knife upload .\"\n", stdout: /USAGE/
1003
- end
1004
- end
1005
- end
1006
- end
1007
-
1008
- # Test upload of an item when the other end doesn't even have the container
1009
- when_the_chef_server "is empty" do
1010
- when_the_repository "has two data bag items" do
1011
- before do
1012
- file "data_bags/x/y.json", {}
1013
- file "data_bags/x/z.json", {}
1014
- end
1015
-
1016
- it "knife upload of one data bag item itself succeeds" do
1017
- knife("upload /data_bags/x/y.json").should_succeed <<~EOM
1018
- Created /data_bags/x
1019
- Created /data_bags/x/y.json
1020
- EOM
1021
- knife("diff --name-status /data_bags").should_succeed <<~EOM
1022
- A\t/data_bags/x/z.json
1023
- EOM
1024
- end
1025
- end
1026
- end
1027
-
1028
- when_the_chef_server "has three data bag items" do
1029
- before do
1030
- data_bag "x", { "deleted" => {}, "modified" => {}, "unmodified" => {} }
1031
- end
1032
- when_the_repository "has a modified, unmodified, added and deleted data bag item" do
1033
- before do
1034
- file "data_bags/x/added.json", {}
1035
- file "data_bags/x/modified.json", { "foo" => "bar" }
1036
- file "data_bags/x/unmodified.json", {}
1037
- end
1038
-
1039
- it "knife upload of the modified file succeeds" do
1040
- knife("upload /data_bags/x/modified.json").should_succeed <<~EOM
1041
- Updated /data_bags/x/modified.json
1042
- EOM
1043
- knife("diff --name-status /data_bags").should_succeed <<~EOM
1044
- D\t/data_bags/x/deleted.json
1045
- A\t/data_bags/x/added.json
1046
- EOM
1047
- end
1048
- it "knife upload of the unmodified file does nothing" do
1049
- knife("upload /data_bags/x/unmodified.json").should_succeed ""
1050
- knife("diff --name-status /data_bags").should_succeed <<~EOM
1051
- D\t/data_bags/x/deleted.json
1052
- M\t/data_bags/x/modified.json
1053
- A\t/data_bags/x/added.json
1054
- EOM
1055
- end
1056
- it "knife upload of the added file succeeds" do
1057
- knife("upload /data_bags/x/added.json").should_succeed <<~EOM
1058
- Created /data_bags/x/added.json
1059
- EOM
1060
- knife("diff --name-status /data_bags").should_succeed <<~EOM
1061
- D\t/data_bags/x/deleted.json
1062
- M\t/data_bags/x/modified.json
1063
- EOM
1064
- end
1065
- it "knife upload of the deleted file does nothing" do
1066
- knife("upload /data_bags/x/deleted.json").should_succeed ""
1067
- knife("diff --name-status /data_bags").should_succeed <<~EOM
1068
- D\t/data_bags/x/deleted.json
1069
- M\t/data_bags/x/modified.json
1070
- A\t/data_bags/x/added.json
1071
- EOM
1072
- end
1073
- it "knife upload --purge of the deleted file deletes it" do
1074
- knife("upload --purge /data_bags/x/deleted.json").should_succeed <<~EOM
1075
- Deleted extra entry /data_bags/x/deleted.json (purge is on)
1076
- EOM
1077
- knife("diff --name-status /data_bags").should_succeed <<~EOM
1078
- M\t/data_bags/x/modified.json
1079
- A\t/data_bags/x/added.json
1080
- EOM
1081
- end
1082
- it "knife upload of the entire data bag uploads everything" do
1083
- knife("upload /data_bags/x").should_succeed <<~EOM
1084
- Created /data_bags/x/added.json
1085
- Updated /data_bags/x/modified.json
1086
- EOM
1087
- knife("diff --name-status /data_bags").should_succeed <<~EOM
1088
- D\t/data_bags/x/deleted.json
1089
- EOM
1090
- end
1091
- it "knife upload --purge of the entire data bag uploads everything" do
1092
- knife("upload --purge /data_bags/x").should_succeed <<~EOM
1093
- Created /data_bags/x/added.json
1094
- Updated /data_bags/x/modified.json
1095
- Deleted extra entry /data_bags/x/deleted.json (purge is on)
1096
- EOM
1097
- knife("diff --name-status /data_bags").should_succeed ""
1098
- end
1099
- context "when cwd is the /data_bags directory" do
1100
- before do
1101
- cwd "data_bags"
1102
- end
1103
- it "knife upload fails" do
1104
- knife("upload").should_fail "FATAL: You must specify at least one argument. If you want to upload everything in this directory, run \"knife upload .\"\n", stdout: /USAGE/
1105
- end
1106
- it "knife upload --purge . uploads everything" do
1107
- knife("upload --purge .").should_succeed <<~EOM
1108
- Created x/added.json
1109
- Updated x/modified.json
1110
- Deleted extra entry x/deleted.json (purge is on)
1111
- EOM
1112
- knife("diff --name-status /data_bags").should_succeed ""
1113
- end
1114
- it "knife upload --purge * uploads everything" do
1115
- knife("upload --purge *").should_succeed <<~EOM
1116
- Created x/added.json
1117
- Updated x/modified.json
1118
- Deleted extra entry x/deleted.json (purge is on)
1119
- EOM
1120
- knife("diff --name-status /data_bags").should_succeed ""
1121
- end
1122
- end
1123
- end
1124
- end
1125
-
1126
- # Cookbook upload is a funny thing ... direct cookbook upload works, but
1127
- # upload of a file is designed not to work at present. Make sure that is the
1128
- # case.
1129
- when_the_chef_server "has a cookbook" do
1130
- before do
1131
- cookbook "x", "1.0.0", { "z.rb" => "" }
1132
- end
1133
-
1134
- when_the_repository "has a modified, extra and missing file for the cookbook" do
1135
- before do
1136
- file "cookbooks/x-1.0.0/metadata.rb", cb_metadata("x", "1.0.0", "#modified")
1137
- file "cookbooks/x-1.0.0/y.rb", "hi"
1138
- end
1139
-
1140
- it "knife upload of any individual file fails" do
1141
- knife("upload /cookbooks/x-1.0.0/metadata.rb").should_fail "ERROR: /cookbooks/x-1.0.0/metadata.rb cannot be updated.\n"
1142
- knife("upload /cookbooks/x-1.0.0/y.rb").should_fail "ERROR: /cookbooks/x-1.0.0 cannot have a child created under it.\n"
1143
- knife("upload --purge /cookbooks/x-1.0.0/z.rb").should_fail "ERROR: /cookbooks/x-1.0.0/z.rb cannot be deleted.\n"
1144
- end
1145
-
1146
- # TODO this is a bit of an inconsistency: if we didn't specify --purge,
1147
- # technically we shouldn't have deleted missing files. But ... cookbooks
1148
- # are a special case.
1149
- it "knife upload of the cookbook itself succeeds" do
1150
- knife("upload /cookbooks/x-1.0.0").should_succeed <<~EOM
1151
- Updated /cookbooks/x-1.0.0
1152
- EOM
1153
- knife("diff --name-status /cookbooks").should_succeed ""
1154
- end
1155
-
1156
- it "knife upload --purge of the cookbook itself succeeds" do
1157
- knife("upload /cookbooks/x-1.0.0").should_succeed <<~EOM
1158
- Updated /cookbooks/x-1.0.0
1159
- EOM
1160
- knife("diff --name-status /cookbooks").should_succeed ""
1161
- end
1162
- end
1163
-
1164
- when_the_repository "has a missing file for the cookbook" do
1165
- before do
1166
- file "cookbooks/x-1.0.0/metadata.rb", cb_metadata("x", "1.0.0")
1167
- end
1168
-
1169
- it "knife upload of the cookbook succeeds" do
1170
- knife("upload /cookbooks/x-1.0.0").should_succeed <<~EOM
1171
- Updated /cookbooks/x-1.0.0
1172
- EOM
1173
- knife("diff --name-status /cookbooks").should_succeed ""
1174
- end
1175
- end
1176
-
1177
- when_the_repository "has an extra file for the cookbook" do
1178
- before do
1179
- file "cookbooks/x-1.0.0/metadata.rb", cb_metadata("x", "1.0.0")
1180
- file "cookbooks/x-1.0.0/z.rb", ""
1181
- file "cookbooks/x-1.0.0/blah.rb", ""
1182
- end
1183
-
1184
- it "knife upload of the cookbook succeeds" do
1185
- knife("upload /cookbooks/x-1.0.0").should_succeed <<~EOM
1186
- Updated /cookbooks/x-1.0.0
1187
- EOM
1188
- knife("diff --name-status /cookbooks").should_succeed ""
1189
- end
1190
- end
1191
- end
1192
-
1193
- when_the_repository "has a cookbook" do
1194
- before do
1195
- file "cookbooks/x-1.0.0/metadata.rb", cb_metadata("x", "1.0.0")
1196
- file "cookbooks/x-1.0.0/onlyin1.0.0.rb", "old_text"
1197
- end
1198
-
1199
- when_the_chef_server "has a later version for the cookbook" do
1200
- before do
1201
- cookbook "x", "1.0.0", { "onlyin1.0.0.rb" => "" }
1202
- cookbook "x", "1.0.1", { "onlyin1.0.1.rb" => "hi" }
1203
- end
1204
-
1205
- it "knife upload /cookbooks uploads the local version" do
1206
- knife("diff --name-status /cookbooks").should_succeed <<~EOM
1207
- M\t/cookbooks/x-1.0.0/onlyin1.0.0.rb
1208
- D\t/cookbooks/x-1.0.1
1209
- EOM
1210
- knife("upload --purge /cookbooks").should_succeed <<~EOM
1211
- Updated /cookbooks/x-1.0.0
1212
- Deleted extra entry /cookbooks/x-1.0.1 (purge is on)
1213
- EOM
1214
- knife("diff --name-status /cookbooks").should_succeed ""
1215
- end
1216
- end
1217
-
1218
- when_the_chef_server "has an earlier version for the cookbook" do
1219
- before do
1220
- cookbook "x", "1.0.0", { "onlyin1.0.0.rb" => "" }
1221
- cookbook "x", "0.9.9", { "onlyin0.9.9.rb" => "hi" }
1222
- end
1223
- it "knife upload /cookbooks uploads the local version" do
1224
- knife("upload --purge /cookbooks").should_succeed <<~EOM
1225
- Updated /cookbooks/x-1.0.0
1226
- Deleted extra entry /cookbooks/x-0.9.9 (purge is on)
1227
- EOM
1228
- knife("diff --name-status /cookbooks").should_succeed ""
1229
- end
1230
- end
1231
-
1232
- when_the_chef_server "has a later version for the cookbook, and no current version" do
1233
- before do
1234
- cookbook "x", "1.0.1", { "onlyin1.0.1.rb" => "hi" }
1235
- end
1236
-
1237
- it "knife upload /cookbooks/x uploads the local version" do
1238
- knife("diff --name-status /cookbooks").should_succeed <<~EOM
1239
- D\t/cookbooks/x-1.0.1
1240
- A\t/cookbooks/x-1.0.0
1241
- EOM
1242
- knife("upload --purge /cookbooks").should_succeed <<~EOM
1243
- Created /cookbooks/x-1.0.0
1244
- Deleted extra entry /cookbooks/x-1.0.1 (purge is on)
1245
- EOM
1246
- knife("diff --name-status /cookbooks").should_succeed ""
1247
- end
1248
- end
1249
-
1250
- when_the_chef_server "has an earlier version for the cookbook, and no current version" do
1251
- before do
1252
- cookbook "x", "0.9.9", { "onlyin0.9.9.rb" => "hi" }
1253
- end
1254
-
1255
- it "knife upload /cookbooks/x uploads the new version" do
1256
- knife("upload --purge /cookbooks").should_succeed <<~EOM
1257
- Created /cookbooks/x-1.0.0
1258
- Deleted extra entry /cookbooks/x-0.9.9 (purge is on)
1259
- EOM
1260
- knife("diff --name-status /cookbooks").should_succeed ""
1261
- end
1262
- end
1263
- end
1264
-
1265
- when_the_chef_server "has an environment" do
1266
- before do
1267
- environment "x", {}
1268
- end
1269
-
1270
- when_the_repository "has the same environment with the wrong name in the file" do
1271
- before do
1272
- file "environments/x.json", { "name" => "y" }
1273
- end
1274
- it "knife upload fails" do
1275
- knife("upload /environments/x.json").should_fail "ERROR: /environments/x.json failed to write: Name must be 'x' (is 'y')\n"
1276
- knife("diff --name-status /environments/x.json").should_succeed "M\t/environments/x.json\n"
1277
- end
1278
- end
1279
-
1280
- when_the_repository "has the same environment with no name in the file" do
1281
- before do
1282
- file "environments/x.json", { "description" => "hi" }
1283
- end
1284
- it "knife upload succeeds" do
1285
- knife("upload /environments/x.json").should_succeed "Updated /environments/x.json\n"
1286
- knife("diff --name-status /environments/x.json").should_succeed ""
1287
- end
1288
- end
1289
- end
1290
-
1291
- when_the_chef_server "is empty" do
1292
-
1293
- when_the_repository "has an environment with the wrong name in the file" do
1294
- before do
1295
- file "environments/x.json", { "name" => "y" }
1296
- end
1297
- it "knife upload fails" do
1298
- knife("upload /environments/x.json").should_fail "ERROR: /environments failed to create_child: Error creating 'x.json': Name must be 'x' (is 'y')\n"
1299
- knife("diff --name-status /environments/x.json").should_succeed "A\t/environments/x.json\n"
1300
- end
1301
- end
1302
-
1303
- when_the_repository "has an environment with no name in the file" do
1304
- before do
1305
- file "environments/x.json", { "description" => "hi" }
1306
- end
1307
- it "knife upload succeeds" do
1308
- knife("upload /environments/x.json").should_succeed "Created /environments/x.json\n"
1309
- knife("diff --name-status /environments/x.json").should_succeed ""
1310
- end
1311
- end
1312
-
1313
- when_the_repository "has a data bag with no id in the file" do
1314
- before do
1315
- file "data_bags/bag/x.json", { "foo" => "bar" }
1316
- end
1317
- it "knife upload succeeds" do
1318
- knife("upload /data_bags/bag/x.json").should_succeed "Created /data_bags/bag\nCreated /data_bags/bag/x.json\n"
1319
- knife("diff --name-status /data_bags/bag/x.json").should_succeed ""
1320
- end
1321
- end
1322
- end
1323
-
1324
- when_the_chef_server "is empty" do
1325
- when_the_repository "has a cookbook with an invalid chef_version constraint in it" do
1326
- before do
1327
- file "cookbooks/x-1.0.0/metadata.rb", cb_metadata("x", "1.0.0", "\nchef_version '~> 999.0'")
1328
- end
1329
- it "knife upload succeeds" do
1330
- knife("upload /cookbooks/x-1.0.0").should_succeed <<~EOM
1331
- Created /cookbooks/x-1.0.0
1332
- EOM
1333
- knife("diff --name-status /cookbooks").should_succeed ""
1334
- end
1335
- end
1336
- end
1337
- end # with versioned cookbooks
1338
-
1339
- when_the_chef_server "has a user" do
1340
- before do
1341
- user "x", {}
1342
- end
1343
-
1344
- when_the_repository "has the same user with json_class in it" do
1345
- before do
1346
- file "users/x.json", { "admin" => true, "json_class" => "Chef::WebUIUser" }
1347
- end
1348
- it "knife upload /users/x.json succeeds" do
1349
- knife("upload /users/x.json").should_succeed "Updated /users/x.json\n"
1350
- end
1351
- end
1352
- end
1353
-
1354
- when_the_chef_server "is in Enterprise mode", osc_compat: false, single_org: false do
1355
- before do
1356
- user "foo", {}
1357
- user "bar", {}
1358
- user "foobar", {}
1359
- organization "foo", { "full_name" => "Something" }
1360
- end
1361
-
1362
- before :each do
1363
- Chef::Config.chef_server_url = URI.join(Chef::Config.chef_server_url, "/organizations/foo")
1364
- end
1365
-
1366
- context "and has nothing but a single group named blah" do
1367
- group "blah", {}
1368
-
1369
- when_the_repository "has at least one of each thing" do
1370
-
1371
- before do
1372
- # TODO We have to upload acls for an existing group due to a lack of
1373
- # dependency detection during upload. Fix that!
1374
- file "acls/groups/blah.json", {}
1375
- file "clients/x.json", { "public_key" => ChefZero::PUBLIC_KEY }
1376
- file "containers/x.json", {}
1377
- file "cookbooks/x/metadata.rb", cb_metadata("x", "1.0.0")
1378
- file "cookbook_artifacts/x-1x1/metadata.rb", cb_metadata("x", "1.0.0")
1379
- file "data_bags/x/y.json", {}
1380
- file "environments/x.json", {}
1381
- file "groups/x.json", {}
1382
- file "invitations.json", [ "foo" ]
1383
- file "members.json", [ "bar" ]
1384
- file "org.json", { "full_name" => "wootles" }
1385
- file "nodes/x.json", { "normal" => { "tags" => [] } }
1386
- file "policies/x-1.0.0.json", { "policy_group_list" => [ "x" ] }
1387
- file "policies/blah-1.0.0.json", { "policy_group_list" => [ "x" ] }
1388
- file "policy_groups/x.json", { "policies" => { "x" => { "revision_id" => "1.0.0" }, "blah" => { "revision_id" => "1.0.0" } } }
1389
- file "roles/x.json", {}
1390
- end
1391
-
1392
- it "knife upload / uploads everything" do
1393
- knife("upload /").should_succeed <<~EOM
1394
- Updated /acls/groups/blah.json
1395
- Created /clients/x.json
1396
- Created /containers/x.json
1397
- Created /cookbook_artifacts/x-1x1
1398
- Created /cookbooks/x
1399
- Created /data_bags/x
1400
- Created /data_bags/x/y.json
1401
- Created /environments/x.json
1402
- Created /groups/x.json
1403
- Updated /invitations.json
1404
- Updated /members.json
1405
- Created /nodes/x.json
1406
- Updated /org.json
1407
- Created /policies/blah-1.0.0.json
1408
- Created /policies/x-1.0.0.json
1409
- Created /policy_groups/x.json
1410
- Created /roles/x.json
1411
- EOM
1412
- expect(api.get("association_requests").map { |a| a["username"] }).to eq([ "foo" ])
1413
- expect(api.get("users").map { |a| a["user"]["username"] }).to eq([ "bar" ])
1414
- knife("diff --name-status --diff-filter=AMT /").should_succeed ""
1415
- end
1416
-
1417
- context "When the chef server has an identical copy of each thing" do
1418
- before do
1419
- file "invitations.json", [ "foo" ]
1420
- file "members.json", [ "bar" ]
1421
- file "org.json", { "full_name" => "Something" }
1422
-
1423
- # acl_for %w(organizations foo groups blah)
1424
- client "x", {}
1425
- cookbook "x", "1.0.0"
1426
- cookbook_artifact "x", "1x1", "metadata.rb" => cb_metadata("x", "1.0.0")
1427
- container "x", {}
1428
- data_bag "x", { "y" => {} }
1429
- environment "x", {}
1430
- group "x", {}
1431
- org_invite "foo"
1432
- org_member "bar"
1433
- node "x", {}
1434
- policy "x", "1.0.0", {}
1435
- policy "blah", "1.0.0", {}
1436
- policy_group "x", {
1437
- "policies" => {
1438
- "x" => { "revision_id" => "1.0.0" },
1439
- "blah" => { "revision_id" => "1.0.0" },
1440
- },
1441
- }
1442
- role "x", {}
1443
- end
1444
-
1445
- it "knife upload makes no changes" do
1446
- knife("upload /").should_succeed <<~EOM
1447
- Updated /acls/groups/blah.json
1448
- EOM
1449
- end
1450
- end
1451
-
1452
- context "When the chef server has a slightly different copy of the policy revision" do
1453
- before do
1454
- policy "x", "1.0.0", { "run_list" => [ "blah" ] }
1455
- end
1456
-
1457
- it "should fail because policies are not updateable" do
1458
- knife("upload /policies/x-1.0.0.json").should_fail <<~EOM
1459
- ERROR: /policies/x-1.0.0.json cannot be updated: policy revisions are immutable once uploaded. If you want to change the policy, create a new revision with your changes.
1460
- EOM
1461
- end
1462
- end
1463
-
1464
- context "When the chef server has a slightly different copy of the cookbook artifact" do
1465
- before do
1466
- cookbook_artifact "x", "1x1", { "recipes" => { "default.rb" => "" } }
1467
- end
1468
-
1469
- it "should fail because cookbook_artifacts cannot be updated" do
1470
- knife("upload /cookbook_artifacts/x-1x1").should_fail <<~EOM
1471
- ERROR: /cookbook_artifacts/x-1x1 cannot be updated: cookbook artifacts are immutable once uploaded.
1472
- EOM
1473
- end
1474
- end
1475
-
1476
- context "When the chef server has a slightly different copy of each thing (except policy revisions)" do
1477
- before do
1478
- # acl_for %w(organizations foo groups blah)
1479
- client "x", { "validator" => true }
1480
- container "x", {}
1481
- cookbook "x", "1.0.0", { "recipes" => { "default.rb" => "" } }
1482
- cookbook_artifact "x", "1x1", { "metadata.rb" => cb_metadata("x", "1.0.0") }
1483
- data_bag "x", { "y" => { "a" => "b" } }
1484
- environment "x", { "description" => "foo" }
1485
- group "x", { "groups" => [ "admin" ] }
1486
- node "x", { "run_list" => [ "blah" ] }
1487
- policy "x", "1.0.0", { "policy_group_list" => [ "x" ] }
1488
- policy "y", "1.0.0", { "policy_group_list" => [ "x" ] }
1489
- policy_group "x", {
1490
- "policies" => {
1491
- "x" => { "revision_id" => "1.0.0" },
1492
- "y" => { "revision_id" => "1.0.0" },
1493
- },
1494
- }
1495
- role "x", { "run_list" => [ "blah" ] }
1496
- end
1497
-
1498
- it "knife upload updates everything" do
1499
- knife("upload /").should_succeed <<~EOM
1500
- Updated /acls/groups/blah.json
1501
- Updated /clients/x.json
1502
- Updated /cookbooks/x
1503
- Updated /data_bags/x/y.json
1504
- Updated /environments/x.json
1505
- Updated /groups/x.json
1506
- Updated /invitations.json
1507
- Updated /members.json
1508
- Updated /nodes/x.json
1509
- Updated /org.json
1510
- Created /policies/blah-1.0.0.json
1511
- Updated /policy_groups/x.json
1512
- Updated /roles/x.json
1513
- EOM
1514
- knife("diff --name-status --diff-filter=AMT /").should_succeed ""
1515
- end
1516
- end
1517
- end
1518
-
1519
- when_the_repository "has an org.json that does not change full_name" do
1520
- before do
1521
- file "org.json", { "full_name" => "Something" }
1522
- end
1523
-
1524
- it "knife upload / emits a warning for bar and adds foo and foobar" do
1525
- knife("upload /").should_succeed ""
1526
- expect(api.get("/")["full_name"]).to eq("Something")
1527
- end
1528
- end
1529
-
1530
- when_the_repository "has an org.json that changes full_name" do
1531
- before do
1532
- file "org.json", { "full_name" => "Something Else" }
1533
- end
1534
-
1535
- it "knife upload / emits a warning for bar and adds foo and foobar" do
1536
- knife("upload /").should_succeed "Updated /org.json\n"
1537
- expect(api.get("/")["full_name"]).to eq("Something Else")
1538
- end
1539
- end
1540
-
1541
- context "and has invited foo and bar is already a member" do
1542
- org_invite "foo"
1543
- org_member "bar"
1544
-
1545
- when_the_repository "wants to invite foo, bar and foobar" do
1546
- before do
1547
- file "invitations.json", %w{foo bar foobar}
1548
- end
1549
-
1550
- it "knife upload / emits a warning for bar and invites foobar" do
1551
- knife("upload /").should_succeed "Updated /invitations.json\n", stderr: "WARN: Could not invite bar to organization foo: User bar is already in organization foo\n"
1552
- expect(api.get("association_requests").map { |a| a["username"] }).to eq(%w{foo foobar})
1553
- expect(api.get("users").map { |a| a["user"]["username"] }).to eq([ "bar" ])
1554
- end
1555
- end
1556
-
1557
- when_the_repository "wants to make foo, bar and foobar members" do
1558
- before do
1559
- file "members.json", %w{foo bar foobar}
1560
- end
1561
-
1562
- it "knife upload / emits a warning for bar and adds foo and foobar" do
1563
- knife("upload /").should_succeed "Updated /members.json\n"
1564
- expect(api.get("association_requests").map { |a| a["username"] }).to eq([ ])
1565
- expect(api.get("users").map { |a| a["user"]["username"] }).to eq(%w{bar foo foobar})
1566
- end
1567
- end
1568
-
1569
- when_the_repository "wants to invite foo and have bar as a member" do
1570
- before do
1571
- file "invitations.json", [ "foo" ]
1572
- file "members.json", [ "bar" ]
1573
- end
1574
-
1575
- it "knife upload / does nothing" do
1576
- knife("upload /").should_succeed ""
1577
- expect(api.get("association_requests").map { |a| a["username"] }).to eq([ "foo" ])
1578
- expect(api.get("users").map { |a| a["user"]["username"] }).to eq([ "bar" ])
1579
- end
1580
- end
1581
- end
1582
-
1583
- context "and has invited bar and foo" do
1584
- org_invite "bar", "foo"
1585
-
1586
- when_the_repository "wants to invite foo and bar (different order)" do
1587
- before do
1588
- file "invitations.json", %w{foo bar}
1589
- end
1590
-
1591
- it "knife upload / does nothing" do
1592
- knife("upload /").should_succeed ""
1593
- expect(api.get("association_requests").map { |a| a["username"] }).to eq(%w{bar foo})
1594
- expect(api.get("users").map { |a| a["user"]["username"] }).to eq([ ])
1595
- end
1596
- end
1597
- end
1598
-
1599
- context "and has already added bar and foo as members of the org" do
1600
- org_member "bar", "foo"
1601
-
1602
- when_the_repository "wants to add foo and bar (different order)" do
1603
- before do
1604
- file "members.json", %w{foo bar}
1605
- end
1606
-
1607
- it "knife upload / does nothing" do
1608
- knife("upload /").should_succeed ""
1609
- expect(api.get("association_requests").map { |a| a["username"] }).to eq([ ])
1610
- expect(api.get("users").map { |a| a["user"]["username"] }).to eq(%w{bar foo})
1611
- end
1612
- end
1613
- end
1614
- end
1615
- end
1616
- end
1
+ #
2
+ # Author:: John Keiser (<jkeiser@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
+ require "knife_spec_helper"
19
+ require "support/shared/integration/integration_helper"
20
+ require "chef/knife/upload"
21
+ require "chef/knife/diff"
22
+ require "chef/knife/raw"
23
+ require "chef/json_compat"
24
+
25
+ describe "knife upload", :workstation do
26
+ include IntegrationSupport
27
+ include KnifeSupport
28
+
29
+ context "without versioned cookbooks" do
30
+
31
+ when_the_chef_server "has one of each thing" do
32
+
33
+ before do
34
+ client "x", {}
35
+ cookbook "x", "1.0.0"
36
+ data_bag "x", { "y" => {} }
37
+ environment "x", {}
38
+ node "x", {}
39
+ role "x", {}
40
+ user "x", {}
41
+ end
42
+
43
+ when_the_repository "has only top-level directories" do
44
+ before do
45
+ directory "clients"
46
+ directory "cookbooks"
47
+ directory "data_bags"
48
+ directory "environments"
49
+ directory "nodes"
50
+ directory "roles"
51
+ directory "users"
52
+ end
53
+
54
+ it "knife upload does nothing" do
55
+ knife("upload /").should_succeed ""
56
+ knife("diff --name-status /").should_succeed <<~EOM
57
+ D\t/clients/chef-validator.json
58
+ D\t/clients/chef-webui.json
59
+ D\t/clients/x.json
60
+ D\t/cookbooks/x
61
+ D\t/data_bags/x
62
+ D\t/environments/_default.json
63
+ D\t/environments/x.json
64
+ D\t/nodes/x.json
65
+ D\t/roles/x.json
66
+ D\t/users/admin.json
67
+ D\t/users/x.json
68
+ EOM
69
+ end
70
+
71
+ it "knife upload --purge deletes everything" do
72
+ knife("upload --purge /").should_succeed(<<~EOM, stderr: "WARNING: /environments/_default.json cannot be deleted (default environment cannot be modified).\n")
73
+ Deleted extra entry /clients/chef-validator.json (purge is on)
74
+ Deleted extra entry /clients/chef-webui.json (purge is on)
75
+ Deleted extra entry /clients/x.json (purge is on)
76
+ Deleted extra entry /cookbooks/x (purge is on)
77
+ Deleted extra entry /data_bags/x (purge is on)
78
+ Deleted extra entry /environments/x.json (purge is on)
79
+ Deleted extra entry /nodes/x.json (purge is on)
80
+ Deleted extra entry /roles/x.json (purge is on)
81
+ Deleted extra entry /users/admin.json (purge is on)
82
+ Deleted extra entry /users/x.json (purge is on)
83
+ EOM
84
+ knife("diff --name-status /").should_succeed <<~EOM
85
+ D\t/environments/_default.json
86
+ EOM
87
+ end
88
+ end
89
+
90
+ when_the_repository "has an identical copy of each thing" do
91
+
92
+ before do
93
+ file "clients/chef-validator.json", { "validator" => true, "public_key" => ChefZero::PUBLIC_KEY }
94
+ file "clients/chef-webui.json", { "admin" => true, "public_key" => ChefZero::PUBLIC_KEY }
95
+ file "clients/x.json", { "public_key" => ChefZero::PUBLIC_KEY }
96
+ file "cookbooks/x/metadata.rb", cb_metadata("x", "1.0.0")
97
+ file "data_bags/x/y.json", {}
98
+ file "environments/_default.json", { "description" => "The default Chef environment" }
99
+ file "environments/x.json", {}
100
+ file "nodes/x.json", { "normal" => { "tags" => [] } }
101
+ file "roles/x.json", {}
102
+ file "users/admin.json", { "admin" => true, "public_key" => ChefZero::PUBLIC_KEY }
103
+ file "users/x.json", { "public_key" => ChefZero::PUBLIC_KEY }
104
+ end
105
+
106
+ it "knife upload makes no changes" do
107
+ knife("upload /cookbooks/x").should_succeed ""
108
+ knife("diff --name-status /").should_succeed ""
109
+ end
110
+
111
+ it "knife upload --purge makes no changes" do
112
+ knife("upload --purge /").should_succeed ""
113
+ knife("diff --name-status /").should_succeed ""
114
+ end
115
+
116
+ context "except the role file" do
117
+ before do
118
+ file "roles/x.json", { "description" => "blarghle" }
119
+ end
120
+
121
+ it "knife upload changes the role" do
122
+ knife("upload /").should_succeed "Updated /roles/x.json\n"
123
+ knife("diff --name-status /").should_succeed ""
124
+ end
125
+ it "knife upload --no-diff does not change the role" do
126
+ knife("upload --no-diff /").should_succeed ""
127
+ knife("diff --name-status /").should_succeed "M\t/roles/x.json\n"
128
+ end
129
+ end
130
+
131
+ context "except the role file is textually different, but not ACTUALLY different" do
132
+ before do
133
+ file "roles/x.json", <<~EOM
134
+ {
135
+ "chef_type": "role",
136
+ "default_attributes": {
137
+ },
138
+ "env_run_lists": {
139
+ },
140
+ "json_class": "Chef::Role",
141
+ "name": "x",
142
+ "description": "",
143
+ "override_attributes": {
144
+ },
145
+ "run_list": [
146
+
147
+ ]
148
+ }
149
+ EOM
150
+ end
151
+
152
+ it "knife upload / does not change anything" do
153
+ knife("upload /").should_succeed ""
154
+ knife("diff --name-status /").should_succeed ""
155
+ end
156
+ end
157
+
158
+ context "the role is in ruby" do
159
+ before do
160
+ file "roles/x.rb", <<~EOM
161
+ name "x"
162
+ description "blargle"
163
+ EOM
164
+ end
165
+
166
+ it "knife upload changes the role" do
167
+ knife("upload /").should_succeed "Updated /roles/x.json\n"
168
+ knife("diff --name-status /").should_succeed ""
169
+ end
170
+
171
+ it "knife upload --no-diff does not change the role" do
172
+ knife("upload --no-diff /").should_succeed ""
173
+ knife("diff --name-status /").should_succeed "M\t/roles/x.rb\n"
174
+ end
175
+ end
176
+
177
+ context "when cookbook metadata has a self-dependency" do
178
+ before do
179
+ file "cookbooks/x/metadata.rb", "name 'x'; version '1.0.0'; depends 'x'"
180
+ end
181
+
182
+ it "fails with RuntimeError" do
183
+ expect { knife("upload /cookbooks") }.to raise_error RuntimeError, /Cookbook depends on itself/
184
+ end
185
+ end
186
+
187
+ context "as well as one extra copy of each thing" do
188
+ before do
189
+ file "clients/y.json", { "public_key" => ChefZero::PUBLIC_KEY }
190
+ file "cookbooks/x/blah.rb", ""
191
+ file "cookbooks/y/metadata.rb", cb_metadata("y", "1.0.0")
192
+ file "data_bags/x/z.json", {}
193
+ file "data_bags/y/zz.json", {}
194
+ file "environments/y.json", {}
195
+ file "nodes/y.json", {}
196
+ file "roles/y.json", {}
197
+ file "users/y.json", { "public_key" => ChefZero::PUBLIC_KEY }
198
+ end
199
+
200
+ it "knife upload adds the new files" do
201
+ knife("upload /").should_succeed <<~EOM
202
+ Created /clients/y.json
203
+ Updated /cookbooks/x
204
+ Created /cookbooks/y
205
+ Created /data_bags/x/z.json
206
+ Created /data_bags/y
207
+ Created /data_bags/y/zz.json
208
+ Created /environments/y.json
209
+ Created /nodes/y.json
210
+ Created /roles/y.json
211
+ Created /users/y.json
212
+ EOM
213
+ knife("diff --name-status /").should_succeed <<~EOM
214
+ D\t/cookbooks/x/metadata.json
215
+ D\t/cookbooks/y/metadata.json
216
+ EOM
217
+ end
218
+
219
+ it "knife upload --no-diff adds the new files" do
220
+ knife("upload --no-diff /").should_succeed <<~EOM
221
+ Created /clients/y.json
222
+ Updated /cookbooks/x
223
+ Created /cookbooks/y
224
+ Created /data_bags/x/z.json
225
+ Created /data_bags/y
226
+ Created /data_bags/y/zz.json
227
+ Created /environments/y.json
228
+ Created /nodes/y.json
229
+ Created /roles/y.json
230
+ Created /users/y.json
231
+ EOM
232
+ knife("diff --name-status /").should_succeed <<~EOM
233
+ D\t/cookbooks/x/metadata.json
234
+ D\t/cookbooks/y/metadata.json
235
+ EOM
236
+ end
237
+ end
238
+ end
239
+
240
+ when_the_repository "is empty" do
241
+ it "knife upload does nothing" do
242
+ knife("upload /").should_succeed ""
243
+ knife("diff --name-status /").should_succeed <<~EOM
244
+ D\t/clients
245
+ D\t/cookbooks
246
+ D\t/data_bags
247
+ D\t/environments
248
+ D\t/nodes
249
+ D\t/roles
250
+ D\t/users
251
+ EOM
252
+ end
253
+
254
+ it "knife upload --purge deletes nothing" do
255
+ knife("upload --purge /").should_fail <<~EOM
256
+ ERROR: /clients cannot be deleted.
257
+ ERROR: /cookbooks cannot be deleted.
258
+ ERROR: /data_bags cannot be deleted.
259
+ ERROR: /environments cannot be deleted.
260
+ ERROR: /nodes cannot be deleted.
261
+ ERROR: /roles cannot be deleted.
262
+ ERROR: /users cannot be deleted.
263
+ EOM
264
+ knife("diff --name-status /").should_succeed <<~EOM
265
+ D\t/clients
266
+ D\t/cookbooks
267
+ D\t/data_bags
268
+ D\t/environments
269
+ D\t/nodes
270
+ D\t/roles
271
+ D\t/users
272
+ EOM
273
+ end
274
+
275
+ context "when current directory is top level" do
276
+ before do
277
+ cwd "."
278
+ end
279
+
280
+ it "knife upload with no parameters reports an error" do
281
+ knife("upload").should_fail "FATAL: You must specify at least one argument. If you want to upload everything in this directory, run \"knife upload .\"\n", stdout: /USAGE/
282
+ end
283
+ end
284
+ end
285
+ end
286
+
287
+ when_the_chef_server "is empty" do
288
+ when_the_repository "has a data bag item" do
289
+
290
+ before do
291
+ file "data_bags/x/y.json", { "foo" => "bar" }
292
+ end
293
+
294
+ it "knife upload of the data bag uploads only the values in the data bag item and no other" do
295
+ knife("upload /data_bags/x/y.json").should_succeed <<~EOM
296
+ Created /data_bags/x
297
+ Created /data_bags/x/y.json
298
+ EOM
299
+ knife("diff --name-status /data_bags").should_succeed <<~EOM
300
+ EOM
301
+ expect(Chef::JSONCompat.parse(knife("raw /data/x/y").stdout, create_additions: false).keys.sort).to eq(%w{foo id})
302
+ end
303
+
304
+ it "knife upload /data_bags/x /data_bags/x/y.json uploads x once" do
305
+ knife("upload /data_bags/x /data_bags/x/y.json").should_succeed <<~EOM
306
+ Created /data_bags/x
307
+ Created /data_bags/x/y.json
308
+ EOM
309
+ end
310
+ end
311
+
312
+ when_the_repository "has a data bag item with keys chef_type and data_bag" do
313
+
314
+ before do
315
+ file "data_bags/x/y.json", { "chef_type" => "aaa", "data_bag" => "bbb" }
316
+ end
317
+
318
+ it "upload preserves chef_type and data_bag" do
319
+ knife("upload /data_bags/x/y.json").should_succeed <<~EOM
320
+ Created /data_bags/x
321
+ Created /data_bags/x/y.json
322
+ EOM
323
+ knife("diff --name-status /data_bags").should_succeed ""
324
+ result = Chef::JSONCompat.parse(knife("raw /data/x/y").stdout, create_additions: false)
325
+ expect(result.keys.sort).to eq(%w{chef_type data_bag id})
326
+ expect(result["chef_type"]).to eq("aaa")
327
+ expect(result["data_bag"]).to eq("bbb")
328
+ end
329
+ end
330
+
331
+ # Test upload of an item when the other end doesn't even have the container
332
+ when_the_repository "has two data bag items" do
333
+ before do
334
+ file "data_bags/x/y.json", {}
335
+ file "data_bags/x/z.json", {}
336
+ end
337
+ it "knife upload of one data bag item itself succeeds" do
338
+ knife("upload /data_bags/x/y.json").should_succeed <<~EOM
339
+ Created /data_bags/x
340
+ Created /data_bags/x/y.json
341
+ EOM
342
+ knife("diff --name-status /data_bags").should_succeed <<~EOM
343
+ A\t/data_bags/x/z.json
344
+ EOM
345
+ end
346
+ end
347
+ end
348
+
349
+ when_the_chef_server "has three data bag items" do
350
+
351
+ before do
352
+ data_bag "x", { "deleted" => {}, "modified" => {}, "unmodified" => {} }
353
+ end
354
+
355
+ when_the_repository "has a modified, unmodified, added and deleted data bag item" do
356
+ before do
357
+ file "data_bags/x/added.json", {}
358
+ file "data_bags/x/modified.json", { "foo" => "bar" }
359
+ file "data_bags/x/unmodified.json", {}
360
+ end
361
+
362
+ it "knife upload of the modified file succeeds" do
363
+ knife("upload /data_bags/x/modified.json").should_succeed <<~EOM
364
+ Updated /data_bags/x/modified.json
365
+ EOM
366
+ knife("diff --name-status /data_bags").should_succeed <<~EOM
367
+ D\t/data_bags/x/deleted.json
368
+ A\t/data_bags/x/added.json
369
+ EOM
370
+ end
371
+ it "knife upload of the unmodified file does nothing" do
372
+ knife("upload /data_bags/x/unmodified.json").should_succeed ""
373
+ knife("diff --name-status /data_bags").should_succeed <<~EOM
374
+ D\t/data_bags/x/deleted.json
375
+ M\t/data_bags/x/modified.json
376
+ A\t/data_bags/x/added.json
377
+ EOM
378
+ end
379
+ it "knife upload of the added file succeeds" do
380
+ knife("upload /data_bags/x/added.json").should_succeed <<~EOM
381
+ Created /data_bags/x/added.json
382
+ EOM
383
+ knife("diff --name-status /data_bags").should_succeed <<~EOM
384
+ D\t/data_bags/x/deleted.json
385
+ M\t/data_bags/x/modified.json
386
+ EOM
387
+ end
388
+ it "knife upload of the deleted file does nothing" do
389
+ knife("upload /data_bags/x/deleted.json").should_succeed ""
390
+ knife("diff --name-status /data_bags").should_succeed <<~EOM
391
+ D\t/data_bags/x/deleted.json
392
+ M\t/data_bags/x/modified.json
393
+ A\t/data_bags/x/added.json
394
+ EOM
395
+ end
396
+ it "knife upload --purge of the deleted file deletes it" do
397
+ knife("upload --purge /data_bags/x/deleted.json").should_succeed <<~EOM
398
+ Deleted extra entry /data_bags/x/deleted.json (purge is on)
399
+ EOM
400
+ knife("diff --name-status /data_bags").should_succeed <<~EOM
401
+ M\t/data_bags/x/modified.json
402
+ A\t/data_bags/x/added.json
403
+ EOM
404
+ end
405
+ it "knife upload of the entire data bag uploads everything" do
406
+ knife("upload /data_bags/x").should_succeed <<~EOM
407
+ Created /data_bags/x/added.json
408
+ Updated /data_bags/x/modified.json
409
+ EOM
410
+ knife("diff --name-status /data_bags").should_succeed <<~EOM
411
+ D\t/data_bags/x/deleted.json
412
+ EOM
413
+ end
414
+ it "knife upload --purge of the entire data bag uploads everything" do
415
+ knife("upload --purge /data_bags/x").should_succeed <<~EOM
416
+ Created /data_bags/x/added.json
417
+ Updated /data_bags/x/modified.json
418
+ Deleted extra entry /data_bags/x/deleted.json (purge is on)
419
+ EOM
420
+ knife("diff --name-status /data_bags").should_succeed ""
421
+ end
422
+ context "when cwd is the /data_bags directory" do
423
+
424
+ before do
425
+ cwd "data_bags"
426
+ end
427
+
428
+ it "knife upload fails" do
429
+ knife("upload").should_fail "FATAL: You must specify at least one argument. If you want to upload everything in this directory, run \"knife upload .\"\n", stdout: /USAGE/
430
+ end
431
+
432
+ it "knife upload --purge . uploads everything" do
433
+ knife("upload --purge .").should_succeed <<~EOM
434
+ Created x/added.json
435
+ Updated x/modified.json
436
+ Deleted extra entry x/deleted.json (purge is on)
437
+ EOM
438
+ knife("diff --name-status /data_bags").should_succeed ""
439
+ end
440
+ it "knife upload --purge * uploads everything" do
441
+ knife("upload --purge *").should_succeed <<~EOM
442
+ Created x/added.json
443
+ Updated x/modified.json
444
+ Deleted extra entry x/deleted.json (purge is on)
445
+ EOM
446
+ knife("diff --name-status /data_bags").should_succeed ""
447
+ end
448
+ end
449
+ end
450
+ end
451
+
452
+ # Cookbook upload is a funny thing ... direct cookbook upload works, but
453
+ # upload of a file is designed not to work at present. Make sure that is the
454
+ # case.
455
+ when_the_chef_server "has a cookbook" do
456
+ before do
457
+ cookbook "x", "1.0.0", { "z.rb" => "" }
458
+ end
459
+
460
+ when_the_repository "does not have metadata file" do
461
+ before do
462
+ file "cookbooks/x/y.rb", "hi"
463
+ end
464
+
465
+ it "raises MetadataNotFound exception" do
466
+ expect { knife("upload /cookbooks/x") }.to raise_error(Chef::Exceptions::MetadataNotFound)
467
+ end
468
+ end
469
+
470
+ when_the_repository "does not have valid metadata" do
471
+ before do
472
+ file "cookbooks/x/metadata.rb", cb_metadata(nil, "1.0.0")
473
+ end
474
+
475
+ it "raises exception for invalid metadata" do
476
+ expect { knife("upload /cookbooks/x") }.to raise_error(Chef::Exceptions::MetadataNotValid)
477
+ end
478
+ end
479
+
480
+ when_the_repository "has a modified, extra and missing file for the cookbook" do
481
+ before do
482
+ file "cookbooks/x/metadata.rb", cb_metadata("x", "1.0.0", "#modified")
483
+ file "cookbooks/x/y.rb", "hi"
484
+ end
485
+
486
+ it "knife upload of any individual file fails" do
487
+ knife("upload /cookbooks/x/metadata.rb").should_fail "ERROR: /cookbooks/x/metadata.rb cannot be updated.\n"
488
+ knife("upload /cookbooks/x/y.rb").should_fail "ERROR: /cookbooks/x cannot have a child created under it.\n"
489
+ knife("upload --purge /cookbooks/x/z.rb").should_fail "ERROR: /cookbooks/x/z.rb cannot be deleted.\n"
490
+ end
491
+
492
+ # TODO this is a bit of an inconsistency: if we didn't specify --purge,
493
+ # technically we shouldn't have deleted missing files. But ... cookbooks
494
+ # are a special case.
495
+ it "knife upload of the cookbook itself succeeds" do
496
+ knife("upload /cookbooks/x").should_succeed <<~EOM
497
+ Updated /cookbooks/x
498
+ EOM
499
+ knife("diff --name-status /cookbooks").should_succeed <<~EOM
500
+ D\t/cookbooks/x/metadata.json
501
+ EOM
502
+ end
503
+
504
+ it "knife upload --purge of the cookbook itself succeeds" do
505
+ knife("upload /cookbooks/x").should_succeed <<~EOM
506
+ Updated /cookbooks/x
507
+ EOM
508
+ knife("diff --name-status /cookbooks").should_succeed <<~EOM
509
+ D\t/cookbooks/x/metadata.json
510
+ EOM
511
+ end
512
+ end
513
+ when_the_repository "has a missing file for the cookbook" do
514
+
515
+ before do
516
+ file "cookbooks/x/metadata.rb", cb_metadata("x", "1.0.0")
517
+ end
518
+
519
+ it "knife upload of the cookbook succeeds" do
520
+ knife("upload /cookbooks/x").should_succeed <<~EOM
521
+ Updated /cookbooks/x
522
+ EOM
523
+ knife("diff --name-status /cookbooks").should_succeed <<~EOM
524
+ D\t/cookbooks/x/metadata.json
525
+ EOM
526
+ end
527
+ end
528
+ when_the_repository "has an extra file for the cookbook" do
529
+
530
+ before do
531
+ file "cookbooks/x/metadata.rb", cb_metadata("x", "1.0.0")
532
+ file "cookbooks/x/z.rb", ""
533
+ file "cookbooks/x/blah.rb", ""
534
+ end
535
+
536
+ it "knife upload of the cookbook succeeds" do
537
+ knife("upload /cookbooks/x").should_succeed <<~EOM
538
+ Updated /cookbooks/x
539
+ EOM
540
+ knife("diff --name-status /cookbooks").should_succeed <<~EOM
541
+ D\t/cookbooks/x/metadata.json
542
+ EOM
543
+ end
544
+ end
545
+
546
+ when_the_repository "has a different file in the cookbook" do
547
+ before do
548
+ file "cookbooks/x/metadata.rb", cb_metadata("x", "1.0.0")
549
+ end
550
+
551
+ it "knife upload --freeze freezes the cookbook" do
552
+ knife("upload --freeze /cookbooks/x").should_succeed <<~EOM
553
+ Updated /cookbooks/x
554
+ EOM
555
+ # Modify a file and attempt to upload
556
+ file "cookbooks/x/metadata.rb", 'name "x"; version "1.0.0"#different'
557
+ knife("upload /cookbooks/x").should_fail "ERROR: /cookbooks failed to write: Cookbook x is frozen\n"
558
+ end
559
+ end
560
+ end
561
+
562
+ when_the_chef_server "has a frozen cookbook" do
563
+ before do
564
+ cookbook "frozencook", "1.0.0", {}, frozen: true
565
+ end
566
+
567
+ when_the_repository "has an update to said cookbook" do
568
+
569
+ before do
570
+ file "cookbooks/frozencook/metadata.rb", cb_metadata("frozencook", "1.0.0", "# This is different")
571
+ end
572
+
573
+ it "knife upload fails to upload the frozen cookbook" do
574
+ knife("upload /cookbooks/frozencook").should_fail "ERROR: /cookbooks failed to write: Cookbook frozencook is frozen\n"
575
+ end
576
+ it "knife upload --force uploads the frozen cookbook" do
577
+ knife("upload --force /cookbooks/frozencook").should_succeed <<~EOM
578
+ Updated /cookbooks/frozencook
579
+ EOM
580
+ end
581
+ end
582
+ end
583
+
584
+ when_the_repository "has a cookbook" do
585
+ before do
586
+ file "cookbooks/x/metadata.rb", cb_metadata("x", "1.0.0")
587
+ file "cookbooks/x/metadata.json", { name: "x", version: "1.0.0" }
588
+ file "cookbooks/x/onlyin1.0.0.rb", "old_text"
589
+ end
590
+
591
+ when_the_chef_server "has a later version for the cookbook" do
592
+ before do
593
+ cookbook "x", "1.0.0", { "onlyin1.0.0.rb" => "" }
594
+ cookbook "x", "1.0.1", { "onlyin1.0.1.rb" => "hi" }
595
+ end
596
+
597
+ it "knife upload /cookbooks/x uploads the local version" do
598
+ knife("diff --name-status /cookbooks").should_succeed <<~EOM
599
+ M\t/cookbooks/x/metadata.rb
600
+ D\t/cookbooks/x/onlyin1.0.1.rb
601
+ A\t/cookbooks/x/metadata.json
602
+ A\t/cookbooks/x/onlyin1.0.0.rb
603
+ EOM
604
+ knife("upload --purge /cookbooks/x").should_succeed <<~EOM
605
+ Updated /cookbooks/x
606
+ EOM
607
+ knife("diff --name-status /cookbooks").should_succeed <<~EOM
608
+ M\t/cookbooks/x/metadata.rb
609
+ D\t/cookbooks/x/onlyin1.0.1.rb
610
+ A\t/cookbooks/x/metadata.json
611
+ A\t/cookbooks/x/onlyin1.0.0.rb
612
+ EOM
613
+ end
614
+ end
615
+ end
616
+
617
+ when_the_repository "has a cookbook" do
618
+ before do
619
+ file "cookbooks/x/metadata.rb", cb_metadata("x", "1.0.0")
620
+ file "cookbooks/x/onlyin1.0.0.rb", "old_text"
621
+ end
622
+
623
+ when_the_chef_server "has a later version for the cookbook" do
624
+ before do
625
+ cookbook "x", "1.0.0", { "onlyin1.0.0.rb" => "" }
626
+ cookbook "x", "1.0.1", { "onlyin1.0.1.rb" => "hi" }
627
+ end
628
+
629
+ it "knife upload /cookbooks/x uploads the local version and generates metadata.json from metadata.rb and uploads it." do
630
+ knife("diff --name-status /cookbooks").should_succeed <<~EOM
631
+ M\t/cookbooks/x/metadata.rb
632
+ D\t/cookbooks/x/onlyin1.0.1.rb
633
+ A\t/cookbooks/x/onlyin1.0.0.rb
634
+ EOM
635
+ knife("upload --purge /cookbooks/x").should_succeed <<~EOM
636
+ Updated /cookbooks/x
637
+ EOM
638
+ knife("diff --name-status /cookbooks").should_succeed <<~EOM
639
+ M\t/cookbooks/x/metadata.rb
640
+ D\t/cookbooks/x/onlyin1.0.1.rb
641
+ A\t/cookbooks/x/onlyin1.0.0.rb
642
+ EOM
643
+ end
644
+ end
645
+
646
+ when_the_chef_server "has an earlier version for the cookbook" do
647
+ before do
648
+ cookbook "x", "1.0.0", { "onlyin1.0.0.rb" => "" }
649
+ cookbook "x", "0.9.9", { "onlyin0.9.9.rb" => "hi" }
650
+ end
651
+
652
+ it "knife upload /cookbooks/x uploads the local version generates metadata.json and uploads it." do
653
+ knife("upload --purge /cookbooks/x").should_succeed <<~EOM
654
+ Updated /cookbooks/x
655
+ EOM
656
+ knife("diff --name-status /cookbooks").should_succeed <<~EOM
657
+ D\t/cookbooks/x/metadata.json
658
+ EOM
659
+ end
660
+ end
661
+
662
+ when_the_chef_server "has a later version for the cookbook, and no current version" do
663
+ before do
664
+ cookbook "x", "1.0.1", { "onlyin1.0.1.rb" => "hi" }
665
+ end
666
+
667
+ it "knife upload /cookbooks/x uploads the local version and generates metadata.json before upload and uploads it." do
668
+ knife("diff --name-status /cookbooks").should_succeed <<~EOM
669
+ M\t/cookbooks/x/metadata.rb
670
+ D\t/cookbooks/x/onlyin1.0.1.rb
671
+ A\t/cookbooks/x/onlyin1.0.0.rb
672
+ EOM
673
+ knife("upload --purge /cookbooks/x").should_succeed <<~EOM
674
+ Updated /cookbooks/x
675
+ EOM
676
+ knife("diff --name-status /cookbooks").should_succeed <<~EOM
677
+ M\t/cookbooks/x/metadata.rb
678
+ D\t/cookbooks/x/onlyin1.0.1.rb
679
+ A\t/cookbooks/x/onlyin1.0.0.rb
680
+ EOM
681
+ end
682
+ end
683
+
684
+ when_the_chef_server "has an earlier version for the cookbook, and no current version" do
685
+ before do
686
+ cookbook "x", "0.9.9", { "onlyin0.9.9.rb" => "hi" }
687
+ end
688
+
689
+ it "knife upload /cookbooks/x uploads the new version" do
690
+ knife("upload --purge /cookbooks/x").should_succeed <<~EOM
691
+ Updated /cookbooks/x
692
+ EOM
693
+ knife("diff --name-status /cookbooks").should_succeed <<~EOM
694
+ D\t/cookbooks/x/metadata.json
695
+ EOM
696
+ end
697
+ end
698
+ end
699
+
700
+ when_the_chef_server "has an environment" do
701
+ before do
702
+ environment "x", {}
703
+ end
704
+
705
+ when_the_repository "has an environment with bad JSON" do
706
+ before do
707
+ file "environments/x.json", "{"
708
+ end
709
+
710
+ it "knife upload tries and fails" do
711
+ error1 = <<~EOH
712
+ WARN: Parse error reading #{path_to("environments/x.json")} as JSON: parse error: premature EOF
713
+ {
714
+ (right here) ------^
715
+
716
+ ERROR: /environments/x.json failed to write: Parse error reading JSON: parse error: premature EOF
717
+ {
718
+ (right here) ------^
719
+ EOH
720
+
721
+ warn = <<~EOH
722
+ WARN: Parse error reading #{path_to("environments/x.json")} as JSON: parse error: premature EOF
723
+ {
724
+ (right here) ------^
725
+
726
+ EOH
727
+ knife("upload /environments/x.json").should_fail(error1)
728
+ knife("diff --name-status /environments/x.json").should_succeed("M\t/environments/x.json\n", stderr: warn)
729
+ end
730
+ end
731
+
732
+ when_the_repository "has the same environment with the wrong name in the file" do
733
+ before do
734
+ file "environments/x.json", { "name" => "y" }
735
+ end
736
+ it "knife upload fails" do
737
+ knife("upload /environments/x.json").should_fail "ERROR: /environments/x.json failed to write: Name must be 'x' (is 'y')\n"
738
+ knife("diff --name-status /environments/x.json").should_succeed "M\t/environments/x.json\n"
739
+ end
740
+ end
741
+
742
+ when_the_repository "has the same environment with no name in the file" do
743
+ before do
744
+ file "environments/x.json", { "description" => "hi" }
745
+ end
746
+ it "knife upload succeeds" do
747
+ knife("upload /environments/x.json").should_succeed "Updated /environments/x.json\n"
748
+ knife("diff --name-status /environments/x.json").should_succeed ""
749
+ end
750
+ end
751
+ end
752
+
753
+ when_the_chef_server "is empty" do
754
+
755
+ when_the_repository "has an environment with the wrong name in the file" do
756
+ before do
757
+ file "environments/x.json", { "name" => "y" }
758
+ end
759
+ it "knife upload fails" do
760
+ knife("upload /environments/x.json").should_fail "ERROR: /environments failed to create_child: Error creating 'x.json': Name must be 'x' (is 'y')\n"
761
+ knife("diff --name-status /environments/x.json").should_succeed "A\t/environments/x.json\n"
762
+ end
763
+ end
764
+
765
+ when_the_repository "has an environment with no name in the file" do
766
+
767
+ before do
768
+ file "environments/x.json", { "description" => "hi" }
769
+ end
770
+ it "knife upload succeeds" do
771
+ knife("upload /environments/x.json").should_succeed "Created /environments/x.json\n"
772
+ knife("diff --name-status /environments/x.json").should_succeed ""
773
+ end
774
+ end
775
+
776
+ when_the_repository "has a data bag with no id in the file" do
777
+ before do
778
+ file "data_bags/bag/x.json", { "foo" => "bar" }
779
+ end
780
+ it "knife upload succeeds" do
781
+ knife("upload /data_bags/bag/x.json").should_succeed "Created /data_bags/bag\nCreated /data_bags/bag/x.json\n"
782
+ knife("diff --name-status /data_bags/bag/x.json").should_succeed ""
783
+ end
784
+ end
785
+ end
786
+ when_the_chef_server "is empty" do
787
+ when_the_repository "has a cookbook with an invalid chef_version constraint in it" do
788
+ before do
789
+ file "cookbooks/x/metadata.rb", cb_metadata("x", "1.0.0", "\nchef_version '~> 999.0'")
790
+ end
791
+ it "knife upload succeeds" do
792
+ knife("upload /cookbooks/x").should_succeed <<~EOM
793
+ Created /cookbooks/x
794
+ EOM
795
+ knife("diff --name-status /cookbooks").should_succeed <<~EOM
796
+ D\t/cookbooks/x/metadata.json
797
+ EOM
798
+ end
799
+ end
800
+ end
801
+ end # without versioned cookbooks
802
+
803
+ context "with versioned cookbooks" do
804
+ before { Chef::Config[:versioned_cookbooks] = true }
805
+
806
+ when_the_chef_server "has one of each thing" do
807
+
808
+ before do
809
+ client "x", {}
810
+ cookbook "x", "1.0.0"
811
+ data_bag "x", { "y" => {} }
812
+ environment "x", {}
813
+ node "x", {}
814
+ role "x", {}
815
+ user "x", {}
816
+ end
817
+
818
+ when_the_repository "has only top-level directories" do
819
+ before do
820
+ directory "clients"
821
+ directory "cookbooks"
822
+ directory "data_bags"
823
+ directory "environments"
824
+ directory "nodes"
825
+ directory "roles"
826
+ directory "users"
827
+ end
828
+
829
+ it "knife upload does nothing" do
830
+ knife("upload /").should_succeed ""
831
+ knife("diff --name-status /").should_succeed <<~EOM
832
+ D\t/clients/chef-validator.json
833
+ D\t/clients/chef-webui.json
834
+ D\t/clients/x.json
835
+ D\t/cookbooks/x-1.0.0
836
+ D\t/data_bags/x
837
+ D\t/environments/_default.json
838
+ D\t/environments/x.json
839
+ D\t/nodes/x.json
840
+ D\t/roles/x.json
841
+ D\t/users/admin.json
842
+ D\t/users/x.json
843
+ EOM
844
+ end
845
+
846
+ it "knife upload --purge deletes everything" do
847
+ knife("upload --purge /").should_succeed(<<~EOM, stderr: "WARNING: /environments/_default.json cannot be deleted (default environment cannot be modified).\n")
848
+ Deleted extra entry /clients/chef-validator.json (purge is on)
849
+ Deleted extra entry /clients/chef-webui.json (purge is on)
850
+ Deleted extra entry /clients/x.json (purge is on)
851
+ Deleted extra entry /cookbooks/x-1.0.0 (purge is on)
852
+ Deleted extra entry /data_bags/x (purge is on)
853
+ Deleted extra entry /environments/x.json (purge is on)
854
+ Deleted extra entry /nodes/x.json (purge is on)
855
+ Deleted extra entry /roles/x.json (purge is on)
856
+ Deleted extra entry /users/admin.json (purge is on)
857
+ Deleted extra entry /users/x.json (purge is on)
858
+ EOM
859
+ knife("diff --name-status /").should_succeed <<~EOM
860
+ D\t/environments/_default.json
861
+ EOM
862
+ end
863
+ end
864
+
865
+ when_the_repository "has an identical copy of each thing" do
866
+ before do
867
+ file "clients/chef-validator.json", { "validator" => true, "public_key" => ChefZero::PUBLIC_KEY }
868
+ file "clients/chef-webui.json", { "admin" => true, "public_key" => ChefZero::PUBLIC_KEY }
869
+ file "clients/x.json", { "public_key" => ChefZero::PUBLIC_KEY }
870
+ file "cookbooks/x-1.0.0/metadata.rb", cb_metadata("x", "1.0.0")
871
+ file "data_bags/x/y.json", {}
872
+ file "environments/_default.json", { "description" => "The default Chef environment" }
873
+ file "environments/x.json", {}
874
+ file "nodes/x.json", { "normal" => { "tags" => [] } }
875
+ file "roles/x.json", {}
876
+ file "users/admin.json", { "admin" => true, "public_key" => ChefZero::PUBLIC_KEY }
877
+ file "users/x.json", { "public_key" => ChefZero::PUBLIC_KEY }
878
+ end
879
+
880
+ it "knife upload makes no changes" do
881
+ knife("upload /cookbooks/x-1.0.0").should_succeed ""
882
+ knife("diff --name-status /").should_succeed ""
883
+ end
884
+
885
+ it "knife upload --purge makes no changes" do
886
+ knife("upload --purge /").should_succeed ""
887
+ knife("diff --name-status /").should_succeed ""
888
+ end
889
+
890
+ context "except the role file" do
891
+ before do
892
+ file "roles/x.json", { "description" => "blarghle" }
893
+ end
894
+
895
+ it "knife upload changes the role" do
896
+ knife("upload /").should_succeed "Updated /roles/x.json\n"
897
+ knife("diff --name-status /").should_succeed ""
898
+ end
899
+ end
900
+
901
+ context "except the role file is textually different, but not ACTUALLY different" do
902
+
903
+ before do
904
+ file "roles/x.json", <<~EOM
905
+ {
906
+ "chef_type": "role",
907
+ "default_attributes": {
908
+ },
909
+ "env_run_lists": {
910
+ },
911
+ "json_class": "Chef::Role",
912
+ "name": "x",
913
+ "description": "",
914
+ "override_attributes": {
915
+ },
916
+ "run_list": [
917
+
918
+ ]
919
+ }
920
+ EOM
921
+ end
922
+
923
+ it "knife upload / does not change anything" do
924
+ knife("upload /").should_succeed ""
925
+ knife("diff --name-status /").should_succeed ""
926
+ end
927
+ end
928
+
929
+ context "as well as one extra copy of each thing" do
930
+ before do
931
+ file "clients/y.json", { "public_key" => ChefZero::PUBLIC_KEY }
932
+ file "cookbooks/x-1.0.0/blah.rb", ""
933
+ file "cookbooks/x-2.0.0/metadata.rb", cb_metadata("x", "2.0.0")
934
+ file "cookbooks/y-1.0.0/metadata.rb", cb_metadata("y", "1.0.0")
935
+ file "data_bags/x/z.json", {}
936
+ file "data_bags/y/zz.json", {}
937
+ file "environments/y.json", {}
938
+ file "nodes/y.json", {}
939
+ file "roles/y.json", {}
940
+ file "users/y.json", { "public_key" => ChefZero::PUBLIC_KEY }
941
+ end
942
+
943
+ it "knife upload adds the new files" do
944
+ knife("upload /").should_succeed <<~EOM
945
+ Created /clients/y.json
946
+ Updated /cookbooks/x-1.0.0
947
+ Created /cookbooks/x-2.0.0
948
+ Created /cookbooks/y-1.0.0
949
+ Created /data_bags/x/z.json
950
+ Created /data_bags/y
951
+ Created /data_bags/y/zz.json
952
+ Created /environments/y.json
953
+ Created /nodes/y.json
954
+ Created /roles/y.json
955
+ Created /users/y.json
956
+ EOM
957
+ knife("diff --name-status /").should_succeed ""
958
+ end
959
+ end
960
+ end
961
+
962
+ when_the_repository "is empty" do
963
+ it "knife upload does nothing" do
964
+ knife("upload /").should_succeed ""
965
+ knife("diff --name-status /").should_succeed <<~EOM
966
+ D\t/clients
967
+ D\t/cookbooks
968
+ D\t/data_bags
969
+ D\t/environments
970
+ D\t/nodes
971
+ D\t/roles
972
+ D\t/users
973
+ EOM
974
+ end
975
+
976
+ it "knife upload --purge deletes nothing" do
977
+ knife("upload --purge /").should_fail <<~EOM
978
+ ERROR: /clients cannot be deleted.
979
+ ERROR: /cookbooks cannot be deleted.
980
+ ERROR: /data_bags cannot be deleted.
981
+ ERROR: /environments cannot be deleted.
982
+ ERROR: /nodes cannot be deleted.
983
+ ERROR: /roles cannot be deleted.
984
+ ERROR: /users cannot be deleted.
985
+ EOM
986
+ knife("diff --name-status /").should_succeed <<~EOM
987
+ D\t/clients
988
+ D\t/cookbooks
989
+ D\t/data_bags
990
+ D\t/environments
991
+ D\t/nodes
992
+ D\t/roles
993
+ D\t/users
994
+ EOM
995
+ end
996
+
997
+ context "when current directory is top level" do
998
+ before do
999
+ cwd "."
1000
+ end
1001
+ it "knife upload with no parameters reports an error" do
1002
+ knife("upload").should_fail "FATAL: You must specify at least one argument. If you want to upload everything in this directory, run \"knife upload .\"\n", stdout: /USAGE/
1003
+ end
1004
+ end
1005
+ end
1006
+ end
1007
+
1008
+ # Test upload of an item when the other end doesn't even have the container
1009
+ when_the_chef_server "is empty" do
1010
+ when_the_repository "has two data bag items" do
1011
+ before do
1012
+ file "data_bags/x/y.json", {}
1013
+ file "data_bags/x/z.json", {}
1014
+ end
1015
+
1016
+ it "knife upload of one data bag item itself succeeds" do
1017
+ knife("upload /data_bags/x/y.json").should_succeed <<~EOM
1018
+ Created /data_bags/x
1019
+ Created /data_bags/x/y.json
1020
+ EOM
1021
+ knife("diff --name-status /data_bags").should_succeed <<~EOM
1022
+ A\t/data_bags/x/z.json
1023
+ EOM
1024
+ end
1025
+ end
1026
+ end
1027
+
1028
+ when_the_chef_server "has three data bag items" do
1029
+ before do
1030
+ data_bag "x", { "deleted" => {}, "modified" => {}, "unmodified" => {} }
1031
+ end
1032
+ when_the_repository "has a modified, unmodified, added and deleted data bag item" do
1033
+ before do
1034
+ file "data_bags/x/added.json", {}
1035
+ file "data_bags/x/modified.json", { "foo" => "bar" }
1036
+ file "data_bags/x/unmodified.json", {}
1037
+ end
1038
+
1039
+ it "knife upload of the modified file succeeds" do
1040
+ knife("upload /data_bags/x/modified.json").should_succeed <<~EOM
1041
+ Updated /data_bags/x/modified.json
1042
+ EOM
1043
+ knife("diff --name-status /data_bags").should_succeed <<~EOM
1044
+ D\t/data_bags/x/deleted.json
1045
+ A\t/data_bags/x/added.json
1046
+ EOM
1047
+ end
1048
+ it "knife upload of the unmodified file does nothing" do
1049
+ knife("upload /data_bags/x/unmodified.json").should_succeed ""
1050
+ knife("diff --name-status /data_bags").should_succeed <<~EOM
1051
+ D\t/data_bags/x/deleted.json
1052
+ M\t/data_bags/x/modified.json
1053
+ A\t/data_bags/x/added.json
1054
+ EOM
1055
+ end
1056
+ it "knife upload of the added file succeeds" do
1057
+ knife("upload /data_bags/x/added.json").should_succeed <<~EOM
1058
+ Created /data_bags/x/added.json
1059
+ EOM
1060
+ knife("diff --name-status /data_bags").should_succeed <<~EOM
1061
+ D\t/data_bags/x/deleted.json
1062
+ M\t/data_bags/x/modified.json
1063
+ EOM
1064
+ end
1065
+ it "knife upload of the deleted file does nothing" do
1066
+ knife("upload /data_bags/x/deleted.json").should_succeed ""
1067
+ knife("diff --name-status /data_bags").should_succeed <<~EOM
1068
+ D\t/data_bags/x/deleted.json
1069
+ M\t/data_bags/x/modified.json
1070
+ A\t/data_bags/x/added.json
1071
+ EOM
1072
+ end
1073
+ it "knife upload --purge of the deleted file deletes it" do
1074
+ knife("upload --purge /data_bags/x/deleted.json").should_succeed <<~EOM
1075
+ Deleted extra entry /data_bags/x/deleted.json (purge is on)
1076
+ EOM
1077
+ knife("diff --name-status /data_bags").should_succeed <<~EOM
1078
+ M\t/data_bags/x/modified.json
1079
+ A\t/data_bags/x/added.json
1080
+ EOM
1081
+ end
1082
+ it "knife upload of the entire data bag uploads everything" do
1083
+ knife("upload /data_bags/x").should_succeed <<~EOM
1084
+ Created /data_bags/x/added.json
1085
+ Updated /data_bags/x/modified.json
1086
+ EOM
1087
+ knife("diff --name-status /data_bags").should_succeed <<~EOM
1088
+ D\t/data_bags/x/deleted.json
1089
+ EOM
1090
+ end
1091
+ it "knife upload --purge of the entire data bag uploads everything" do
1092
+ knife("upload --purge /data_bags/x").should_succeed <<~EOM
1093
+ Created /data_bags/x/added.json
1094
+ Updated /data_bags/x/modified.json
1095
+ Deleted extra entry /data_bags/x/deleted.json (purge is on)
1096
+ EOM
1097
+ knife("diff --name-status /data_bags").should_succeed ""
1098
+ end
1099
+ context "when cwd is the /data_bags directory" do
1100
+ before do
1101
+ cwd "data_bags"
1102
+ end
1103
+ it "knife upload fails" do
1104
+ knife("upload").should_fail "FATAL: You must specify at least one argument. If you want to upload everything in this directory, run \"knife upload .\"\n", stdout: /USAGE/
1105
+ end
1106
+ it "knife upload --purge . uploads everything" do
1107
+ knife("upload --purge .").should_succeed <<~EOM
1108
+ Created x/added.json
1109
+ Updated x/modified.json
1110
+ Deleted extra entry x/deleted.json (purge is on)
1111
+ EOM
1112
+ knife("diff --name-status /data_bags").should_succeed ""
1113
+ end
1114
+ it "knife upload --purge * uploads everything" do
1115
+ knife("upload --purge *").should_succeed <<~EOM
1116
+ Created x/added.json
1117
+ Updated x/modified.json
1118
+ Deleted extra entry x/deleted.json (purge is on)
1119
+ EOM
1120
+ knife("diff --name-status /data_bags").should_succeed ""
1121
+ end
1122
+ end
1123
+ end
1124
+ end
1125
+
1126
+ # Cookbook upload is a funny thing ... direct cookbook upload works, but
1127
+ # upload of a file is designed not to work at present. Make sure that is the
1128
+ # case.
1129
+ when_the_chef_server "has a cookbook" do
1130
+ before do
1131
+ cookbook "x", "1.0.0", { "z.rb" => "" }
1132
+ end
1133
+
1134
+ when_the_repository "has a modified, extra and missing file for the cookbook" do
1135
+ before do
1136
+ file "cookbooks/x-1.0.0/metadata.rb", cb_metadata("x", "1.0.0", "#modified")
1137
+ file "cookbooks/x-1.0.0/y.rb", "hi"
1138
+ end
1139
+
1140
+ it "knife upload of any individual file fails" do
1141
+ knife("upload /cookbooks/x-1.0.0/metadata.rb").should_fail "ERROR: /cookbooks/x-1.0.0/metadata.rb cannot be updated.\n"
1142
+ knife("upload /cookbooks/x-1.0.0/y.rb").should_fail "ERROR: /cookbooks/x-1.0.0 cannot have a child created under it.\n"
1143
+ knife("upload --purge /cookbooks/x-1.0.0/z.rb").should_fail "ERROR: /cookbooks/x-1.0.0/z.rb cannot be deleted.\n"
1144
+ end
1145
+
1146
+ # TODO this is a bit of an inconsistency: if we didn't specify --purge,
1147
+ # technically we shouldn't have deleted missing files. But ... cookbooks
1148
+ # are a special case.
1149
+ it "knife upload of the cookbook itself succeeds" do
1150
+ knife("upload /cookbooks/x-1.0.0").should_succeed <<~EOM
1151
+ Updated /cookbooks/x-1.0.0
1152
+ EOM
1153
+ knife("diff --name-status /cookbooks").should_succeed ""
1154
+ end
1155
+
1156
+ it "knife upload --purge of the cookbook itself succeeds" do
1157
+ knife("upload /cookbooks/x-1.0.0").should_succeed <<~EOM
1158
+ Updated /cookbooks/x-1.0.0
1159
+ EOM
1160
+ knife("diff --name-status /cookbooks").should_succeed ""
1161
+ end
1162
+ end
1163
+
1164
+ when_the_repository "has a missing file for the cookbook" do
1165
+ before do
1166
+ file "cookbooks/x-1.0.0/metadata.rb", cb_metadata("x", "1.0.0")
1167
+ end
1168
+
1169
+ it "knife upload of the cookbook succeeds" do
1170
+ knife("upload /cookbooks/x-1.0.0").should_succeed <<~EOM
1171
+ Updated /cookbooks/x-1.0.0
1172
+ EOM
1173
+ knife("diff --name-status /cookbooks").should_succeed ""
1174
+ end
1175
+ end
1176
+
1177
+ when_the_repository "has an extra file for the cookbook" do
1178
+ before do
1179
+ file "cookbooks/x-1.0.0/metadata.rb", cb_metadata("x", "1.0.0")
1180
+ file "cookbooks/x-1.0.0/z.rb", ""
1181
+ file "cookbooks/x-1.0.0/blah.rb", ""
1182
+ end
1183
+
1184
+ it "knife upload of the cookbook succeeds" do
1185
+ knife("upload /cookbooks/x-1.0.0").should_succeed <<~EOM
1186
+ Updated /cookbooks/x-1.0.0
1187
+ EOM
1188
+ knife("diff --name-status /cookbooks").should_succeed ""
1189
+ end
1190
+ end
1191
+ end
1192
+
1193
+ when_the_repository "has a cookbook" do
1194
+ before do
1195
+ file "cookbooks/x-1.0.0/metadata.rb", cb_metadata("x", "1.0.0")
1196
+ file "cookbooks/x-1.0.0/onlyin1.0.0.rb", "old_text"
1197
+ end
1198
+
1199
+ when_the_chef_server "has a later version for the cookbook" do
1200
+ before do
1201
+ cookbook "x", "1.0.0", { "onlyin1.0.0.rb" => "" }
1202
+ cookbook "x", "1.0.1", { "onlyin1.0.1.rb" => "hi" }
1203
+ end
1204
+
1205
+ it "knife upload /cookbooks uploads the local version" do
1206
+ knife("diff --name-status /cookbooks").should_succeed <<~EOM
1207
+ M\t/cookbooks/x-1.0.0/onlyin1.0.0.rb
1208
+ D\t/cookbooks/x-1.0.1
1209
+ EOM
1210
+ knife("upload --purge /cookbooks").should_succeed <<~EOM
1211
+ Updated /cookbooks/x-1.0.0
1212
+ Deleted extra entry /cookbooks/x-1.0.1 (purge is on)
1213
+ EOM
1214
+ knife("diff --name-status /cookbooks").should_succeed ""
1215
+ end
1216
+ end
1217
+
1218
+ when_the_chef_server "has an earlier version for the cookbook" do
1219
+ before do
1220
+ cookbook "x", "1.0.0", { "onlyin1.0.0.rb" => "" }
1221
+ cookbook "x", "0.9.9", { "onlyin0.9.9.rb" => "hi" }
1222
+ end
1223
+ it "knife upload /cookbooks uploads the local version" do
1224
+ knife("upload --purge /cookbooks").should_succeed <<~EOM
1225
+ Updated /cookbooks/x-1.0.0
1226
+ Deleted extra entry /cookbooks/x-0.9.9 (purge is on)
1227
+ EOM
1228
+ knife("diff --name-status /cookbooks").should_succeed ""
1229
+ end
1230
+ end
1231
+
1232
+ when_the_chef_server "has a later version for the cookbook, and no current version" do
1233
+ before do
1234
+ cookbook "x", "1.0.1", { "onlyin1.0.1.rb" => "hi" }
1235
+ end
1236
+
1237
+ it "knife upload /cookbooks/x uploads the local version" do
1238
+ knife("diff --name-status /cookbooks").should_succeed <<~EOM
1239
+ D\t/cookbooks/x-1.0.1
1240
+ A\t/cookbooks/x-1.0.0
1241
+ EOM
1242
+ knife("upload --purge /cookbooks").should_succeed <<~EOM
1243
+ Created /cookbooks/x-1.0.0
1244
+ Deleted extra entry /cookbooks/x-1.0.1 (purge is on)
1245
+ EOM
1246
+ knife("diff --name-status /cookbooks").should_succeed ""
1247
+ end
1248
+ end
1249
+
1250
+ when_the_chef_server "has an earlier version for the cookbook, and no current version" do
1251
+ before do
1252
+ cookbook "x", "0.9.9", { "onlyin0.9.9.rb" => "hi" }
1253
+ end
1254
+
1255
+ it "knife upload /cookbooks/x uploads the new version" do
1256
+ knife("upload --purge /cookbooks").should_succeed <<~EOM
1257
+ Created /cookbooks/x-1.0.0
1258
+ Deleted extra entry /cookbooks/x-0.9.9 (purge is on)
1259
+ EOM
1260
+ knife("diff --name-status /cookbooks").should_succeed ""
1261
+ end
1262
+ end
1263
+ end
1264
+
1265
+ when_the_chef_server "has an environment" do
1266
+ before do
1267
+ environment "x", {}
1268
+ end
1269
+
1270
+ when_the_repository "has the same environment with the wrong name in the file" do
1271
+ before do
1272
+ file "environments/x.json", { "name" => "y" }
1273
+ end
1274
+ it "knife upload fails" do
1275
+ knife("upload /environments/x.json").should_fail "ERROR: /environments/x.json failed to write: Name must be 'x' (is 'y')\n"
1276
+ knife("diff --name-status /environments/x.json").should_succeed "M\t/environments/x.json\n"
1277
+ end
1278
+ end
1279
+
1280
+ when_the_repository "has the same environment with no name in the file" do
1281
+ before do
1282
+ file "environments/x.json", { "description" => "hi" }
1283
+ end
1284
+ it "knife upload succeeds" do
1285
+ knife("upload /environments/x.json").should_succeed "Updated /environments/x.json\n"
1286
+ knife("diff --name-status /environments/x.json").should_succeed ""
1287
+ end
1288
+ end
1289
+ end
1290
+
1291
+ when_the_chef_server "is empty" do
1292
+
1293
+ when_the_repository "has an environment with the wrong name in the file" do
1294
+ before do
1295
+ file "environments/x.json", { "name" => "y" }
1296
+ end
1297
+ it "knife upload fails" do
1298
+ knife("upload /environments/x.json").should_fail "ERROR: /environments failed to create_child: Error creating 'x.json': Name must be 'x' (is 'y')\n"
1299
+ knife("diff --name-status /environments/x.json").should_succeed "A\t/environments/x.json\n"
1300
+ end
1301
+ end
1302
+
1303
+ when_the_repository "has an environment with no name in the file" do
1304
+ before do
1305
+ file "environments/x.json", { "description" => "hi" }
1306
+ end
1307
+ it "knife upload succeeds" do
1308
+ knife("upload /environments/x.json").should_succeed "Created /environments/x.json\n"
1309
+ knife("diff --name-status /environments/x.json").should_succeed ""
1310
+ end
1311
+ end
1312
+
1313
+ when_the_repository "has a data bag with no id in the file" do
1314
+ before do
1315
+ file "data_bags/bag/x.json", { "foo" => "bar" }
1316
+ end
1317
+ it "knife upload succeeds" do
1318
+ knife("upload /data_bags/bag/x.json").should_succeed "Created /data_bags/bag\nCreated /data_bags/bag/x.json\n"
1319
+ knife("diff --name-status /data_bags/bag/x.json").should_succeed ""
1320
+ end
1321
+ end
1322
+ end
1323
+
1324
+ when_the_chef_server "is empty" do
1325
+ when_the_repository "has a cookbook with an invalid chef_version constraint in it" do
1326
+ before do
1327
+ file "cookbooks/x-1.0.0/metadata.rb", cb_metadata("x", "1.0.0", "\nchef_version '~> 999.0'")
1328
+ end
1329
+ it "knife upload succeeds" do
1330
+ knife("upload /cookbooks/x-1.0.0").should_succeed <<~EOM
1331
+ Created /cookbooks/x-1.0.0
1332
+ EOM
1333
+ knife("diff --name-status /cookbooks").should_succeed ""
1334
+ end
1335
+ end
1336
+ end
1337
+ end # with versioned cookbooks
1338
+
1339
+ when_the_chef_server "has a user" do
1340
+ before do
1341
+ user "x", {}
1342
+ end
1343
+
1344
+ when_the_repository "has the same user with json_class in it" do
1345
+ before do
1346
+ file "users/x.json", { "admin" => true, "json_class" => "Chef::WebUIUser" }
1347
+ end
1348
+ it "knife upload /users/x.json succeeds" do
1349
+ knife("upload /users/x.json").should_succeed "Updated /users/x.json\n"
1350
+ end
1351
+ end
1352
+ end
1353
+
1354
+ when_the_chef_server "is in Enterprise mode", osc_compat: false, single_org: false do
1355
+ before do
1356
+ user "foo", {}
1357
+ user "bar", {}
1358
+ user "foobar", {}
1359
+ organization "foo", { "full_name" => "Something" }
1360
+ end
1361
+
1362
+ before :each do
1363
+ Chef::Config.chef_server_url = URI.join(Chef::Config.chef_server_url, "/organizations/foo")
1364
+ end
1365
+
1366
+ context "and has nothing but a single group named blah" do
1367
+ group "blah", {}
1368
+
1369
+ when_the_repository "has at least one of each thing" do
1370
+
1371
+ before do
1372
+ # TODO We have to upload acls for an existing group due to a lack of
1373
+ # dependency detection during upload. Fix that!
1374
+ file "acls/groups/blah.json", {}
1375
+ file "clients/x.json", { "public_key" => ChefZero::PUBLIC_KEY }
1376
+ file "containers/x.json", {}
1377
+ file "cookbooks/x/metadata.rb", cb_metadata("x", "1.0.0")
1378
+ file "cookbook_artifacts/x-1x1/metadata.rb", cb_metadata("x", "1.0.0")
1379
+ file "data_bags/x/y.json", {}
1380
+ file "environments/x.json", {}
1381
+ file "groups/x.json", {}
1382
+ file "invitations.json", [ "foo" ]
1383
+ file "members.json", [ "bar" ]
1384
+ file "org.json", { "full_name" => "wootles" }
1385
+ file "nodes/x.json", { "normal" => { "tags" => [] } }
1386
+ file "policies/x-1.0.0.json", { "policy_group_list" => [ "x" ] }
1387
+ file "policies/blah-1.0.0.json", { "policy_group_list" => [ "x" ] }
1388
+ file "policy_groups/x.json", { "policies" => { "x" => { "revision_id" => "1.0.0" }, "blah" => { "revision_id" => "1.0.0" } } }
1389
+ file "roles/x.json", {}
1390
+ end
1391
+
1392
+ it "knife upload / uploads everything" do
1393
+ knife("upload /").should_succeed <<~EOM
1394
+ Updated /acls/groups/blah.json
1395
+ Created /clients/x.json
1396
+ Created /containers/x.json
1397
+ Created /cookbook_artifacts/x-1x1
1398
+ Created /cookbooks/x
1399
+ Created /data_bags/x
1400
+ Created /data_bags/x/y.json
1401
+ Created /environments/x.json
1402
+ Created /groups/x.json
1403
+ Updated /invitations.json
1404
+ Updated /members.json
1405
+ Created /nodes/x.json
1406
+ Updated /org.json
1407
+ Created /policies/blah-1.0.0.json
1408
+ Created /policies/x-1.0.0.json
1409
+ Created /policy_groups/x.json
1410
+ Created /roles/x.json
1411
+ EOM
1412
+ expect(api.get("association_requests").map { |a| a["username"] }).to eq([ "foo" ])
1413
+ expect(api.get("users").map { |a| a["user"]["username"] }).to eq([ "bar" ])
1414
+ knife("diff --name-status --diff-filter=AMT /").should_succeed ""
1415
+ end
1416
+
1417
+ context "When the chef server has an identical copy of each thing" do
1418
+ before do
1419
+ file "invitations.json", [ "foo" ]
1420
+ file "members.json", [ "bar" ]
1421
+ file "org.json", { "full_name" => "Something" }
1422
+
1423
+ # acl_for %w(organizations foo groups blah)
1424
+ client "x", {}
1425
+ cookbook "x", "1.0.0"
1426
+ cookbook_artifact "x", "1x1", "metadata.rb" => cb_metadata("x", "1.0.0")
1427
+ container "x", {}
1428
+ data_bag "x", { "y" => {} }
1429
+ environment "x", {}
1430
+ group "x", {}
1431
+ org_invite "foo"
1432
+ org_member "bar"
1433
+ node "x", {}
1434
+ policy "x", "1.0.0", {}
1435
+ policy "blah", "1.0.0", {}
1436
+ policy_group "x", {
1437
+ "policies" => {
1438
+ "x" => { "revision_id" => "1.0.0" },
1439
+ "blah" => { "revision_id" => "1.0.0" },
1440
+ },
1441
+ }
1442
+ role "x", {}
1443
+ end
1444
+
1445
+ it "knife upload makes no changes" do
1446
+ knife("upload /").should_succeed <<~EOM
1447
+ Updated /acls/groups/blah.json
1448
+ EOM
1449
+ end
1450
+ end
1451
+
1452
+ context "When the chef server has a slightly different copy of the policy revision" do
1453
+ before do
1454
+ policy "x", "1.0.0", { "run_list" => [ "blah" ] }
1455
+ end
1456
+
1457
+ it "should fail because policies are not updateable" do
1458
+ knife("upload /policies/x-1.0.0.json").should_fail <<~EOM
1459
+ ERROR: /policies/x-1.0.0.json cannot be updated: policy revisions are immutable once uploaded. If you want to change the policy, create a new revision with your changes.
1460
+ EOM
1461
+ end
1462
+ end
1463
+
1464
+ context "When the chef server has a slightly different copy of the cookbook artifact" do
1465
+ before do
1466
+ cookbook_artifact "x", "1x1", { "recipes" => { "default.rb" => "" } }
1467
+ end
1468
+
1469
+ it "should fail because cookbook_artifacts cannot be updated" do
1470
+ knife("upload /cookbook_artifacts/x-1x1").should_fail <<~EOM
1471
+ ERROR: /cookbook_artifacts/x-1x1 cannot be updated: cookbook artifacts are immutable once uploaded.
1472
+ EOM
1473
+ end
1474
+ end
1475
+
1476
+ context "When the chef server has a slightly different copy of each thing (except policy revisions)" do
1477
+ before do
1478
+ # acl_for %w(organizations foo groups blah)
1479
+ client "x", { "validator" => true }
1480
+ container "x", {}
1481
+ cookbook "x", "1.0.0", { "recipes" => { "default.rb" => "" } }
1482
+ cookbook_artifact "x", "1x1", { "metadata.rb" => cb_metadata("x", "1.0.0") }
1483
+ data_bag "x", { "y" => { "a" => "b" } }
1484
+ environment "x", { "description" => "foo" }
1485
+ group "x", { "groups" => [ "admin" ] }
1486
+ node "x", { "run_list" => [ "blah" ] }
1487
+ policy "x", "1.0.0", { "policy_group_list" => [ "x" ] }
1488
+ policy "y", "1.0.0", { "policy_group_list" => [ "x" ] }
1489
+ policy_group "x", {
1490
+ "policies" => {
1491
+ "x" => { "revision_id" => "1.0.0" },
1492
+ "y" => { "revision_id" => "1.0.0" },
1493
+ },
1494
+ }
1495
+ role "x", { "run_list" => [ "blah" ] }
1496
+ end
1497
+
1498
+ it "knife upload updates everything" do
1499
+ knife("upload /").should_succeed <<~EOM
1500
+ Updated /acls/groups/blah.json
1501
+ Updated /clients/x.json
1502
+ Updated /cookbooks/x
1503
+ Updated /data_bags/x/y.json
1504
+ Updated /environments/x.json
1505
+ Updated /groups/x.json
1506
+ Updated /invitations.json
1507
+ Updated /members.json
1508
+ Updated /nodes/x.json
1509
+ Updated /org.json
1510
+ Created /policies/blah-1.0.0.json
1511
+ Updated /policy_groups/x.json
1512
+ Updated /roles/x.json
1513
+ EOM
1514
+ knife("diff --name-status --diff-filter=AMT /").should_succeed ""
1515
+ end
1516
+ end
1517
+ end
1518
+
1519
+ when_the_repository "has an org.json that does not change full_name" do
1520
+ before do
1521
+ file "org.json", { "full_name" => "Something" }
1522
+ end
1523
+
1524
+ it "knife upload / emits a warning for bar and adds foo and foobar" do
1525
+ knife("upload /").should_succeed ""
1526
+ expect(api.get("/")["full_name"]).to eq("Something")
1527
+ end
1528
+ end
1529
+
1530
+ when_the_repository "has an org.json that changes full_name" do
1531
+ before do
1532
+ file "org.json", { "full_name" => "Something Else" }
1533
+ end
1534
+
1535
+ it "knife upload / emits a warning for bar and adds foo and foobar" do
1536
+ knife("upload /").should_succeed "Updated /org.json\n"
1537
+ expect(api.get("/")["full_name"]).to eq("Something Else")
1538
+ end
1539
+ end
1540
+
1541
+ context "and has invited foo and bar is already a member" do
1542
+ org_invite "foo"
1543
+ org_member "bar"
1544
+
1545
+ when_the_repository "wants to invite foo, bar and foobar" do
1546
+ before do
1547
+ file "invitations.json", %w{foo bar foobar}
1548
+ end
1549
+
1550
+ it "knife upload / emits a warning for bar and invites foobar" do
1551
+ knife("upload /").should_succeed "Updated /invitations.json\n", stderr: "WARN: Could not invite bar to organization foo: User bar is already in organization foo\n"
1552
+ expect(api.get("association_requests").map { |a| a["username"] }).to eq(%w{foo foobar})
1553
+ expect(api.get("users").map { |a| a["user"]["username"] }).to eq([ "bar" ])
1554
+ end
1555
+ end
1556
+
1557
+ when_the_repository "wants to make foo, bar and foobar members" do
1558
+ before do
1559
+ file "members.json", %w{foo bar foobar}
1560
+ end
1561
+
1562
+ it "knife upload / emits a warning for bar and adds foo and foobar" do
1563
+ knife("upload /").should_succeed "Updated /members.json\n"
1564
+ expect(api.get("association_requests").map { |a| a["username"] }).to eq([ ])
1565
+ expect(api.get("users").map { |a| a["user"]["username"] }).to eq(%w{bar foo foobar})
1566
+ end
1567
+ end
1568
+
1569
+ when_the_repository "wants to invite foo and have bar as a member" do
1570
+ before do
1571
+ file "invitations.json", [ "foo" ]
1572
+ file "members.json", [ "bar" ]
1573
+ end
1574
+
1575
+ it "knife upload / does nothing" do
1576
+ knife("upload /").should_succeed ""
1577
+ expect(api.get("association_requests").map { |a| a["username"] }).to eq([ "foo" ])
1578
+ expect(api.get("users").map { |a| a["user"]["username"] }).to eq([ "bar" ])
1579
+ end
1580
+ end
1581
+ end
1582
+
1583
+ context "and has invited bar and foo" do
1584
+ org_invite "bar", "foo"
1585
+
1586
+ when_the_repository "wants to invite foo and bar (different order)" do
1587
+ before do
1588
+ file "invitations.json", %w{foo bar}
1589
+ end
1590
+
1591
+ it "knife upload / does nothing" do
1592
+ knife("upload /").should_succeed ""
1593
+ expect(api.get("association_requests").map { |a| a["username"] }).to eq(%w{bar foo})
1594
+ expect(api.get("users").map { |a| a["user"]["username"] }).to eq([ ])
1595
+ end
1596
+ end
1597
+ end
1598
+
1599
+ context "and has already added bar and foo as members of the org" do
1600
+ org_member "bar", "foo"
1601
+
1602
+ when_the_repository "wants to add foo and bar (different order)" do
1603
+ before do
1604
+ file "members.json", %w{foo bar}
1605
+ end
1606
+
1607
+ it "knife upload / does nothing" do
1608
+ knife("upload /").should_succeed ""
1609
+ expect(api.get("association_requests").map { |a| a["username"] }).to eq([ ])
1610
+ expect(api.get("users").map { |a| a["user"]["username"] }).to eq(%w{bar foo})
1611
+ end
1612
+ end
1613
+ end
1614
+ end
1615
+ end
1616
+ end