chef 10.34.6 → 11.0.0.beta.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (506) hide show
  1. data/CONTRIBUTING.md +155 -0
  2. data/README.md +89 -0
  3. data/Rakefile +4 -12
  4. data/bin/chef-apply +25 -0
  5. data/bin/chef-shell +34 -0
  6. data/bin/shef +6 -5
  7. data/distro/common/html/chef-client.8.html +4 -4
  8. data/distro/common/html/chef-expander.8.html +4 -4
  9. data/distro/common/html/chef-expanderctl.8.html +4 -4
  10. data/distro/common/html/chef-server-webui.8.html +4 -4
  11. data/distro/common/html/chef-server.8.html +4 -4
  12. data/distro/common/html/{shef.1.html → chef-shell.1.html} +49 -46
  13. data/distro/common/html/chef-solo.8.html +18 -12
  14. data/distro/common/html/chef-solr.8.html +4 -4
  15. data/distro/common/html/knife-bootstrap.1.html +4 -4
  16. data/distro/common/html/knife-client.1.html +4 -4
  17. data/distro/common/html/knife-configure.1.html +4 -4
  18. data/distro/common/html/knife-cookbook-site.1.html +4 -4
  19. data/distro/common/html/knife-cookbook.1.html +10 -7
  20. data/distro/common/html/knife-data-bag.1.html +10 -7
  21. data/distro/common/html/knife-environment.1.html +8 -6
  22. data/distro/common/html/knife-exec.1.html +9 -9
  23. data/distro/common/html/knife-index.1.html +4 -4
  24. data/distro/common/html/knife-node.1.html +4 -4
  25. data/distro/common/html/knife-role.1.html +4 -4
  26. data/distro/common/html/knife-search.1.html +4 -4
  27. data/distro/common/html/knife-ssh.1.html +4 -4
  28. data/distro/common/html/knife-status.1.html +4 -4
  29. data/distro/common/html/knife-tag.1.html +4 -4
  30. data/distro/common/html/knife.1.html +8 -13
  31. data/distro/common/man/man1/{shef.1 → chef-shell.1} +21 -57
  32. data/distro/common/man/man1/knife-bootstrap.1 +1 -1
  33. data/distro/common/man/man1/knife-client.1 +1 -1
  34. data/distro/common/man/man1/knife-configure.1 +1 -1
  35. data/distro/common/man/man1/knife-cookbook-site.1 +1 -1
  36. data/distro/common/man/man1/knife-cookbook.1 +15 -2
  37. data/distro/common/man/man1/knife-data-bag.1 +15 -2
  38. data/distro/common/man/man1/knife-environment.1 +12 -2
  39. data/distro/common/man/man1/knife-exec.1 +4 -7
  40. data/distro/common/man/man1/knife-index.1 +1 -1
  41. data/distro/common/man/man1/knife-node.1 +1 -1
  42. data/distro/common/man/man1/knife-role.1 +1 -1
  43. data/distro/common/man/man1/knife-search.1 +1 -1
  44. data/distro/common/man/man1/knife-ssh.1 +1 -1
  45. data/distro/common/man/man1/knife-status.1 +1 -1
  46. data/distro/common/man/man1/knife-tag.1 +1 -1
  47. data/distro/common/man/man1/knife.1 +3 -6
  48. data/distro/common/man/man8/chef-client.8 +1 -1
  49. data/distro/common/man/man8/chef-expander.8 +1 -1
  50. data/distro/common/man/man8/chef-expanderctl.8 +1 -1
  51. data/distro/common/man/man8/chef-server-webui.8 +1 -1
  52. data/distro/common/man/man8/chef-server.8 +1 -1
  53. data/distro/common/man/man8/chef-solo.8 +36 -4
  54. data/distro/common/man/man8/chef-solr.8 +1 -1
  55. data/distro/common/markdown/man1/{shef.mkd → chef-shell.mkd} +49 -43
  56. data/distro/common/markdown/man1/knife-exec.mkd +11 -6
  57. data/distro/common/markdown/man1/knife.mkd +4 -9
  58. data/distro/debian/etc/default/chef-client +0 -1
  59. data/distro/debian/etc/init.d/chef-client +2 -2
  60. data/lib/chef.rb +2 -5
  61. data/lib/chef/api_client.rb +20 -130
  62. data/lib/chef/api_client/registration.rb +126 -0
  63. data/lib/chef/application.rb +71 -14
  64. data/lib/chef/application/apply.rb +160 -0
  65. data/lib/chef/application/client.rb +25 -18
  66. data/lib/chef/application/knife.rb +0 -2
  67. data/lib/chef/application/solo.rb +23 -8
  68. data/lib/chef/application/windows_service.rb +5 -2
  69. data/lib/chef/applications.rb +1 -0
  70. data/lib/chef/chef_fs.rb +11 -0
  71. data/lib/chef/chef_fs/command_line.rb +232 -0
  72. data/lib/chef/chef_fs/file_pattern.rb +312 -0
  73. data/lib/chef/chef_fs/file_system.rb +358 -0
  74. data/lib/chef/chef_fs/file_system/base_fs_dir.rb +47 -0
  75. data/lib/chef/chef_fs/file_system/base_fs_object.rb +121 -0
  76. data/lib/chef/chef_fs/file_system/chef_repository_file_system_entry.rb +109 -0
  77. data/{spec/unit/monkey_patches/uri_spec.rb → lib/chef/chef_fs/file_system/chef_repository_file_system_root_dir.rb} +12 -15
  78. data/lib/chef/chef_fs/file_system/chef_server_root_dir.rb +84 -0
  79. data/lib/chef/chef_fs/file_system/cookbook_dir.rb +188 -0
  80. data/lib/chef/chef_fs/file_system/cookbook_file.rb +78 -0
  81. data/lib/chef/chef_fs/file_system/cookbook_subdir.rb +54 -0
  82. data/lib/chef/chef_fs/file_system/cookbooks_dir.rb +68 -0
  83. data/lib/chef/chef_fs/file_system/data_bag_dir.rb +78 -0
  84. data/lib/chef/chef_fs/file_system/data_bag_item.rb +59 -0
  85. data/lib/chef/chef_fs/file_system/data_bags_dir.rb +66 -0
  86. data/lib/chef/chef_fs/file_system/file_system_entry.rb +90 -0
  87. data/lib/chef/{index_queue.rb → chef_fs/file_system/file_system_error.rb} +14 -12
  88. data/lib/chef/{resource/whyrun_safe_ruby_block.rb → chef_fs/file_system/file_system_root_dir.rb} +10 -10
  89. data/lib/chef/chef_fs/file_system/must_delete_recursively_error.rb +31 -0
  90. data/lib/chef/chef_fs/file_system/nodes_dir.rb +47 -0
  91. data/lib/chef/{provider/whyrun_safe_ruby_block.rb → chef_fs/file_system/nonexistent_fs_object.rb} +19 -9
  92. data/lib/chef/chef_fs/file_system/not_found_error.rb +31 -0
  93. data/lib/chef/chef_fs/file_system/rest_list_dir.rb +84 -0
  94. data/lib/chef/chef_fs/file_system/rest_list_entry.rb +123 -0
  95. data/lib/chef/chef_fs/knife.rb +77 -0
  96. data/lib/chef/chef_fs/path_utils.rb +64 -0
  97. data/lib/chef/client.rb +44 -21
  98. data/lib/chef/config.rb +52 -43
  99. data/lib/chef/cookbook/synchronizer.rb +6 -8
  100. data/lib/chef/cookbook/syntax_check.rb +61 -14
  101. data/lib/chef/cookbook_loader.rb +39 -26
  102. data/lib/chef/cookbook_uploader.rb +17 -19
  103. data/lib/chef/cookbook_version.rb +3 -302
  104. data/lib/chef/daemon.rb +3 -18
  105. data/lib/chef/data_bag.rb +4 -97
  106. data/lib/chef/data_bag_item.rb +2 -65
  107. data/lib/chef/digester.rb +73 -0
  108. data/lib/chef/dsl.rb +6 -0
  109. data/lib/chef/dsl/data_query.rb +66 -0
  110. data/lib/chef/dsl/include_attribute.rb +60 -0
  111. data/lib/chef/dsl/include_recipe.rb +42 -0
  112. data/lib/chef/dsl/platform_introspection.rb +213 -0
  113. data/lib/chef/dsl/recipe.rb +84 -0
  114. data/lib/chef/dsl/registry_helper.rb +59 -0
  115. data/lib/chef/encrypted_data_bag_item.rb +74 -19
  116. data/lib/chef/environment.rb +9 -180
  117. data/lib/chef/exceptions.rb +87 -14
  118. data/lib/chef/formatters/base.rb +4 -1
  119. data/lib/chef/formatters/error_inspectors/registration_error_inspector.rb +0 -4
  120. data/lib/chef/json_compat.rb +1 -97
  121. data/lib/chef/knife.rb +90 -41
  122. data/lib/chef/knife/bootstrap/archlinux-gems.erb +2 -2
  123. data/lib/chef/knife/bootstrap/centos5-gems.erb +2 -2
  124. data/lib/chef/knife/bootstrap/chef-full.erb +3 -3
  125. data/lib/chef/knife/bootstrap/fedora13-gems.erb +2 -2
  126. data/lib/chef/knife/bootstrap/ubuntu10.04-apt.erb +2 -2
  127. data/lib/chef/knife/bootstrap/ubuntu10.04-gems.erb +2 -2
  128. data/lib/chef/knife/bootstrap/ubuntu12.04-gems.erb +2 -2
  129. data/lib/chef/knife/configure.rb +1 -2
  130. data/lib/chef/knife/cookbook_metadata.rb +1 -0
  131. data/lib/chef/knife/cookbook_test.rb +3 -2
  132. data/lib/chef/knife/cookbook_upload.rb +12 -7
  133. data/lib/chef/knife/core/bootstrap_context.rb +1 -1
  134. data/lib/chef/knife/core/generic_presenter.rb +26 -13
  135. data/lib/chef/knife/core/node_editor.rb +36 -16
  136. data/lib/chef/knife/core/node_presenter.rb +1 -1
  137. data/lib/chef/knife/core/text_formatter.rb +23 -37
  138. data/lib/chef/knife/core/ui.rb +15 -9
  139. data/lib/chef/knife/delete.rb +39 -0
  140. data/lib/chef/knife/diff.rb +46 -0
  141. data/lib/chef/knife/download.rb +50 -0
  142. data/lib/chef/knife/environment_show.rb +7 -0
  143. data/lib/chef/knife/exec.rb +5 -5
  144. data/lib/chef/knife/help_topics.rb +1 -1
  145. data/lib/chef/knife/index_rebuild.rb +91 -7
  146. data/lib/chef/knife/list.rb +109 -0
  147. data/lib/chef/knife/raw.rb +108 -0
  148. data/lib/chef/knife/search.rb +40 -22
  149. data/lib/chef/knife/show.rb +32 -0
  150. data/lib/chef/knife/ssh.rb +6 -2
  151. data/lib/chef/knife/upload.rb +50 -0
  152. data/lib/chef/mixin/checksum.rb +3 -3
  153. data/lib/chef/mixin/deep_merge.rb +55 -197
  154. data/lib/chef/mixin/language.rb +9 -222
  155. data/lib/chef/mixin/language_include_attribute.rb +6 -38
  156. data/lib/chef/mixin/language_include_recipe.rb +3 -35
  157. data/lib/chef/mixin/params_validate.rb +6 -19
  158. data/lib/chef/mixin/recipe_definition_dsl_core.rb +8 -61
  159. data/lib/chef/mixin/securable.rb +32 -7
  160. data/lib/chef/mixin/template.rb +40 -0
  161. data/lib/chef/mixins.rb +0 -4
  162. data/lib/chef/monkey_patches/net_http.rb +0 -34
  163. data/lib/chef/node.rb +133 -309
  164. data/lib/chef/node/attribute.rb +333 -473
  165. data/lib/chef/node/attribute_collections.rb +199 -0
  166. data/lib/chef/node/immutable_collections.rb +186 -0
  167. data/lib/chef/platform.rb +7 -22
  168. data/lib/chef/provider.rb +2 -49
  169. data/lib/chef/provider/breakpoint.rb +6 -6
  170. data/lib/chef/provider/cookbook_file.rb +5 -33
  171. data/lib/chef/provider/deploy.rb +2 -1
  172. data/lib/chef/provider/directory.rb +14 -17
  173. data/lib/chef/provider/file.rb +19 -52
  174. data/lib/chef/provider/group.rb +31 -51
  175. data/lib/chef/provider/group/dscl.rb +13 -53
  176. data/lib/chef/provider/group/gpasswd.rb +19 -14
  177. data/lib/chef/provider/group/groupadd.rb +1 -41
  178. data/lib/chef/provider/group/groupmod.rb +36 -46
  179. data/lib/chef/provider/group/pw.rb +16 -59
  180. data/lib/chef/provider/group/suse.rb +13 -16
  181. data/lib/chef/provider/group/usermod.rb +18 -40
  182. data/lib/chef/provider/group/windows.rb +6 -13
  183. data/lib/chef/provider/http_request.rb +25 -42
  184. data/lib/chef/provider/link.rb +2 -0
  185. data/lib/chef/provider/lwrp_base.rb +150 -0
  186. data/lib/chef/provider/package/portage.rb +4 -9
  187. data/lib/chef/provider/package/rpm.rb +2 -2
  188. data/lib/chef/provider/package/rubygems.rb +9 -41
  189. data/lib/chef/provider/package/yum.rb +12 -19
  190. data/lib/chef/provider/registry_key.rb +156 -0
  191. data/lib/chef/provider/remote_directory.rb +2 -0
  192. data/lib/chef/provider/remote_file.rb +21 -12
  193. data/lib/chef/provider/ruby_block.rb +5 -2
  194. data/lib/chef/provider/service.rb +15 -0
  195. data/lib/chef/provider/service/init.rb +9 -7
  196. data/lib/chef/provider/service/macosx.rb +15 -73
  197. data/lib/chef/provider/service/simple.rb +1 -1
  198. data/lib/chef/provider/service/solaris.rb +3 -3
  199. data/lib/chef/provider/template.rb +22 -25
  200. data/lib/chef/provider/template_finder.rb +61 -0
  201. data/lib/chef/provider/user.rb +0 -1
  202. data/lib/chef/provider/user/dscl.rb +175 -568
  203. data/lib/chef/provider/user/useradd.rb +30 -47
  204. data/lib/chef/providers.rb +3 -2
  205. data/lib/chef/recipe.rb +14 -8
  206. data/lib/chef/resource.rb +13 -154
  207. data/lib/chef/resource/group.rb +1 -11
  208. data/lib/chef/resource/http_request.rb +2 -1
  209. data/lib/chef/resource/lwrp_base.rb +127 -0
  210. data/lib/chef/resource/mount.rb +10 -11
  211. data/lib/chef/resource/registry_key.rb +86 -0
  212. data/lib/chef/resource/remote_directory.rb +6 -5
  213. data/lib/chef/resource/remote_file.rb +22 -31
  214. data/lib/chef/resource/ruby_block.rb +2 -2
  215. data/lib/chef/resource/service.rb +14 -0
  216. data/lib/chef/resource/user.rb +0 -18
  217. data/lib/chef/resource_collection.rb +25 -21
  218. data/lib/chef/resources.rb +2 -1
  219. data/lib/chef/rest.rb +50 -131
  220. data/lib/chef/rest/auth_credentials.rb +4 -20
  221. data/lib/chef/rest/rest_request.rb +2 -7
  222. data/lib/chef/role.rb +1 -97
  223. data/lib/chef/run_context.rb +108 -130
  224. data/lib/chef/run_context/cookbook_compiler.rb +280 -0
  225. data/lib/chef/run_list.rb +0 -2
  226. data/lib/chef/run_list/run_list_expansion.rb +0 -15
  227. data/lib/chef/run_lock.rb +90 -0
  228. data/lib/chef/runner.rb +28 -5
  229. data/lib/chef/sandbox.rb +15 -148
  230. data/lib/chef/scan_access_control.rb +2 -4
  231. data/lib/chef/shef/ext.rb +3 -575
  232. data/lib/chef/{shef.rb → shell.rb} +35 -40
  233. data/lib/chef/shell/ext.rb +593 -0
  234. data/lib/chef/{shef → shell}/model_wrapper.rb +3 -3
  235. data/lib/chef/{shef/shef_rest.rb → shell/shell_rest.rb} +4 -4
  236. data/lib/chef/{shef/shef_session.rb → shell/shell_session.rb} +17 -15
  237. data/lib/chef/shell_out.rb +7 -0
  238. data/lib/chef/util/windows/net_group.rb +1 -5
  239. data/lib/chef/version.rb +3 -3
  240. data/lib/chef/win32/api/process.rb +0 -1
  241. data/lib/chef/win32/handle.rb +1 -8
  242. data/lib/chef/win32/registry.rb +371 -0
  243. data/spec/data/big_json.json +1 -2
  244. data/spec/data/big_json_plus_one.json +1 -2
  245. data/spec/data/cookbooks/openldap/attributes/default.rb +10 -9
  246. data/spec/data/cookbooks/openldap/attributes/smokey.rb +1 -1
  247. data/spec/data/lwrp/providers/inline_compiler.rb +26 -0
  248. data/spec/data/nodes/default.rb +3 -3
  249. data/spec/data/nodes/test.example.com.rb +3 -3
  250. data/spec/data/nodes/test.rb +3 -3
  251. data/spec/data/partial_one.erb +1 -0
  252. data/spec/data/run_context/cookbooks/circular-dep1/attributes/default.rb +4 -0
  253. data/spec/data/run_context/cookbooks/circular-dep1/definitions/circular_dep1_res.rb +1 -0
  254. data/spec/data/run_context/cookbooks/circular-dep1/libraries/lib.rb +2 -0
  255. data/spec/data/run_context/cookbooks/circular-dep1/metadata.rb +2 -0
  256. data/spec/data/run_context/cookbooks/circular-dep1/providers/provider.rb +1 -0
  257. data/spec/data/{knife-home/.chef/plugins/knife/example_home_subcommand.rb → run_context/cookbooks/circular-dep1/recipes/default.rb} +0 -0
  258. data/spec/data/run_context/cookbooks/circular-dep1/resources/resource.rb +1 -0
  259. data/spec/data/run_context/cookbooks/circular-dep2/attributes/default.rb +3 -0
  260. data/spec/data/run_context/cookbooks/circular-dep2/definitions/circular_dep2_res.rb +1 -0
  261. data/spec/data/run_context/cookbooks/circular-dep2/libraries/lib.rb +2 -0
  262. data/spec/data/run_context/cookbooks/circular-dep2/metadata.rb +2 -0
  263. data/spec/data/run_context/cookbooks/circular-dep2/providers/provider.rb +1 -0
  264. data/spec/data/{lwrp_const_scoping/resources/conflict.rb → run_context/cookbooks/circular-dep2/recipes/default.rb} +0 -0
  265. data/spec/data/run_context/cookbooks/circular-dep2/resources/resource.rb +1 -0
  266. data/spec/data/run_context/cookbooks/dependency1/attributes/aa_first.rb +2 -0
  267. data/spec/data/run_context/cookbooks/dependency1/attributes/default.rb +2 -0
  268. data/spec/data/run_context/cookbooks/dependency1/attributes/zz_last.rb +3 -0
  269. data/spec/data/run_context/cookbooks/dependency1/definitions/dependency1_res.rb +1 -0
  270. data/spec/data/run_context/cookbooks/dependency1/libraries/lib.rb +2 -0
  271. data/spec/data/run_context/cookbooks/dependency1/providers/provider.rb +1 -0
  272. data/spec/data/run_context/cookbooks/dependency1/recipes/default.rb +0 -0
  273. data/spec/data/run_context/cookbooks/dependency1/resources/resource.rb +1 -0
  274. data/spec/data/run_context/cookbooks/dependency2/attributes/default.rb +3 -0
  275. data/spec/data/run_context/cookbooks/dependency2/definitions/dependency2_res.rb +1 -0
  276. data/spec/data/run_context/cookbooks/dependency2/libraries/lib.rb +2 -0
  277. data/spec/data/run_context/cookbooks/dependency2/providers/provider.rb +1 -0
  278. data/spec/data/run_context/cookbooks/dependency2/recipes/default.rb +0 -0
  279. data/spec/data/run_context/cookbooks/dependency2/resources/resource.rb +1 -0
  280. data/spec/data/run_context/cookbooks/no-default-attr/attributes/server.rb +3 -0
  281. data/spec/data/run_context/cookbooks/no-default-attr/definitions/no_default-attr_res.rb +1 -0
  282. data/spec/data/run_context/cookbooks/no-default-attr/providers/provider.rb +1 -0
  283. data/spec/data/run_context/cookbooks/no-default-attr/recipes/default.rb +0 -0
  284. data/spec/data/run_context/cookbooks/no-default-attr/resources/resource.rb +1 -0
  285. data/spec/data/run_context/cookbooks/test-with-circular-deps/attributes/default.rb +3 -0
  286. data/spec/data/run_context/cookbooks/test-with-circular-deps/definitions/test_with-circular-deps_res.rb +1 -0
  287. data/spec/data/run_context/cookbooks/test-with-circular-deps/libraries/lib.rb +2 -0
  288. data/spec/data/run_context/cookbooks/test-with-circular-deps/metadata.rb +2 -0
  289. data/spec/data/run_context/cookbooks/test-with-circular-deps/providers/provider.rb +1 -0
  290. data/spec/data/run_context/cookbooks/test-with-circular-deps/recipes/default.rb +0 -0
  291. data/spec/data/run_context/cookbooks/test-with-circular-deps/resources/resource.rb +1 -0
  292. data/spec/data/run_context/cookbooks/test-with-deps/attributes/default.rb +3 -0
  293. data/spec/data/run_context/cookbooks/test-with-deps/definitions/test_with-deps_res.rb +1 -0
  294. data/spec/data/run_context/cookbooks/test-with-deps/libraries/lib.rb +1 -0
  295. data/spec/data/run_context/cookbooks/test-with-deps/metadata.rb +3 -0
  296. data/spec/data/run_context/cookbooks/test-with-deps/providers/provider.rb +1 -0
  297. data/spec/data/run_context/cookbooks/test-with-deps/recipes/default.rb +0 -0
  298. data/spec/data/run_context/cookbooks/test-with-deps/recipes/server.rb +0 -0
  299. data/spec/data/run_context/cookbooks/test-with-deps/resources/resource.rb +1 -0
  300. data/spec/data/run_context/cookbooks/test/attributes/default.rb +0 -0
  301. data/spec/data/run_context/cookbooks/test/attributes/george.rb +1 -1
  302. data/spec/data/run_context/cookbooks/test/definitions/test_res.rb +1 -0
  303. data/spec/data/run_context/cookbooks/test/providers/provider.rb +1 -0
  304. data/spec/data/run_context/cookbooks/test/resources/resource.rb +1 -0
  305. data/spec/data/shef-config.rb +7 -0
  306. data/spec/functional/dsl/registry_helper_spec.rb +63 -0
  307. data/spec/functional/knife/cookbook_delete_spec.rb +1 -1
  308. data/spec/functional/knife/exec_spec.rb +2 -2
  309. data/spec/functional/knife/ssh_spec.rb +5 -1
  310. data/spec/functional/resource/cookbook_file_spec.rb +7 -19
  311. data/spec/functional/resource/directory_spec.rb +4 -0
  312. data/spec/functional/resource/file_spec.rb +56 -22
  313. data/spec/functional/resource/link_spec.rb +2 -0
  314. data/spec/functional/resource/registry_spec.rb +576 -0
  315. data/spec/functional/resource/remote_directory_spec.rb +142 -36
  316. data/spec/functional/resource/remote_file_spec.rb +18 -0
  317. data/spec/functional/resource/template_spec.rb +23 -2
  318. data/spec/functional/run_lock_spec.rb +106 -0
  319. data/spec/functional/shell_spec.rb +100 -0
  320. data/spec/functional/win32/registry_helper_spec.rb +632 -0
  321. data/spec/spec_helper.rb +5 -29
  322. data/spec/stress/win32/security_spec.rb +1 -1
  323. data/spec/support/chef_helpers.rb +0 -2
  324. data/spec/support/platform_helpers.rb +8 -15
  325. data/spec/support/shared/functional/directory_resource.rb +84 -22
  326. data/spec/support/shared/functional/file_resource.rb +169 -71
  327. data/spec/support/shared/functional/securable_resource.rb +143 -119
  328. data/spec/support/shared/functional/securable_resource_with_reporting.rb +375 -0
  329. data/spec/support/shared/unit/file_system_support.rb +110 -0
  330. data/spec/support/shared/unit/platform_introspector.rb +162 -0
  331. data/spec/unit/api_client/registration_spec.rb +175 -0
  332. data/spec/unit/api_client_spec.rb +78 -156
  333. data/spec/unit/application/apply.rb +84 -0
  334. data/spec/unit/application/client_spec.rb +1 -37
  335. data/spec/unit/application/knife_spec.rb +5 -0
  336. data/spec/unit/application_spec.rb +57 -2
  337. data/spec/unit/checksum/storage/filesystem_spec.rb +1 -1
  338. data/spec/unit/chef_fs/diff_spec.rb +328 -0
  339. data/spec/unit/chef_fs/file_pattern_spec.rb +526 -0
  340. data/spec/unit/chef_fs/file_system/chef_server_root_dir_spec.rb +237 -0
  341. data/spec/unit/chef_fs/file_system/cookbooks_dir_spec.rb +568 -0
  342. data/spec/unit/chef_fs/file_system/data_bags_dir_spec.rb +220 -0
  343. data/spec/unit/chef_fs/file_system_spec.rb +136 -0
  344. data/spec/unit/client_spec.rb +124 -33
  345. data/spec/unit/config_spec.rb +46 -13
  346. data/spec/unit/cookbook/synchronizer_spec.rb +1 -49
  347. data/spec/unit/cookbook/syntax_check_spec.rb +48 -109
  348. data/spec/unit/cookbook_loader_spec.rb +153 -91
  349. data/spec/unit/cookbook_manifest_spec.rb +81 -81
  350. data/spec/unit/cookbook_spec.rb +3 -20
  351. data/spec/unit/cookbook_version_spec.rb +23 -122
  352. data/spec/unit/daemon_spec.rb +3 -24
  353. data/spec/unit/data_bag_spec.rb +6 -4
  354. data/spec/unit/digester_spec.rb +50 -0
  355. data/spec/unit/dsl/data_query_spec.rb +66 -0
  356. data/spec/unit/dsl/platform_introspection_spec.rb +130 -0
  357. data/spec/unit/dsl/regsitry_helper_spec.rb +55 -0
  358. data/spec/unit/encrypted_data_bag_item_spec.rb +50 -105
  359. data/spec/unit/environment_spec.rb +0 -130
  360. data/spec/unit/exceptions_spec.rb +2 -3
  361. data/spec/unit/formatters/error_inspectors/resource_failure_inspector_spec.rb +3 -3
  362. data/spec/unit/json_compat_spec.rb +15 -7
  363. data/spec/unit/knife/bootstrap_spec.rb +2 -0
  364. data/spec/unit/knife/configure_spec.rb +20 -14
  365. data/spec/unit/knife/cookbook_metadata_spec.rb +11 -4
  366. data/spec/unit/knife/cookbook_test_spec.rb +1 -0
  367. data/spec/unit/knife/cookbook_upload_spec.rb +43 -8
  368. data/spec/unit/knife/core/bootstrap_context_spec.rb +1 -1
  369. data/spec/unit/knife/core/ui_spec.rb +156 -125
  370. data/spec/unit/knife/data_bag_create_spec.rb +9 -0
  371. data/spec/unit/knife/data_bag_edit_spec.rb +1 -4
  372. data/spec/unit/knife/data_bag_from_file_spec.rb +4 -6
  373. data/spec/unit/knife/data_bag_show_spec.rb +11 -4
  374. data/spec/unit/knife/index_rebuild_spec.rb +96 -33
  375. data/spec/unit/knife/knife_help.rb +7 -7
  376. data/spec/unit/knife/node_edit_spec.rb +6 -33
  377. data/spec/unit/knife/node_run_list_remove_spec.rb +2 -1
  378. data/spec/unit/knife/ssh_spec.rb +12 -15
  379. data/spec/unit/knife/status_spec.rb +2 -2
  380. data/spec/unit/knife_spec.rb +53 -0
  381. data/spec/unit/lwrp_spec.rb +59 -42
  382. data/spec/unit/mixin/checksum_spec.rb +2 -2
  383. data/spec/unit/mixin/deep_merge_spec.rb +101 -799
  384. data/spec/unit/mixin/enforce_ownership_and_permissions_spec.rb +6 -1
  385. data/spec/unit/mixin/params_validate_spec.rb +4 -37
  386. data/spec/unit/mixin/securable_spec.rb +5 -3
  387. data/spec/unit/mixin/template_spec.rb +119 -0
  388. data/spec/unit/node/attribute_spec.rb +195 -173
  389. data/spec/unit/node/immutable_collections_spec.rb +139 -0
  390. data/spec/unit/node_spec.rb +366 -370
  391. data/spec/unit/platform_spec.rb +9 -10
  392. data/spec/unit/provider/breakpoint_spec.rb +8 -8
  393. data/spec/unit/provider/cookbook_file_spec.rb +4 -8
  394. data/spec/unit/provider/directory_spec.rb +96 -64
  395. data/spec/unit/provider/env_spec.rb +2 -2
  396. data/spec/unit/provider/file_spec.rb +48 -39
  397. data/spec/unit/provider/group/dscl_spec.rb +0 -36
  398. data/spec/unit/provider/group/gpasswd_spec.rb +9 -16
  399. data/spec/unit/provider/group/groupadd_spec.rb +4 -3
  400. data/spec/unit/provider/group/groupmod_spec.rb +1 -0
  401. data/spec/unit/provider/group/pw_spec.rb +15 -12
  402. data/spec/unit/provider/group/usermod_spec.rb +6 -21
  403. data/spec/unit/provider/group/windows_spec.rb +8 -0
  404. data/spec/unit/provider/group_spec.rb +6 -28
  405. data/spec/unit/provider/http_request_spec.rb +28 -69
  406. data/spec/unit/provider/ifconfig_spec.rb +2 -2
  407. data/spec/unit/provider/ohai_spec.rb +4 -4
  408. data/spec/unit/provider/package/apt_spec.rb +0 -1
  409. data/spec/unit/provider/package/ips_spec.rb +0 -1
  410. data/spec/unit/provider/package/portage_spec.rb +0 -44
  411. data/spec/unit/provider/package/rpm_spec.rb +0 -12
  412. data/spec/unit/provider/package/rubygems_spec.rb +1 -44
  413. data/spec/unit/provider/package/yum_spec.rb +39 -36
  414. data/spec/unit/provider/package_spec.rb +7 -5
  415. data/spec/unit/provider/registry_key_spec.rb +269 -0
  416. data/spec/unit/provider/remote_directory_spec.rb +7 -3
  417. data/spec/unit/provider/remote_file_spec.rb +36 -0
  418. data/spec/unit/provider/route_spec.rb +4 -3
  419. data/spec/unit/provider/ruby_block_spec.rb +8 -0
  420. data/spec/unit/provider/service/arch_service_spec.rb +5 -5
  421. data/spec/unit/provider/service/debian_service_spec.rb +1 -1
  422. data/spec/unit/provider/service/freebsd_service_spec.rb +5 -5
  423. data/spec/unit/provider/service/init_service_spec.rb +27 -4
  424. data/spec/unit/provider/service/insserv_service_spec.rb +1 -1
  425. data/spec/unit/provider/service/invokercd_service_spec.rb +4 -4
  426. data/spec/unit/provider/service/macosx_spec.rb +11 -66
  427. data/spec/unit/provider/service/redhat_spec.rb +1 -1
  428. data/spec/unit/provider/service/simple_service_spec.rb +3 -3
  429. data/spec/unit/provider/service/upstart_service_spec.rb +9 -9
  430. data/spec/unit/provider/subversion_spec.rb +1 -1
  431. data/spec/unit/provider/template_spec.rb +35 -11
  432. data/spec/unit/provider/user/dscl_spec.rb +285 -681
  433. data/spec/unit/provider/user/useradd_spec.rb +1 -22
  434. data/spec/unit/provider/user_spec.rb +1 -1
  435. data/spec/unit/recipe_spec.rb +10 -8
  436. data/spec/unit/registry_helper_spec.rb +374 -0
  437. data/spec/unit/resource/mount_spec.rb +0 -11
  438. data/spec/unit/resource/registry_key_spec.rb +171 -0
  439. data/spec/unit/resource/remote_file_spec.rb +21 -23
  440. data/spec/unit/resource/ruby_block_spec.rb +7 -3
  441. data/spec/unit/resource/service_spec.rb +11 -0
  442. data/spec/unit/resource_spec.rb +4 -19
  443. data/spec/unit/rest/auth_credentials_spec.rb +2 -19
  444. data/spec/unit/rest_spec.rb +130 -284
  445. data/spec/unit/run_context/cookbook_compiler_spec.rb +181 -0
  446. data/spec/unit/run_context_spec.rb +18 -4
  447. data/spec/unit/run_list_spec.rb +0 -209
  448. data/spec/unit/run_lock_spec.rb +37 -0
  449. data/spec/unit/runner_spec.rb +101 -2
  450. data/spec/unit/scan_access_control_spec.rb +4 -4
  451. data/spec/unit/{shef → shell}/model_wrapper_spec.rb +5 -5
  452. data/spec/unit/{shef/shef_ext_spec.rb → shell/shell_ext_spec.rb} +21 -21
  453. data/spec/unit/{shef/shef_session_spec.rb → shell/shell_session_spec.rb} +14 -69
  454. data/spec/unit/shell_out_spec.rb +18 -0
  455. data/spec/unit/{shef_spec.rb → shell_spec.rb} +20 -20
  456. metadata +275 -234
  457. checksums.yaml +0 -15
  458. data/README.rdoc +0 -177
  459. data/distro/common/html/knife-recipe.1.html +0 -92
  460. data/lib/chef/certificate.rb +0 -161
  461. data/lib/chef/checksum.rb +0 -167
  462. data/lib/chef/checksum_cache.rb +0 -190
  463. data/lib/chef/cookbook_version_selector.rb +0 -168
  464. data/lib/chef/couchdb.rb +0 -246
  465. data/lib/chef/index_queue/amqp_client.rb +0 -116
  466. data/lib/chef/index_queue/consumer.rb +0 -76
  467. data/lib/chef/index_queue/indexable.rb +0 -109
  468. data/lib/chef/knife/bootstrap/ubuntu12.10-gems.erb +0 -60
  469. data/lib/chef/monkey_patches/moneta.rb +0 -50
  470. data/lib/chef/monkey_patches/uri.rb +0 -70
  471. data/lib/chef/openid_registration.rb +0 -187
  472. data/lib/chef/provider/user/solaris.rb +0 -90
  473. data/lib/chef/solr_query.rb +0 -187
  474. data/lib/chef/solr_query/lucene.treetop +0 -150
  475. data/lib/chef/solr_query/lucene_nodes.rb +0 -285
  476. data/lib/chef/solr_query/query_transform.rb +0 -65
  477. data/lib/chef/solr_query/solr_http_request.rb +0 -132
  478. data/lib/chef/webui_user.rb +0 -231
  479. data/spec/data/cookbooks/openldap/files/default/.dotfile +0 -1
  480. data/spec/data/cookbooks/openldap/files/default/.ssh/id_rsa +0 -1
  481. data/spec/data/cookbooks/openldap/files/default/remotedir/.a_dotdir/.a_dotfile_in_a_dotdir +0 -1
  482. data/spec/data/cookbooks/openldap/files/default/remotedir/remotesubdir/.a_dotfile +0 -1
  483. data/spec/data/mac_users/10.7-8.plist.xml +0 -559
  484. data/spec/data/mac_users/10.7-8.shadow.xml +0 -11
  485. data/spec/data/mac_users/10.7.plist.xml +0 -559
  486. data/spec/data/mac_users/10.7.shadow.xml +0 -11
  487. data/spec/data/mac_users/10.8.plist.xml +0 -559
  488. data/spec/data/mac_users/10.8.shadow.xml +0 -21
  489. data/spec/data/mac_users/10.9.plist.xml +0 -560
  490. data/spec/data/mac_users/10.9.shadow.xml +0 -21
  491. data/spec/functional/resource/base.rb +0 -40
  492. data/spec/functional/resource/group_spec.rb +0 -343
  493. data/spec/functional/resource/user/dscl_spec.rb +0 -199
  494. data/spec/unit/certificate_spec.rb +0 -76
  495. data/spec/unit/checksum_cache_spec.rb +0 -209
  496. data/spec/unit/checksum_spec.rb +0 -94
  497. data/spec/unit/couchdb_spec.rb +0 -274
  498. data/spec/unit/index_queue_spec.rb +0 -391
  499. data/spec/unit/mixin/language_spec.rb +0 -305
  500. data/spec/unit/openid_registration_spec.rb +0 -153
  501. data/spec/unit/provider/user/solaris_spec.rb +0 -414
  502. data/spec/unit/provider/whyrun_safe_ruby_block_spec.rb +0 -47
  503. data/spec/unit/solr_query/query_transform_spec.rb +0 -454
  504. data/spec/unit/solr_query/solr_http_request_spec.rb +0 -244
  505. data/spec/unit/solr_query_spec.rb +0 -203
  506. data/spec/unit/webui_user_spec.rb +0 -238
