chef 11.4.4 → 11.6.0.hotfix.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (504) hide show
  1. data/README.md +1 -1
  2. data/Rakefile +18 -1
  3. data/bin/chef-service-manager +37 -0
  4. data/distro/common/html/chef-client.8.html +4 -4
  5. data/distro/common/html/chef-expander.8.html +4 -4
  6. data/distro/common/html/chef-expanderctl.8.html +4 -4
  7. data/distro/common/html/chef-server-webui.8.html +4 -4
  8. data/distro/common/html/chef-server.8.html +4 -4
  9. data/distro/common/html/chef-shell.1.html +4 -4
  10. data/distro/common/html/chef-solo.8.html +12 -18
  11. data/distro/common/html/chef-solr.8.html +4 -4
  12. data/distro/common/html/knife-bootstrap.1.html +4 -4
  13. data/distro/common/html/knife-client.1.html +4 -4
  14. data/distro/common/html/knife-configure.1.html +4 -4
  15. data/distro/common/html/knife-cookbook-site.1.html +4 -4
  16. data/distro/common/html/knife-cookbook.1.html +7 -10
  17. data/distro/common/html/knife-data-bag.1.html +7 -10
  18. data/distro/common/html/knife-environment.1.html +6 -8
  19. data/distro/common/html/knife-exec.1.html +4 -4
  20. data/distro/common/html/knife-index.1.html +4 -4
  21. data/distro/common/html/knife-node.1.html +4 -4
  22. data/distro/common/html/knife-role.1.html +4 -4
  23. data/distro/common/html/knife-search.1.html +4 -4
  24. data/distro/common/html/knife-ssh.1.html +4 -4
  25. data/distro/common/html/knife-status.1.html +4 -4
  26. data/distro/common/html/knife-tag.1.html +4 -4
  27. data/distro/common/html/knife.1.html +4 -4
  28. data/distro/common/man/man1/chef-shell.1 +1 -1
  29. data/distro/common/man/man1/knife-bootstrap.1 +1 -1
  30. data/distro/common/man/man1/knife-client.1 +1 -1
  31. data/distro/common/man/man1/knife-configure.1 +1 -1
  32. data/distro/common/man/man1/knife-cookbook-site.1 +1 -1
  33. data/distro/common/man/man1/knife-cookbook.1 +2 -15
  34. data/distro/common/man/man1/knife-data-bag.1 +2 -15
  35. data/distro/common/man/man1/knife-environment.1 +2 -12
  36. data/distro/common/man/man1/knife-exec.1 +1 -1
  37. data/distro/common/man/man1/knife-index.1 +1 -1
  38. data/distro/common/man/man1/knife-node.1 +1 -1
  39. data/distro/common/man/man1/knife-role.1 +1 -1
  40. data/distro/common/man/man1/knife-search.1 +1 -1
  41. data/distro/common/man/man1/knife-ssh.1 +1 -1
  42. data/distro/common/man/man1/knife-status.1 +1 -1
  43. data/distro/common/man/man1/knife-tag.1 +1 -1
  44. data/distro/common/man/man1/knife.1 +1 -1
  45. data/distro/common/man/man8/chef-client.8 +1 -1
  46. data/distro/common/man/man8/chef-expander.8 +1 -1
  47. data/distro/common/man/man8/chef-expanderctl.8 +1 -1
  48. data/distro/common/man/man8/chef-server-webui.8 +1 -1
  49. data/distro/common/man/man8/chef-server.8 +1 -1
  50. data/distro/common/man/man8/chef-solo.8 +4 -36
  51. data/distro/common/man/man8/chef-solr.8 +1 -1
  52. data/distro/debian/etc/init.d/chef-client +4 -2
  53. data/distro/windows/service_manager.rb +2 -146
  54. data/lib/chef.rb +1 -1
  55. data/lib/chef/application.rb +5 -12
  56. data/lib/chef/application/apply.rb +2 -0
  57. data/lib/chef/application/client.rb +12 -12
  58. data/lib/chef/application/knife.rb +2 -2
  59. data/lib/chef/application/solo.rb +4 -5
  60. data/lib/chef/application/windows_service.rb +113 -56
  61. data/lib/chef/application/windows_service_manager.rb +179 -0
  62. data/lib/chef/chef_fs.rb +2 -4
  63. data/lib/chef/chef_fs/chef_fs_data_store.rb +371 -0
  64. data/lib/chef/chef_fs/command_line.rb +145 -93
  65. data/lib/chef/chef_fs/config.rb +205 -0
  66. data/lib/chef/chef_fs/data_handler/acl_data_handler.rb +26 -0
  67. data/lib/chef/chef_fs/data_handler/client_data_handler.rb +38 -0
  68. data/lib/chef/chef_fs/data_handler/container_data_handler.rb +29 -0
  69. data/lib/chef/chef_fs/data_handler/cookbook_data_handler.rb +38 -0
  70. data/lib/chef/chef_fs/data_handler/data_bag_item_data_handler.rb +56 -0
  71. data/lib/chef/chef_fs/data_handler/data_handler_base.rb +128 -0
  72. data/lib/chef/chef_fs/data_handler/environment_data_handler.rb +40 -0
  73. data/lib/chef/chef_fs/data_handler/group_data_handler.rb +51 -0
  74. data/lib/chef/chef_fs/data_handler/node_data_handler.rb +36 -0
  75. data/lib/chef/chef_fs/data_handler/role_data_handler.rb +40 -0
  76. data/lib/chef/chef_fs/data_handler/user_data_handler.rb +27 -0
  77. data/lib/chef/chef_fs/file_system.rb +195 -127
  78. data/lib/chef/chef_fs/file_system/acl_dir.rb +64 -0
  79. data/lib/chef/chef_fs/file_system/acl_entry.rb +58 -0
  80. data/lib/chef/chef_fs/file_system/acls_dir.rb +68 -0
  81. data/lib/chef/chef_fs/file_system/already_exists_error.rb +31 -0
  82. data/lib/chef/chef_fs/file_system/base_fs_object.rb +98 -39
  83. data/lib/chef/chef_fs/file_system/chef_repository_file_system_cookbook_dir.rb +85 -0
  84. data/lib/chef/chef_fs/file_system/chef_repository_file_system_cookbook_entry.rb +71 -0
  85. data/lib/chef/chef_fs/file_system/chef_repository_file_system_cookbooks_dir.rb +55 -0
  86. data/lib/chef/chef_fs/file_system/chef_repository_file_system_data_bags_dir.rb +36 -0
  87. data/lib/chef/chef_fs/file_system/chef_repository_file_system_entry.rb +14 -63
  88. data/lib/chef/chef_fs/file_system/chef_repository_file_system_root_dir.rb +93 -3
  89. data/lib/chef/chef_fs/file_system/chef_server_root_dir.rb +35 -9
  90. data/lib/chef/chef_fs/file_system/cookbook_dir.rb +67 -32
  91. data/lib/chef/chef_fs/file_system/cookbook_file.rb +13 -6
  92. data/lib/chef/chef_fs/file_system/cookbook_frozen_error.rb +31 -0
  93. data/lib/chef/chef_fs/file_system/cookbooks_acl_dir.rb +41 -0
  94. data/lib/chef/chef_fs/file_system/cookbooks_dir.rb +102 -21
  95. data/lib/chef/chef_fs/file_system/data_bag_dir.rb +12 -21
  96. data/lib/chef/chef_fs/file_system/data_bags_dir.rb +14 -8
  97. data/lib/chef/chef_fs/file_system/default_environment_cannot_be_modified_error.rb +36 -0
  98. data/lib/chef/chef_fs/file_system/environments_dir.rb +60 -0
  99. data/lib/chef/chef_fs/file_system/file_system_entry.rb +8 -8
  100. data/lib/chef/chef_fs/file_system/file_system_error.rb +3 -1
  101. data/lib/chef/chef_fs/file_system/memory_dir.rb +52 -0
  102. data/lib/chef/chef_fs/file_system/memory_file.rb +17 -0
  103. data/lib/chef/chef_fs/file_system/memory_root.rb +21 -0
  104. data/lib/chef/chef_fs/file_system/multiplexed_dir.rb +48 -0
  105. data/lib/chef/chef_fs/file_system/must_delete_recursively_error.rb +2 -2
  106. data/lib/chef/chef_fs/file_system/nodes_dir.rb +17 -9
  107. data/lib/chef/chef_fs/file_system/nonexistent_fs_object.rb +0 -4
  108. data/lib/chef/chef_fs/file_system/not_found_error.rb +2 -2
  109. data/lib/chef/chef_fs/file_system/operation_failed_error.rb +34 -0
  110. data/lib/chef/chef_fs/file_system/operation_not_allowed_error.rb +48 -0
  111. data/lib/chef/chef_fs/file_system/rest_list_dir.rb +42 -13
  112. data/lib/chef/chef_fs/file_system/rest_list_entry.rb +81 -27
  113. data/lib/chef/chef_fs/knife.rb +68 -29
  114. data/lib/chef/chef_fs/parallelizer.rb +129 -0
  115. data/lib/chef/chef_fs/path_utils.rb +29 -3
  116. data/lib/chef/chef_fs/raw_request.rb +79 -0
  117. data/lib/chef/client.rb +46 -20
  118. data/lib/chef/config.rb +59 -61
  119. data/lib/chef/cookbook/chefignore.rb +2 -1
  120. data/lib/chef/cookbook/synchronizer.rb +8 -6
  121. data/lib/chef/cookbook/syntax_check.rb +17 -2
  122. data/lib/chef/cookbook_uploader.rb +10 -1
  123. data/lib/chef/cookbook_version.rb +0 -109
  124. data/lib/chef/data_bag.rb +15 -6
  125. data/lib/chef/deprecation/mixin/template.rb +49 -0
  126. data/lib/chef/deprecation/provider/cookbook_file.rb +55 -0
  127. data/lib/chef/deprecation/provider/file.rb +197 -0
  128. data/lib/chef/deprecation/provider/remote_file.rb +86 -0
  129. data/lib/chef/deprecation/provider/template.rb +63 -0
  130. data/lib/chef/deprecation/warnings.rb +38 -0
  131. data/lib/chef/encrypted_data_bag_item.rb +153 -61
  132. data/lib/chef/environment.rb +34 -3
  133. data/lib/chef/event_dispatch/base.rb +3 -0
  134. data/lib/chef/exceptions.rb +27 -2
  135. data/lib/chef/file_access_control/unix.rb +64 -7
  136. data/lib/chef/file_access_control/windows.rb +22 -11
  137. data/lib/chef/file_content_management/content_base.rb +56 -0
  138. data/lib/chef/file_content_management/deploy.rb +38 -0
  139. data/lib/chef/file_content_management/deploy/cp.rb +48 -0
  140. data/lib/chef/file_content_management/deploy/mv_unix.rb +77 -0
  141. data/lib/chef/file_content_management/deploy/mv_windows.rb +95 -0
  142. data/lib/chef/file_content_management/tempfile.rb +61 -0
  143. data/lib/chef/formatters/doc.rb +1 -1
  144. data/lib/chef/formatters/error_descriptor.rb +5 -4
  145. data/lib/chef/formatters/error_inspectors/cookbook_resolve_error_inspector.rb +23 -3
  146. data/lib/chef/formatters/error_inspectors/registration_error_inspector.rb +4 -0
  147. data/lib/chef/json_compat.rb +3 -0
  148. data/lib/chef/knife.rb +12 -3
  149. data/lib/chef/knife/bootstrap.rb +46 -2
  150. data/lib/chef/knife/bootstrap/archlinux-gems.erb +3 -3
  151. data/lib/chef/knife/bootstrap/centos5-gems.erb +3 -3
  152. data/lib/chef/knife/bootstrap/chef-full.erb +4 -4
  153. data/lib/chef/knife/bootstrap/fedora13-gems.erb +3 -3
  154. data/lib/chef/knife/bootstrap/ubuntu10.04-apt.erb +3 -3
  155. data/lib/chef/knife/bootstrap/ubuntu10.04-gems.erb +3 -3
  156. data/lib/chef/knife/bootstrap/ubuntu12.04-gems.erb +3 -3
  157. data/lib/chef/knife/client_show.rb +2 -5
  158. data/lib/chef/knife/configure.rb +3 -3
  159. data/lib/chef/knife/cookbook_create.rb +6 -5
  160. data/lib/chef/knife/cookbook_download.rb +13 -5
  161. data/lib/chef/knife/cookbook_site_share.rb +1 -0
  162. data/lib/chef/knife/cookbook_test.rb +1 -0
  163. data/lib/chef/knife/cookbook_upload.rb +4 -9
  164. data/lib/chef/knife/core/bootstrap_context.rb +10 -4
  165. data/lib/chef/knife/core/generic_presenter.rb +16 -0
  166. data/lib/chef/knife/core/node_editor.rb +1 -1
  167. data/lib/chef/knife/core/subcommand_loader.rb +43 -12
  168. data/lib/chef/knife/core/ui.rb +19 -4
  169. data/lib/chef/knife/delete.rb +76 -9
  170. data/lib/chef/knife/deps.rb +139 -0
  171. data/lib/chef/knife/diff.rb +22 -5
  172. data/lib/chef/knife/download.rb +16 -3
  173. data/lib/chef/knife/edit.rb +76 -0
  174. data/lib/chef/knife/environment_show.rb +2 -7
  175. data/lib/chef/knife/list.rb +91 -47
  176. data/lib/chef/knife/node_run_list_set.rb +66 -0
  177. data/lib/chef/knife/node_show.rb +1 -7
  178. data/lib/chef/knife/raw.rb +20 -64
  179. data/lib/chef/knife/role_show.rb +2 -4
  180. data/lib/chef/knife/search.rb +5 -6
  181. data/lib/chef/knife/show.rb +34 -11
  182. data/lib/chef/knife/ssh.rb +8 -0
  183. data/lib/chef/knife/upload.rb +23 -4
  184. data/lib/chef/knife/user_show.rb +2 -5
  185. data/lib/chef/knife/xargs.rb +265 -0
  186. data/lib/chef/log.rb +2 -2
  187. data/lib/chef/mixin/file_class.rb +0 -9
  188. data/lib/chef/mixin/language_include_recipe.rb +1 -1
  189. data/lib/chef/mixin/params_validate.rb +19 -9
  190. data/lib/chef/mixin/template.rb +126 -36
  191. data/lib/chef/mixin/windows_architecture_helper.rb +91 -0
  192. data/lib/chef/monkey_patches/file.rb +26 -0
  193. data/lib/chef/monkey_patches/net-ssh-multi.rb +140 -0
  194. data/lib/chef/monkey_patches/securerandom.rb +44 -0
  195. data/lib/chef/monologger.rb +93 -0
  196. data/lib/chef/node.rb +4 -0
  197. data/lib/chef/platform.rb +4 -490
  198. data/lib/chef/platform/provider_mapping.rb +529 -0
  199. data/lib/chef/{monkey_patches/dir.rb → platform/query_helpers.rb} +25 -19
  200. data/lib/chef/provider.rb +12 -9
  201. data/lib/chef/provider/batch.rb +35 -0
  202. data/lib/chef/provider/cookbook_file.rb +9 -78
  203. data/lib/chef/provider/cookbook_file/content.rb +49 -0
  204. data/lib/chef/provider/deploy.rb +24 -20
  205. data/lib/chef/provider/deploy/revision.rb +27 -0
  206. data/lib/chef/provider/directory.rb +19 -22
  207. data/lib/chef/provider/execute.rb +22 -5
  208. data/lib/chef/provider/file.rb +299 -217
  209. data/lib/chef/provider/file/content.rb +39 -0
  210. data/lib/chef/provider/git.rb +76 -43
  211. data/lib/chef/provider/group/usermod.rb +2 -2
  212. data/lib/chef/provider/ifconfig.rb +25 -35
  213. data/lib/chef/provider/ifconfig/debian.rb +71 -0
  214. data/lib/chef/provider/ifconfig/redhat.rb +47 -0
  215. data/lib/chef/provider/link.rb +10 -3
  216. data/lib/chef/provider/mount.rb +1 -1
  217. data/lib/chef/provider/mount/mount.rb +8 -3
  218. data/lib/chef/provider/mount/windows.rb +4 -1
  219. data/lib/chef/provider/package/portage.rb +9 -4
  220. data/lib/chef/provider/package/rubygems.rb +45 -10
  221. data/lib/chef/provider/package/smartos.rb +47 -36
  222. data/lib/chef/provider/package/yum.rb +19 -12
  223. data/lib/chef/provider/package/zypper.rb +45 -55
  224. data/lib/chef/provider/powershell_script.rb +77 -0
  225. data/lib/chef/provider/remote_directory.rb +5 -6
  226. data/lib/chef/provider/remote_file.rb +12 -108
  227. data/lib/chef/provider/remote_file/cache_control_data.rb +165 -0
  228. data/lib/chef/provider/remote_file/content.rb +75 -0
  229. data/lib/chef/provider/remote_file/fetcher.rb +43 -0
  230. data/lib/chef/provider/remote_file/ftp.rb +183 -0
  231. data/lib/chef/provider/remote_file/http.rb +124 -0
  232. data/lib/chef/provider/remote_file/local_file.rb +47 -0
  233. data/lib/chef/provider/route.rb +6 -2
  234. data/lib/chef/provider/script.rb +14 -2
  235. data/lib/chef/provider/service/macosx.rb +16 -10
  236. data/lib/chef/provider/service/solaris.rb +6 -5
  237. data/lib/chef/provider/template.rb +16 -78
  238. data/lib/chef/provider/template/content.rb +61 -0
  239. data/lib/chef/provider/user/solaris.rb +90 -0
  240. data/lib/chef/provider/user/useradd.rb +76 -63
  241. data/lib/chef/provider/windows_script.rb +73 -0
  242. data/lib/chef/providers.rb +16 -0
  243. data/lib/chef/resource.rb +23 -2
  244. data/lib/chef/resource/batch.rb +31 -0
  245. data/lib/chef/resource/conditional.rb +4 -0
  246. data/lib/chef/resource/conditional_action_not_nothing.rb +48 -0
  247. data/lib/chef/resource/file.rb +31 -3
  248. data/lib/chef/resource/link.rb +17 -0
  249. data/lib/chef/resource/lwrp_base.rb +1 -1
  250. data/lib/chef/resource/mount.rb +29 -2
  251. data/lib/chef/resource/powershell_script.rb +31 -0
  252. data/lib/chef/resource/remote_file.rb +47 -1
  253. data/lib/chef/resource/route.rb +1 -1
  254. data/lib/chef/resource/template.rb +145 -0
  255. data/lib/chef/resource/windows_script.rb +62 -0
  256. data/lib/chef/resource_collection.rb +45 -11
  257. data/lib/chef/resource_reporter.rb +81 -52
  258. data/lib/chef/resources.rb +2 -0
  259. data/lib/chef/rest.rb +13 -4
  260. data/lib/chef/rest/rest_request.rb +5 -1
  261. data/lib/chef/run_context/cookbook_compiler.rb +3 -3
  262. data/lib/chef/run_list/run_list_expansion.rb +1 -1
  263. data/lib/chef/run_lock.rb +7 -1
  264. data/lib/chef/runner.rb +0 -1
  265. data/lib/chef/scan_access_control.rb +6 -1
  266. data/lib/chef/search/query.rb +2 -2
  267. data/lib/chef/shell/shell_session.rb +2 -2
  268. data/lib/chef/util/backup.rb +84 -0
  269. data/lib/chef/util/diff.rb +145 -0
  270. data/lib/chef/util/file_edit.rb +1 -1
  271. data/lib/chef/util/selinux.rb +100 -0
  272. data/lib/chef/util/windows/net_user.rb +14 -1
  273. data/lib/chef/util/windows/volume.rb +2 -2
  274. data/lib/chef/version.rb +1 -1
  275. data/lib/chef/version/platform.rb +42 -0
  276. data/lib/chef/version_class.rb +1 -1
  277. data/lib/chef/version_constraint.rb +6 -5
  278. data/lib/chef/version_constraint/platform.rb +26 -0
  279. data/lib/chef/win32/api/file.rb +8 -2
  280. data/lib/chef/win32/version.rb +25 -8
  281. data/spec/data/apt/chef-integration-test-1.0/debian/changelog +5 -0
  282. data/spec/data/apt/chef-integration-test-1.0/debian/compat +1 -0
  283. data/spec/data/apt/chef-integration-test-1.0/debian/control +13 -0
  284. data/spec/data/apt/chef-integration-test-1.0/debian/copyright +34 -0
  285. data/spec/data/apt/chef-integration-test-1.0/debian/files +1 -0
  286. data/spec/data/apt/chef-integration-test-1.0/debian/rules +13 -0
  287. data/spec/data/apt/chef-integration-test-1.0/debian/source/format +1 -0
  288. data/spec/data/apt/chef-integration-test-1.1/debian/changelog +11 -0
  289. data/spec/data/apt/chef-integration-test-1.1/debian/compat +1 -0
  290. data/spec/data/apt/chef-integration-test-1.1/debian/control +13 -0
  291. data/spec/data/apt/chef-integration-test-1.1/debian/copyright +34 -0
  292. data/spec/data/apt/chef-integration-test-1.1/debian/files +1 -0
  293. data/spec/data/apt/chef-integration-test-1.1/debian/rules +13 -0
  294. data/spec/data/apt/chef-integration-test-1.1/debian/source/format +1 -0
  295. data/spec/data/apt/chef-integration-test_1.0-1_amd64.changes +22 -0
  296. data/spec/data/apt/chef-integration-test_1.0-1_amd64.deb +0 -0
  297. data/spec/data/apt/chef-integration-test_1.0.orig.tar.gz +0 -0
  298. data/spec/data/apt/chef-integration-test_1.1-1_amd64.changes +22 -0
  299. data/spec/data/apt/chef-integration-test_1.1-1_amd64.deb +0 -0
  300. data/spec/data/apt/chef-integration-test_1.1.orig.tar.gz +0 -0
  301. data/spec/data/apt/var/www/apt/conf/distributions +7 -0
  302. data/spec/data/apt/var/www/apt/conf/incoming +4 -0
  303. data/spec/data/apt/var/www/apt/conf/pulls +3 -0
  304. data/spec/data/apt/var/www/apt/db/checksums.db +0 -0
  305. data/spec/data/apt/var/www/apt/db/contents.cache.db +0 -0
  306. data/spec/data/apt/var/www/apt/db/packages.db +0 -0
  307. data/spec/data/apt/var/www/apt/db/references.db +0 -0
  308. data/spec/data/apt/var/www/apt/db/release.caches.db +0 -0
  309. data/spec/data/apt/var/www/apt/db/version +4 -0
  310. data/spec/data/apt/var/www/apt/dists/sid/Release +19 -0
  311. data/spec/data/apt/var/www/apt/dists/sid/main/binary-amd64/Packages +16 -0
  312. data/spec/data/apt/var/www/apt/dists/sid/main/binary-amd64/Packages.gz +0 -0
  313. data/spec/data/apt/var/www/apt/dists/sid/main/binary-amd64/Release +5 -0
  314. data/spec/data/apt/var/www/apt/dists/sid/main/binary-i386/Packages +0 -0
  315. data/spec/data/apt/var/www/apt/pool/main/c/chef-integration-test/chef-integration-test_1.0-1_amd64.deb +0 -0
  316. data/spec/data/apt/var/www/apt/pool/main/c/chef-integration-test/chef-integration-test_1.1-1_amd64.deb +0 -0
  317. data/spec/data/bootstrap/encrypted_data_bag_secret +1 -0
  318. data/spec/data/bootstrap/secret.erb +9 -0
  319. data/spec/data/cookbooks/ignorken/recipes/default.rb +1 -0
  320. data/spec/data/cookbooks/ignorken/recipes/ignoreme.rb +2 -0
  321. data/spec/data/cookbooks/openldap/files/default/.dotfile +1 -0
  322. data/spec/data/cookbooks/openldap/files/default/.ssh/id_rsa +1 -0
  323. data/spec/data/cookbooks/openldap/files/default/remotedir/.a_dotdir/.a_dotfile_in_a_dotdir +1 -0
  324. data/spec/data/cookbooks/openldap/files/default/remotedir/remotesubdir/.a_dotfile +1 -0
  325. data/spec/data/cookbooks/openldap/templates/default/all_windows_line_endings.erb +4 -0
  326. data/spec/data/cookbooks/openldap/templates/default/helper_test.erb +1 -0
  327. data/spec/data/cookbooks/openldap/templates/default/helpers_via_partial_test.erb +1 -0
  328. data/spec/data/cookbooks/openldap/templates/default/no_windows_line_endings.erb +4 -0
  329. data/spec/data/cookbooks/openldap/templates/default/some_windows_line_endings.erb +4 -0
  330. data/spec/data/cookbooks/preseed/files/default/preseed-file.seed +1 -0
  331. data/spec/data/cookbooks/preseed/templates/default/preseed-template.seed +1 -0
  332. data/spec/data/file-providers-method-snapshot-chef-11-4.json +127 -0
  333. data/spec/data/git_bundles/example-repo.gitbundle +0 -0
  334. data/spec/data/knife-home/.chef/plugins/knife/example_home_subcommand.rb +0 -0
  335. data/spec/data/knife_subcommand/test_yourself.rb +8 -0
  336. data/spec/data/null_config.rb +1 -0
  337. data/spec/data/partial_one.erb +1 -1
  338. data/spec/data/remote_file/nyan_cat.png.gz +0 -0
  339. data/spec/functional/file_content_management/deploy_strategies_spec.rb +238 -0
  340. data/spec/functional/knife/exec_spec.rb +2 -2
  341. data/spec/functional/provider/remote_file/cache_control_data_spec.rb +101 -0
  342. data/spec/functional/resource/batch_spec.rb +64 -0
  343. data/spec/functional/resource/cookbook_file_spec.rb +2 -3
  344. data/spec/functional/resource/deploy_revision_spec.rb +180 -0
  345. data/spec/functional/resource/directory_spec.rb +2 -2
  346. data/spec/functional/resource/file_spec.rb +17 -1
  347. data/spec/functional/resource/git_spec.rb +259 -0
  348. data/spec/functional/resource/link_spec.rb +422 -388
  349. data/spec/functional/resource/package_spec.rb +297 -0
  350. data/spec/functional/resource/powershell_spec.rb +188 -0
  351. data/spec/functional/resource/registry_spec.rb +8 -4
  352. data/spec/functional/resource/remote_directory_spec.rb +2 -2
  353. data/spec/functional/resource/remote_file_spec.rb +97 -29
  354. data/spec/functional/resource/template_spec.rb +173 -17
  355. data/spec/functional/resource/user_spec.rb +547 -0
  356. data/spec/functional/run_lock_spec.rb +5 -0
  357. data/spec/functional/shell_spec.rb +2 -1
  358. data/spec/functional/win32/service_manager_spec.rb +269 -0
  359. data/spec/functional/win32/versions_spec.rb +78 -0
  360. data/spec/integration/knife/chef_repo_path_spec.rb +805 -0
  361. data/spec/integration/knife/chef_repository_file_system_spec.rb +276 -0
  362. data/spec/integration/knife/chefignore_spec.rb +271 -0
  363. data/spec/integration/knife/delete_spec.rb +944 -0
  364. data/spec/integration/knife/deps_spec.rb +648 -0
  365. data/spec/integration/knife/diff_spec.rb +536 -0
  366. data/spec/integration/knife/download_spec.rb +962 -0
  367. data/spec/integration/knife/list_spec.rb +633 -0
  368. data/spec/integration/knife/raw_spec.rb +166 -0
  369. data/spec/integration/knife/redirection_spec.rb +57 -0
  370. data/spec/integration/knife/show_spec.rb +158 -0
  371. data/spec/integration/knife/upload_spec.rb +1060 -0
  372. data/spec/integration/solo/solo_spec.rb +41 -0
  373. data/spec/spec_helper.rb +55 -1
  374. data/spec/support/chef_helpers.rb +32 -0
  375. data/spec/support/platform_helpers.rb +40 -0
  376. data/spec/support/platforms/win32/spec_service.rb +59 -0
  377. data/spec/support/shared/functional/directory_resource.rb +43 -16
  378. data/spec/support/shared/functional/file_resource.rb +661 -20
  379. data/spec/support/shared/functional/securable_resource.rb +109 -8
  380. data/spec/support/shared/functional/securable_resource_with_reporting.rb +39 -31
  381. data/spec/support/shared/integration/integration_helper.rb +166 -0
  382. data/spec/support/shared/integration/knife_support.rb +171 -0
  383. data/spec/support/shared/unit/execute_resource.rb +125 -0
  384. data/spec/support/shared/unit/file_system_support.rb +8 -48
  385. data/spec/support/shared/unit/provider/file.rb +609 -0
  386. data/spec/support/shared/unit/provider/useradd_based_user_provider.rb +407 -0
  387. data/spec/support/shared/unit/script_resource.rb +52 -0
  388. data/spec/support/shared/unit/windows_script_resource.rb +48 -0
  389. data/spec/tiny_server.rb +13 -11
  390. data/spec/unit/application/client_spec.rb +39 -1
  391. data/spec/unit/application/knife_spec.rb +12 -0
  392. data/spec/unit/application/solo_spec.rb +1 -1
  393. data/spec/unit/application_spec.rb +57 -2
  394. data/spec/unit/chef_fs/diff_spec.rb +30 -31
  395. data/spec/unit/chef_fs/file_pattern_spec.rb +2 -2
  396. data/spec/unit/chef_fs/file_system_spec.rb +2 -3
  397. data/spec/unit/client_spec.rb +20 -1
  398. data/spec/unit/config_spec.rb +70 -52
  399. data/spec/unit/cookbook/synchronizer_spec.rb +49 -1
  400. data/spec/unit/cookbook/syntax_check_spec.rb +28 -3
  401. data/spec/unit/cookbook_loader_spec.rb +3 -2
  402. data/spec/unit/daemon_spec.rb +7 -7
  403. data/spec/unit/data_bag_spec.rb +7 -0
  404. data/spec/unit/deprecation_spec.rb +86 -0
  405. data/spec/unit/encrypted_data_bag_item_spec.rb +183 -88
  406. data/spec/unit/environment_spec.rb +98 -0
  407. data/spec/unit/exceptions_spec.rb +6 -1
  408. data/spec/unit/file_access_control_spec.rb +21 -1
  409. data/spec/unit/file_content_management/deploy/cp_spec.rb +46 -0
  410. data/spec/unit/file_content_management/deploy/mv_unix_spec.rb +103 -0
  411. data/spec/unit/file_content_management/deploy/mv_windows_spec.rb +179 -0
  412. data/spec/unit/formatters/error_inspectors/cookbook_resolve_error_inspector_spec.rb +38 -2
  413. data/spec/unit/formatters/error_inspectors/resource_failure_inspector_spec.rb +3 -2
  414. data/spec/unit/knife/bootstrap_spec.rb +128 -29
  415. data/spec/unit/knife/configure_spec.rb +42 -26
  416. data/spec/unit/knife/cookbook_download_spec.rb +24 -3
  417. data/spec/unit/knife/cookbook_upload_spec.rb +8 -4
  418. data/spec/unit/knife/core/bootstrap_context_spec.rb +78 -61
  419. data/spec/unit/knife/core/subcommand_loader_spec.rb +20 -0
  420. data/spec/unit/knife/core/ui_spec.rb +41 -0
  421. data/spec/unit/knife/node_run_list_set_spec.rb +140 -0
  422. data/spec/unit/knife_spec.rb +21 -0
  423. data/spec/unit/mixin/enforce_ownership_and_permissions_spec.rb +1 -0
  424. data/spec/unit/mixin/params_validate_spec.rb +35 -0
  425. data/spec/unit/mixin/template_spec.rb +69 -57
  426. data/spec/unit/mixin/windows_architecture_helper_spec.rb +83 -0
  427. data/spec/unit/node_spec.rb +7 -0
  428. data/spec/unit/platform_spec.rb +15 -1
  429. data/spec/unit/provider/cookbook_file/content_spec.rb +40 -0
  430. data/spec/unit/provider/cookbook_file_spec.rb +26 -183
  431. data/spec/unit/provider/cron/solaris_spec.rb +1 -1
  432. data/spec/unit/provider/deploy/revision_spec.rb +19 -11
  433. data/spec/unit/provider/deploy_spec.rb +2 -2
  434. data/spec/unit/provider/directory_spec.rb +23 -23
  435. data/spec/unit/provider/execute_spec.rb +27 -1
  436. data/spec/unit/provider/file/content_spec.rb +101 -0
  437. data/spec/unit/provider/file_spec.rb +25 -484
  438. data/spec/unit/provider/git_spec.rb +224 -28
  439. data/spec/unit/provider/group/usermod_spec.rb +3 -1
  440. data/spec/unit/provider/ifconfig/debian_spec.rb +89 -0
  441. data/spec/unit/provider/ifconfig/redhat_spec.rb +71 -0
  442. data/spec/unit/provider/ifconfig_spec.rb +0 -33
  443. data/spec/unit/provider/mount/mount_spec.rb +33 -2
  444. data/spec/unit/provider/mount/windows_spec.rb +4 -1
  445. data/spec/unit/provider/mount_spec.rb +16 -6
  446. data/spec/unit/provider/package/portage_spec.rb +44 -0
  447. data/spec/unit/provider/package/rubygems_spec.rb +44 -1
  448. data/spec/unit/provider/package/smartos_spec.rb +3 -2
  449. data/spec/unit/provider/package/yum_spec.rb +36 -39
  450. data/spec/unit/provider/package/zypper_spec.rb +84 -22
  451. data/spec/unit/provider/package_spec.rb +0 -4
  452. data/spec/unit/provider/powershell_spec.rb +38 -0
  453. data/spec/unit/provider/remote_directory_spec.rb +0 -4
  454. data/spec/unit/provider/remote_file/cache_control_data_spec.rb +211 -0
  455. data/spec/unit/provider/remote_file/content_spec.rb +230 -0
  456. data/spec/unit/provider/remote_file/fetcher_spec.rb +75 -0
  457. data/spec/unit/provider/remote_file/ftp_spec.rb +224 -0
  458. data/spec/unit/provider/remote_file/http_spec.rb +319 -0
  459. data/spec/unit/provider/remote_file/local_file_spec.rb +60 -0
  460. data/spec/unit/provider/remote_file_spec.rb +33 -295
  461. data/spec/unit/provider/route_spec.rb +25 -9
  462. data/spec/unit/provider/service/macosx_spec.rb +176 -152
  463. data/spec/unit/provider/service/solaris_smf_service_spec.rb +21 -18
  464. data/spec/unit/provider/service/systemd_service_spec.rb +2 -2
  465. data/spec/unit/provider/service/upstart_service_spec.rb +2 -2
  466. data/spec/unit/provider/service_spec.rb +3 -3
  467. data/spec/unit/provider/template/content_spec.rb +78 -0
  468. data/spec/unit/provider/template_spec.rb +52 -184
  469. data/spec/unit/provider/user/solaris_spec.rb +80 -0
  470. data/spec/unit/provider/user/useradd_spec.rb +12 -358
  471. data/spec/unit/resource/batch_spec.rb +48 -0
  472. data/spec/unit/resource/conditional_action_not_nothing_spec.rb +45 -0
  473. data/spec/unit/resource/execute_spec.rb +3 -101
  474. data/spec/unit/resource/file_spec.rb +0 -5
  475. data/spec/unit/resource/group_spec.rb +9 -0
  476. data/spec/unit/resource/ifconfig_spec.rb +60 -1
  477. data/spec/unit/resource/link_spec.rb +1 -0
  478. data/spec/unit/resource/mount_spec.rb +37 -0
  479. data/spec/unit/resource/powershell_spec.rb +48 -0
  480. data/spec/unit/resource/remote_file_spec.rb +44 -4
  481. data/spec/unit/resource/route_spec.rb +1 -1
  482. data/spec/unit/resource/script_spec.rb +13 -36
  483. data/spec/unit/resource/template_spec.rb +111 -8
  484. data/spec/unit/resource/user_spec.rb +7 -0
  485. data/spec/unit/resource_collection_spec.rb +61 -32
  486. data/spec/unit/resource_reporter_spec.rb +115 -102
  487. data/spec/unit/resource_spec.rb +170 -1
  488. data/spec/unit/rest/auth_credentials_spec.rb +2 -2
  489. data/spec/unit/rest_spec.rb +6 -2
  490. data/spec/unit/run_context/cookbook_compiler_spec.rb +9 -0
  491. data/spec/unit/runner_spec.rb +1 -1
  492. data/spec/unit/scan_access_control_spec.rb +4 -2
  493. data/spec/unit/shell/shell_session_spec.rb +15 -2
  494. data/spec/unit/util/backup_spec.rb +149 -0
  495. data/spec/unit/util/diff_spec.rb +596 -0
  496. data/spec/unit/util/selinux_spec.rb +172 -0
  497. data/spec/unit/version/platform_spec.rb +61 -0
  498. data/spec/unit/version_constraint/platform_spec.rb +46 -0
  499. data/spec/unit/version_constraint_spec.rb +5 -0
  500. metadata +233 -10
  501. data/lib/chef/chef_fs/file_system/data_bag_item.rb +0 -59
  502. data/spec/unit/chef_fs/file_system/chef_server_root_dir_spec.rb +0 -237
  503. data/spec/unit/chef_fs/file_system/cookbooks_dir_spec.rb +0 -568
  504. data/spec/unit/chef_fs/file_system/data_bags_dir_spec.rb +0 -220
