tecnh-chef 0.9.9.vpc

Sign up to get free protection for your applications and to get access to all the features.
Files changed (337) hide show
  1. data/LICENSE +201 -0
  2. data/README.rdoc +172 -0
  3. data/bin/chef-client +26 -0
  4. data/bin/chef-solo +25 -0
  5. data/bin/knife +26 -0
  6. data/bin/shef +34 -0
  7. data/distro/README +2 -0
  8. data/distro/common/man/man1/chef-indexer.1 +42 -0
  9. data/distro/common/man/man1/chef-server-webui.1 +106 -0
  10. data/distro/common/man/man1/chef-server.1 +107 -0
  11. data/distro/common/man/man1/chef-solr-indexer.1 +55 -0
  12. data/distro/common/man/man1/chef-solr.1 +55 -0
  13. data/distro/common/man/man8/chef-client.8 +63 -0
  14. data/distro/common/man/man8/chef-solo.8 +57 -0
  15. data/distro/common/man/man8/chef-solr-rebuild.8 +37 -0
  16. data/distro/common/man/man8/knife.8 +1339 -0
  17. data/distro/common/man/man8/shef.8 +45 -0
  18. data/distro/common/markdown/README +3 -0
  19. data/distro/common/markdown/knife.mkd +825 -0
  20. data/distro/debian/etc/default/chef-client +4 -0
  21. data/distro/debian/etc/default/chef-server +6 -0
  22. data/distro/debian/etc/default/chef-server-webui +6 -0
  23. data/distro/debian/etc/default/chef-solr +4 -0
  24. data/distro/debian/etc/default/chef-solr-indexer +4 -0
  25. data/distro/debian/etc/init.d/chef-client +175 -0
  26. data/distro/debian/etc/init.d/chef-server +120 -0
  27. data/distro/debian/etc/init.d/chef-server-webui +121 -0
  28. data/distro/debian/etc/init.d/chef-solr +177 -0
  29. data/distro/debian/etc/init.d/chef-solr-indexer +176 -0
  30. data/distro/debian/etc/init/chef-client.conf +17 -0
  31. data/distro/debian/etc/init/chef-server-webui.conf +17 -0
  32. data/distro/debian/etc/init/chef-server.conf +17 -0
  33. data/distro/debian/etc/init/chef-solr-indexer.conf +17 -0
  34. data/distro/debian/etc/init/chef-solr.conf +17 -0
  35. data/distro/redhat/etc/init.d/chef-client +106 -0
  36. data/distro/redhat/etc/init.d/chef-server +112 -0
  37. data/distro/redhat/etc/init.d/chef-server-webui +112 -0
  38. data/distro/redhat/etc/init.d/chef-solr +104 -0
  39. data/distro/redhat/etc/init.d/chef-solr-indexer +104 -0
  40. data/distro/redhat/etc/logrotate.d/chef-client +8 -0
  41. data/distro/redhat/etc/logrotate.d/chef-server +8 -0
  42. data/distro/redhat/etc/logrotate.d/chef-server-webui +8 -0
  43. data/distro/redhat/etc/logrotate.d/chef-solr +8 -0
  44. data/distro/redhat/etc/logrotate.d/chef-solr-indexer +8 -0
  45. data/distro/redhat/etc/sysconfig/chef-client +15 -0
  46. data/distro/redhat/etc/sysconfig/chef-server +14 -0
  47. data/distro/redhat/etc/sysconfig/chef-server-webui +14 -0
  48. data/distro/redhat/etc/sysconfig/chef-solr +8 -0
  49. data/distro/redhat/etc/sysconfig/chef-solr-indexer +7 -0
  50. data/lib/chef.rb +39 -0
  51. data/lib/chef/api_client.rb +264 -0
  52. data/lib/chef/application.rb +125 -0
  53. data/lib/chef/application/agent.rb +18 -0
  54. data/lib/chef/application/client.rb +237 -0
  55. data/lib/chef/application/knife.rb +160 -0
  56. data/lib/chef/application/solo.rb +216 -0
  57. data/lib/chef/applications.rb +4 -0
  58. data/lib/chef/cache.rb +61 -0
  59. data/lib/chef/cache/checksum.rb +91 -0
  60. data/lib/chef/certificate.rb +194 -0
  61. data/lib/chef/checksum.rb +118 -0
  62. data/lib/chef/client.rb +340 -0
  63. data/lib/chef/config.rb +240 -0
  64. data/lib/chef/cookbook/cookbook_collection.rb +44 -0
  65. data/lib/chef/cookbook/file_system_file_vendor.rb +54 -0
  66. data/lib/chef/cookbook/file_vendor.rb +48 -0
  67. data/lib/chef/cookbook/metadata.rb +500 -0
  68. data/lib/chef/cookbook/metadata/version.rb +87 -0
  69. data/lib/chef/cookbook/remote_file_vendor.rb +80 -0
  70. data/lib/chef/cookbook/syntax_check.rb +136 -0
  71. data/lib/chef/cookbook_loader.rb +225 -0
  72. data/lib/chef/cookbook_site_streaming_uploader.rb +215 -0
  73. data/lib/chef/cookbook_uploader.rb +102 -0
  74. data/lib/chef/cookbook_version.rb +877 -0
  75. data/lib/chef/couchdb.rb +239 -0
  76. data/lib/chef/daemon.rb +172 -0
  77. data/lib/chef/data_bag.rb +215 -0
  78. data/lib/chef/data_bag_item.rb +228 -0
  79. data/lib/chef/exceptions.rb +61 -0
  80. data/lib/chef/file_access_control.rb +140 -0
  81. data/lib/chef/file_cache.rb +218 -0
  82. data/lib/chef/handler.rb +163 -0
  83. data/lib/chef/handler/json_file.rb +58 -0
  84. data/lib/chef/index_queue.rb +29 -0
  85. data/lib/chef/index_queue/amqp_client.rb +113 -0
  86. data/lib/chef/index_queue/consumer.rb +76 -0
  87. data/lib/chef/index_queue/indexable.rb +76 -0
  88. data/lib/chef/knife.rb +496 -0
  89. data/lib/chef/knife/bootstrap.rb +167 -0
  90. data/lib/chef/knife/bootstrap/centos5-gems.erb +50 -0
  91. data/lib/chef/knife/bootstrap/fedora13-gems.erb +47 -0
  92. data/lib/chef/knife/bootstrap/ubuntu10.04-apt.erb +41 -0
  93. data/lib/chef/knife/bootstrap/ubuntu10.04-gems.erb +53 -0
  94. data/lib/chef/knife/client_bulk_delete.rb +40 -0
  95. data/lib/chef/knife/client_create.rb +70 -0
  96. data/lib/chef/knife/client_delete.rb +45 -0
  97. data/lib/chef/knife/client_edit.rb +45 -0
  98. data/lib/chef/knife/client_list.rb +40 -0
  99. data/lib/chef/knife/client_reregister.rb +56 -0
  100. data/lib/chef/knife/client_show.rb +50 -0
  101. data/lib/chef/knife/configure.rb +140 -0
  102. data/lib/chef/knife/configure_client.rb +50 -0
  103. data/lib/chef/knife/cookbook_bulk_delete.rb +58 -0
  104. data/lib/chef/knife/cookbook_create.rb +209 -0
  105. data/lib/chef/knife/cookbook_delete.rb +143 -0
  106. data/lib/chef/knife/cookbook_download.rb +130 -0
  107. data/lib/chef/knife/cookbook_list.rb +41 -0
  108. data/lib/chef/knife/cookbook_metadata.rb +82 -0
  109. data/lib/chef/knife/cookbook_metadata_from_file.rb +40 -0
  110. data/lib/chef/knife/cookbook_show.rb +98 -0
  111. data/lib/chef/knife/cookbook_site_download.rb +58 -0
  112. data/lib/chef/knife/cookbook_site_list.rb +56 -0
  113. data/lib/chef/knife/cookbook_site_search.rb +51 -0
  114. data/lib/chef/knife/cookbook_site_share.rb +108 -0
  115. data/lib/chef/knife/cookbook_site_show.rb +57 -0
  116. data/lib/chef/knife/cookbook_site_unshare.rb +52 -0
  117. data/lib/chef/knife/cookbook_site_vendor.rb +127 -0
  118. data/lib/chef/knife/cookbook_test.rb +82 -0
  119. data/lib/chef/knife/cookbook_upload.rb +95 -0
  120. data/lib/chef/knife/data_bag_create.rb +59 -0
  121. data/lib/chef/knife/data_bag_delete.rb +48 -0
  122. data/lib/chef/knife/data_bag_edit.rb +50 -0
  123. data/lib/chef/knife/data_bag_from_file.rb +44 -0
  124. data/lib/chef/knife/data_bag_list.rb +43 -0
  125. data/lib/chef/knife/data_bag_show.rb +41 -0
  126. data/lib/chef/knife/ec2_instance_data.rb +46 -0
  127. data/lib/chef/knife/ec2_server_create.rb +211 -0
  128. data/lib/chef/knife/ec2_server_delete.rb +90 -0
  129. data/lib/chef/knife/ec2_server_list.rb +88 -0
  130. data/lib/chef/knife/index_rebuild.rb +51 -0
  131. data/lib/chef/knife/node_bulk_delete.rb +43 -0
  132. data/lib/chef/knife/node_create.rb +47 -0
  133. data/lib/chef/knife/node_delete.rb +44 -0
  134. data/lib/chef/knife/node_edit.rb +44 -0
  135. data/lib/chef/knife/node_from_file.rb +42 -0
  136. data/lib/chef/knife/node_list.rb +41 -0
  137. data/lib/chef/knife/node_run_list_add.rb +64 -0
  138. data/lib/chef/knife/node_run_list_remove.rb +45 -0
  139. data/lib/chef/knife/node_show.rb +54 -0
  140. data/lib/chef/knife/rackspace_server_create.rb +156 -0
  141. data/lib/chef/knife/rackspace_server_delete.rb +57 -0
  142. data/lib/chef/knife/rackspace_server_list.rb +59 -0
  143. data/lib/chef/knife/recipe_list.rb +32 -0
  144. data/lib/chef/knife/role_bulk_delete.rb +44 -0
  145. data/lib/chef/knife/role_create.rb +52 -0
  146. data/lib/chef/knife/role_delete.rb +44 -0
  147. data/lib/chef/knife/role_edit.rb +45 -0
  148. data/lib/chef/knife/role_from_file.rb +46 -0
  149. data/lib/chef/knife/role_list.rb +40 -0
  150. data/lib/chef/knife/role_show.rb +51 -0
  151. data/lib/chef/knife/search.rb +94 -0
  152. data/lib/chef/knife/slicehost_images_list.rb +53 -0
  153. data/lib/chef/knife/slicehost_server_create.rb +103 -0
  154. data/lib/chef/knife/slicehost_server_delete.rb +61 -0
  155. data/lib/chef/knife/slicehost_server_list.rb +64 -0
  156. data/lib/chef/knife/ssh.rb +311 -0
  157. data/lib/chef/knife/status.rb +70 -0
  158. data/lib/chef/knife/terremark_server_create.rb +152 -0
  159. data/lib/chef/knife/terremark_server_delete.rb +87 -0
  160. data/lib/chef/knife/terremark_server_list.rb +77 -0
  161. data/lib/chef/log.rb +61 -0
  162. data/lib/chef/mixin/check_helper.rb +31 -0
  163. data/lib/chef/mixin/checksum.rb +32 -0
  164. data/lib/chef/mixin/command.rb +220 -0
  165. data/lib/chef/mixin/command/unix.rb +215 -0
  166. data/lib/chef/mixin/command/windows.rb +72 -0
  167. data/lib/chef/mixin/convert_to_class_name.rb +63 -0
  168. data/lib/chef/mixin/create_path.rb +56 -0
  169. data/lib/chef/mixin/deep_merge.rb +225 -0
  170. data/lib/chef/mixin/deprecation.rb +65 -0
  171. data/lib/chef/mixin/find_preferred_file.rb +92 -0
  172. data/lib/chef/mixin/from_file.rb +50 -0
  173. data/lib/chef/mixin/language.rb +156 -0
  174. data/lib/chef/mixin/language_include_attribute.rb +61 -0
  175. data/lib/chef/mixin/language_include_recipe.rb +52 -0
  176. data/lib/chef/mixin/params_validate.rb +225 -0
  177. data/lib/chef/mixin/recipe_definition_dsl_core.rb +81 -0
  178. data/lib/chef/mixin/shell_out.rb +38 -0
  179. data/lib/chef/mixin/template.rb +95 -0
  180. data/lib/chef/mixin/xml_escape.rb +140 -0
  181. data/lib/chef/mixins.rb +16 -0
  182. data/lib/chef/monkey_patches/dir.rb +36 -0
  183. data/lib/chef/monkey_patches/string.rb +28 -0
  184. data/lib/chef/monkey_patches/tempfile.rb +64 -0
  185. data/lib/chef/node.rb +605 -0
  186. data/lib/chef/node/attribute.rb +450 -0
  187. data/lib/chef/openid_registration.rb +187 -0
  188. data/lib/chef/platform.rb +354 -0
  189. data/lib/chef/provider.rb +124 -0
  190. data/lib/chef/provider/breakpoint.rb +36 -0
  191. data/lib/chef/provider/cookbook_file.rb +96 -0
  192. data/lib/chef/provider/cron.rb +186 -0
  193. data/lib/chef/provider/deploy.rb +319 -0
  194. data/lib/chef/provider/deploy/revision.rb +73 -0
  195. data/lib/chef/provider/deploy/timestamped.rb +33 -0
  196. data/lib/chef/provider/directory.rb +72 -0
  197. data/lib/chef/provider/env.rb +152 -0
  198. data/lib/chef/provider/env/windows.rb +75 -0
  199. data/lib/chef/provider/erl_call.rb +72 -0
  200. data/lib/chef/provider/execute.rb +58 -0
  201. data/lib/chef/provider/file.rb +212 -0
  202. data/lib/chef/provider/git.rb +203 -0
  203. data/lib/chef/provider/group.rb +133 -0
  204. data/lib/chef/provider/group/dscl.rb +121 -0
  205. data/lib/chef/provider/group/gpasswd.rb +53 -0
  206. data/lib/chef/provider/group/groupadd.rb +78 -0
  207. data/lib/chef/provider/group/pw.rb +84 -0
  208. data/lib/chef/provider/group/usermod.rb +57 -0
  209. data/lib/chef/provider/group/windows.rb +79 -0
  210. data/lib/chef/provider/http_request.rb +106 -0
  211. data/lib/chef/provider/ifconfig.rb +132 -0
  212. data/lib/chef/provider/link.rb +156 -0
  213. data/lib/chef/provider/log.rb +53 -0
  214. data/lib/chef/provider/mdadm.rb +91 -0
  215. data/lib/chef/provider/mount.rb +117 -0
  216. data/lib/chef/provider/mount/mount.rb +227 -0
  217. data/lib/chef/provider/mount/windows.rb +80 -0
  218. data/lib/chef/provider/package.rb +160 -0
  219. data/lib/chef/provider/package/apt.rb +110 -0
  220. data/lib/chef/provider/package/dpkg.rb +112 -0
  221. data/lib/chef/provider/package/easy_install.rb +114 -0
  222. data/lib/chef/provider/package/freebsd.rb +123 -0
  223. data/lib/chef/provider/package/macports.rb +105 -0
  224. data/lib/chef/provider/package/pacman.rb +101 -0
  225. data/lib/chef/provider/package/portage.rb +124 -0
  226. data/lib/chef/provider/package/rpm.rb +101 -0
  227. data/lib/chef/provider/package/rubygems.rb +462 -0
  228. data/lib/chef/provider/package/solaris.rb +127 -0
  229. data/lib/chef/provider/package/yum-dump.py +125 -0
  230. data/lib/chef/provider/package/yum.rb +222 -0
  231. data/lib/chef/provider/package/zypper.rb +133 -0
  232. data/lib/chef/provider/remote_directory.rb +134 -0
  233. data/lib/chef/provider/remote_file.rb +120 -0
  234. data/lib/chef/provider/route.rb +195 -0
  235. data/lib/chef/provider/ruby_block.rb +33 -0
  236. data/lib/chef/provider/script.rb +55 -0
  237. data/lib/chef/provider/service.rb +128 -0
  238. data/lib/chef/provider/service/arch.rb +109 -0
  239. data/lib/chef/provider/service/debian.rb +105 -0
  240. data/lib/chef/provider/service/freebsd.rb +156 -0
  241. data/lib/chef/provider/service/gentoo.rb +54 -0
  242. data/lib/chef/provider/service/init.rb +71 -0
  243. data/lib/chef/provider/service/redhat.rb +60 -0
  244. data/lib/chef/provider/service/simple.rb +118 -0
  245. data/lib/chef/provider/service/solaris.rb +85 -0
  246. data/lib/chef/provider/service/upstart.rb +192 -0
  247. data/lib/chef/provider/service/windows.rb +129 -0
  248. data/lib/chef/provider/subversion.rb +156 -0
  249. data/lib/chef/provider/template.rb +105 -0
  250. data/lib/chef/provider/user.rb +187 -0
  251. data/lib/chef/provider/user/dscl.rb +280 -0
  252. data/lib/chef/provider/user/pw.rb +113 -0
  253. data/lib/chef/provider/user/useradd.rb +109 -0
  254. data/lib/chef/provider/user/windows.rb +124 -0
  255. data/lib/chef/providers.rb +89 -0
  256. data/lib/chef/recipe.rb +130 -0
  257. data/lib/chef/resource.rb +417 -0
  258. data/lib/chef/resource/apt_package.rb +34 -0
  259. data/lib/chef/resource/bash.rb +33 -0
  260. data/lib/chef/resource/breakpoint.rb +35 -0
  261. data/lib/chef/resource/cookbook_file.rb +45 -0
  262. data/lib/chef/resource/cron.rb +188 -0
  263. data/lib/chef/resource/csh.rb +33 -0
  264. data/lib/chef/resource/deploy.rb +371 -0
  265. data/lib/chef/resource/deploy_revision.rb +35 -0
  266. data/lib/chef/resource/directory.rb +76 -0
  267. data/lib/chef/resource/dpkg_package.rb +34 -0
  268. data/lib/chef/resource/easy_install_package.rb +41 -0
  269. data/lib/chef/resource/env.rb +58 -0
  270. data/lib/chef/resource/erl_call.rb +83 -0
  271. data/lib/chef/resource/execute.rb +127 -0
  272. data/lib/chef/resource/file.rb +92 -0
  273. data/lib/chef/resource/freebsd_package.rb +35 -0
  274. data/lib/chef/resource/gem_package.rb +49 -0
  275. data/lib/chef/resource/git.rb +36 -0
  276. data/lib/chef/resource/group.rb +70 -0
  277. data/lib/chef/resource/http_request.rb +52 -0
  278. data/lib/chef/resource/ifconfig.rb +134 -0
  279. data/lib/chef/resource/link.rb +78 -0
  280. data/lib/chef/resource/log.rb +62 -0
  281. data/lib/chef/resource/macports_package.rb +29 -0
  282. data/lib/chef/resource/mdadm.rb +82 -0
  283. data/lib/chef/resource/mount.rb +135 -0
  284. data/lib/chef/resource/package.rb +80 -0
  285. data/lib/chef/resource/pacman_package.rb +33 -0
  286. data/lib/chef/resource/perl.rb +33 -0
  287. data/lib/chef/resource/portage_package.rb +33 -0
  288. data/lib/chef/resource/python.rb +33 -0
  289. data/lib/chef/resource/remote_directory.rb +100 -0
  290. data/lib/chef/resource/remote_file.rb +83 -0
  291. data/lib/chef/resource/route.rb +135 -0
  292. data/lib/chef/resource/ruby.rb +33 -0
  293. data/lib/chef/resource/ruby_block.rb +40 -0
  294. data/lib/chef/resource/scm.rb +146 -0
  295. data/lib/chef/resource/script.rb +51 -0
  296. data/lib/chef/resource/service.rb +160 -0
  297. data/lib/chef/resource/solaris_package.rb +36 -0
  298. data/lib/chef/resource/subversion.rb +36 -0
  299. data/lib/chef/resource/template.rb +69 -0
  300. data/lib/chef/resource/timestamped_deploy.rb +31 -0
  301. data/lib/chef/resource/user.rb +130 -0
  302. data/lib/chef/resource/yum_package.rb +43 -0
  303. data/lib/chef/resource_collection.rb +216 -0
  304. data/lib/chef/resource_collection/stepable_iterator.rb +124 -0
  305. data/lib/chef/resource_definition.rb +67 -0
  306. data/lib/chef/resource_definition_list.rb +38 -0
  307. data/lib/chef/resources.rb +62 -0
  308. data/lib/chef/rest.rb +388 -0
  309. data/lib/chef/rest/auth_credentials.rb +78 -0
  310. data/lib/chef/rest/cookie_jar.rb +31 -0
  311. data/lib/chef/rest/rest_request.rb +186 -0
  312. data/lib/chef/role.rb +287 -0
  313. data/lib/chef/run_context.rb +110 -0
  314. data/lib/chef/run_list.rb +150 -0
  315. data/lib/chef/run_list/run_list_expansion.rb +172 -0
  316. data/lib/chef/run_list/run_list_item.rb +78 -0
  317. data/lib/chef/run_status.rb +121 -0
  318. data/lib/chef/runner.rb +131 -0
  319. data/lib/chef/sandbox.rb +153 -0
  320. data/lib/chef/search/query.rb +60 -0
  321. data/lib/chef/shef.rb +325 -0
  322. data/lib/chef/shef/ext.rb +565 -0
  323. data/lib/chef/shef/model_wrapper.rb +120 -0
  324. data/lib/chef/shef/shef_rest.rb +28 -0
  325. data/lib/chef/shef/shef_session.rb +271 -0
  326. data/lib/chef/shell_out.rb +413 -0
  327. data/lib/chef/streaming_cookbook_uploader.rb +201 -0
  328. data/lib/chef/tasks/chef_repo.rake +256 -0
  329. data/lib/chef/util/file_edit.rb +122 -0
  330. data/lib/chef/util/windows.rb +56 -0
  331. data/lib/chef/util/windows/net_group.rb +101 -0
  332. data/lib/chef/util/windows/net_use.rb +121 -0
  333. data/lib/chef/util/windows/net_user.rb +198 -0
  334. data/lib/chef/util/windows/volume.rb +59 -0
  335. data/lib/chef/version.rb +21 -0
  336. data/lib/chef/webui_user.rb +232 -0
  337. metadata +614 -0