@@ -6,9 +6,9 @@
6
6
  # Licensed under the Apache License, Version 2.0 (the "License");
7
7
  # you may not use this file except in compliance with the License.
8
8
  # You may obtain a copy of the License at
9
- #
9
+ #
10
10
  # http://www.apache.org/licenses/LICENSE-2.0
11
- #
11
+ #
12
12
  # Unless required by applicable law or agreed to in writing, software
13
13
  # distributed under the License is distributed on an "AS IS" BASIS,
14
14
  # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
@@ -17,288 +17,246 @@
17
17
  #
18
18
 
19
19
  require 'chef/mixin/shell_out'
20
- require 'mixlib/shellout'
21
20
  require 'chef/provider/user'
22
21
  require 'openssl'
23
- require 'plist'
24
22
 
25
23
  class Chef
26
24
  class Provider
27
25
  class User
28
- include Chef::Mixin::ShellOut
29
-
30
- #
31
- # The most tricky bit of this provider is the way it deals with user passwords.
32
- # Mac OS X has different password shadow calculations based on the version.
33
- # < 10.7 => password shadow calculation format SALTED-SHA1
34
- # => stored in: /var/db/shadow/hash/#{guid}
35
- # => shadow binary length 68 bytes
36
- # => First 4 bytes salt / Next 64 bytes shadow value
37
- # = 10.7 => password shadow calculation format SALTED-SHA512
38
- # => stored in: /var/db/dslocal/nodes/Default/users/#{name}.plist
39
- # => shadow binary length 68 bytes
40
- # => First 4 bytes salt / Next 64 bytes shadow value
41
- # > 10.7 => password shadow calculation format SALTED-SHA512-PBKDF2
42
- # => stored in: /var/db/dslocal/nodes/Default/users/#{name}.plist
43
- # => shadow binary length 128 bytes
44
- # => Salt / Iterations are stored seperately in the same file
45
- #
46
- # This provider only supports Mac OSX versions 10.7 and above
47
26
  class Dscl < Chef::Provider::User