@@ -0,0 +1,171 @@
1
+ #
2
+ # Author:: John Keiser (<jkeiser@opscode.com>)
3
+ # Copyright:: Copyright (c) 2013 Opscode, 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 'chef/config'
19
+ require 'chef/knife'
20
+ require 'chef/application/knife'
21
+ require 'logger'
22
+ require 'chef/log'
23
+
24
+ module KnifeSupport
25
+ DEBUG = ENV['DEBUG']
26
+ def knife(*args, &block)
27
+ # Allow knife('role from file roles/blah.json') rather than requiring the
28
+ # arguments to be split like knife('role', 'from', 'file', 'roles/blah.json')
29
+ # If any argument will have actual spaces in it, the long form is required.
30
+ # (Since knife commands always start with the command name, and command
31
+ # names with spaces are always multiple args, this is safe.)
32
+ if args.length == 1
33
+ args = args[0].split(/\s+/)
34
+ end
35
+
36
+ # Make output stable
37
+ Chef::Config[:concurrency] = 1
38
+
39
+ # Work on machines where we can't access /var
40
+ checksums_cache_dir = Dir.mktmpdir('checksums') do |checksums_cache_dir|
41
+ old_cache_options = Chef::Config[:cache_options]
42
+ Chef::Config[:cache_options] = {
43
+ :path => checksums_cache_dir,
44
+ :skip_expires => true
45
+ }
46
+
47
+ # This is Chef::Knife.run without load_commands--we'll load stuff
48
+ # ourselves, thank you very much
49
+ stdout = StringIO.new
50
+ stderr = StringIO.new
51
+ old_loggers = Chef::Log.loggers
52
+ old_log_level = Chef::Log.level
53
+ begin
54
+ puts "knife: #{args.join(' ')}" if DEBUG
55
+ subcommand_class = Chef::Knife.subcommand_class_from(args)
56
+ subcommand_class.options = Chef::Application::Knife.options.merge(subcommand_class.options)
57
+ subcommand_class.load_deps
58
+ instance = subcommand_class.new(args)
59
+
60
+ # Capture stdout/stderr
61
+ instance.ui = Chef::Knife::UI.new(stdout, stderr, STDIN, {})
62
+
63
+ # Don't print stuff
64
+ Chef::Config[:verbosity] = ( DEBUG ? 2 : 0 )
65
+ instance.config[:config_file] = File.join(CHEF_SPEC_DATA, "null_config.rb")
66
+
67
+
68
+ # Configure chef with a (mostly) blank knife.rb
69
+ # We set a global and then mutate it in our stub knife.rb so we can be
70
+ # extra sure that we're not loading someone's real knife.rb and then
71
+ # running test scenarios against a real chef server. If things don't
72
+ # smell right, abort.
73
+
74
+ $__KNIFE_INTEGRATION_FAILSAFE_CHECK = "ole"
75
+ instance.configure_chef
76
+
77
+ unless $__KNIFE_INTEGRATION_FAILSAFE_CHECK == "ole ole"
78
+ raise Exception, "Potential misconfiguration of integration tests detected. Aborting test."
79
+ end
80
+
81
+ logger = Logger.new(stderr)
82
+ logger.formatter = proc { |severity, datetime, progname, msg| "#{severity}: #{msg}\n" }
83
+ Chef::Log.use_log_devices([logger])
84
+ Chef::Log.level = ( DEBUG ? :debug : :warn )
85
+ Chef::Log::Formatter.show_time = false
86
+
87
+ instance.run
88
+
89
+ exit_code = 0
90
+
91
+ # This is how rspec catches exit()
92
+ rescue SystemExit => e
93
+ exit_code = e.status
94
+ ensure
95
+ Chef::Log.use_log_devices(old_loggers)
96
+ Chef::Log.level = old_log_level
97
+ Chef::Config[:cache_options] = old_cache_options
98
+ end
99
+
100
+ KnifeResult.new(stdout.string, stderr.string, exit_code)
101
+ end
102
+ end
103
+
104
+ private
105
+
106
+ class KnifeResult
107
+ def initialize(stdout, stderr, exit_code)
108
+ @stdout = stdout
109
+ @stderr = stderr
110
+ @exit_code = exit_code
111
+ end
112
+
113
+ attr_reader :stdout
114
+ attr_reader :stderr
115
+ attr_reader :exit_code
116
+
117
+ def should_fail(*args)
118
+ expected = {}
119
+ args.each do |arg|
120
+ if arg.is_a?(Hash)
121
+ expected.merge!(arg)
122
+ elsif arg.is_a?(Integer)
123
+ expected[:exit_code] = arg
124
+ else
125
+ expected[:stderr] = arg
126
+ end
127
+ end
128
+ expected[:exit_code] = 1 if !expected[:exit_code]
129
+ should_result_in(expected)
130
+ end
131
+
132
+ def should_succeed(*args)
133
+ expected = {}
134
+ args.each do |arg|
135
+ if arg.is_a?(Hash)
136
+ expected.merge!(arg)
137
+ else
138
+ expected[:stdout] = arg
139
+ end
140
+ end
141
+ should_result_in(expected)
142
+ end
143
+
144
+ private
145
+
146
+ def should_result_in(expected)
147
+ expected[:stdout] = '' if !expected[:stdout]
148
+ expected[:stderr] = '' if !expected[:stderr]
149
+ expected[:exit_code] = 0 if !expected[:exit_code]
150
+ # TODO make this go away
151
+ stderr_actual = @stderr.sub(/^WARNING: No knife configuration file found\n/, '')
152
+
153
+ if expected[:stderr].is_a?(Regexp)
154
+ stderr_actual.should =~ expected[:stderr]
155
+ else
156
+ stderr_actual.should == expected[:stderr]
157
+ end
158
+ stdout_actual = @stdout
159
+ if Chef::Platform.windows?
160
+ stderr_actual = stderr_actual.gsub("\r\n", "\n")
161
+ stdout_actual = stdout_actual.gsub("\r\n", "\n")
162
+ end
163
+ @exit_code.should == expected[:exit_code]
164
+ if expected[:stdout].is_a?(Regexp)
165
+ stdout_actual.should =~ expected[:stdout]
166
+ else
167
+ stdout_actual.should == expected[:stdout]
168
+ end
169
+ end
170
+ end
171
+ end
@@ -0,0 +1,125 @@
1
+ #
2
+ # Author:: Adam Jacob (<adam@opscode.com>)
3
+ # Author:: Tyler Cloke (<tyler@opscode.com>)
4
+ # Copyright:: Copyright (c) 2008 Opscode, Inc.
5
+ # License:: Apache License, Version 2.0
6
+ #
7
+ # Licensed under the Apache License, Version 2.0 (the "License");
8
+ # you may not use this file except in compliance with the License.
9
+ # You may obtain a copy of the License at
10
+ #
11
+ # http://www.apache.org/licenses/LICENSE-2.0
12
+ #
13
+ # Unless required by applicable law or agreed to in writing, software
14
+ # distributed under the License is distributed on an "AS IS" BASIS,
15
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16
+ # See the License for the specific language governing permissions and
17
+ # limitations under the License.
18
+ #
19
+
20
+ require 'spec_helper'
21
+
22
+ shared_examples_for "an execute resource" do
23
+
24
+ before(:each) do
25
+ @resource = execute_resource
26
+ end
27
+
28
+ it "should create a new Chef::Resource::Execute" do
29
+ @resource.should be_a_kind_of(Chef::Resource)
30
+ @resource.should be_a_kind_of(Chef::Resource::Execute)
31
+ end
32
+
33
+ it "should set the command to the first argument to new" do
34
+ @resource.command.should eql(resource_instance_name)
35
+ end
36
+
37
+ it "should accept an array on instantiation, too" do
38
+ resource = Chef::Resource::Execute.new(%w{something else})
39
+ resource.should be_a_kind_of(Chef::Resource)
40
+ resource.should be_a_kind_of(Chef::Resource::Execute)
41
+ resource.command.should eql(%w{something else})
42
+ end
43
+
44
+ it "should accept a string for the command to run" do
45
+ @resource.command "something"
46
+ @resource.command.should eql("something")
47
+ end
48
+
49
+ it "should accept an array for the command to run" do
50
+ @resource.command %w{something else}
51
+ @resource.command.should eql(%w{something else})
52
+ end
53
+
54
+ it "should accept a string for the cwd" do
55
+ @resource.cwd "something"
56
+ @resource.cwd.should eql("something")
57
+ end
58
+
59
+ it "should accept a hash for the environment" do
60
+ test_hash = { :one => :two }
61
+ @resource.environment(test_hash)
62
+ @resource.environment.should eql(test_hash)
63
+ end
64
+
65
+ it "allows the environment to be specified with #env" do
66
+ @resource.should respond_to(:env)
67
+ end
68
+
69
+ it "should accept a string for the group" do
70
+ @resource.group "something"
71
+ @resource.group.should eql("something")
72
+ end
73
+
74
+ it "should accept an integer for the group" do
75
+ @resource.group 1
76
+ @resource.group.should eql(1)
77
+ end
78
+
79
+ it "should accept an array for the execution path" do
80
+ @resource.path ["woot"]
81
+ @resource.path.should eql(["woot"])
82
+ end
83
+
84
+ it "should accept an integer for the return code" do
85
+ @resource.returns 1
86
+ @resource.returns.should eql(1)
87
+ end
88
+
89
+ it "should accept an integer for the timeout" do
90
+ @resource.timeout 1
91
+ @resource.timeout.should eql(1)
92
+ end
93
+
94
+ it "should accept a string for the user" do
95
+ @resource.user "something"
96
+ @resource.user.should eql("something")
97
+ end
98
+
99
+ it "should accept an integer for the user" do
100
+ @resource.user 1
101
+ @resource.user.should eql(1)
102
+ end
103
+
104
+ it "should accept a string for creates" do
105
+ @resource.creates "something"
106
+ @resource.creates.should eql("something")
107
+ end
108
+
109
+ describe "when it has cwd, environment, group, path, return value, and a user" do
110
+ before do
111
+ @resource.command("grep")
112
+ @resource.cwd("/tmp/")
113
+ @resource.environment({ :one => :two })
114
+ @resource.group("legos")
115
+ @resource.path(["/var/local/"])
116
+ @resource.returns(1)
117
+ @resource.user("root")
118
+ end
119
+
120
+ it "returns the command as its identity" do
121
+ @resource.identity.should == "grep"
122
+ end
123
+ end
124
+ end
125
+
@@ -17,56 +17,16 @@
17
17
  #