data/lib/chef/shef.rb ADDED
@@ -0,0 +1,325 @@
1
+ # Author:: Daniel DeLeo (<dan@kallistec.com>)
2
+ # Copyright:: Copyright (c) 2009 Daniel DeLeo
3
+ # License:: Apache License, Version 2.0
4
+ #
5
+ # Licensed under the Apache License, Version 2.0 (the "License");
6
+ # you may not use this file except in compliance with the License.
7
+ # You may obtain a copy of the License at
8
+ #
9
+ # http://www.apache.org/licenses/LICENSE-2.0
10
+ #
11
+ # Unless required by applicable law or agreed to in writing, software
12
+ # distributed under the License is distributed on an "AS IS" BASIS,
13
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14
+ # See the License for the specific language governing permissions and
15
+ # limitations under the License.
16
+ #
17
+
18
+ require "singleton"
19
+ require "pp"
20
+ require "etc"
21
+ require "mixlib/cli"
22
+
23
+ require 'chef/version'
24
+ require "chef/client"
25
+ require "chef/config"
26
+
27
+ require "chef/shef/shef_session"
28
+ require "chef/shef/ext"
29
+
30
+ # = Shef
31
+ # Shef is Chef in an IRB session. Shef can interact with a Chef server via the
32
+ # REST API, and run and debug recipes interactively.
33
+ module Shef
34
+ LEADERS = Hash.new("")
35
+ LEADERS[Chef::Recipe] = ":recipe"
36
+ LEADERS[Chef::Node] = ":attributes"
37
+
38
+ class << self
39
+ attr_accessor :client_type
40
+ attr_accessor :options
41
+ attr_accessor :env
42
+ attr_writer :editor
43
+ end
44
+
45
+ # Start the irb REPL with shef's customizations
46
+ def self.start
47
+ setup_logger
48
+ # FUGLY HACK: irb gives us no other choice.
49
+ irb_help = [:help, :irb_help, IRB::ExtendCommandBundle::NO_OVERRIDE]
50
+ IRB::ExtendCommandBundle.instance_variable_get(:@ALIASES).delete(irb_help)
51
+
52
+ parse_opts
53
+
54
+ # HACK: this duplicates the functions of IRB.start, but we have to do it
55
+ # to get access to the main object before irb starts.
56
+ ::IRB.setup(nil)
57
+
58
+ irb = IRB::Irb.new
59
+
60
+ init(irb.context.main)
61
+
62
+
63
+ irb_conf[:IRB_RC].call(irb.context) if irb_conf[:IRB_RC]
64
+ irb_conf[:MAIN_CONTEXT] = irb.context
65
+
66
+ trap("SIGINT") do
67
+ irb.signal_handle
68
+ end
69
+
70
+ catch(:IRB_EXIT) do
71
+ irb.eval_input
72
+ end
73
+ end
74
+
75
+ def self.setup_logger
76
+ Chef::Config[:log_level] ||= :warn
77
+ Chef::Log.init(STDERR)
78
+ Mixlib::Authentication::Log.logger = Ohai::Log.logger = Chef::Log.logger
79
+ Chef::Log.level = Chef::Config[:log_level] || :warn
80
+ end
81
+
82
+ # Shef assumes it's running whenever it is defined
83
+ def self.running?
84
+ true
85
+ end
86
+
87
+ # Set the irb_conf object to something other than IRB.conf
88
+ # usful for testing.
89
+ def self.irb_conf=(conf_hash)
90
+ @irb_conf = conf_hash
91
+ end
92
+
93
+ def self.irb_conf
94
+ @irb_conf || IRB.conf
95
+ end
96
+
97
+ def self.configure_irb
98
+ irb_conf[:HISTORY_FILE] = "~/.shef_history"
99
+ irb_conf[:SAVE_HISTORY] = 1000
100
+
101
+ irb_conf[:IRB_RC] = lambda do |conf|
102
+ m = conf.main
103
+
104
+ conf.prompt_c = "chef#{leader(m)} > "
105
+ conf.return_format = " => %s \n"
106
+ conf.prompt_i = "chef#{leader(m)} > "
107
+ conf.prompt_n = "chef#{leader(m)} ?> "
108
+ conf.prompt_s = "chef#{leader(m)}%l> "
109
+ end
110
+ end
111
+
112
+ def self.leader(main_object)
113
+ env_string = Shef.env ? " (#{Shef.env})" : ""
114
+ LEADERS[main_object.class] + env_string
115
+ end
116
+
117
+ def self.session
118
+ unless client_type.instance.node_built?
119
+ puts "Session type: #{client_type.session_type}"
120
+ client_type.instance.reset!
121
+ end
122
+ client_type.instance
123
+ end
124
+
125
+ def self.init(main)
126
+ parse_json
127
+ configure_irb
128
+
129
+ session # trigger ohai run + session load
130
+
131
+ session.node.consume_attributes(@json_attribs)
132
+
133
+ Extensions.extend_context_object(main)
134
+
135
+ main.version
136
+ puts
137
+
138
+ puts "run `help' for help, `exit' or ^D to quit."
139
+ puts
140
+ puts "Ohai2u#{greeting}!"
141
+ end
142
+
143
+ def self.greeting
144
+ " #{Etc.getlogin}@#{Shef.session.node.fqdn}"
145
+ rescue NameError
146
+ ""
147
+ end
148
+
149
+ def self.parse_json
150
+ # HACK: copied verbatim from chef/application/client, because it's not
151
+ # reusable as written there :(
152
+ if Chef::Config[:json_attribs]
153
+ begin
154
+ json_io = open(Chef::Config[:json_attribs])
155
+ rescue SocketError => error
156
+ fatal!("I cannot connect to #{Chef::Config[:json_attribs]}", 2)
157
+ rescue Errno::ENOENT => error
158
+ fatal!("I cannot find #{Chef::Config[:json_attribs]}", 2)
159
+ rescue Errno::EACCES => error
160
+ fatal!("Permissions are incorrect on #{Chef::Config[:json_attribs]}. Please chmod a+r #{Chef::Config[:json_attribs]}", 2)
161
+ rescue Exception => error
162
+ fatal!("Got an unexpected error reading #{Chef::Config[:json_attribs]}: #{error.message}", 2)
163
+ end
164
+
165
+ begin
166
+ @json_attribs = JSON.parse(json_io.read)
167
+ rescue JSON::ParserError => error
168
+ fatal!("Could not parse the provided JSON file (#{Chef::Config[:json_attribs]})!: " + error.message, 2)
169
+ end
170
+ end
171
+ end
172
+
173
+ def self.fatal!(message, exit_status)
174
+ Chef::Log.fatal(message)
175
+ exit exit_status
176
+ end
177
+
178
+ def self.client_type
179
+ type = Shef::StandAloneSession
180
+ type = Shef::SoloSession if Chef::Config[:shef_solo]
181
+ type = Shef::ClientSession if Chef::Config[:client]
182
+ type = Shef::DoppelGangerSession if Chef::Config[:doppelganger]
183
+ type
184
+ end
185
+
186
+ def self.parse_opts
187
+ @options = Options.new
188
+ @options.parse_opts
189
+ end
190
+
191
+ def self.editor
192
+ @editor || Chef::Config[:editor] || ENV['EDITOR']
193
+ end
194
+
195
+ class Options
196
+ include Mixlib::CLI
197
+
198
+ def self.footer(text=nil)
199
+ @footer = text if text
200
+ @footer
201
+ end
202
+
203
+ banner("shef #{Chef::VERSION}\n\nUsage: shef [NAMED_CONF] (OPTIONS)")
204
+
205
+ footer(<<-FOOTER)
206
+ When no CONFIG is specified, shef attempts to load a default configuration file:
207
+ * If a NAMED_CONF is given, shef will load ~/.chef/NAMED_CONF/shef.rb
208
+ * If no NAMED_CONF is given shef will load ~/.chef/shef.rb if it exists
209
+ * Shef falls back to loading /etc/chef/client.rb or /etc/chef/solo.rb if -z or
210
+ -s options are given and no shef.rb can be found.
211
+ FOOTER
212
+
213
+ option :config_file,
214
+ :short => "-c CONFIG",
215
+ :long => "--config CONFIG",
216
+ :description => "The configuration file to use"
217
+
218
+ option :help,
219
+ :short => "-h",
220
+ :long => "--help",
221
+ :description => "Show this message",
222
+ :on => :tail,
223
+ :boolean => true,
224
+ :proc => proc { print_help }
225
+
226
+ option :log_level,
227
+ :short => "-l LOG_LEVEL",
228
+ :long => '--log-level LOG_LEVEL',
229
+ :description => "Set the logging level",
230
+ :proc => proc { |level| Chef::Log.level = level.to_sym }
231
+
232
+ option :standalone,
233
+ :short => "-a",
234
+ :long => "--standalone",
235
+ :description => "standalone shef session",
236
+ :default => true,
237
+ :boolean => true
238
+
239
+ option :shef_solo,
240
+ :short => "-s",
241
+ :long => "--solo",
242
+ :description => "chef-solo shef session",
243
+ :boolean => true,
244
+ :proc => proc {Chef::Config[:solo] = true}
245
+
246
+ option :client,
247
+ :short => "-z",
248
+ :long => "--client",
249
+ :description => "chef-client shef session",
250
+ :boolean => true
251
+
252
+ option :json_attribs,
253
+ :short => "-j JSON_ATTRIBS",
254
+ :long => "--json-attributes JSON_ATTRIBS",
255
+ :description => "Load attributes from a JSON file or URL",
256
+ :proc => nil
257
+
258
+ option :chef_server_url,
259
+ :short => "-S CHEFSERVERURL",
260
+ :long => "--server CHEFSERVERURL",
261
+ :description => "The chef server URL",
262
+ :proc => nil
263
+
264
+ option :version,
265
+ :short => "-v",
266
+ :long => "--version",
267
+ :description => "Show chef version",
268
+ :boolean => true,
269
+ :proc => lambda {|v| puts "Chef: #{::Chef::VERSION}"},
270
+ :exit => 0
271
+
272
+ def self.print_help
273
+ instance = new
274
+ instance.parse_options([])
275
+ puts instance.opt_parser
276
+ puts
277
+ puts footer
278
+ puts
279
+ exit 1
280
+ end
281
+
282
+ def self.setup!
283
+ self.new.parse_opts
284
+ end
285
+
286
+ def parse_opts
287
+ remainder = parse_options
288
+ environment = remainder.first
289
+ # We have to nuke ARGV to make sure irb's option parser never sees it.
290
+ # otherwise, IRB complains about command line switches it doesn't recognize.
291
+ ARGV.clear
292
+ config[:config_file] = config_file_for_shef_mode(environment)
293
+ config_msg = config[:config_file] || "none (standalone shef session)"
294
+ puts "loading configuration: #{config_msg}"
295
+ Chef::Config.from_file(config[:config_file]) if !config[:config_file].nil? && File.exists?(config[:config_file]) && File.readable?(config[:config_file])
296
+ Chef::Config.merge!(config)
297
+ end
298
+
299
+ private
300
+
301
+ def config_file_for_shef_mode(environment)
302
+ if config[:config_file]
303
+ config[:config_file]
304
+ elsif environment
305
+ Shef.env = environment
306
+ config_file_to_try = ::File.join(ENV['HOME'], '.chef', environment, 'shef.rb')
307
+ unless ::File.exist?(config_file_to_try)
308
+ puts "could not find shef config for environment #{environment} at #{config_file_to_try}"
309
+ exit 1
310
+ end
311
+ config_file_to_try
312
+ elsif ENV['HOME'] && ::File.exist?(File.join(ENV['HOME'], '.chef', 'shef.rb'))
313
+ File.join(ENV['HOME'], '.chef', 'shef.rb')
314
+ elsif config[:solo]
315
+ "/etc/chef/solo.rb"
316
+ elsif config[:client]
317
+ "/etc/chef/client.rb"
318
+ else
319
+ nil
320
+ end
321
+ end
322
+
323
+ end
324
+
325
+ end
@@ -0,0 +1,565 @@
1
+ #--
2
+ # Author:: Daniel DeLeo (<dan@kallistec.com>)
3
+ # Copyright:: Copyright (c) 2009 Daniel DeLeo
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 'tempfile'
20
+ require 'chef/recipe'
21
+ require 'fileutils'
22
+ require 'chef/version'
23
+ require 'chef/shef/shef_session'
24
+ require 'chef/shef/model_wrapper'
25
+ require 'chef/shef/shef_rest'
26
+
27
+ module Shef
28
+ module Extensions
29
+
30
+ Help = Struct.new(:cmd, :desc, :explanation)
31
+
32
+ # Extensions to be included in every 'main' object in shef. These objects
33
+ # are extended with this module.
34
+ module ObjectCoreExtensions
35
+
36
+ def ensure_session_select_defined
37
+ # irb breaks if you prematurely define IRB::JobMangager
38
+ # so these methods need to be defined at the latest possible time.
39
+ unless jobs.respond_to?(:select_session_by_context)
40
+ def jobs.select_session_by_context(&block)
41
+ @jobs.select { |job| block.call(job[1].context.main)}
42
+ end
43
+ end
44
+
45
+ unless jobs.respond_to?(:session_select)
46
+ def jobs.select_shef_session(target_context)
47
+ session = if target_context.kind_of?(Class)
48
+ select_session_by_context { |main| main.kind_of?(target_context) }
49
+ else
50
+ select_session_by_context { |main| main.equal?(target_context) }
51
+ end
52
+ Array(session.first)[1]
53
+ end
54
+ end
55
+ end
56
+
57
+ def find_or_create_session_for(context_obj)
58
+ ensure_session_select_defined
59
+ if subsession = jobs.select_shef_session(context_obj)
60
+ jobs.switch(subsession)
61
+ else
62
+ irb(context_obj)
63
+ end
64
+ end
65
+
66
+ def help_banner
67
+ banner = []
68
+ banner << ""
69
+ banner << "Shef Help"
70
+ banner << "".ljust(80, "=")
71
+ banner << "| " + "Command".ljust(25) + "| " + "Description"
72
+ banner << "".ljust(80, "=")
73
+
74
+ self.all_help_descriptions.each do |help_text|
75
+ banner << "| " + help_text.cmd.ljust(25) + "| " + help_text.desc
76
+ end
77
+ banner << "".ljust(80, "=")
78
+ banner << "\n"
79
+ banner << "Use help(:command) to get detailed help with individual commands"
80
+ banner << "\n"
81
+ banner.join("\n")
82
+ end
83
+
84
+ def explain_command(method_name)
85
+ help = self.all_help_descriptions.find { |h| h.cmd.to_s == method_name.to_s }
86
+ if help
87
+ puts ""
88
+ puts "Command: #{method_name}"
89
+ puts "".ljust(80, "=")
90
+ puts help.explanation || help.desc
91
+ puts "".ljust(80, "=")
92
+ puts ""
93
+ else
94
+ puts ""
95
+ puts "command #{method_name} not found or no help available"
96
+ puts ""
97
+ end
98
+ end
99
+
100
+ # helpfully returns +:on+ so we can have sugary syntax like `tracing on'
101
+ def on
102
+ :on
103
+ end
104
+
105
+ # returns +:off+ so you can just do `tracing off'
106
+ def off
107
+ :off
108
+ end
109
+
110
+ def help_descriptions
111
+ @help_descriptions ||= []
112
+ end
113
+
114
+ def all_help_descriptions
115
+ help_descriptions
116
+ end
117
+
118
+ def desc(help_text)
119
+ @desc = help_text
120
+ end
121
+
122
+ def explain(explain_text)
123
+ @explain = explain_text
124
+ end
125
+
126
+ def subcommands(subcommand_help={})
127
+ @subcommand_help = subcommand_help
128
+ end
129
+
130
+ def singleton_method_added(mname)
131
+ if @desc
132
+ help_descriptions << Help.new(mname.to_s, @desc.to_s, @explain)
133
+ @desc, @explain = nil, nil
134
+ end
135
+ if @subcommand_help
136
+ @subcommand_help.each do |subcommand, text|
137
+ help_descriptions << Help.new("#{mname}.#{subcommand}", text.to_s, nil)
138
+ end
139
+ end
140
+ @subcommand_help = {}
141
+ end
142
+
143
+ end
144
+
145
+ module String
146
+ def on_off_to_bool
147
+ case self
148
+ when "on"
149
+ true
150
+ when "off"
151
+ false
152
+ else
153
+ self
154
+ end
155
+ end
156
+ end
157
+
158
+ module Symbol
159
+ def on_off_to_bool
160
+ self.to_s.on_off_to_bool
161
+ end
162
+ end
163
+
164
+ module TrueClass
165
+ def to_on_off_str
166
+ "on"
167
+ end
168
+
169
+ def on_off_to_bool
170
+ self
171
+ end
172
+ end
173
+
174
+ module FalseClass
175
+ def to_on_off_str
176
+ "off"
177
+ end
178
+
179
+ def on_off_to_bool
180
+ self
181
+ end
182
+ end
183
+
184
+ # Methods that have associated help text need to be dynamically added
185
+ # to the main irb objects, so we define them in a proc and later
186
+ # instance_eval the proc in the object.
187
+ ObjectUIExtensions = Proc.new do
188
+ extend Shef::Extensions::ObjectCoreExtensions
189
+
190
+ desc "prints this help message"
191
+ explain(<<-E)
192
+ ## SUMMARY ##
193
+ When called with no argument, +help+ prints a table of all shef commands. When
194
+ called with an argument COMMAND, +help+ prints a detailed explanation of the
195
+ command if available, or the description if no explanation is available.
196
+ E
197
+ def help(commmand=nil)
198
+ if commmand
199
+ explain_command(commmand)
200
+ else
201
+ puts help_banner
202
+ end
203
+ :ucanhaz_halp
204
+ end
205
+ alias :halp :help
206
+
207
+ desc "prints information about chef"
208
+ def version
209
+ puts "This is shef, the Chef shell.\n" +
210
+ " Chef Version: #{::Chef::VERSION}\n" +
211
+ " http://www.opscode.com/chef\n" +
212
+ " http://wiki.opscode.com/display/chef/Home"
213
+ :ucanhaz_automation
214
+ end
215
+ alias :shef :version
216
+
217
+ desc "switch to recipe mode"
218
+ def recipe
219
+ find_or_create_session_for Shef.session.recipe
220
+ :recipe
221
+ end
222
+
223
+ desc "switch to attributes mode"
224
+ def attributes
225
+ find_or_create_session_for Shef.session.node
226
+ :attributes
227
+ end
228
+
229
+ desc "returns the current node (i.e., this host)"
230
+ def node
231
+ Shef.session.node
232
+ end
233
+
234
+ desc "pretty print the node's attributes"
235
+ def ohai(key=nil)
236
+ pp(key ? node.attribute[key] : node.attribute)
237
+ end
238
+
239
+ desc "run chef using the current recipe"
240
+ def run_chef
241
+ Chef::Log.level = :debug
242
+ session = Shef.session
243
+ runrun = Chef::Runner.new(session.run_context).converge
244
+ Chef::Log.level = :info
245
+ runrun
246
+ end
247
+
248
+ desc "returns an object to control a paused chef run"
249
+ subcommands :resume => "resume the chef run",
250
+ :step => "run only the next resource",
251
+ :skip_back => "move back in the run list",
252
+ :skip_forward => "move forward in the run list"
253
+ def chef_run
254
+ Shef.session.resource_collection.iterator
255
+ end
256
+
257
+ desc "resets the current recipe"
258
+ def reset
259
+ Shef.session.reset!
260
+ end
261
+
262
+ desc "assume the identity of another node."
263
+ def become_node(node_name)
264
+ Shef::DoppelGangerSession.instance.assume_identity(node_name)
265
+ :doppelganger
266
+ end
267
+ alias :doppelganger :become_node
268
+
269
+ desc "turns printout of return values on or off"
270
+ def echo(on_or_off)
271
+ conf.echo = on_or_off.on_off_to_bool
272
+ end
273
+
274
+ desc "says if echo is on or off"
275
+ def echo?
276
+ puts "echo is #{conf.echo.to_on_off_str}"
277
+ end
278
+
279
+ desc "turns on or off tracing of execution. *verbose*"
280
+ def tracing(on_or_off)
281
+ conf.use_tracer = on_or_off.on_off_to_bool
282
+ tracing?
283
+ end
284
+ alias :trace :tracing
285
+
286
+ desc "says if tracing is on or off"
287
+ def tracing?
288
+ puts "tracing is #{conf.use_tracer.to_on_off_str}"
289
+ end
290
+ alias :trace? :tracing?
291
+
292
+ desc "simple ls style command"
293
+ def ls(directory)
294
+ Dir.entries(directory)
295
+ end
296
+ end
297
+
298
+ RESTApiExtensions = Proc.new do
299
+ desc "edit an object in your EDITOR"
300
+ explain(<<-E)
301
+ ## SUMMARY ##
302
+ +edit(object)+ allows you to edit any object that can be converted to JSON.
303
+ When finished editing, this method will return the edited object:
304
+
305
+ new_node = edit(existing_node)
306
+
307
+ ## EDITOR SELECTION ##
308
+ Shef looks for an editor using the following logic
309
+ 1. Looks for an EDITOR set by Shef.editor = "EDITOR"
310
+ 2. Looks for an EDITOR configured in your shef config file
311
+ 3. Uses the value of the EDITOR environment variable
312
+ E
313
+ def edit(object)
314
+ unless Shef.editor
315
+ puts "Please set your editor with Shef.editor = \"vim|emacs|mate|ed\""
316
+ return :failburger
317
+ end
318
+
319
+ filename = "shef-edit-#{object.class.name}-"
320
+ if object.respond_to?(:name)
321
+ filename += object.name
322
+ elsif object.respond_to?(:id)
323
+ filename += object.id
324
+ end
325
+
326
+ edited_data = Tempfile.open([filename, ".js"]) do |tempfile|
327
+ tempfile.sync = true
328
+ tempfile.puts object.to_json
329
+ system("#{Shef.editor.to_s} #{tempfile.path}")
330
+ tempfile.rewind
331
+ tempfile.read
332
+ end
333
+
334
+ JSON.parse(edited_data)
335
+ end
336
+
337
+ desc "Find and edit API clients"
338
+ explain(<<-E)
339
+ ## SUMMARY ##
340
+ +clients+ allows you to query you chef server for information about your api
341
+ clients.
342
+
343
+ ## LIST ALL CLIENTS ##
344
+ To see all clients on the system, use
345
+
346
+ clients.all #=> [<Chef::ApiClient...>, ...]
347
+
348
+ If the output from all is too verbose, or you're only interested in a specific
349
+ value from each of the objects, you can give a code block to +all+:
350
+
351
+ clients.all { |client| client.name } #=> [CLIENT1_NAME, CLIENT2_NAME, ...]
352
+
353
+ ## SHOW ONE CLIENT ##
354
+ To see a specific client, use
355
+
356
+ clients.show(CLIENT_NAME)
357
+
358
+ ## SEARCH FOR CLIENTS ##
359
+ You can also search for clients using +find+ or +search+. You can use the
360
+ familiar string search syntax:
361
+
362
+ clients.search("KEY:VALUE")
363
+
364
+ Just as the +all+ subcommand, the +search+ subcommand can use a code block to
365
+ filter or transform the information returned from the search:
366
+
367
+ clients.search("KEY:VALUE") { |c| c.name }
368
+
369
+ You can also use a Hash based syntax, multiple search conditions will be
370
+ joined with AND.
371
+
372
+ clients.find :KEY => :VALUE, :KEY2 => :VALUE2, ...
373
+
374
+ ## BULK-EDIT CLIENTS ##
375
+ **BE CAREFUL, THIS IS DESTRUCTIVE**
376
+ You can bulk edit API Clients using the +transform+ subcommand, which requires
377
+ a code block. Each client will be saved after the code block is run. If the
378
+ code block returns +nil+ or +false+, that client will be skipped:
379
+
380
+ clients.transform("*:*") do |client|
381
+ if client.name =~ /borat/i
382
+ client.admin(false)
383
+ true
384
+ else
385
+ nil
386
+ end
387
+ end
388
+
389
+ This will strip the admin privileges from any client named after borat.
390
+ E
391
+ subcommands :all => "list all api clients",
392
+ :show => "load an api client by name",
393
+ :search => "search for API clients",
394
+ :transform => "edit all api clients via a code block and save them"
395
+ def clients
396
+ @clients ||= Shef::ModelWrapper.new(Chef::ApiClient, :client)
397
+ end
398
+
399
+ desc "Find and edit cookbooks"
400
+ subcommands :all => "list all cookbooks",
401
+ :show => "load a cookbook by name",
402
+ :transform => "edit all cookbooks via a code block and save them"
403
+ def cookbooks
404
+ @cookbooks ||= Shef::ModelWrapper.new(Chef::CookbookVersion)
405
+ end
406
+
407
+ desc "Find and edit nodes via the API"
408
+ explain(<<-E)
409
+ ## SUMMARY ##
410
+ +nodes+ Allows you to query your chef server for information about your nodes.
411
+
412
+ ## LIST ALL NODES ##
413
+ You can list all nodes using +all+ or +list+
414
+
415
+ nodes.all #=> [<Chef::Node...>, <Chef::Node...>, ...]
416
+
417
+ To limit the information returned for each node, pass a code block to the +all+
418
+ subcommand:
419
+
420
+ nodes.all { |node| node.name } #=> [NODE1_NAME, NODE2_NAME, ...]
421
+
422
+ ## SHOW ONE NODE ##
423
+ You can show the data for a single node using the +show+ subcommand:
424
+
425
+ nodes.show("NODE_NAME") => <Chef::Node @name="NODE_NAME" ...>
426
+
427
+ ## SEARCH FOR NODES ##
428
+ You can search for nodes using the +search+ or +find+ subcommands:
429
+
430
+ nodes.find(:name => "app*") #=> [<Chef::Node @name="app1.example.com" ...>, ...]
431
+
432
+ Similarly to +all+, you can pass a code block to limit or transform the
433
+ information returned:
434
+
435
+ nodes.find(:name => "app#") { |node| node.ec2 }
436
+
437
+ ## BULK EDIT NODES ##
438
+ **BE CAREFUL, THIS OPERATION IS DESTRUCTIVE**
439
+
440
+ Bulk edit nodes by passing a code block to the +transform+ or +bulk_edit+
441
+ subcommand. The block will be applied to each matching node, and then the node
442
+ will be saved. If the block returns +nil+ or +false+, that node will be
443
+ skipped.
444
+
445
+ nodes.transform do |node|
446
+ if node.fqdn =~ /.*\\.preprod\\.example\\.com/
447
+ node.set[:environment] = "preprod"
448
+ end
449
+ end
450
+
451
+ This will assign the attribute to every node with a FQDN matching the regex.
452
+ E
453
+ subcommands :all => "list all nodes",
454
+ :show => "load a node by name",
455
+ :search => "search for nodes",
456
+ :transform => "edit all nodes via a code block and save them"
457
+ def nodes
458
+ @nodes ||= Shef::ModelWrapper.new(Chef::Node)
459
+ end
460
+
461
+ desc "Find and edit roles via the API"
462
+ explain(<<-E)
463
+ ## SUMMARY ##
464
+ +roles+ allows you to query and edit roles on your Chef server.
465
+
466
+ ## SUBCOMMANDS ##
467
+ * all (list)
468
+ * show (load)
469
+ * search (find)
470
+ * transform (bulk_edit)
471
+
472
+ ## SEE ALSO ##
473
+ See the help for +nodes+ for more information about the subcommands.
474
+ E
475
+ subcommands :all => "list all roles",
476
+ :show => "load a role by name",
477
+ :search => "search for roles",
478
+ :transform => "edit all roles via a code block and save them"
479
+ def roles
480
+ @roles ||= Shef::ModelWrapper.new(Chef::Role)
481
+ end
482
+
483
+ desc "Find and edit +databag_name+ via the api"
484
+ explain(<<-E)
485
+ ## SUMMARY ##
486
+ +databags(DATABAG_NAME)+ allows you to query and edit data bag items on your
487
+ Chef server. Unlike other commands for working with data on the server,
488
+ +databags+ requires the databag name as an argument, for example:
489
+ databags(:users).all
490
+
491
+ ## SUBCOMMANDS ##
492
+ * all (list)
493
+ * show (load)
494
+ * search (find)
495
+ * transform (bulk_edit)
496
+
497
+ ## SEE ALSO ##
498
+ See the help for +nodes+ for more information about the subcommands.
499
+
500
+ E
501
+ subcommands :all => "list all items in the data bag",
502
+ :show => "load a data bag item by id",
503
+ :search => "search for items in the data bag",
504
+ :transform => "edit all items via a code block and save them"
505
+ def databags(databag_name)
506
+ @named_databags_wrappers ||= {}
507
+ @named_databags_wrappers[databag_name] ||= Shef::NamedDataBagWrapper.new(databag_name)
508
+ end
509
+
510
+ desc "A REST Client configured to authenticate with the API"
511
+ def api
512
+ @rest = Shef::ShefREST.new(Chef::Config[:chef_server_url])
513
+ end
514
+
515
+ end
516
+
517
+ RecipeUIExtensions = Proc.new do
518
+ alias :original_resources :resources
519
+
520
+ desc "list all the resources on the current recipe"
521
+ def resources(*args)
522
+ if args.empty?
523
+ pp run_context.resource_collection.instance_variable_get(:@resources_by_name).keys
524
+ else
525
+ pp resources = original_resources(*args)
526
+ resources
527
+ end
528
+ end
529
+ end
530
+
531
+ def self.extend_context_object(obj)
532
+ obj.instance_eval(&ObjectUIExtensions)
533
+ obj.instance_eval(&RESTApiExtensions)
534
+ obj.extend(FileUtils)
535
+ obj.extend(Chef::Mixin::Language)
536
+ end
537
+
538
+ def self.extend_context_node(node_obj)
539
+ node_obj.instance_eval(&ObjectUIExtensions)
540
+ end
541
+
542
+ def self.extend_context_recipe(recipe_obj)
543
+ recipe_obj.instance_eval(&ObjectUIExtensions)
544
+ recipe_obj.instance_eval(&RecipeUIExtensions)
545
+ end
546
+
547
+ end
548
+ end
549
+
550
+ class String
551
+ include Shef::Extensions::String
552
+ end
553
+
554
+ class Symbol
555
+ include Shef::Extensions::Symbol
556
+ end
557
+
558
+ class TrueClass
559
+ include Shef::Extensions::TrueClass
560
+ end
561
+
562
+ class FalseClass
563
+ include Shef::Extensions::FalseClass
564
+ end
565
+