27
+ include Chef::Mixin::ShellOut
28
+
29
+ NFS_HOME_DIRECTORY = %r{^NFSHomeDirectory: (.*)$}
30
+ AUTHENTICATION_AUTHORITY = %r{^AuthenticationAuthority: (.*)$}
31
+
32
+ def dscl(*args)
33
+ shell_out("dscl . -#{args.join(' ')}")
34
+ end
48
35
 
49
- def define_resource_requirements
50
- super
51
-
52
- requirements.assert(:all_actions) do |a|
53
- a.assertion { mac_osx_version_less_than_10_7? == false }
54
- a.failure_message(Chef::Exceptions::User, "Chef::Provider::User::Dscl only supports Mac OS X versions 10.7 and above.")
55
- end
56
-
57
- requirements.assert(:all_actions) do |a|
58
- a.assertion { ::File.exists?("/usr/bin/dscl") }
59
- a.failure_message(Chef::Exceptions::User, "Cannot find binary '/usr/bin/dscl' on the system for #{@new_resource}!")
60
- end
36
+ def safe_dscl(*args)
37
+ result = dscl(*args)
38
+ return "" if ( args.first =~ /^delete/ ) && ( result.exitstatus != 0 )
39
+ raise(Chef::Exceptions::DsclCommandFailed,"dscl error: #{result.inspect}") unless result.exitstatus == 0
40
+ raise(Chef::Exceptions::DsclCommandFailed,"dscl error: #{result.inspect}") if result.stdout =~ /No such key: /
41
+ return result.stdout
42
+ end
61
43
 