18
18
 
19
19
  require 'chef/chef_fs/file_system'
20
- require 'chef/chef_fs/file_system/base_fs_dir'
21
- require 'chef/chef_fs/file_system/base_fs_object'
20
+ require 'chef/chef_fs/file_system/memory_root'
21
+ require 'chef/chef_fs/file_system/memory_dir'
22
+ require 'chef/chef_fs/file_system/memory_file'
22
23
 
23
24
  module FileSystemSupport
24
- class MemoryFile < Chef::ChefFS::FileSystem::BaseFSObject
25
- def initialize(name, parent, value)
26
- super(name, parent)
27
- @value = value
28
- end
29
- def read
30
- return @value
31
- end
32
- end
33
-
34
- class MemoryDir < Chef::ChefFS::FileSystem::BaseFSDir
35
- def initialize(name, parent)
36
- super(name, parent)
37
- @children = []
38
- end
39
- attr_reader :children
40
- def child(name)
41
- @children.select { |child| child.name == name }.first || Chef::ChefFS::FileSystem::NonexistentFSObject.new(name, self)
42
- end
43
- def add_child(child)
44
- @children.push(child)
45
- end
46
- def can_have_child?(name, is_dir)
47
- root.cannot_be_in_regex ? (name !~ root.cannot_be_in_regex) : true
48
- end
49
- end
50
-
51
- class MemoryRoot < MemoryDir
52
- def initialize(pretty_name, cannot_be_in_regex = nil)
53
- super('', nil)
54
- @pretty_name = pretty_name
55
- @cannot_be_in_regex = cannot_be_in_regex
56
- end
57
-
58
- attr_reader :cannot_be_in_regex
59
-
60
- def path_for_printing
61
- @pretty_name
62
- end
63
- end
64
-
65
25
  def memory_fs(pretty_name, value, cannot_be_in_regex = nil)