62
- requirements.assert(:all_actions) do |a|
63
- a.assertion { ::File.exists?("/usr/bin/plutil") }
64
- a.failure_message(Chef::Exceptions::User, "Cannot find binary '/usr/bin/plutil' on the system for #{@new_resource}!")
65
- end
44
+ # This is handled in providers/group.rb by Etc.getgrnam()
45
+ # def user_exists?(user)
46
+ # users = safe_dscl("list /Users")
47
+ # !! ( users =~ Regexp.new("\n#{user}\n") )
48
+ # end
66
49
 
67
- requirements.assert(:create, :modify, :manage) do |a|
68
- a.assertion do
69
- if @new_resource.password && mac_osx_version_greater_than_10_7?
70
- # SALTED-SHA512 password shadow hashes are not supported on 10.8 and above.
71
- !salted_sha512?(@new_resource.password)
72
- else
73
- true
74
- end
50
+ # get a free UID greater than 200
51
+ def get_free_uid(search_limit=1000)
52
+ uid = nil; next_uid_guess = 200
53
+ users_uids = safe_dscl("list /Users uid")
54
+ while(next_uid_guess < search_limit + 200)
55
+ if users_uids =~ Regexp.new("#{Regexp.escape(next_uid_guess.to_s)}\n")
56
+ next_uid_guess += 1
57
+ else
58
+ uid = next_uid_guess
59
+ break
75
60
  end
76
- a.failure_message(Chef::Exceptions::User, "SALTED-SHA512 passwords are not supported on Mac 10.8 and above. \
77
- If you want to set the user password using shadow info make sure you specify a SALTED-SHA512-PBKDF2 shadow hash \
78
- in 'password', with the associated 'salt' and 'iterations'.")
79
61
  end
62
+ return uid || raise("uid not found. Exhausted. Searched #{search_limit} times")
63
+ end
80
64
 
81
- requirements.assert(:create, :modify, :manage) do |a|
82
- a.assertion do
83
- if @new_resource.password && mac_osx_version_greater_than_10_7? && salted_sha512_pbkdf2?(@new_resource.password)
84
- # salt and iterations should be specified when
85
- # SALTED-SHA512-PBKDF2 password shadow hash is given
86
- !@new_resource.salt.nil? && !@new_resource.iterations.nil?
87
- else
88
- true
89
- end
90
- end
91
- a.failure_message(Chef::Exceptions::User, "SALTED-SHA512-PBKDF2 shadow hash is given without associated \
92
- 'salt' and 'iterations'. Please specify 'salt' and 'iterations' in order to set the user password using shadow hash.")
65
+ def uid_used?(uid)
66
+ return false unless uid
67
+ users_uids = safe_dscl("list /Users uid")
68
+ !! ( users_uids =~ Regexp.new("#{Regexp.escape(uid.to_s)}\n") )
69
+ end
70
+
71
+ def set_uid
72
+ @new_resource.uid(get_free_uid) if (@new_resource.uid.nil? || @new_resource.uid == '')
73
+ if uid_used?(@new_resource.uid)
74
+ raise(Chef::Exceptions::RequestedUIDUnavailable, "uid #{@new_resource.uid} is already in use")
93
75
  end
76
+ safe_dscl("create /Users/#{@new_resource.username} UniqueID #{@new_resource.uid}")
77
+ end
94
78
 
95
- requirements.assert(:create, :modify, :manage) do |a|
96
- a.assertion do
97
- if @new_resource.password && !mac_osx_version_greater_than_10_7?
98
- # On 10.7 SALTED-SHA512-PBKDF2 is not supported
99
- !salted_sha512_pbkdf2?(@new_resource.password)
100
- else
101
- true
102
- end
79
+ def modify_home
80
+ return safe_dscl("delete /Users/#{@new_resource.username} NFSHomeDirectory") if (@new_resource.home.nil? || @new_resource.home.empty?)
81
+ if @new_resource.supports[:manage_home]
82
+ validate_home_dir_specification!
83
+
84
+ if (@current_resource.home == @new_resource.home) && !new_home_exists?
85
+ ditto_home
86
+ elsif !current_home_exists? && !new_home_exists?
87
+ ditto_home
88
+ elsif current_home_exists?
89
+ move_home
103
90
  end
104
- a.failure_message(Chef::Exceptions::User, "SALTED-SHA512-PBKDF2 shadow hashes are not supported on \
105
- Mac OS X version 10.7. Please specify a SALTED-SHA512 shadow hash in 'password' attribute to set the \
106
- user password using shadow hash.")
107
91
  end
108
-
92
+ safe_dscl("create /Users/#{@new_resource.username} NFSHomeDirectory '#{@new_resource.home}'")
109
93
  end
110
94
 
111
- def load_current_resource
112
- @current_resource = Chef::Resource::User.new(@new_resource.username)
113
- @current_resource.username(@new_resource.username)
95
+ def osx_shadow_hash?(string)
96
+ return !! ( string =~ /^[[:xdigit:]]{1240}$/ )
97
+ end
114
98
 
115
- @user_info = read_user_info
116
- if @user_info
117
- @current_resource.uid(dscl_get(@user_info, :uid))
118
- @current_resource.gid(dscl_get(@user_info, :gid))
119
- @current_resource.home(dscl_get(@user_info, :home))
120
- @current_resource.shell(dscl_get(@user_info, :shell))
121
- @current_resource.comment(dscl_get(@user_info, :comment))
122
- @authentication_authority = dscl_get(@user_info, :auth_authority)
99
+ def osx_salted_sha1?(string)
100
+ return !! ( string =~ /^[[:xdigit:]]{48}$/ )
101
+ end
123
102
 
124
- if @new_resource.password && dscl_get(@user_info, :password) == "********"
125
- # A password is set. Let's get the password information from shadow file
126
- shadow_hash_binary = dscl_get(@user_info, :shadow_hash)
103
+ def guid
104
+ safe_dscl("read /Users/#{@new_resource.username} GeneratedUID").gsub(/GeneratedUID: /,"").strip
105
+ end
127
106
 
128
- # Calling shell_out directly since we want to give an input stream
129
- shadow_hash_xml = convert_binary_plist_to_xml(shadow_hash_binary.string)
130
- shadow_hash = Plist::parse_xml(shadow_hash_xml)
107
+ def shadow_hash_set?
108
+ user_data = safe_dscl("read /Users/#{@new_resource.username}")
109
+ if user_data =~ /AuthenticationAuthority: / && user_data =~ /ShadowHash/
110
+ true
111
+ else
112
+ false
113
+ end
114
+ end
131
115
 
132
- if shadow_hash["SALTED-SHA512"]
133
- # Convert the shadow value from Base64 encoding to hex before consuming them
134
- @password_shadow_conversion_algorithm = "SALTED-SHA512"
135
- @current_resource.password(shadow_hash["SALTED-SHA512"].string.unpack('H*').first)
136
- elsif shadow_hash["SALTED-SHA512-PBKDF2"]
137
- @password_shadow_conversion_algorithm = "SALTED-SHA512-PBKDF2"
138
- # Convert the entropy from Base64 encoding to hex before consuming them
139
- @current_resource.password(shadow_hash["SALTED-SHA512-PBKDF2"]["entropy"].string.unpack('H*').first)
140
- @current_resource.iterations(shadow_hash["SALTED-SHA512-PBKDF2"]["iterations"])
141
- # Convert the salt from Base64 encoding to hex before consuming them
142
- @current_resource.salt(shadow_hash["SALTED-SHA512-PBKDF2"]["salt"].string.unpack('H*').first)
116
+ def modify_password
117
+ if @new_resource.password
118
+ shadow_hash = nil
119
+
120
+ Chef::Log.debug("#{new_resource} updating password")
121
+ if osx_shadow_hash?(@new_resource.password)
122
+ shadow_hash = @new_resource.password.upcase
123
+ else
124
+ if osx_salted_sha1?(@new_resource.password)
125
+ salted_sha1 = @new_resource.password.upcase
143
126
  else
144
- raise(Chef::Exceptions::User,"Unknown shadow_hash format: #{shadow_hash.keys.join(' ')}")
127
+ hex_salt = ""
128
+ OpenSSL::Random.random_bytes(10).each_byte { |b| hex_salt << b.to_i.to_s(16) }
129
+ hex_salt = hex_salt.slice(0...8)
130
+ salt = [hex_salt].pack("H*")
131
+ sha1 = ::OpenSSL::Digest::SHA1.hexdigest(salt+@new_resource.password)
132
+ salted_sha1 = (hex_salt+sha1).upcase
145
133
  end
134
+ shadow_hash = String.new("00000000"*155)
135
+ shadow_hash[168] = salted_sha1
136
+ end
137
+
138
+ ::File.open("/var/db/shadow/hash/#{guid}",'w',0600) do |output|
139
+ output.puts shadow_hash
140
+ end
141
+
142
+ unless shadow_hash_set?
143
+ safe_dscl("append /Users/#{@new_resource.username} AuthenticationAuthority ';ShadowHash;'")
146
144
  end
147
-
148
- convert_group_name if @new_resource.gid
149
- else
150
- @user_exists = false
151
- Chef::Log.debug("#{@new_resource} user does not exist")
152
145
  end
153
-
154
- @current_resource
155
146
  end
156
147
 
157
- #
158
- # Provider Actions
159
- #
148
+ def load_current_resource
149
+ super
150
+ raise Chef::Exceptions::User, "Could not find binary /usr/bin/dscl for #{@new_resource}" unless ::File.exists?("/usr/bin/dscl")
151
+ end
160
152
 
161
153
  def create_user
162
154
  dscl_create_user
163
- # set_password modifies the plist file of the user directly. So update
164
- # the password first before making any modifications to the user.
165
- set_password
166
155
  dscl_create_comment
167
- dscl_set_uid
156
+ set_uid
168
157
  dscl_set_gid
169
- dscl_set_home
158
+ modify_home
170
159
  dscl_set_shell
160
+ modify_password
171
161
  end
172
-
162
+
173
163
  def manage_user
174
- # set_password modifies the plist file of the user directly. So update
175
- # the password first before making any modifications to the user.
176
- set_password if diverged_password?
177
164
  dscl_create_user if diverged?(:username)
178
165
  dscl_create_comment if diverged?(:comment)
179
- dscl_set_uid if diverged?(:uid)
166
+ set_uid if diverged?(:uid)
180
167
  dscl_set_gid if diverged?(:gid)
181
- dscl_set_home if diverged?(:home)
168
+ modify_home if diverged?(:home)
182
169
  dscl_set_shell if diverged?(:shell)
170
+ modify_password if diverged?(:password)
183
171
  end
184
-
185
- #
186
- # Action Helpers
187
- #
188
-
189
- #
190
- # Create a user using dscl
191
- #
172
+
192
173
  def dscl_create_user
193
- run_dscl("create /Users/#{@new_resource.username}")
174
+ safe_dscl("create /Users/#{@new_resource.username}")
194
175
  end
195
-
196
- #
197
- # Saves the specified Chef user `comment` into RealName attribute
198
- # of Mac user.
199
- #
176
+
200
177
  def dscl_create_comment
201
- run_dscl("create /Users/#{@new_resource.username} RealName '#{@new_resource.comment}'")
202
- end
203
-
204
- #
205
- # Sets the user id for the user using dscl.
206
- # If a `uid` is not specified, it finds the next available one starting
207
- # from 200 if `system` is set, 500 otherwise.
208
- #
209
- def dscl_set_uid
210
- @new_resource.uid(get_free_uid) if (@new_resource.uid.nil? || @new_resource.uid == '')
211
-
212
- if uid_used?(@new_resource.uid)
213
- raise(Chef::Exceptions::RequestedUIDUnavailable, "uid #{@new_resource.uid} is already in use")
214
- end
215
-
216
- run_dscl("create /Users/#{@new_resource.username} UniqueID #{@new_resource.uid}")
178
+ safe_dscl("create /Users/#{@new_resource.username} RealName '#{@new_resource.comment}'")
217
179
  end
218
-
219
- #
220
- # Find the next available uid on the system. starting with 200 if `system` is set,
221
- # 500 otherwise.
222
- #
223
- def get_free_uid(search_limit=1000)
224
- uid = nil
225
- base_uid = @new_resource.system ? 200 : 500
226
- next_uid_guess = base_uid
227
- users_uids = run_dscl("list /Users uid")
228
- while(next_uid_guess < search_limit + base_uid)
229
- if users_uids =~ Regexp.new("#{Regexp.escape(next_uid_guess.to_s)}\n")
230
- next_uid_guess += 1
231
- else
232
- uid = next_uid_guess
233
- break
234
- end
235
- end
236
- return uid || raise("uid not found. Exhausted. Searched #{search_limit} times")
237
- end
238
-
239
- #
240
- # Returns true if uid is in use by a different account, false otherwise.
241
- #
242
- def uid_used?(uid)
243
- return false unless uid
244
- users_uids = run_dscl("list /Users uid")
245
- !! ( users_uids =~ Regexp.new("#{Regexp.escape(uid.to_s)}\n") )
246
- end
247
-
248
- #
249
- # Sets the group id for the user using dscl. Fails if a group doesn't
250
- # exist on the system with given group id.
251
- #
180
+
252
181
  def dscl_set_gid
253
182
  unless @new_resource.gid && @new_resource.gid.to_s.match(/^\d+$/)
254
183
  begin
255
- possible_gid = run_dscl("read /Groups/#{@new_resource.gid} PrimaryGroupID").split(" ").last
184
+ possible_gid = safe_dscl("read /Groups/#{@new_resource.gid} PrimaryGroupID").split(" ").last
256
185
  rescue Chef::Exceptions::DsclCommandFailed => e
257
186
  raise Chef::Exceptions::GroupIDNotFound.new("Group not found for #{@new_resource.gid} when creating user #{@new_resource.username}")
258
187
  end
259
188
  @new_resource.gid(possible_gid) if possible_gid && possible_gid.match(/^\d+$/)
260
189
  end
261
- run_dscl("create /Users/#{@new_resource.username} PrimaryGroupID '#{@new_resource.gid}'")
190
+ safe_dscl("create /Users/#{@new_resource.username} PrimaryGroupID '#{@new_resource.gid}'")
262
191
  end
263
-
264
- #
265
- # Sets the home directory for the user. If `:manage_home` is set home
266
- # directory is managed (moved / created) for the user.
267
- #
268
- def dscl_set_home
269
- if @new_resource.home.nil? || @new_resource.home.empty?
270
- run_dscl("delete /Users/#{@new_resource.username} NFSHomeDirectory")
271
- return
192
+
193
+ def dscl_set_shell
194
+ if @new_resource.password || ::File.exists?("#{@new_resource.shell}")
195
+ safe_dscl("create /Users/#{@new_resource.username} UserShell '#{@new_resource.shell}'")
196
+ else
197
+ safe_dscl("create /Users/#{@new_resource.username} UserShell '/usr/bin/false'")
272
198
  end
273
-
199
+ end
200
+
201
+ def remove_user
274
202
  if @new_resource.supports[:manage_home]
275
- validate_home_dir_specification!
276
-
277
- if (@current_resource.home == @new_resource.home) && !new_home_exists?
278
- ditto_home
279
- elsif !current_home_exists? && !new_home_exists?
280
- ditto_home
281
- elsif current_home_exists?
282
- move_home
203
+ user_info = safe_dscl("read /Users/#{@new_resource.username}")
204
+ if nfs_home_match = user_info.match(NFS_HOME_DIRECTORY)
205
+ #nfs_home = safe_dscl("read /Users/#{@new_resource.username} NFSHomeDirectory")
206
+ #nfs_home.gsub!(/NFSHomeDirectory: /,"").gsub!(/\n$/,"")
207
+ nfs_home = nfs_home_match[1]
208
+ FileUtils.rm_rf(nfs_home)
283
209
  end
284
210
  end
285
- run_dscl("create /Users/#{@new_resource.username} NFSHomeDirectory '#{@new_resource.home}'")
211
+ # remove the user from its groups
212
+ groups = []
213
+ Etc.group do |group|
214
+ groups << group.name if group.mem.include?(@new_resource.username)
215
+ end
216
+ groups.each do |group_name|
217
+ safe_dscl("delete /Groups/#{group_name} GroupMembership '#{@new_resource.username}'")
218
+ end
219
+ # remove user account
220
+ safe_dscl("delete /Users/#{@new_resource.username}")
221
+ end
222
+
223
+ def locked?
224
+ user_info = safe_dscl("read /Users/#{@new_resource.username}")
225
+ if auth_authority_md = AUTHENTICATION_AUTHORITY.match(user_info)
226
+ !!(auth_authority_md[1] =~ /DisabledUser/ )
227
+ else
228
+ false
229
+ end
230
+ end
231
+
232
+ def check_lock
233
+ return @locked = locked?
286
234
  end
287
235
 
236
+ def lock_user
237
+ safe_dscl("append /Users/#{@new_resource.username} AuthenticationAuthority ';DisabledUser;'")
238
+ end
239
+
240
+ def unlock_user
241
+ auth_info = safe_dscl("read /Users/#{@new_resource.username} AuthenticationAuthority")
242
+ auth_string = auth_info.gsub(/AuthenticationAuthority: /,"").gsub(/;DisabledUser;/,"").strip#.gsub!(/[; ]*$/,"")
243
+ safe_dscl("create /Users/#{@new_resource.username} AuthenticationAuthority '#{auth_string}'")
244
+ end
245
+
288
246
  def validate_home_dir_specification!
289
247
  unless @new_resource.home =~ /^\//
290
- raise(Chef::Exceptions::InvalidHomeDirectory,"invalid path spec for User: '#{@new_resource.username}', home directory: '#{@new_resource.home}'")
248
+ raise(Chef::Exceptions::InvalidHomeDirectory,"invalid path spec for User: '#{@new_resource.username}', home directory: '#{@new_resource.home}'")
291
249
  end
292
250
  end
293
-
251
+
294
252
  def current_home_exists?
295
253
  ::File.exist?("#{@current_resource.home}")
296
254
  end
297
-
255
+
298
256
  def new_home_exists?
299
- ::File.exist?("#{@new_resource.home}")
257
+ ::File.exist?("#{@new_resource.home}")
300
258
  end
301
-
259
+
302
260
  def ditto_home
303
261
  skel = "/System/Library/User Template/English.lproj"
304
262
  raise(Chef::Exceptions::User,"can't find skel at: #{skel}") unless ::File.exists?(skel)
@@ -308,7 +266,7 @@ user password using shadow hash.")
308
266
 
309
267
  def move_home
310
268
  Chef::Log.debug("#{@new_resource} moving #{self} home from #{@current_resource.home} to #{@new_resource.home}")
311
-
269
+
312
270
  src = @current_resource.home
313
271
  FileUtils.mkdir_p(@new_resource.home)
314
272
  files = ::Dir.glob("#{src}/*", ::File::FNM_DOTMATCH) - ["#{src}/.","#{src}/.."]
@@ -316,365 +274,14 @@ user password using shadow hash.")
316
274
  ::FileUtils.rmdir(src)
317
275
  ::FileUtils.chown_R(@new_resource.username,@new_resource.gid.to_s,@new_resource.home)
318
276
  end
319
-
320
- #
321
- # Sets the shell for the user using dscl.
322
- #
323
- def dscl_set_shell
324
- if @new_resource.shell || ::File.exists?("#{@new_resource.shell}")
325
- run_dscl("create /Users/#{@new_resource.username} UserShell '#{@new_resource.shell}'")
326
- else
327
- run_dscl("create /Users/#{@new_resource.username} UserShell '/usr/bin/false'")
328
- end
329
- end
330
-
331
- #
332
- # Sets the password for the user based on given password parameters.
333
- # Chef supports specifying plain-text passwords and password shadow
334
- # hash data.
335
- #
336
- def set_password
337
- # Return if there is no password to set
338
- return if @new_resource.password.nil?
339
-
340
- shadow_info = prepare_password_shadow_info
341
-
342
- # Shadow info is saved as binary plist. Convert the info to binary plist.
343
- shadow_info_binary = StringIO.new
344
- command = Mixlib::ShellOut.new("plutil -convert binary1 -o - -",
345
- :input => shadow_info.to_plist, :live_stream => shadow_info_binary)
346
- command.run_command
347
-
348
- if @user_info.nil?
349
- # User is just created. read_user_info() will read the fresh information
350
- # for the user with a cache flush. However with experimentation we've seen
351
- # that dscl cache is not immediately updated after the creation of the user
352
- # This is odd and needs to be investigated further.
353
- sleep 3
354
- @user_info = read_user_info
355
- end
356
-
357
- # Replace the shadow info in user's plist
358
- dscl_set(@user_info, :shadow_hash, shadow_info_binary)
359
- save_user_info(@user_info)
360
- end
361
-
362
- #
363
- # Prepares the password shadow info based on the platform version.
364
- #
365
- def prepare_password_shadow_info
366
- shadow_info = { }
367
- entropy = nil
368
- salt = nil
369
- iterations = nil
370
-
371
- if mac_osx_version_10_7?
372
- hash_value = if salted_sha512?(@new_resource.password)
373
- @new_resource.password
374
- else
375
- # Create a random 4 byte salt
376
- salt = OpenSSL::Random.random_bytes(4)
377
- encoded_password = OpenSSL::Digest::SHA512.hexdigest(salt + @new_resource.password)
378
- hash_value = salt.unpack('H*').first + encoded_password
379
- end
380
-
381
- shadow_info["SALTED-SHA512"] = StringIO.new
382
- shadow_info["SALTED-SHA512"].string = convert_to_binary(hash_value)
383
- shadow_info
384
- else
385
- if salted_sha512_pbkdf2?(@new_resource.password)
386
- entropy = convert_to_binary(@new_resource.password)
387
- salt = convert_to_binary(@new_resource.salt)
388
- iterations = @new_resource.iterations
389
- else
390
- salt = OpenSSL::Random.random_bytes(32)
391
- iterations = @new_resource.iterations # Use the default if not specified by the user
392
-
393
- entropy = OpenSSL::PKCS5::pbkdf2_hmac(
394
- @new_resource.password,
395
- salt,
396
- iterations,
397
- 128,
398
- OpenSSL::Digest::SHA512.new
399
- )
400
- end
401
-
402
- pbkdf_info = { }
403
- pbkdf_info["entropy"] = StringIO.new
404
- pbkdf_info["entropy"].string = entropy
405
- pbkdf_info["salt"] = StringIO.new
406
- pbkdf_info["salt"].string = salt
407
- pbkdf_info["iterations"] = iterations
408
-
409
- shadow_info["SALTED-SHA512-PBKDF2"] = pbkdf_info
410
- end
411
-
412
- shadow_info
413
- end
414
-
415
- #
416
- # Removes the user from the system after removing user from his groups
417
- # and deleting home directory if needed.
418
- #
419
- def remove_user
420
- if @new_resource.supports[:manage_home]
421
- # Remove home directory
422
- FileUtils.rm_rf(@current_resource.home)
423
- end
424
-
425
- # Remove the user from its groups
426
- run_dscl("list /Groups").each_line do |group|
427
- if member_of_group?(group.chomp)
428
- run_dscl("delete /Groups/#{group.chomp} GroupMembership '#{@new_resource.username}'")
429
- end
430
- end
431
-
432
- # Remove user account
433
- run_dscl("delete /Users/#{@new_resource.username}")
434
- end
435
-
436
- #
437
- # Locks the user.
438
- #
439
- def lock_user
440
- run_dscl("append /Users/#{@new_resource.username} AuthenticationAuthority ';DisabledUser;'")
441
- end
442
-
443
- #
444
- # Unlocks the user
445
- #
446
- def unlock_user
447
- auth_string = @authentication_authority.gsub(/AuthenticationAuthority: /,"").gsub(/;DisabledUser;/,"").strip
448
- run_dscl("create /Users/#{@new_resource.username} AuthenticationAuthority '#{auth_string}'")
449
- end
450
-
451
- #
452
- # Returns true if the user is locked, false otherwise.
453
- #
454
- def locked?
455
- if @authentication_authority
456
- !!(@authentication_authority =~ /DisabledUser/ )
457
- else
458
- false
459
- end
460
- end
461
-
462
- #
463
- # This is the interface base User provider requires to provide idempotency.
464
- #
465
- def check_lock
466
- return @locked = locked?
467
- end
468
-
469
- #
470
- # Helper functions
471
- #
472
-
473
- #
474
- # Returns true if the system state and desired state is different for
475
- # given attribute.
476
- #
277
+
477
278
  def diverged?(parameter)
478
279
  parameter_updated?(parameter) && (not @new_resource.send(parameter).nil?)
479
280
  end
480
-
281
+
481
282
  def parameter_updated?(parameter)
482
283
  not (@new_resource.send(parameter) == @current_resource.send(parameter))
483
284
  end
484
-
485
- #
486
- # We need a special check function for password since we support both
487
- # plain text and shadow hash data.
488
- #
489
- # Checks if password needs update based on platform version and the
490
- # type of the password specified.
491
- #
492
- def diverged_password?
493
- return false if @new_resource.password.nil?
494
-
495
- # Dscl provider supports both plain text passwords and shadow hashes.
496
- if mac_osx_version_10_7?
497
- if salted_sha512?(@new_resource.password)
498
- diverged?(:password)
499
- else
500
- !salted_sha512_password_match?
501
- end
502
- else
503
- # When a system is upgraded to a version 10.7+ shadow hashes of the users
504
- # will be updated when the user logs in. So it's possible that we will have
505
- # SALTED-SHA512 password in the current_resource. In that case we will force
506
- # password to be updated.
507
- return true if salted_sha512?(@current_resource.password)
508
-
509
- if salted_sha512_pbkdf2?(@new_resource.password)
510
- diverged?(:password) || diverged?(:salt) || diverged?(:iterations)
511
- else
512
- !salted_sha512_pbkdf2_password_match?
513
- end
514
- end
515
- end
516
-
517
- #
518
- # Returns true if user is member of the specified group, false otherwise.
519
- #
520
- def member_of_group?(group_name)
521
- membership_info = ""
522
- begin
523
- membership_info = run_dscl("read /Groups/#{group_name}")
524
- rescue Chef::Exceptions::DsclCommandFailed
525
- # Raised if the group doesn't contain any members
526
- end
527
- # Output is something like:
528
- # GroupMembership: root admin etc
529
- members = membership_info.split(" ")
530
- members.shift # Get rid of GroupMembership: string
531
- members.include?(@new_resource.username)
532
- end
533
-
534
- #
535
- # DSCL Helper functions
536
- #
537
-
538
- # A simple map of Chef's terms to DSCL's terms.
539
- DSCL_PROPERTY_MAP = {
540
- :uid => "generateduid",
541
- :gid => "gid",
542
- :home => "home",
543
- :shell => "shell",
544
- :comment => "realname",
545
- :password => "passwd",
546
- :auth_authority => "authentication_authority",
547
- :shadow_hash => "ShadowHashData"
548
- }.freeze
549
-
550
- # Directory where the user plist files are stored for versions 10.7 and above
551
- USER_PLIST_DIRECTORY = "/var/db/dslocal/nodes/Default/users".freeze
552
-
553
- #
554
- # Reads the user plist and returns a hash keyed with DSCL properties specified
555
- # in DSCL_PROPERTY_MAP. Return nil if the user is not found.
556
- #
557
- def read_user_info
558
- user_info = nil
559
-
560
- # We flush the cache here in order to make sure that we read fresh information
561
- # for the user.
562
- shell_out("dscacheutil '-flushcache'")
563
-
564
- begin
565
- user_plist_file = "#{USER_PLIST_DIRECTORY}/#{@new_resource.username}.plist"
566
- user_plist_info = run_plutil("convert xml1 -o - #{user_plist_file}")
567
- user_info = Plist::parse_xml(user_plist_info)
568
- rescue Chef::Exceptions::PlistUtilCommandFailed
569
- end
570
-
571
- user_info
572
- end
573
-
574
- #
575
- # Saves the given hash keyed with DSCL properties specified
576
- # in DSCL_PROPERTY_MAP to the disk.
577
- #
578
- def save_user_info(user_info)
579
- user_plist_file = "#{USER_PLIST_DIRECTORY}/#{@new_resource.username}.plist"
580
- Plist::Emit.save_plist(user_info, user_plist_file)
581
- run_plutil("convert binary1 #{user_plist_file}")
582
- end
583
-
584
- #
585
- # Sets a value in user information hash using Chef attributes as keys.
586
- #
587
- def dscl_set(user_hash, key, value)
588
- raise "Unknown dscl key #{key}" unless DSCL_PROPERTY_MAP.keys.include?(key)
589
- user_hash[DSCL_PROPERTY_MAP[key]] = [ value ]
590
- user_hash
591
- end
592
-
593
- #
594
- # Gets a value from user information hash using Chef attributes as keys.
595
- #
596
- def dscl_get(user_hash, key)
597
- raise "Unknown dscl key #{key}" unless DSCL_PROPERTY_MAP.keys.include?(key)
598
- # DSCL values are set as arrays
599
- value = user_hash[DSCL_PROPERTY_MAP[key]]
600
- value.nil? ? value : value.first
601
- end
602
-
603
- #
604
- # System Helpets
605
- #
606
-
607
- def mac_osx_version
608
- # This provider will only be invoked on node[:platform] == "mac_os_x"
609
- # We do not check or assert that here.
610
- node[:platform_version]
611
- end
612
-
613
- def mac_osx_version_10_7?
614
- mac_osx_version.start_with?("10.7.")
615
- end
616
-
617
- def mac_osx_version_less_than_10_7?
618
- versions = mac_osx_version.split(".")
619
- # Make integer comparison in order not to report 10.10 less than 10.7
620
- (versions[0].to_i <= 10 && versions[1].to_i < 7)
621
- end
622
-
623
- def mac_osx_version_greater_than_10_7?
624
- versions = mac_osx_version.split(".")
625
- # Make integer comparison in order not to report 10.10 less than 10.7
626
- (versions[0].to_i >= 10 && versions[1].to_i > 7)
627
- end
628
-
629
- def run_dscl(*args)
630
- result = shell_out("dscl . -#{args.join(' ')}")
631
- return "" if ( args.first =~ /^delete/ ) && ( result.exitstatus != 0 )
632
- raise(Chef::Exceptions::DsclCommandFailed,"dscl error: #{result.inspect}") unless result.exitstatus == 0
633
- raise(Chef::Exceptions::DsclCommandFailed,"dscl error: #{result.inspect}") if result.stdout =~ /No such key: /
634
- result.stdout
635
- end
636
-
637
- def run_plutil(*args)
638
- result = shell_out("plutil -#{args.join(' ')}")
639
- raise(Chef::Exceptions::PlistUtilCommandFailed,"plutil error: #{result.inspect}") unless result.exitstatus == 0
640
- result.stdout
641
- end
642
-
643
- def convert_binary_plist_to_xml(binary_plist_string)
644
- Mixlib::ShellOut.new("plutil -convert xml1 -o - -", :input => binary_plist_string).run_command.stdout
645
- end
646
-
647
- def convert_to_binary(string)
648
- string.unpack('a2'*(string.size/2)).collect { |i| i.hex.chr }.join
649
- end
650
-
651
- def salted_sha512?(string)
652
- !!(string =~ /^[[:xdigit:]]{136}$/)
653
- end
654
-
655
- def salted_sha512_password_match?
656
- # Salt is included in the first 4 bytes of shadow data
657
- salt = @current_resource.password.slice(0,8)
658
- shadow = OpenSSL::Digest::SHA512.hexdigest(convert_to_binary(salt) + @new_resource.password)
659
- @current_resource.password == salt + shadow
660
- end
661
-
662
- def salted_sha512_pbkdf2?(string)
663
- !!(string =~ /^[[:xdigit:]]{256}$/)
664
- end
665
-
666
- def salted_sha512_pbkdf2_password_match?
667
- salt = convert_to_binary(@current_resource.salt)
668
-
669
- OpenSSL::PKCS5::pbkdf2_hmac(
670
- @new_resource.password,
671
- salt,
672
- @current_resource.iterations,
673
- 128,
674
- OpenSSL::Digest::SHA512.new
675
- ).unpack('H*').first == @current_resource.password
676
- end
677
-
678
285
  end
679
286
  end
680
287
  end