66
26
  if !value.is_a?(Hash)
67
27
  raise "memory_fs() must take a Hash"
68
28
  end
69
- dir = MemoryRoot.new(pretty_name, cannot_be_in_regex)
29
+ dir = Chef::ChefFS::FileSystem::MemoryRoot.new(pretty_name, cannot_be_in_regex)
70
30
  value.each do |key, child|
71
31
  dir.add_child(memory_fs_value(child, key.to_s, dir))
72
32
  end
@@ -75,13 +35,13 @@ module FileSystemSupport
75
35
 
76
36
  def memory_fs_value(value, name = '', parent = nil)
77
37
  if value.is_a?(Hash)
78
- dir = MemoryDir.new(name, parent)
38
+ dir = Chef::ChefFS::FileSystem::MemoryDir.new(name, parent)
79
39
  value.each do |key, child|
80
40
  dir.add_child(memory_fs_value(child, key.to_s, dir))
81
41
  end
82
42
  dir
83
43
  else
84
- MemoryFile.new(name, parent, value || "#{name}\n")
44
+ Chef::ChefFS::FileSystem::MemoryFile.new(name, parent, value || "#{name}\n")
85
45
  end
86
46
  end
87
47
 
@@ -94,7 +54,7 @@ module FileSystemSupport
94
54
  end
95
55
 
96
56
  def no_blocking_calls_allowed
97
- [ MemoryFile, MemoryDir ].each do |c|
57
+ [ Chef::ChefFS::FileSystem::MemoryFile, Chef::ChefFS::FileSystem::MemoryDir ].each do |c|
98
58
  [ :children, :exists?, :read ].each do |m|
99
59
  c.any_instance.stub(m).and_raise("#{m.to_s} should not be called")
100
60
  end
@@ -103,7 +63,7 @@ module FileSystemSupport
103
63
 
104
64
  def list_should_yield_paths(fs, pattern_str, *expected_paths)
105
65
  result_paths = []
106
- Chef::ChefFS::FileSystem.list(fs, pattern(pattern_str)) { |result| result_paths << result.path }
66
+ Chef::ChefFS::FileSystem.list(fs, pattern(pattern_str)).each { |result| result_paths << result.path }
107
67
  result_paths.should =~ expected_paths
108
68
  end
109
69
  end
@@ -0,0 +1,609 @@
1
+ #
2
+ # Author:: Lamont Granquist (<lamont@opscode.com>)
3
+ # Copyright:: Copyright (c) 2013 Opscode, Inc.
4
+ # License:: Apache License, Version 2.0
5
+ #
6
+ # Licensed under the Apache License, Version 2.0 (the "License");
7
+ # you may not use this file except in compliance with the License.
8
+ # You may obtain a copy of the License at
9
+ #
10
+ # http://www.apache.org/licenses/LICENSE-2.0
11
+ #
12
+ # Unless required by applicable law or agreed to in writing, software
13
+ # distributed under the License is distributed on an "AS IS" BASIS,
14
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15
+ # See the License for the specific language governing permissions and
16
+ # limitations under the License.
17
+ #
18
+
19
+ require 'spec_helper'
20
+ require 'tmpdir'
21
+ if windows?
22
+ require 'chef/win32/file'
23
+ end
24
+
25
+ # Filesystem stubs
26
+ def file_symlink_class
27
+ if windows?
28
+ Chef::ReservedNames::Win32::File
29
+ else
30
+ File
31
+ end
32
+ end
33
+
34
+ def normalized_path
35
+ File.expand_path(resource_path)
36
+ end
37
+
38
+ def setup_normal_file
39
+ File.stub!(:exists?).with(resource_path).and_return(true)
40
+ File.stub!(:directory?).with(resource_path).and_return(false)
41
+ File.stub!(:directory?).with(enclosing_directory).and_return(true)
42
+ File.stub!(:writable?).with(resource_path).and_return(true)
43
+ file_symlink_class.stub!(:symlink?).with(resource_path).and_return(false)
44
+ file_symlink_class.stub!(:symlink?).with(normalized_path).and_return(false)
45
+ end
46
+
47
+ def setup_missing_file
48
+ File.stub!(:exists?).with(resource_path).and_return(false)
49
+ File.stub!(:directory?).with(resource_path).and_return(false)
50
+ File.stub!(:directory?).with(enclosing_directory).and_return(true)
51
+ File.stub!(:writable?).with(resource_path).and_return(false)
52
+ file_symlink_class.stub!(:symlink?).with(resource_path).and_return(false)
53
+ end
54
+
55
+ def setup_symlink
56
+ File.stub!(:exists?).with(resource_path).and_return(true)
57
+ File.stub!(:directory?).with(normalized_path).and_return(false)
58
+ File.stub!(:directory?).with(enclosing_directory).and_return(true)
59
+ File.stub!(:writable?).with(resource_path).and_return(true)
60
+ file_symlink_class.stub!(:symlink?).with(resource_path).and_return(true)
61
+ file_symlink_class.stub!(:symlink?).with(normalized_path).and_return(true)
62
+ end
63
+
64
+ def setup_unwritable_file
65
+ File.stub!(:exists?).with(resource_path).and_return(true)
66
+ File.stub!(:directory?).with(resource_path).and_return(false)
67
+ File.stub!(:directory?).with(enclosing_directory).and_return(true)
68
+ File.stub!(:writable?).with(resource_path).and_return(false)
69
+ file_symlink_class.stub!(:symlink?).with(resource_path).and_return(false)
70
+ end
71
+
72
+ def setup_missing_enclosing_directory
73
+ File.stub!(:exists?).with(resource_path).and_return(false)
74
+ File.stub!(:directory?).with(resource_path).and_return(false)
75
+ File.stub!(:directory?).with(enclosing_directory).and_return(false)
76
+ File.stub!(:writable?).with(resource_path).and_return(false)
77
+ file_symlink_class.stub!(:symlink?).with(resource_path).and_return(false)
78
+ end
79
+
80
+ shared_examples_for Chef::Provider::File do
81
+
82
+ it "should return a #{described_class}" do
83
+ provider.should be_a_kind_of(described_class)
84
+ end
85
+
86
+ it "should store the resource passed to new as new_resource" do
87
+ provider.new_resource.should eql(resource)
88
+ end
89
+
90
+ it "should store the node passed to new as node" do
91
+ provider.node.should eql(node)
92
+ end
93
+
94
+ context "when loading the current resource" do
95
+
96
+ context "when running load_current_resource and the file exists" do
97
+ before do
98
+ setup_normal_file
99
+ provider.load_current_resource
100
+ end
101
+
102
+ it "should load a current resource based on the one specified at construction" do
103
+ provider.current_resource.should be_a_kind_of(Chef::Resource::File)
104
+ end
105
+
106
+ it "the loaded current_resource name should be the same as the resource name" do
107
+ provider.current_resource.name.should eql(resource.name)
108
+ end
109
+
110
+ it "the loaded current_resource path should be the same as the resoure path" do
111
+ provider.current_resource.path.should eql(resource.path)
112
+ end
113
+
114
+ it "the loaded current_resource content should be nil" do
115
+ provider.current_resource.content.should eql(nil)
116
+ end
117
+ end
118
+
119
+ context "when running load_current_resource and the file does not exist" do
120
+ before do
121
+ setup_missing_file
122
+ provider.load_current_resource
123
+ end
124
+
125
+ it "the current_resource should be a Chef::Resource::File" do
126
+ provider.current_resource.should be_a_kind_of(Chef::Resource::File)
127
+ end
128
+
129
+ it "the current_resource name should be the same as the resource name" do
130
+ provider.current_resource.name.should eql(resource.name)
131
+ end
132
+
133
+ it "the current_resource path should be the same as the resource path" do
134
+ provider.current_resource.path.should eql(resource.path)
135
+ end
136
+
137
+ it "the loaded current_resource content should be nil" do
138
+ provider.current_resource.content.should eql(nil)
139
+ end
140
+ end
141
+
142
+ context "examining file security metadata on Unix with a file that exists" do
143
+ before do
144
+ # fake that we're on unix even if we're on windows
145
+ Chef::Platform.stub!(:windows?).and_return(false)
146
+ # mock up the filesystem to behave like unix
147
+ setup_normal_file
148
+ stat_struct = mock("::File.stat", :mode => 0600, :uid => 0, :gid => 0, :mtime => 10000)
149
+ resource_real_path = File.realpath(resource.path)
150
+ File.should_receive(:stat).with(resource_real_path).at_least(:once).and_return(stat_struct)
151
+ Etc.stub!(:getgrgid).with(0).and_return(mock("Group Ent", :name => "wheel"))
152
+ Etc.stub!(:getpwuid).with(0).and_return(mock("User Ent", :name => "root"))
153
+ end
154
+
155
+ context "when the new_resource does not specify any state" do
156
+ before do
157
+ provider.load_current_resource
158
+ end
159
+
160
+ it "should load the permissions into the current_resource" do
161
+ provider.current_resource.mode.should == "0600"
162
+ provider.current_resource.owner.should == "root"
163
+ provider.current_resource.group.should == "wheel"
164
+ end
165
+
166
+ it "should not set the new_resource permissions" do
167
+ provider.new_resource.group.should be_nil
168
+ provider.new_resource.owner.should be_nil
169
+ provider.new_resource.mode.should be_nil
170
+ end
171
+ end
172
+
173
+ context "when the new_resource explicitly specifies resource state as numbers" do
174
+ before do
175
+ resource.owner(1)
176
+ resource.group(1)
177
+ resource.mode(0644)
178
+ provider.load_current_resource
179
+ end
180
+
181
+ it "should load the permissions into the current_resource as numbers" do
182
+ # Mode is always loaded as string for reporting purposes.
183
+ provider.current_resource.mode.should == "0600"
184
+ provider.current_resource.owner.should == 0
185
+ provider.current_resource.group.should == 0
186
+ end
187
+
188
+ it "should not set the new_resource permissions" do
189
+ provider.new_resource.group.should == 1
190
+ provider.new_resource.owner.should == 1
191
+ provider.new_resource.mode.should == 0644
192
+ end
193
+ end
194
+
195
+ context "when the new_resource explicitly specifies resource state as symbols" do
196
+ before do
197
+ resource.owner("macklemore")
198
+ resource.group("seattlehiphop")
199
+ resource.mode("0321")
200
+ provider.load_current_resource
201
+ end
202
+
203
+ it "should load the permissions into the current_resource as symbols" do
204
+ provider.current_resource.mode.should == "0600"
205
+ provider.current_resource.owner.should == "root"
206
+ provider.current_resource.group.should == "wheel"
207
+ end
208
+
209
+ it "should not set the new_resource permissions" do
210
+ provider.new_resource.group.should == "seattlehiphop"
211
+ provider.new_resource.owner.should == "macklemore"
212
+ provider.new_resource.mode.should == "0321"
213
+ end
214
+ end
215
+
216
+ end
217
+
218
+ context "examining file security metadata on Unix with a file that does not exist" do
219
+ before do
220
+ # fake that we're on unix even if we're on windows
221
+ Chef::Platform.stub!(:windows?).and_return(false)
222
+ setup_missing_file
223
+ end
224
+
225
+ context "when the new_resource does not specify any state" do
226
+ before do
227
+ provider.load_current_resource
228
+ end
229
+
230
+ it "the current_resource permissions should be nil" do
231
+ provider.current_resource.mode.should be_nil
232
+ provider.current_resource.owner.should be_nil
233
+ provider.current_resource.group.should be_nil
234
+ end
235
+
236
+ it "should not set the new_resource permissions" do
237
+ provider.new_resource.group.should be_nil
238
+ provider.new_resource.owner.should be_nil
239
+ provider.new_resource.mode.should be_nil
240
+ end
241
+ end
242
+
243
+ context "when the new_resource explicitly specifies resource state" do
244
+ before do
245
+ resource.owner(63945)
246
+ resource.group(51948)
247
+ resource.mode(0123)
248
+ provider.load_current_resource
249
+ end
250
+
251
+ it "the current_resource permissions should be nil" do
252
+ provider.current_resource.mode.should be_nil
253
+ provider.current_resource.owner.should be_nil
254
+ provider.current_resource.group.should be_nil
255
+ end
256
+
257
+ it "should not set the new_resource permissions" do
258
+ provider.new_resource.group.should == 51948
259
+ provider.new_resource.owner.should == 63945
260
+ provider.new_resource.mode.should == 0123
261
+ end
262
+ end
263
+ end
264
+ end
265
+
266
+ context "when loading the new_resource after the run" do
267
+
268
+ before do
269
+ # fake that we're on unix even if we're on windows
270
+ Chef::Platform.stub!(:windows?).and_return(false)
271
+ # mock up the filesystem to behave like unix
272
+ setup_normal_file
273
+ stat_struct = mock("::File.stat", :mode => 0600, :uid => 0, :gid => 0, :mtime => 10000)
274
+ resource_real_path = File.realpath(resource.path)
275
+ File.stub!(:stat).with(resource_real_path).and_return(stat_struct)
276
+ Etc.stub!(:getgrgid).with(0).and_return(mock("Group Ent", :name => "wheel"))
277
+ Etc.stub!(:getpwuid).with(0).and_return(mock("User Ent", :name => "root"))
278
+ provider.send(:load_resource_attributes_from_file, resource)
279
+ end
280
+
281
+ it "new_resource should record the new permission information" do
282
+ provider.new_resource.group.should == "wheel"
283
+ provider.new_resource.owner.should == "root"
284
+ provider.new_resource.mode.should == "0600"
285
+ end
286
+ end
287
+
288
+ context "when reporting security metadata on windows" do
289
+ it "records the file owner" do
290
+ pending
291
+ end
292
+
293
+ it "records rights for each user in the ACL" do
294
+ pending
295
+ end
296
+
297
+ it "records deny_rights for each user in the ACL" do
298
+ pending
299
+ end
300
+ end
301
+
302
+ context "define_resource_requirements" do
303
+ context "when the enclosing directory does not exist" do
304
+ before { setup_missing_enclosing_directory }
305
+
306
+ [:create, :create_if_missing, :touch].each do |action|
307
+ context "action #{action}" do
308
+ it "raises EnclosingDirectoryDoesNotExist" do
309
+ lambda {provider.run_action(action)}.should raise_error(Chef::Exceptions::EnclosingDirectoryDoesNotExist)
310
+ end
311
+
312
+ it "does not raise an exception in why-run mode" do
313
+ Chef::Config[:why_run] = true
314
+ lambda {provider.run_action(action)}.should_not raise_error(Chef::Exceptions::EnclosingDirectoryDoesNotExist)
315
+ Chef::Config[:why_run] = false
316
+ end
317
+ end
318
+ end
319
+ end
320
+
321
+ context "when the file exists but is not deletable" do
322
+ before { setup_unwritable_file }
323
+
324
+ it "action delete raises InsufficientPermissions" do
325
+ lambda {provider.run_action(:delete)}.should raise_error(Chef::Exceptions::InsufficientPermissions)
326
+ end
327
+
328
+ it "action delete also raises InsufficientPermissions in why-run mode" do
329
+ Chef::Config[:why_run] = true
330
+ lambda {provider.run_action(:delete)}.should raise_error(Chef::Exceptions::InsufficientPermissions)
331
+ Chef::Config[:why_run] = false
332
+ end
333
+ end
334
+ end
335
+
336
+ context "action create" do
337
+ it "should create the file, update its contents and then set the acls on the file" do
338
+ setup_missing_file
339
+ provider.should_receive(:do_create_file)
340
+ provider.should_receive(:do_contents_changes)
341
+ provider.should_receive(:do_acl_changes)
342
+ provider.should_receive(:load_resource_attributes_from_file)
343
+ provider.run_action(:create)
344
+ end
345
+
346
+ context "do_create_file" do
347
+ context "when the file exists" do
348
+ before { setup_normal_file }
349
+ it "should not create the file" do
350
+ provider.deployment_strategy.should_not_receive(:create).with(resource_path)
351
+ provider.send(:do_create_file)
352
+ provider.send(:file_created?).should == false
353
+ end
354
+ end
355
+ context "when the file does not exist" do
356
+ before { setup_missing_file }
357
+ it "should create the file" do
358
+ provider.deployment_strategy.should_receive(:create).with(resource_path)
359
+ provider.send(:do_create_file)
360
+ provider.send(:file_created?).should == true
361
+ end
362
+ end
363
+ end
364
+
365
+ context "do_contents_changes" do
366
+ context "when there is content to deploy" do
367
+ before do
368
+ setup_normal_file
369
+ provider.load_current_resource
370
+ tempfile = double('Tempfile', :path => "/tmp/foo-bar-baz")
371
+ content.stub!(:tempfile).and_return(tempfile)
372
+ File.should_receive(:exists?).with("/tmp/foo-bar-baz").and_return(true)
373
+ tempfile.should_receive(:unlink).once
374
+ end
375
+
376
+ context "when the contents have changed" do
377
+ let(:tempfile_path) { "/tmp/foo-bar-baz" }
378
+ let(:tempfile_sha256) { "42971f0ddce0cb20cf7660a123ffa1a1543beb2f1e7cd9d65858764a27f3201d" }
379
+ let(:diff_for_reporting) { "+++\n---\n+foo\n-bar\n" }
380
+ before do
381
+ provider.stub!(:contents_changed?).and_return(true)
382
+ diff = double('Diff', :for_output => ['+++','---','+foo','-bar'],
383
+ :for_reporting => diff_for_reporting )
384
+ diff.stub!(:diff).with(resource_path, tempfile_path).and_return(true)
385
+ provider.should_receive(:diff).at_least(:once).and_return(diff)
386
+ provider.should_receive(:checksum).with(tempfile_path).and_return(tempfile_sha256)
387
+ provider.should_receive(:checksum).with(resource_path).and_return(tempfile_sha256)
388
+ provider.deployment_strategy.should_receive(:deploy).with(tempfile_path, normalized_path)
389
+ end
390
+ context "when the file was created" do
391
+ before { provider.should_receive(:file_created?).at_least(:once).and_return(true) }
392
+ it "does not backup the file and does not produce a diff for reporting" do
393
+ provider.should_not_receive(:do_backup)
394
+ provider.send(:do_contents_changes)
395
+ resource.diff.should be_nil
396
+ end
397
+ end
398
+ context "when the file was not created" do
399
+ before { provider.should_receive(:file_created?).at_least(:once).and_return(false) }
400
+ it "backs up the file and produces a diff for reporting" do
401
+ provider.should_receive(:do_backup)
402
+ provider.send(:do_contents_changes)
403
+ resource.diff.should == diff_for_reporting
404
+ end
405
+ end
406
+ end
407
+
408
+ it "does nothing when the contents have not changed" do
409
+ provider.stub!(:contents_changed?).and_return(false)
410
+ provider.should_not_receive(:diff)
411
+ provider.send(:do_contents_changes)
412
+ end
413
+ end
414
+
415
+ it "does nothing when there is no content to deploy (tempfile returned from contents is nil)" do
416
+ provider.send(:content).should_receive(:tempfile).at_least(:once).and_return(nil)
417
+ provider.should_not_receive(:diff)
418
+ lambda{ provider.send(:do_contents_changes) }.should_not raise_error
419
+ end
420
+
421
+ it "raises an exception when the content object returns a tempfile with a nil path" do
422
+ tempfile = double('Tempfile', :path => nil)
423
+ provider.send(:content).should_receive(:tempfile).at_least(:once).and_return(tempfile)
424
+ lambda{ provider.send(:do_contents_changes) }.should raise_error
425
+ end
426
+
427
+ it "raises an exception when the content object returns a tempfile that does not exist" do
428
+ tempfile = double('Tempfile', :path => "/tmp/foo-bar-baz")
429
+ provider.send(:content).should_receive(:tempfile).at_least(:once).and_return(tempfile)
430
+ File.should_receive(:exists?).with("/tmp/foo-bar-baz").and_return(false)
431
+ lambda{ provider.send(:do_contents_changes) }.should raise_error
432
+ end
433
+ end
434
+
435
+ context "do_acl_changes" do
436
+ it "needs tests" do
437
+ pending
438
+ end
439
+ end
440
+
441
+ context "do_selinux" do
442
+ context "when resource is updated" do
443
+ before do
444
+ setup_normal_file
445
+ provider.load_current_resource
446
+ provider.stub!(:resource_updated?).and_return(true)
447
+ end
448
+
449
+ it "should check for selinux_enabled? by default" do
450
+ provider.should_receive(:selinux_enabled?)
451
+ provider.send(:do_selinux)
452
+ end
453
+
454
+ context "when selinux fixup is enabled in the config" do
455
+ before do
456
+ @original_selinux_fixup = Chef::Config[:enable_selinux_file_permission_fixup]
457
+ Chef::Config[:enable_selinux_file_permission_fixup] = true
458
+ end
459
+
460
+ after do
461
+ Chef::Config[:enable_selinux_file_permission_fixup] = @original_selinux_fixup
462
+ end
463
+
464
+ context "when selinux is enabled on the system" do
465
+ before do
466
+ provider.should_receive(:selinux_enabled?).and_return(true)
467
+ end
468
+
469
+ it "restores security context on the file" do
470
+ provider.should_receive(:restore_security_context).with(normalized_path, false)
471
+ provider.send(:do_selinux)
472
+ end
473
+
474
+ it "restores security context recursively when told so" do
475
+ provider.should_receive(:restore_security_context).with(normalized_path, true)
476
+ provider.send(:do_selinux, true)
477
+ end
478
+ end
479
+
480
+ context "when selinux is disabled on the system" do
481
+ before do
482
+ provider.should_receive(:selinux_enabled?).and_return(false)
483
+ end
484
+
485
+ it "should not restore security context" do
486
+ provider.should_not_receive(:restore_security_context)
487
+ provider.send(:do_selinux)
488
+ end
489
+ end
490
+ end
491
+
492
+ context "when selinux fixup is disabled in the config" do
493
+ before do
494
+ @original_selinux_fixup = Chef::Config[:enable_selinux_file_permission_fixup]
495
+ Chef::Config[:enable_selinux_file_permission_fixup] = false
496
+ end
497
+
498
+ after do
499
+ Chef::Config[:enable_selinux_file_permission_fixup] = @original_selinux_fixup
500
+ end
501
+
502
+ it "should not check for selinux_enabled?" do
503
+ provider.should_not_receive(:selinux_enabled?)
504
+ provider.send(:do_selinux)
505
+ end
506
+ end
507
+ end
508
+
509
+ context "when resource is not updated" do
510
+ before do
511
+ provider.stub!(:resource_updated?).and_return(false)
512
+ end
513
+
514
+ it "should not check for selinux_enabled?" do
515
+ provider.should_not_receive(:selinux_enabled?)
516
+ provider.send(:do_selinux)
517
+ end
518
+ end
519
+ end
520
+
521
+ end
522
+
523
+ context "action delete" do
524
+ context "when the file exists" do
525
+ context "when the file is writable" do
526
+ context "when the file is not a symlink" do
527
+ before { setup_normal_file }
528
+ it "should backup and delete the file and be updated by the last action" do
529
+ provider.should_receive(:do_backup).at_least(:once).and_return(true)
530
+ File.should_receive(:delete).with(resource_path).and_return(true)
531
+ provider.run_action(:delete)
532
+ resource.should be_updated_by_last_action
533
+ end
534
+ end
535
+ context "when the file is a symlink" do
536
+ before { setup_symlink }
537
+ it "should not backup the symlink" do
538
+ provider.should_not_receive(:do_backup)
539
+ File.should_receive(:delete).with(resource_path).and_return(true)
540
+ provider.run_action(:delete)
541
+ resource.should be_updated_by_last_action
542
+ end
543
+ end
544
+ end
545
+ context "when the file is not writable" do
546
+ before { setup_unwritable_file }
547
+ it "should not try to backup or delete the file, and should not be updated by last action" do
548
+ provider.should_not_receive(:do_backup)
549
+ File.should_not_receive(:delete)
550
+ lambda { provider.run_action(:delete) }.should raise_error()
551
+ resource.should_not be_updated_by_last_action
552
+ end
553
+ end
554
+ end
555
+
556
+ context "when the file does not exist" do
557
+ before { setup_missing_file }
558
+
559
+ it "should not try to backup or delete the file, and should not be updated by last action" do
560
+ provider.should_not_receive(:do_backup)
561
+ File.should_not_receive(:delete)
562
+ lambda { provider.run_action(:delete) }.should_not raise_error()
563
+ resource.should_not be_updated_by_last_action
564
+ end
565
+ end
566
+ end
567
+
568
+ context "action touch" do
569
+ context "when the file does not exist" do
570
+ before { setup_missing_file }
571
+ it "should update the atime/mtime on action_touch" do
572
+ File.should_receive(:utime).once
573
+ provider.should_receive(:action_create)
574
+ provider.run_action(:touch)
575
+ resource.should be_updated_by_last_action
576
+ end
577
+ end
578
+ context "when the file exists" do
579
+ before { setup_normal_file }
580
+ it "should update the atime/mtime on action_touch" do
581
+ File.should_receive(:utime).once
582
+ provider.should_receive(:action_create)
583
+ provider.run_action(:touch)
584
+ resource.should be_updated_by_last_action
585
+ end
586
+ end
587
+ end
588
+
589
+ context "action create_if_missing" do
590
+ context "when the file does not exist" do
591
+ before { setup_missing_file }
592
+ it "should call action_create" do
593
+ provider.should_receive(:action_create)
594
+ provider.run_action(:create_if_missing)
595
+ end
596
+ end
597
+
598
+ context "when the file exists" do
599
+ before { setup_normal_file }
600
+ it "should not call action_create" do
601
+ provider.should_not_receive(:action_create)
602
+ provider.run_action(:create_if_missing)
603
+ end
604
+ end
605
+
606
+ end
607
+
608
+ end
609
+