tnargav 1.3.5 → 1.3.6

Sign up to get free protection for your applications and to get access to all the features.
Files changed (478) hide show
  1. data/lib/vagrant/action.rb +45 -0
  2. data/lib/vagrant/action/builder.rb +174 -0
  3. data/lib/vagrant/action/builtin/box_add.rb +92 -0
  4. data/lib/vagrant/action/builtin/call.rb +67 -0
  5. data/lib/vagrant/action/builtin/config_validate.rb +30 -0
  6. data/lib/vagrant/action/builtin/confirm.rb +39 -0
  7. data/lib/vagrant/action/builtin/destroy_confirm.rb +21 -0
  8. data/lib/vagrant/action/builtin/env_set.rb +24 -0
  9. data/lib/vagrant/action/builtin/graceful_halt.rb +82 -0
  10. data/lib/vagrant/action/builtin/handle_box_url.rb +86 -0
  11. data/lib/vagrant/action/builtin/handle_forwarded_port_collisions.rb +134 -0
  12. data/lib/vagrant/action/builtin/lock.rb +57 -0
  13. data/lib/vagrant/action/builtin/mixin_provisioners.rb +43 -0
  14. data/lib/vagrant/action/builtin/nfs.rb +128 -0
  15. data/lib/vagrant/action/builtin/provision.rb +79 -0
  16. data/lib/vagrant/action/builtin/provisioner_cleanup.rb +35 -0
  17. data/lib/vagrant/action/builtin/set_hostname.rb +27 -0
  18. data/lib/vagrant/action/builtin/ssh_exec.rb +42 -0
  19. data/lib/vagrant/action/builtin/ssh_run.rb +43 -0
  20. data/lib/vagrant/action/builtin/wait_for_communicator.rb +84 -0
  21. data/lib/vagrant/action/general/package.rb +103 -0
  22. data/lib/vagrant/action/hook.rb +107 -0
  23. data/lib/vagrant/action/runner.rb +69 -0
  24. data/lib/vagrant/action/warden.rb +104 -0
  25. data/lib/vagrant/batch_action.rb +113 -0
  26. data/lib/vagrant/box.rb +97 -0
  27. data/lib/vagrant/box_collection.rb +393 -0
  28. data/lib/vagrant/cli.rb +74 -0
  29. data/lib/vagrant/config.rb +61 -0
  30. data/lib/vagrant/config/loader.rb +222 -0
  31. data/lib/vagrant/config/v1.rb +9 -0
  32. data/lib/vagrant/config/v1/dummy_config.rb +13 -0
  33. data/lib/vagrant/config/v1/loader.rb +105 -0
  34. data/lib/vagrant/config/v1/root.rb +60 -0
  35. data/lib/vagrant/config/v2.rb +9 -0
  36. data/lib/vagrant/config/v2/dummy_config.rb +13 -0
  37. data/lib/vagrant/config/v2/loader.rb +141 -0
  38. data/lib/vagrant/config/v2/root.rb +111 -0
  39. data/lib/vagrant/config/v2/util.rb +21 -0
  40. data/lib/vagrant/config/version_base.rb +80 -0
  41. data/lib/vagrant/environment.rb +819 -0
  42. data/lib/vagrant/errors.rb +614 -0
  43. data/lib/vagrant/guest.rb +172 -0
  44. data/lib/vagrant/hosts.rb +28 -0
  45. data/lib/vagrant/machine.rb +319 -0
  46. data/lib/vagrant/machine_state.rb +45 -0
  47. data/lib/vagrant/plugin.rb +6 -0
  48. data/lib/vagrant/plugin/v1.rb +19 -0
  49. data/lib/vagrant/plugin/v1/command.rb +169 -0
  50. data/lib/vagrant/plugin/v1/communicator.rb +98 -0
  51. data/lib/vagrant/plugin/v1/config.rb +112 -0
  52. data/lib/vagrant/plugin/v1/errors.rb +15 -0
  53. data/lib/vagrant/plugin/v1/guest.rb +92 -0
  54. data/lib/vagrant/plugin/v1/host.rb +66 -0
  55. data/lib/vagrant/plugin/v1/manager.rb +131 -0
  56. data/lib/vagrant/plugin/v1/plugin.rb +229 -0
  57. data/lib/vagrant/plugin/v1/provider.rb +68 -0
  58. data/lib/vagrant/plugin/v1/provisioner.rb +50 -0
  59. data/lib/vagrant/plugin/v2.rb +22 -0
  60. data/lib/vagrant/plugin/v2/command.rb +237 -0
  61. data/lib/vagrant/plugin/v2/communicator.rb +119 -0
  62. data/lib/vagrant/plugin/v2/components.rb +47 -0
  63. data/lib/vagrant/plugin/v2/config.rb +136 -0
  64. data/lib/vagrant/plugin/v2/errors.rb +15 -0
  65. data/lib/vagrant/plugin/v2/guest.rb +23 -0
  66. data/lib/vagrant/plugin/v2/host.rb +66 -0
  67. data/lib/vagrant/plugin/v2/manager.rb +173 -0
  68. data/lib/vagrant/plugin/v2/plugin.rb +226 -0
  69. data/lib/vagrant/plugin/v2/provider.rb +69 -0
  70. data/lib/vagrant/plugin/v2/provisioner.rb +53 -0
  71. data/lib/vagrant/registry.rb +78 -0
  72. data/lib/vagrant/ui.rb +215 -0
  73. data/lib/vagrant/util.rb +12 -0
  74. data/lib/vagrant/util/ansi_escape_code_remover.rb +34 -0
  75. data/lib/vagrant/util/busy.rb +59 -0
  76. data/lib/vagrant/util/counter.rb +24 -0
  77. data/lib/vagrant/util/downloader.rb +151 -0
  78. data/lib/vagrant/util/file_checksum.rb +38 -0
  79. data/lib/vagrant/util/file_mode.rb +12 -0
  80. data/lib/vagrant/util/hash_with_indifferent_access.rb +63 -0
  81. data/lib/vagrant/util/is_port_open.rb +38 -0
  82. data/lib/vagrant/util/line_ending_helpers.rb +14 -0
  83. data/lib/vagrant/util/network_ip.rb +28 -0
  84. data/lib/vagrant/util/platform.rb +101 -0
  85. data/lib/vagrant/util/retryable.rb +31 -0
  86. data/lib/vagrant/util/safe_chdir.rb +33 -0
  87. data/lib/vagrant/util/safe_exec.rb +36 -0
  88. data/lib/vagrant/util/safe_puts.rb +31 -0
  89. data/lib/vagrant/util/scoped_hash_override.rb +45 -0
  90. data/lib/vagrant/util/ssh.rb +155 -0
  91. data/lib/vagrant/util/stacked_proc_runner.rb +35 -0
  92. data/lib/vagrant/util/string_block_editor.rb +77 -0
  93. data/lib/vagrant/util/subprocess.rb +294 -0
  94. data/lib/vagrant/util/template_renderer.rb +83 -0
  95. data/lib/vagrant/util/which.rb +43 -0
  96. data/lib/vagrant/version.rb +6 -0
  97. data/plugins/commands/box/command/add.rb +51 -0
  98. data/plugins/commands/box/command/list.rb +41 -0
  99. data/plugins/commands/box/command/remove.rb +57 -0
  100. data/plugins/commands/box/command/repackage.rb +43 -0
  101. data/plugins/commands/box/command/root.rb +75 -0
  102. data/plugins/commands/box/plugin.rb +15 -0
  103. data/plugins/commands/destroy/command.rb +36 -0
  104. data/plugins/commands/destroy/plugin.rb +18 -0
  105. data/plugins/commands/halt/command.rb +33 -0
  106. data/plugins/commands/halt/plugin.rb +18 -0
  107. data/plugins/commands/help/command.rb +12 -0
  108. data/plugins/commands/help/plugin.rb +17 -0
  109. data/plugins/commands/init/command.rb +40 -0
  110. data/plugins/commands/init/plugin.rb +18 -0
  111. data/plugins/commands/package/command.rb +83 -0
  112. data/plugins/commands/package/plugin.rb +18 -0
  113. data/plugins/commands/plugin/action.rb +63 -0
  114. data/plugins/commands/plugin/action/bundler_check.rb +25 -0
  115. data/plugins/commands/plugin/action/install_gem.rb +106 -0
  116. data/plugins/commands/plugin/action/license_plugin.rb +54 -0
  117. data/plugins/commands/plugin/action/list_plugins.rb +54 -0
  118. data/plugins/commands/plugin/action/plugin_exists_check.rb +26 -0
  119. data/plugins/commands/plugin/action/prune_gems.rb +149 -0
  120. data/plugins/commands/plugin/action/uninstall_plugin.rb +23 -0
  121. data/plugins/commands/plugin/command/base.rb +22 -0
  122. data/plugins/commands/plugin/command/install.rb +41 -0
  123. data/plugins/commands/plugin/command/license.rb +31 -0
  124. data/plugins/commands/plugin/command/list.rb +28 -0
  125. data/plugins/commands/plugin/command/mixin_install_opts.rb +30 -0
  126. data/plugins/commands/plugin/command/root.rb +80 -0
  127. data/plugins/commands/plugin/command/uninstall.rb +28 -0
  128. data/plugins/commands/plugin/command/update.rb +41 -0
  129. data/plugins/commands/plugin/gem_helper.rb +74 -0
  130. data/plugins/commands/plugin/plugin.rb +22 -0
  131. data/plugins/commands/plugin/state_file.rb +57 -0
  132. data/plugins/commands/provision/command.rb +34 -0
  133. data/plugins/commands/provision/plugin.rb +18 -0
  134. data/plugins/commands/reload/command.rb +41 -0
  135. data/plugins/commands/reload/plugin.rb +18 -0
  136. data/plugins/commands/resume/command.rb +25 -0
  137. data/plugins/commands/resume/plugin.rb +17 -0
  138. data/plugins/commands/ssh/command.rb +63 -0
  139. data/plugins/commands/ssh/plugin.rb +17 -0
  140. data/plugins/commands/ssh_config/command.rb +49 -0
  141. data/plugins/commands/ssh_config/plugin.rb +18 -0
  142. data/plugins/commands/status/command.rb +44 -0
  143. data/plugins/commands/status/plugin.rb +18 -0
  144. data/plugins/commands/suspend/command.rb +25 -0
  145. data/plugins/commands/suspend/plugin.rb +18 -0
  146. data/plugins/commands/up/command.rb +67 -0
  147. data/plugins/commands/up/plugin.rb +17 -0
  148. data/plugins/commands/up/start_mixins.rb +37 -0
  149. data/plugins/communicators/ssh/communicator.rb +400 -0
  150. data/plugins/communicators/ssh/plugin.rb +19 -0
  151. data/plugins/guests/arch/cap/change_host_name.rb +17 -0
  152. data/plugins/guests/arch/cap/configure_networks.rb +30 -0
  153. data/plugins/guests/arch/guest.rb +11 -0
  154. data/plugins/guests/arch/plugin.rb +25 -0
  155. data/plugins/guests/coreos/cap/change_host_name.rb +15 -0
  156. data/plugins/guests/coreos/cap/configure_networks.rb +72 -0
  157. data/plugins/guests/coreos/guest.rb +9 -0
  158. data/plugins/guests/coreos/plugin.rb +25 -0
  159. data/plugins/guests/darwin/cap/change_host_name.rb +14 -0
  160. data/plugins/guests/darwin/cap/configure_networks.rb +52 -0
  161. data/plugins/guests/darwin/cap/halt.rb +16 -0
  162. data/plugins/guests/darwin/cap/mount_nfs_folder.rb +25 -0
  163. data/plugins/guests/darwin/cap/mount_vmware_shared_folder.rb +36 -0
  164. data/plugins/guests/darwin/cap/shell_expand_guest_path.rb +26 -0
  165. data/plugins/guests/darwin/cap/verify_vmware_hgfs.rb +12 -0
  166. data/plugins/guests/darwin/guest.rb +15 -0
  167. data/plugins/guests/darwin/plugin.rb +50 -0
  168. data/plugins/guests/debian/cap/change_host_name.rb +19 -0
  169. data/plugins/guests/debian/cap/configure_networks.rb +61 -0
  170. data/plugins/guests/debian/guest.rb +9 -0
  171. data/plugins/guests/debian/plugin.rb +25 -0
  172. data/plugins/guests/fedora/cap/configure_networks.rb +59 -0
  173. data/plugins/guests/fedora/cap/network_scripts_dir.rb +15 -0
  174. data/plugins/guests/fedora/guest.rb +11 -0
  175. data/plugins/guests/fedora/plugin.rb +25 -0
  176. data/plugins/guests/freebsd/cap/change_host_name.rb +14 -0
  177. data/plugins/guests/freebsd/cap/configure_networks.rb +39 -0
  178. data/plugins/guests/freebsd/cap/halt.rb +16 -0
  179. data/plugins/guests/freebsd/cap/mount_nfs_folder.rb +14 -0
  180. data/plugins/guests/freebsd/guest.rb +14 -0
  181. data/plugins/guests/freebsd/plugin.rb +35 -0
  182. data/plugins/guests/gentoo/cap/change_host_name.rb +17 -0
  183. data/plugins/guests/gentoo/cap/configure_networks.rb +43 -0
  184. data/plugins/guests/gentoo/guest.rb +9 -0
  185. data/plugins/guests/gentoo/plugin.rb +25 -0
  186. data/plugins/guests/linux/cap/halt.rb +16 -0
  187. data/plugins/guests/linux/cap/mount_nfs.rb +38 -0
  188. data/plugins/guests/linux/cap/mount_virtualbox_shared_folder.rb +59 -0
  189. data/plugins/guests/linux/cap/read_ip_address.rb +17 -0
  190. data/plugins/guests/linux/cap/shell_expand_guest_path.rb +29 -0
  191. data/plugins/guests/linux/guest.rb +11 -0
  192. data/plugins/guests/linux/plugin.rb +40 -0
  193. data/plugins/guests/omnios/cap/change_host_name.rb +17 -0
  194. data/plugins/guests/omnios/guest.rb +11 -0
  195. data/plugins/guests/omnios/plugin.rb +20 -0
  196. data/plugins/guests/openbsd/cap/change_host_name.rb +14 -0
  197. data/plugins/guests/openbsd/cap/configure_networks.rb +40 -0
  198. data/plugins/guests/openbsd/cap/halt.rb +16 -0
  199. data/plugins/guests/openbsd/cap/mount_nfs_folder.rb +14 -0
  200. data/plugins/guests/openbsd/guest.rb +11 -0
  201. data/plugins/guests/openbsd/plugin.rb +35 -0
  202. data/plugins/guests/pld/cap/network_scripts_dir.rb +11 -0
  203. data/plugins/guests/pld/guest.rb +11 -0
  204. data/plugins/guests/pld/plugin.rb +20 -0
  205. data/plugins/guests/redhat/cap/change_host_name.rb +18 -0
  206. data/plugins/guests/redhat/cap/configure_networks.rb +60 -0
  207. data/plugins/guests/redhat/cap/network_scripts_dir.rb +11 -0
  208. data/plugins/guests/redhat/guest.rb +11 -0
  209. data/plugins/guests/redhat/plugin.rb +30 -0
  210. data/plugins/guests/solaris/cap/change_host_name.rb +17 -0
  211. data/plugins/guests/solaris/cap/configure_networks.rb +25 -0
  212. data/plugins/guests/solaris/cap/halt.rb +22 -0
  213. data/plugins/guests/solaris/cap/mount_virtualbox_shared_folder.rb +31 -0
  214. data/plugins/guests/solaris/config.rb +18 -0
  215. data/plugins/guests/solaris/guest.rb +14 -0
  216. data/plugins/guests/solaris/plugin.rb +40 -0
  217. data/plugins/guests/solaris11/cap/change_host_name.rb +25 -0
  218. data/plugins/guests/solaris11/cap/configure_networks.rb +32 -0
  219. data/plugins/guests/solaris11/cap/halt.rb +26 -0
  220. data/plugins/guests/solaris11/cap/mount_virtualbox_shared_folder.rb +35 -0
  221. data/plugins/guests/solaris11/config.rb +22 -0
  222. data/plugins/guests/solaris11/guest.rb +15 -0
  223. data/plugins/guests/solaris11/plugin.rb +44 -0
  224. data/plugins/guests/suse/cap/change_host_name.rb +18 -0
  225. data/plugins/guests/suse/cap/configure_networks.rb +60 -0
  226. data/plugins/guests/suse/cap/network_scripts_dir.rb +11 -0
  227. data/plugins/guests/suse/guest.rb +11 -0
  228. data/plugins/guests/suse/plugin.rb +30 -0
  229. data/plugins/guests/ubuntu/cap/change_host_name.rb +24 -0
  230. data/plugins/guests/ubuntu/cap/mount_nfs.rb +19 -0
  231. data/plugins/guests/ubuntu/cap/mount_virtualbox_shared_folder.rb +14 -0
  232. data/plugins/guests/ubuntu/guest.rb +13 -0
  233. data/plugins/guests/ubuntu/plugin.rb +30 -0
  234. data/plugins/hosts/arch/host.rb +45 -0
  235. data/plugins/hosts/arch/plugin.rb +15 -0
  236. data/plugins/hosts/bsd/host.rb +178 -0
  237. data/plugins/hosts/bsd/plugin.rb +15 -0
  238. data/plugins/hosts/fedora/host.rb +57 -0
  239. data/plugins/hosts/fedora/plugin.rb +15 -0
  240. data/plugins/hosts/freebsd/host.rb +43 -0
  241. data/plugins/hosts/freebsd/plugin.rb +15 -0
  242. data/plugins/hosts/gentoo/host.rb +26 -0
  243. data/plugins/hosts/gentoo/plugin.rb +15 -0
  244. data/plugins/hosts/linux/host.rb +128 -0
  245. data/plugins/hosts/linux/plugin.rb +15 -0
  246. data/plugins/hosts/opensuse/host.rb +36 -0
  247. data/plugins/hosts/opensuse/plugin.rb +15 -0
  248. data/plugins/hosts/slackware/host.rb +26 -0
  249. data/plugins/hosts/slackware/plugin.rb +15 -0
  250. data/plugins/hosts/windows/host.rb +17 -0
  251. data/plugins/hosts/windows/plugin.rb +15 -0
  252. data/plugins/kernel_v1/config/nfs.rb +20 -0
  253. data/plugins/kernel_v1/config/package.rb +17 -0
  254. data/plugins/kernel_v1/config/ssh.rb +46 -0
  255. data/plugins/kernel_v1/config/vagrant.rb +31 -0
  256. data/plugins/kernel_v1/config/vm.rb +187 -0
  257. data/plugins/kernel_v1/plugin.rb +44 -0
  258. data/plugins/kernel_v2/config/nfs.rb +14 -0
  259. data/plugins/kernel_v2/config/package.rb +13 -0
  260. data/plugins/kernel_v2/config/ssh.rb +68 -0
  261. data/plugins/kernel_v2/config/ssh_connect.rb +40 -0
  262. data/plugins/kernel_v2/config/vagrant.rb +13 -0
  263. data/plugins/kernel_v2/config/vm.rb +486 -0
  264. data/plugins/kernel_v2/config/vm_provisioner.rb +55 -0
  265. data/plugins/kernel_v2/config/vm_subvm.rb +30 -0
  266. data/plugins/kernel_v2/plugin.rb +44 -0
  267. data/plugins/providers/virtualbox/action.rb +324 -0
  268. data/plugins/providers/virtualbox/action/boot.rb +23 -0
  269. data/plugins/providers/virtualbox/action/check_accessible.rb +23 -0
  270. data/plugins/providers/virtualbox/action/check_created.rb +21 -0
  271. data/plugins/providers/virtualbox/action/check_guest_additions.rb +45 -0
  272. data/plugins/providers/virtualbox/action/check_running.rb +21 -0
  273. data/plugins/providers/virtualbox/action/check_virtualbox.rb +22 -0
  274. data/plugins/providers/virtualbox/action/clean_machine_folder.rb +43 -0
  275. data/plugins/providers/virtualbox/action/clear_forwarded_ports.rb +18 -0
  276. data/plugins/providers/virtualbox/action/clear_network_interfaces.rb +31 -0
  277. data/plugins/providers/virtualbox/action/clear_shared_folders.rb +17 -0
  278. data/plugins/providers/virtualbox/action/created.rb +20 -0
  279. data/plugins/providers/virtualbox/action/customize.rb +43 -0
  280. data/plugins/providers/virtualbox/action/destroy.rb +19 -0
  281. data/plugins/providers/virtualbox/action/destroy_unused_network_interfaces.rb +23 -0
  282. data/plugins/providers/virtualbox/action/discard_state.rb +20 -0
  283. data/plugins/providers/virtualbox/action/export.rb +57 -0
  284. data/plugins/providers/virtualbox/action/forced_halt.rb +25 -0
  285. data/plugins/providers/virtualbox/action/forward_ports.rb +89 -0
  286. data/plugins/providers/virtualbox/action/import.rb +54 -0
  287. data/plugins/providers/virtualbox/action/is_paused.rb +20 -0
  288. data/plugins/providers/virtualbox/action/is_running.rb +20 -0
  289. data/plugins/providers/virtualbox/action/is_saved.rb +20 -0
  290. data/plugins/providers/virtualbox/action/match_mac_address.rb +21 -0
  291. data/plugins/providers/virtualbox/action/message_already_running.rb +16 -0
  292. data/plugins/providers/virtualbox/action/message_not_created.rb +16 -0
  293. data/plugins/providers/virtualbox/action/message_not_running.rb +16 -0
  294. data/plugins/providers/virtualbox/action/message_will_not_destroy.rb +17 -0
  295. data/plugins/providers/virtualbox/action/network.rb +424 -0
  296. data/plugins/providers/virtualbox/action/package.rb +20 -0
  297. data/plugins/providers/virtualbox/action/package_vagrantfile.rb +33 -0
  298. data/plugins/providers/virtualbox/action/prepare_forwarded_port_collision_params.rb +35 -0
  299. data/plugins/providers/virtualbox/action/prepare_nfs_settings.rb +69 -0
  300. data/plugins/providers/virtualbox/action/prune_nfs_exports.rb +20 -0
  301. data/plugins/providers/virtualbox/action/resume.rb +25 -0
  302. data/plugins/providers/virtualbox/action/sane_defaults.rb +91 -0
  303. data/plugins/providers/virtualbox/action/set_name.rb +52 -0
  304. data/plugins/providers/virtualbox/action/setup_package_files.rb +51 -0
  305. data/plugins/providers/virtualbox/action/share_folders.rb +128 -0
  306. data/plugins/providers/virtualbox/action/suspend.rb +20 -0
  307. data/plugins/providers/virtualbox/config.rb +123 -0
  308. data/plugins/providers/virtualbox/driver/base.rb +374 -0
  309. data/plugins/providers/virtualbox/driver/meta.rb +148 -0
  310. data/plugins/providers/virtualbox/driver/version_4_0.rb +485 -0
  311. data/plugins/providers/virtualbox/driver/version_4_1.rb +485 -0
  312. data/plugins/providers/virtualbox/driver/version_4_2.rb +499 -0
  313. data/plugins/providers/virtualbox/model/forwarded_port.rb +70 -0
  314. data/plugins/providers/virtualbox/plugin.rb +42 -0
  315. data/plugins/providers/virtualbox/provider.rb +92 -0
  316. data/plugins/providers/virtualbox/util/compile_forwarded_ports.rb +35 -0
  317. data/plugins/provisioners/ansible/config.rb +89 -0
  318. data/plugins/provisioners/ansible/plugin.rb +23 -0
  319. data/plugins/provisioners/ansible/provisioner.rb +100 -0
  320. data/plugins/provisioners/cfengine/cap/debian/cfengine_install.rb +19 -0
  321. data/plugins/provisioners/cfengine/cap/linux/cfengine_installed.rb +14 -0
  322. data/plugins/provisioners/cfengine/cap/linux/cfengine_needs_bootstrap.rb +34 -0
  323. data/plugins/provisioners/cfengine/cap/redhat/cfengine_install.rb +24 -0
  324. data/plugins/provisioners/cfengine/config.rb +126 -0
  325. data/plugins/provisioners/cfengine/plugin.rb +42 -0
  326. data/plugins/provisioners/cfengine/provisioner.rb +135 -0
  327. data/plugins/provisioners/chef/config/base.rb +116 -0
  328. data/plugins/provisioners/chef/config/chef_client.rb +66 -0
  329. data/plugins/provisioners/chef/config/chef_solo.rb +94 -0
  330. data/plugins/provisioners/chef/plugin.rb +33 -0
  331. data/plugins/provisioners/chef/provisioner/base.rb +113 -0
  332. data/plugins/provisioners/chef/provisioner/chef_client.rb +126 -0
  333. data/plugins/provisioners/chef/provisioner/chef_solo.rb +195 -0
  334. data/plugins/provisioners/file/config.rb +29 -0
  335. data/plugins/provisioners/file/plugin.rb +23 -0
  336. data/plugins/provisioners/file/provisioner.rb +16 -0
  337. data/plugins/provisioners/puppet/config/puppet.rb +93 -0
  338. data/plugins/provisioners/puppet/config/puppet_server.rb +16 -0
  339. data/plugins/provisioners/puppet/plugin.rb +33 -0
  340. data/plugins/provisioners/puppet/provisioner/puppet.rb +146 -0
  341. data/plugins/provisioners/puppet/provisioner/puppet_server.rb +74 -0
  342. data/plugins/provisioners/salt/bootstrap-salt.sh +6 -0
  343. data/plugins/provisioners/salt/config.rb +105 -0
  344. data/plugins/provisioners/salt/errors.rb +11 -0
  345. data/plugins/provisioners/salt/plugin.rb +22 -0
  346. data/plugins/provisioners/salt/provisioner.rb +249 -0
  347. data/plugins/provisioners/shell/config.rb +65 -0
  348. data/plugins/provisioners/shell/plugin.rb +23 -0
  349. data/plugins/provisioners/shell/provisioner.rb +98 -0
  350. data/templates/commands/init/Vagrantfile.erb +118 -0
  351. data/templates/commands/ssh_config/config.erb +20 -0
  352. data/templates/config/messages.erb +14 -0
  353. data/templates/config/validation_failed.erb +7 -0
  354. data/templates/guests/arch/network_dhcp.erb +4 -0
  355. data/templates/guests/arch/network_static.erb +5 -0
  356. data/templates/guests/coreos/etcd.service.erb +10 -0
  357. data/templates/guests/debian/network_dhcp.erb +11 -0
  358. data/templates/guests/debian/network_static.erb +7 -0
  359. data/templates/guests/fedora/network_dhcp.erb +6 -0
  360. data/templates/guests/fedora/network_static.erb +12 -0
  361. data/templates/guests/freebsd/network_dhcp.erb +3 -0
  362. data/templates/guests/freebsd/network_static.erb +3 -0
  363. data/templates/guests/gentoo/network_dhcp.erb +4 -0
  364. data/templates/guests/gentoo/network_static.erb +4 -0
  365. data/templates/guests/openbsd/network_dhcp.erb +1 -0
  366. data/templates/guests/openbsd/network_static.erb +1 -0
  367. data/templates/guests/redhat/network_dhcp.erb +6 -0
  368. data/templates/guests/redhat/network_static.erb +8 -0
  369. data/templates/guests/suse/network_dhcp.erb +6 -0
  370. data/templates/guests/suse/network_static.erb +10 -0
  371. data/templates/locales/en.yml +1248 -0
  372. data/templates/nfs/exports.erb +5 -0
  373. data/templates/nfs/exports_freebsd.erb +5 -0
  374. data/templates/nfs/exports_linux.erb +7 -0
  375. data/templates/provisioners/chef_client/client.erb +41 -0
  376. data/templates/provisioners/chef_solo/solo.erb +45 -0
  377. data/test/acceptance/base.rb +48 -0
  378. data/test/acceptance/box_test.rb +99 -0
  379. data/test/acceptance/destroy_test.rb +37 -0
  380. data/test/acceptance/halt_test.rb +72 -0
  381. data/test/acceptance/init_test.rb +33 -0
  382. data/test/acceptance/networking/host_only_test.rb +37 -0
  383. data/test/acceptance/networking/port_forward_test.rb +125 -0
  384. data/test/acceptance/package_test.rb +46 -0
  385. data/test/acceptance/provisioning/basic_test.rb +61 -0
  386. data/test/acceptance/provisioning/chef_solo_test.rb +37 -0
  387. data/test/acceptance/provisioning/shell_test.rb +53 -0
  388. data/test/acceptance/resume_test.rb +17 -0
  389. data/test/acceptance/shared_folders_test.rb +84 -0
  390. data/test/acceptance/skeletons/chef_solo_basic/README.md +3 -0
  391. data/test/acceptance/skeletons/chef_solo_basic/cookbooks/basic/recipes/default.rb +5 -0
  392. data/test/acceptance/skeletons/chef_solo_json/README.md +3 -0
  393. data/test/acceptance/skeletons/chef_solo_json/cookbooks/basic/recipes/default.rb +6 -0
  394. data/test/acceptance/skeletons/provisioner_multi/README.md +3 -0
  395. data/test/acceptance/skeletons/provisioner_multi/cookbooks/basic/recipes/default.rb +5 -0
  396. data/test/acceptance/ssh_test.rb +46 -0
  397. data/test/acceptance/support/config.rb +42 -0
  398. data/test/acceptance/support/isolated_environment.rb +118 -0
  399. data/test/acceptance/support/matchers/have_color.rb +9 -0
  400. data/test/acceptance/support/matchers/match_output.rb +14 -0
  401. data/test/acceptance/support/matchers/succeed.rb +14 -0
  402. data/test/acceptance/support/network_tests.rb +29 -0
  403. data/test/acceptance/support/output.rb +95 -0
  404. data/test/acceptance/support/shared/base_context.rb +72 -0
  405. data/test/acceptance/support/shared/command_examples.rb +33 -0
  406. data/test/acceptance/support/virtualbox.rb +36 -0
  407. data/test/acceptance/suspend_test.rb +56 -0
  408. data/test/acceptance/up_basic_test.rb +33 -0
  409. data/test/acceptance/up_with_box_url.rb +40 -0
  410. data/test/acceptance/vagrant_test.rb +47 -0
  411. data/test/acceptance/version_test.rb +15 -0
  412. data/test/config/acceptance_boxes.yml +7 -0
  413. data/test/support/isolated_environment.rb +46 -0
  414. data/test/support/tempdir.rb +43 -0
  415. data/test/unit/base.rb +27 -0
  416. data/test/unit/support/dummy_provider.rb +16 -0
  417. data/test/unit/support/isolated_environment.rb +179 -0
  418. data/test/unit/support/shared/base_context.rb +104 -0
  419. data/test/unit/vagrant/action/builder_test.rb +242 -0
  420. data/test/unit/vagrant/action/builtin/call_test.rb +145 -0
  421. data/test/unit/vagrant/action/builtin/confirm_test.rb +36 -0
  422. data/test/unit/vagrant/action/builtin/env_set_test.rb +20 -0
  423. data/test/unit/vagrant/action/builtin/graceful_halt_test.rb +60 -0
  424. data/test/unit/vagrant/action/builtin/lock_test.rb +98 -0
  425. data/test/unit/vagrant/action/builtin/ssh_exec_test.rb +57 -0
  426. data/test/unit/vagrant/action/hook_test.rb +120 -0
  427. data/test/unit/vagrant/action/runner_test.rb +71 -0
  428. data/test/unit/vagrant/action/warden_test.rb +92 -0
  429. data/test/unit/vagrant/batch_action_test.rb +36 -0
  430. data/test/unit/vagrant/box_collection_test.rb +237 -0
  431. data/test/unit/vagrant/box_test.rb +122 -0
  432. data/test/unit/vagrant/cli_test.rb +28 -0
  433. data/test/unit/vagrant/config/loader_test.rb +184 -0
  434. data/test/unit/vagrant/config/v1/dummy_config_test.rb +24 -0
  435. data/test/unit/vagrant/config/v1/loader_test.rb +145 -0
  436. data/test/unit/vagrant/config/v1/root_test.rb +40 -0
  437. data/test/unit/vagrant/config/v2/dummy_config_test.rb +24 -0
  438. data/test/unit/vagrant/config/v2/loader_test.rb +151 -0
  439. data/test/unit/vagrant/config/v2/root_test.rb +115 -0
  440. data/test/unit/vagrant/config/v2/util_test.rb +21 -0
  441. data/test/unit/vagrant/config_test.rb +66 -0
  442. data/test/unit/vagrant/environment_test.rb +746 -0
  443. data/test/unit/vagrant/guest_test.rb +179 -0
  444. data/test/unit/vagrant/hosts_test.rb +37 -0
  445. data/test/unit/vagrant/machine_state_test.rb +26 -0
  446. data/test/unit/vagrant/machine_test.rb +384 -0
  447. data/test/unit/vagrant/plugin/v1/command_test.rb +143 -0
  448. data/test/unit/vagrant/plugin/v1/communicator_test.rb +9 -0
  449. data/test/unit/vagrant/plugin/v1/config_test.rb +50 -0
  450. data/test/unit/vagrant/plugin/v1/host_test.rb +5 -0
  451. data/test/unit/vagrant/plugin/v1/manager_test.rb +114 -0
  452. data/test/unit/vagrant/plugin/v1/plugin_test.rb +267 -0
  453. data/test/unit/vagrant/plugin/v1/provider_test.rb +18 -0
  454. data/test/unit/vagrant/plugin/v2/command_test.rb +238 -0
  455. data/test/unit/vagrant/plugin/v2/communicator_test.rb +9 -0
  456. data/test/unit/vagrant/plugin/v2/components_test.rb +17 -0
  457. data/test/unit/vagrant/plugin/v2/config_test.rb +60 -0
  458. data/test/unit/vagrant/plugin/v2/host_test.rb +5 -0
  459. data/test/unit/vagrant/plugin/v2/manager_test.rb +174 -0
  460. data/test/unit/vagrant/plugin/v2/plugin_test.rb +305 -0
  461. data/test/unit/vagrant/plugin/v2/provider_test.rb +18 -0
  462. data/test/unit/vagrant/registry_test.rb +128 -0
  463. data/test/unit/vagrant/util/ansi_escape_code_remover_test.rb +16 -0
  464. data/test/unit/vagrant/util/downloader_test.rb +57 -0
  465. data/test/unit/vagrant/util/file_checksum_test.rb +23 -0
  466. data/test/unit/vagrant/util/hash_with_indifferent_access_test.rb +38 -0
  467. data/test/unit/vagrant/util/is_port_open_test.rb +53 -0
  468. data/test/unit/vagrant/util/line_endings_helper_test.rb +16 -0
  469. data/test/unit/vagrant/util/network_ip_test.rb +17 -0
  470. data/test/unit/vagrant/util/retryable_test.rb +106 -0
  471. data/test/unit/vagrant/util/safe_chdir_test.rb +43 -0
  472. data/test/unit/vagrant/util/scoped_hash_override_test.rb +48 -0
  473. data/test/unit/vagrant/util/ssh_test.rb +30 -0
  474. data/test/unit/vagrant/util/string_block_editor_test.rb +106 -0
  475. data/test/unit/vagrant/util/which_test.rb +43 -0
  476. data/test/unit/vagrant_test.rb +74 -0
  477. data/tnargav.gemspec +1 -1
  478. metadata +477 -1
@@ -0,0 +1,9 @@
1
+ module Vagrant
2
+ module Config
3
+ module V2
4
+ autoload :DummyConfig, "vagrant/config/v2/dummy_config"
5
+ autoload :Loader, "vagrant/config/v2/loader"
6
+ autoload :Root, "vagrant/config/v2/root"
7
+ end
8
+ end
9
+ end
@@ -0,0 +1,13 @@
1
+ module Vagrant
2
+ module Config
3
+ module V2
4
+ # This is a configuration object that can have anything done
5
+ # to it. Anything, and it just appears to keep working.
6
+ class DummyConfig
7
+ def method_missing(name, *args, &block)
8
+ DummyConfig.new
9
+ end
10
+ end
11
+ end
12
+ end
13
+ end
@@ -0,0 +1,141 @@
1
+ require "vagrant/config/v2/root"
2
+
3
+ module Vagrant
4
+ module Config
5
+ module V2
6
+ # This is the loader that handles configuration loading for V2
7
+ # configurations.
8
+ class Loader < VersionBase
9
+ # Returns a bare empty configuration object.
10
+ #
11
+ # @return [V2::Root]
12
+ def self.init
13
+ new_root_object
14
+ end
15
+
16
+ # Finalizes the configuration by making sure there is at least
17
+ # one VM defined in it.
18
+ def self.finalize(config)
19
+ # Call the `#finalize` method on each of the configuration keys.
20
+ # They're expected to modify themselves in our case.
21
+ config.finalize!
22
+
23
+ # Return the object
24
+ config
25
+ end
26
+
27
+ # Loads the configuration for the given proc and returns a configuration
28
+ # object.
29
+ #
30
+ # @param [Proc] config_proc
31
+ # @return [Object]
32
+ def self.load(config_proc)
33
+ # Create a root configuration object
34
+ root = new_root_object
35
+
36
+ # Call the proc with the root
37
+ config_proc.call(root)
38
+
39
+ # Return the root object, which doubles as the configuration object
40
+ # we actually use for accessing as well.
41
+ root
42
+ end
43
+
44
+ # Merges two configuration objects.
45
+ #
46
+ # @param [V2::Root] old The older root config.
47
+ # @param [V2::Root] new The newer root config.
48
+ # @return [V2::Root]
49
+ def self.merge(old, new)
50
+ # Grab the internal states, we use these heavily throughout the process
51
+ old_state = old.__internal_state
52
+ new_state = new.__internal_state
53
+
54
+ # The config map for the new object is the old one merged with the
55
+ # new one.
56
+ config_map = old_state["config_map"].merge(new_state["config_map"])
57
+
58
+ # Merge the keys.
59
+ old_keys = old_state["keys"]
60
+ new_keys = new_state["keys"]
61
+ keys = {}
62
+ old_keys.each do |key, old_value|
63
+ if new_keys.has_key?(key)
64
+ # We need to do a merge, which we expect to be available
65
+ # on the config class itself.
66
+ keys[key] = old_value.merge(new_keys[key])
67
+ else
68
+ # We just take the old value, but dup it so that we can modify.
69
+ keys[key] = old_value.dup
70
+ end
71
+ end
72
+
73
+ new_keys.each do |key, new_value|
74
+ # Add in the keys that the new class has that we haven't merged.
75
+ if !keys.has_key?(key)
76
+ keys[key] = new_value.dup
77
+ end
78
+ end
79
+
80
+ # Merge the missing keys
81
+ new_missing_key_calls =
82
+ old_state["missing_key_calls"] + new_state["missing_key_calls"]
83
+
84
+ # Return the final root object
85
+ V2::Root.new(config_map).tap do |result|
86
+ result.__set_internal_state({
87
+ "config_map" => config_map,
88
+ "keys" => keys,
89
+ "missing_key_calls" => new_missing_key_calls
90
+ })
91
+ end
92
+ end
93
+
94
+ # Upgrade a V1 configuration to a V2 configuration. We do this by
95
+ # creating a V2 configuration, and calling "upgrade" on each of the
96
+ # V1 configurations, expecting them to set the right settings on the
97
+ # new root.
98
+ #
99
+ # @param [V1::Root] old
100
+ # @return [Array] A 3-tuple result.
101
+ def self.upgrade(old)
102
+ # Get a new root
103
+ root = new_root_object
104
+
105
+ # Store the warnings/errors
106
+ warnings = []
107
+ errors = []
108
+
109
+ # Go through the old keys and upgrade them if they can be
110
+ old.__internal_state["keys"].each do |_, old_value|
111
+ if old_value.respond_to?(:upgrade)
112
+ result = old_value.upgrade(root)
113
+
114
+ # Sanity check to guard against random return values
115
+ if result.is_a?(Array)
116
+ warnings += result[0]
117
+ errors += result[1]
118
+ end
119
+ end
120
+ end
121
+
122
+ old.__internal_state["missing_key_calls"].to_a.sort.each do |key|
123
+ warnings << I18n.t("vagrant.config.loader.bad_v1_key", :key => key)
124
+ end
125
+
126
+ [root, warnings, errors]
127
+ end
128
+
129
+ protected
130
+
131
+ def self.new_root_object
132
+ # Get all the registered plugins for V2
133
+ config_map = Vagrant.plugin("2").manager.config
134
+
135
+ # Create the configuration root object
136
+ V2::Root.new(config_map)
137
+ end
138
+ end
139
+ end
140
+ end
141
+ end
@@ -0,0 +1,111 @@
1
+ require "set"
2
+
3
+ require "vagrant/config/v2/util"
4
+
5
+ module Vagrant
6
+ module Config
7
+ module V2
8
+ # This is the root configuration class. An instance of this is what
9
+ # is passed into version 1 Vagrant configuration blocks.
10
+ class Root
11
+ # Initializes a root object that maps the given keys to specific
12
+ # configuration classes.
13
+ #
14
+ # @param [Hash] config_map Map of key to config class.
15
+ def initialize(config_map, keys=nil)
16
+ @keys = keys || {}
17
+ @config_map = config_map
18
+ @missing_key_calls = Set.new
19
+ end
20
+
21
+ # We use method_missing as a way to get the configuration that is
22
+ # used for Vagrant and load the proper configuration classes for
23
+ # each.
24
+ def method_missing(name, *args)
25
+ return @keys[name] if @keys.has_key?(name)
26
+
27
+ config_klass = @config_map[name.to_sym]
28
+ if config_klass
29
+ # Instantiate the class and return the instance
30
+ @keys[name] = config_klass.new
31
+ return @keys[name]
32
+ else
33
+ # Record access to a missing key as an error
34
+ @missing_key_calls.add(name.to_s)
35
+ return DummyConfig.new
36
+ end
37
+ end
38
+
39
+ # Called to finalize this object just prior to it being used by
40
+ # the Vagrant system. The "!" signifies that this is expected to
41
+ # mutate itself.
42
+ def finalize!
43
+ @config_map.each do |key, klass|
44
+ if !@keys.has_key?(key)
45
+ @keys[key] = klass.new
46
+ end
47
+ end
48
+
49
+ @keys.each do |_key, instance|
50
+ instance.finalize!
51
+ end
52
+ end
53
+
54
+ # This validates the configuration and returns a hash of error
55
+ # messages by section. If there are no errors, an empty hash
56
+ # is returned.
57
+ #
58
+ # @param [Environment] env
59
+ # @return [Hash]
60
+ def validate(machine)
61
+ # Go through each of the configuration keys and validate
62
+ errors = {}
63
+ @keys.each do |_key, instance|
64
+ if instance.respond_to?(:validate)
65
+ # Validate this single item, and if we have errors then
66
+ # we merge them into our total errors list.
67
+ result = instance.validate(machine)
68
+ if result && !result.empty?
69
+ errors = Util.merge_errors(errors, result)
70
+ end
71
+ end
72
+ end
73
+
74
+ # Go through and delete empty keys
75
+ errors.keys.each do |key|
76
+ errors.delete(key) if errors[key].empty?
77
+ end
78
+
79
+ # If we have missing keys, record those as errors
80
+ if !@missing_key_calls.empty?
81
+ errors["Vagrant"] = @missing_key_calls.to_a.sort.map do |key|
82
+ I18n.t("vagrant.config.root.bad_key", :key => key)
83
+ end
84
+ end
85
+
86
+ errors
87
+ end
88
+
89
+ # Returns the internal state of the root object. This is used
90
+ # by outside classes when merging, and shouldn't be called directly.
91
+ # Note the strange method name is to attempt to avoid any name
92
+ # clashes with potential configuration keys.
93
+ def __internal_state
94
+ {
95
+ "config_map" => @config_map,
96
+ "keys" => @keys,
97
+ "missing_key_calls" => @missing_key_calls
98
+ }
99
+ end
100
+
101
+ # This sets the internal state. This is used by the core to do some
102
+ # merging logic and shouldn't be used by the general public.
103
+ def __set_internal_state(state)
104
+ @config_map = state["config_map"] if state.has_key?("config_map")
105
+ @keys = state["keys"] if state.has_key?("keys")
106
+ @missing_key_calls = state["missing_key_calls"] if state.has_key?("missing_key_calls")
107
+ end
108
+ end
109
+ end
110
+ end
111
+ end
@@ -0,0 +1,21 @@
1
+ module Vagrant
2
+ module Config
3
+ module V2
4
+ class Util
5
+ # This merges two error hashes from validate methods.
6
+ #
7
+ # @param [Hash] first
8
+ # @param [Hash] second
9
+ # @return [Hash] Merged result
10
+ def self.merge_errors(first, second)
11
+ first.dup.tap do |result|
12
+ second.each do |key, value|
13
+ result[key] ||= []
14
+ result[key] += value
15
+ end
16
+ end
17
+ end
18
+ end
19
+ end
20
+ end
21
+ end
@@ -0,0 +1,80 @@
1
+ module Vagrant
2
+ module Config
3
+ # This is the base class for any configuration versions, and includes
4
+ # the stub methods that configuaration versions must implement. Vagrant
5
+ # supports configuration versioning so that backwards compatibility can be
6
+ # maintained for past Vagrantfiles while newer configurations are added.
7
+ # Vagrant only introduces new configuration versions for major versions
8
+ # of Vagrant.
9
+ class VersionBase
10
+ # Returns an empty configuration object. This can be any kind of object,
11
+ # since it is treated as an opaque value on the other side, used only
12
+ # for things like calling into {merge}.
13
+ #
14
+ # @return [Object]
15
+ def self.init
16
+ raise NotImplementedError
17
+ end
18
+
19
+ # This is called just before configuration loading is complete of
20
+ # a potentially completely-merged value to perform final touch-ups
21
+ # to the configuration, if required.
22
+ #
23
+ # This is an optional method to implement. The default implementation
24
+ # will simply return the same object.
25
+ #
26
+ # This will ONLY be called if this is the version that is being
27
+ # used. In the case that an `upgrade` is called, this will never
28
+ # be called.
29
+ #
30
+ # @param [Object] obj Final configuration object.
31
+ # @param [Object] Finalized configuration object.
32
+ def self.finalize(obj)
33
+ obj
34
+ end
35
+
36
+ # Loads the configuration for the given proc and returns a configuration
37
+ # object. The return value is treated as an opaque object, so it can be
38
+ # anything you'd like. The return value is the object that is passed
39
+ # into methods like {merge}, so it should be something you expect.
40
+ #
41
+ # @param [Proc] proc The proc that is to be configured.
42
+ # @return [Object]
43
+ def self.load(proc)
44
+ raise NotImplementedError
45
+ end
46
+
47
+ # Merges two configuration objects, returning the merged object.
48
+ # The values of `old` and `new` are the opaque objects returned by
49
+ # {load} or {init}.
50
+ #
51
+ # Once again, the return object is treated as an opaque value by
52
+ # the Vagrant configuration loader, so it can be anything you'd like.
53
+ #
54
+ # @param [Object] old Old configuration object.
55
+ # @param [Object] new New configuration object.
56
+ # @return [Object] The merged configuration object.
57
+ def self.merge(old, new)
58
+ raise NotImplementedError
59
+ end
60
+
61
+ # This is called if a previous version of configuration needs to be
62
+ # upgraded to this version. Each version of configuration should know
63
+ # how to upgrade the version immediately prior to it. This should be
64
+ # a best effort upgrade that makes many assumptions. The goal is for
65
+ # this to work in almost every case, but perhaps with some warnings.
66
+ # The return value for this is a 3-tuple: `[object, warnings, errors]`,
67
+ # where `object` is the upgraded configuration object, `warnings` is
68
+ # an array of warning messages, and `errors` is an array of error
69
+ # messages.
70
+ #
71
+ # @param [Object] old The version of the configuration object just
72
+ # prior to this one.
73
+ # @return [Array] The 3-tuple result. Please see the above documentation
74
+ # for more information on the exact structure of this object.
75
+ def self.upgrade(old)
76
+ raise NotImplementedError
77
+ end
78
+ end
79
+ end
80
+ end
@@ -0,0 +1,819 @@
1
+ require 'fileutils'
2
+ require 'json'
3
+ require 'pathname'
4
+ require 'set'
5
+ require 'thread'
6
+
7
+ require 'log4r'
8
+
9
+ require 'vagrant/util/file_mode'
10
+ require 'vagrant/util/platform'
11
+
12
+ module Vagrant
13
+ # Represents a single Vagrant environment. A "Vagrant environment" is
14
+ # defined as basically a folder with a "Vagrantfile." This class allows
15
+ # access to the VMs, CLI, etc. all in the scope of this environment.
16
+ class Environment
17
+ DEFAULT_LOCAL_DATA = ".vagrant"
18
+
19
+ # The `cwd` that this environment represents
20
+ attr_reader :cwd
21
+
22
+ # The persistent data directory where global data can be stored. It
23
+ # is up to the creator of the data in this directory to properly
24
+ # remove it when it is no longer needed.
25
+ #
26
+ # @return [Pathname]
27
+ attr_reader :data_dir
28
+
29
+ # The valid name for a Vagrantfile for this environment.
30
+ attr_reader :vagrantfile_name
31
+
32
+ # The {UI} object to communicate with the outside world.
33
+ attr_reader :ui
34
+
35
+ # This is the UI class to use when creating new UIs.
36
+ attr_reader :ui_class
37
+
38
+ # The directory to the "home" folder that Vagrant will use to store
39
+ # global state.
40
+ attr_reader :home_path
41
+
42
+ # The directory to the directory where local, environment-specific
43
+ # data is stored.
44
+ attr_reader :local_data_path
45
+
46
+ # The directory where temporary files for Vagrant go.
47
+ attr_reader :tmp_path
48
+
49
+ # The directory where boxes are stored.
50
+ attr_reader :boxes_path
51
+
52
+ # The path where the plugins are stored (gems)
53
+ attr_reader :gems_path
54
+
55
+ # The path to the default private key
56
+ attr_reader :default_private_key_path
57
+
58
+ # Initializes a new environment with the given options. The options
59
+ # is a hash where the main available key is `cwd`, which defines where
60
+ # the environment represents. There are other options available but
61
+ # they shouldn't be used in general. If `cwd` is nil, then it defaults
62
+ # to the `Dir.pwd` (which is the cwd of the executing process).
63
+ def initialize(opts=nil)
64
+ opts = {
65
+ :cwd => nil,
66
+ :home_path => nil,
67
+ :local_data_path => nil,
68
+ :lock_path => nil,
69
+ :ui_class => nil,
70
+ :vagrantfile_name => nil
71
+ }.merge(opts || {})
72
+
73
+ # Set the default working directory to look for the vagrantfile
74
+ opts[:cwd] ||= ENV["VAGRANT_CWD"] if ENV.has_key?("VAGRANT_CWD")
75
+ opts[:cwd] ||= Dir.pwd
76
+ opts[:cwd] = Pathname.new(opts[:cwd])
77
+ if !opts[:cwd].directory?
78
+ raise Errors::EnvironmentNonExistentCWD, cwd: opts[:cwd].to_s
79
+ end
80
+
81
+ # Set the default ui class
82
+ opts[:ui_class] ||= UI::Silent
83
+
84
+ # Set the Vagrantfile name up. We append "Vagrantfile" and "vagrantfile" so that
85
+ # those continue to work as well, but anything custom will take precedence.
86
+ opts[:vagrantfile_name] ||= ENV["VAGRANT_VAGRANTFILE"] if \
87
+ ENV.has_key?("VAGRANT_VAGRANTFILE")
88
+ opts[:vagrantfile_name] = [opts[:vagrantfile_name]] if \
89
+ opts[:vagrantfile_name] && !opts[:vagrantfile_name].is_a?(Array)
90
+
91
+ # Set instance variables for all the configuration parameters.
92
+ @cwd = opts[:cwd]
93
+ @home_path = opts[:home_path]
94
+ @lock_path = opts[:lock_path]
95
+ @vagrantfile_name = opts[:vagrantfile_name]
96
+ @ui = opts[:ui_class].new
97
+ @ui_class = opts[:ui_class]
98
+
99
+ # This is the batch lock, that enforces that only one {BatchAction}
100
+ # runs at a time from {#batch}.
101
+ @batch_lock = Mutex.new
102
+
103
+ @lock_acquired = false
104
+
105
+ @logger = Log4r::Logger.new("vagrant::environment")
106
+ @logger.info("Environment initialized (#{self})")
107
+ @logger.info(" - cwd: #{cwd}")
108
+
109
+ # Setup the home directory
110
+ setup_home_path
111
+ @boxes_path = @home_path.join("boxes")
112
+ @data_dir = @home_path.join("data")
113
+ @gems_path = @home_path.join("gems")
114
+ @tmp_path = @home_path.join("tmp")
115
+
116
+ # Setup the local data directory. If a configuration path is given,
117
+ # then it is expanded relative to the working directory. Otherwise,
118
+ # we use the default which is expanded relative to the root path.
119
+ opts[:local_data_path] ||= ENV["VAGRANT_DOTFILE_PATH"]
120
+ opts[:local_data_path] ||= root_path.join(DEFAULT_LOCAL_DATA) if !root_path.nil?
121
+ if opts[:local_data_path]
122
+ @local_data_path = Pathname.new(File.expand_path(opts[:local_data_path], @cwd))
123
+ end
124
+
125
+ setup_local_data_path
126
+
127
+ # Setup the default private key
128
+ @default_private_key_path = @home_path.join("insecure_private_key")
129
+ copy_insecure_private_key
130
+
131
+ # Load the plugins
132
+ load_plugins
133
+
134
+ # Call the environment load hooks
135
+ hook(:environment_load)
136
+ end
137
+
138
+ # Return a human-friendly string for pretty printed or inspected
139
+ # instances.
140
+ #
141
+ # @return [String]
142
+ def inspect
143
+ "#<#{self.class}: #{@cwd}>"
144
+ end
145
+
146
+ #---------------------------------------------------------------
147
+ # Helpers
148
+ #---------------------------------------------------------------
149
+
150
+ # Returns a list of machines that this environment is currently
151
+ # managing that physically have been created.
152
+ #
153
+ # An "active" machine is a machine that Vagrant manages that has
154
+ # been created. The machine itself may be in any state such as running,
155
+ # suspended, etc. but if a machine is "active" then it exists.
156
+ #
157
+ # Note that the machines in this array may no longer be present in
158
+ # the Vagrantfile of this environment. In this case the machine can
159
+ # be considered an "orphan." Determining which machines are orphan
160
+ # and which aren't is not currently a supported feature, but will
161
+ # be in a future version.
162
+ #
163
+ # @return [Array<String, Symbol>]
164
+ def active_machines
165
+ machine_folder = @local_data_path.join("machines")
166
+
167
+ # If the machine folder is not a directory then we just return
168
+ # an empty array since no active machines exist.
169
+ return [] if !machine_folder.directory?
170
+
171
+ # Traverse the machines folder accumulate a result
172
+ result = []
173
+
174
+ machine_folder.children(true).each do |name_folder|
175
+ # If this isn't a directory then it isn't a machine
176
+ next if !name_folder.directory?
177
+
178
+ name = name_folder.basename.to_s.to_sym
179
+ name_folder.children(true).each do |provider_folder|
180
+ # If this isn't a directory then it isn't a provider
181
+ next if !provider_folder.directory?
182
+
183
+ # If this machine doesn't have an ID, then ignore
184
+ next if !provider_folder.join("id").file?
185
+
186
+ provider = provider_folder.basename.to_s.to_sym
187
+ result << [name, provider]
188
+ end
189
+ end
190
+
191
+ # Return the results
192
+ result
193
+ end
194
+
195
+ # This creates a new batch action, yielding it, and then running it
196
+ # once the block is called.
197
+ #
198
+ # This handles the case where batch actions are disabled by the
199
+ # VAGRANT_NO_PARALLEL environmental variable.
200
+ def batch(parallel=true)
201
+ parallel = false if ENV["VAGRANT_NO_PARALLEL"]
202
+
203
+ @batch_lock.synchronize do
204
+ BatchAction.new(parallel).tap do |b|
205
+ # Yield it so that the caller can setup actions
206
+ yield b
207
+
208
+ # And run it!
209
+ b.run
210
+ end
211
+ end
212
+ end
213
+
214
+ # This returns the provider name for the default provider for this
215
+ # environment. The provider returned is currently hardcoded to "virtualbox"
216
+ # but one day should be a detected valid, best-case provider for this
217
+ # environment.
218
+ #
219
+ # @return [Symbol] Name of the default provider.
220
+ def default_provider
221
+ (ENV['VAGRANT_DEFAULT_PROVIDER'] || :virtualbox).to_sym
222
+ end
223
+
224
+ # Returns the collection of boxes for the environment.
225
+ #
226
+ # @return [BoxCollection]
227
+ def boxes
228
+ @_boxes ||= BoxCollection.new(boxes_path, temp_dir_root: tmp_path)
229
+ end
230
+
231
+ # This is the global config, comprised of loading configuration from
232
+ # the default, home, and root Vagrantfiles. This configuration is only
233
+ # really useful for reading the list of virtual machines, since each
234
+ # individual VM can override _most_ settings.
235
+ #
236
+ # This is lazy-loaded upon first use.
237
+ #
238
+ # @return [Object]
239
+ def config_global
240
+ return @config_global if @config_global
241
+
242
+ @logger.info("Initializing config...")
243
+
244
+ home_vagrantfile = nil
245
+ root_vagrantfile = nil
246
+ home_vagrantfile = find_vagrantfile(home_path) if home_path
247
+ root_vagrantfile = find_vagrantfile(root_path, @vagrantfile_name) if root_path
248
+
249
+ # Create the configuration loader and set the sources that are global.
250
+ # We use this to load the configuration, and the list of machines we are
251
+ # managing. Then, the actual individual configuration is loaded for
252
+ # each {#machine} call.
253
+ @config_loader = Config::Loader.new(Config::VERSIONS, Config::VERSIONS_ORDER)
254
+ @config_loader.set(:default, File.expand_path("config/default.rb", Vagrant.source_root))
255
+ @config_loader.set(:home, home_vagrantfile) if home_vagrantfile
256
+ @config_loader.set(:root, root_vagrantfile) if root_vagrantfile
257
+
258
+ # Make the initial call to get the "global" config. This is mostly
259
+ # only useful to get the list of machines that we are managing.
260
+ # Because of this, we ignore any warnings or errors.
261
+ @config_global, _ = @config_loader.load([:default, :home, :root])
262
+
263
+ # Return the config
264
+ @config_global
265
+ end
266
+
267
+ # This defines a hook point where plugin action hooks that are registered
268
+ # against the given name will be run in the context of this environment.
269
+ #
270
+ # @param [Symbol] name Name of the hook.
271
+ def hook(name)
272
+ @logger.info("Running hook: #{name}")
273
+ callable = Action::Builder.new
274
+ action_runner.run(
275
+ callable,
276
+ :action_name => name,
277
+ :env => self)
278
+ end
279
+
280
+ # This returns a machine with the proper provider for this environment.
281
+ # The machine named by `name` must be in this environment.
282
+ #
283
+ # @param [Symbol] name Name of the machine (as configured in the
284
+ # Vagrantfile).
285
+ # @param [Symbol] provider The provider that this machine should be
286
+ # backed by.
287
+ # @param [Boolean] refresh If true, then if there is a cached version
288
+ # it is reloaded.
289
+ # @return [Machine]
290
+ def machine(name, provider, refresh=false)
291
+ @logger.info("Getting machine: #{name} (#{provider})")
292
+
293
+ # Compose the cache key of the name and provider, and return from
294
+ # the cache if we have that.
295
+ cache_key = [name, provider]
296
+ @machines ||= {}
297
+ if refresh
298
+ @logger.info("Refreshing machine (busting cache): #{name} (#{provider})")
299
+ @machines.delete(cache_key)
300
+ end
301
+
302
+ if @machines.has_key?(cache_key)
303
+ @logger.info("Returning cached machine: #{name} (#{provider})")
304
+ return @machines[cache_key]
305
+ end
306
+
307
+ @logger.info("Uncached load of machine.")
308
+ sub_vm = config_global.vm.defined_vms[name]
309
+ if !sub_vm
310
+ raise Errors::MachineNotFound, :name => name, :provider => provider
311
+ end
312
+
313
+ provider_plugin = Vagrant.plugin("2").manager.providers[provider]
314
+ if !provider_plugin
315
+ raise Errors::ProviderNotFound, :machine => name, :provider => provider
316
+ end
317
+
318
+ # Extra the provider class and options from the plugin data
319
+ provider_cls = provider_plugin[0]
320
+ provider_options = provider_plugin[1]
321
+
322
+ # Build the machine configuration. This requires two passes: The first pass
323
+ # loads in the machine sub-configuration. Since this can potentially
324
+ # define a new box to base the machine from, we then make a second pass
325
+ # with the box Vagrantfile (if it has one).
326
+ vm_config_key = "vm_#{name}".to_sym
327
+ @config_loader.set(vm_config_key, sub_vm.config_procs)
328
+ config, config_warnings, config_errors = \
329
+ @config_loader.load([:default, :home, :root, vm_config_key])
330
+
331
+ # Determine the possible box formats for any boxes and find the box
332
+ box_formats = provider_options[:box_format] || provider
333
+ box = nil
334
+
335
+ # Set this variable in order to keep track of if the box changes
336
+ # too many times.
337
+ original_box = config.vm.box
338
+ box_changed = false
339
+
340
+ load_box_and_overrides = lambda do
341
+ box = nil
342
+ if config.vm.box
343
+ begin
344
+ box = boxes.find(config.vm.box, box_formats)
345
+ rescue Errors::BoxUpgradeRequired
346
+ # Upgrade the box if we must
347
+ @logger.info("Upgrading box during config load: #{config.vm.box}")
348
+ boxes.upgrade(config.vm.box)
349
+ retry
350
+ end
351
+ end
352
+
353
+ # If a box was found, then we attempt to load the Vagrantfile for
354
+ # that box. We don't require a box since we allow providers to download
355
+ # boxes and so on.
356
+ if box
357
+ box_vagrantfile = find_vagrantfile(box.directory)
358
+ if box_vagrantfile
359
+ # The box has a custom Vagrantfile, so we load that into the config
360
+ # as well.
361
+ @logger.info("Box exists with Vagrantfile. Reloading machine config.")
362
+ box_config_key = "box_#{box.name}_#{box.provider}".to_sym
363
+ @config_loader.set(box_config_key, box_vagrantfile)
364
+ config, config_warnings, config_errors = \
365
+ @config_loader.load([:default, box_config_key, :home, :root, vm_config_key])
366
+ end
367
+ end
368
+
369
+ # If there are provider overrides for the machine, then we run
370
+ # those as well.
371
+ provider_overrides = config.vm.get_provider_overrides(provider)
372
+ if provider_overrides.length > 0
373
+ @logger.info("Applying #{provider_overrides.length} provider overrides. Reloading config.")
374
+ provider_override_key = "vm_#{name}_#{config.vm.box}_#{provider}".to_sym
375
+ @config_loader.set(provider_override_key, provider_overrides)
376
+ config, config_warnings, config_errors = \
377
+ @config_loader.load([:default, box_config_key, :home, :root, vm_config_key, provider_override_key])
378
+ end
379
+
380
+ if config.vm.box && original_box != config.vm.box
381
+ if box_changed
382
+ # We already changed boxes once, so report an error that a
383
+ # box is attempting to change boxes again.
384
+ raise Errors::BoxConfigChangingBox
385
+ end
386
+
387
+ # The box changed, probably due to the provider override. Let's
388
+ # run the configuration one more time with the new box.
389
+ @logger.info("Box changed to: #{config.vm.box}. Reloading configurations.")
390
+ original_box = config.vm.box
391
+ box_changed = true
392
+
393
+ # Recurse so that we reload all the configurations
394
+ load_box_and_overrides.call
395
+ end
396
+ end
397
+
398
+ # Load the box and overrides configuration
399
+ load_box_and_overrides.call
400
+
401
+ # Get the provider configuration from the final loaded configuration
402
+ provider_config = config.vm.get_provider_config(provider)
403
+
404
+ # Determine the machine data directory and pass it to the machine.
405
+ # XXX: Permissions error here.
406
+ machine_data_path = @local_data_path.join("machines/#{name}/#{provider}")
407
+ FileUtils.mkdir_p(machine_data_path)
408
+
409
+ # If there were warnings or errors we want to output them
410
+ if !config_warnings.empty? || !config_errors.empty?
411
+ # The color of the output depends on whether we have warnings
412
+ # or errors...
413
+ level = config_errors.empty? ? :warn : :error
414
+ output = Util::TemplateRenderer.render(
415
+ "config/messages",
416
+ :warnings => config_warnings,
417
+ :errors => config_errors).chomp
418
+ @ui.send(level, I18n.t("vagrant.general.config_upgrade_messages",
419
+ name: name,
420
+ :output => output))
421
+
422
+ # If we had errors, then we bail
423
+ raise Errors::ConfigUpgradeErrors if !config_errors.empty?
424
+ end
425
+
426
+ # Create the machine and cache it for future calls. This will also
427
+ # return the machine from this method.
428
+ @machines[cache_key] = Machine.new(name, provider, provider_cls, provider_config,
429
+ provider_options, config, machine_data_path, box, self)
430
+ end
431
+
432
+ # This returns a list of the configured machines for this environment.
433
+ # Each of the names returned by this method is valid to be used with
434
+ # the {#machine} method.
435
+ #
436
+ # @return [Array<Symbol>] Configured machine names.
437
+ def machine_names
438
+ config_global.vm.defined_vm_keys.dup
439
+ end
440
+
441
+ # This returns the name of the machine that is the "primary." In the
442
+ # case of a single-machine environment, this is just the single machine
443
+ # name. In the case of a multi-machine environment, then this can
444
+ # potentially be nil if no primary machine is specified.
445
+ #
446
+ # @return [Symbol]
447
+ def primary_machine_name
448
+ # If it is a single machine environment, then return the name
449
+ return machine_names.first if machine_names.length == 1
450
+
451
+ # If it is a multi-machine environment, then return the primary
452
+ config_global.vm.defined_vms.each do |name, subvm|
453
+ return name if subvm.options[:primary]
454
+ end
455
+
456
+ # If no primary was specified, nil it is
457
+ nil
458
+ end
459
+
460
+ # Unload the environment, running completion hooks. The environment
461
+ # should not be used after this (but CAN be, technically). It is
462
+ # recommended to always immediately set the variable to `nil` after
463
+ # running this so you can't accidentally run any more methods. Example:
464
+ #
465
+ # env.unload
466
+ # env = nil
467
+ #
468
+ def unload
469
+ hook(:environment_unload)
470
+ end
471
+
472
+ # Makes a call to the CLI with the given arguments as if they
473
+ # came from the real command line (sometimes they do!). An example:
474
+ #
475
+ # env.cli("package", "--vagrantfile", "Vagrantfile")
476
+ #
477
+ def cli(*args)
478
+ CLI.new(args.flatten, self).execute
479
+ end
480
+
481
+ # Returns the host object associated with this environment.
482
+ #
483
+ # @return [Class]
484
+ def host
485
+ return @host if defined?(@host)
486
+
487
+ # Attempt to figure out the host class. Note that the order
488
+ # matters here, so please don't touch. Specifically: The symbol
489
+ # check is done after the detect check because the symbol check
490
+ # will return nil, and we don't want to trigger a detect load.
491
+ host_klass = config_global.vagrant.host
492
+ if host_klass.nil? || host_klass == :detect
493
+ hosts = Vagrant.plugin("2").manager.hosts.to_hash
494
+
495
+ # Get the flattened list of available hosts
496
+ host_klass = Hosts.detect(hosts)
497
+ end
498
+
499
+ # If no host class is detected, we use the base class.
500
+ host_klass ||= Vagrant.plugin("2", :host)
501
+
502
+ @host ||= host_klass.new(@ui)
503
+ end
504
+
505
+ # Action runner for executing actions in the context of this environment.
506
+ #
507
+ # @return [Action::Runner]
508
+ def action_runner
509
+ @action_runner ||= Action::Runner.new do
510
+ {
511
+ :action_runner => action_runner,
512
+ :box_collection => boxes,
513
+ :global_config => config_global,
514
+ :host => host,
515
+ :gems_path => gems_path,
516
+ :home_path => home_path,
517
+ :root_path => root_path,
518
+ :tmp_path => tmp_path,
519
+ :ui => @ui
520
+ }
521
+ end
522
+ end
523
+
524
+ # The root path is the path where the top-most (loaded last)
525
+ # Vagrantfile resides. It can be considered the project root for
526
+ # this environment.
527
+ #
528
+ # @return [String]
529
+ def root_path
530
+ return @root_path if defined?(@root_path)
531
+
532
+ root_finder = lambda do |path|
533
+ # Note: To remain compatible with Ruby 1.8, we have to use
534
+ # a `find` here instead of an `each`.
535
+ vf = find_vagrantfile(path, @vagrantfile_name)
536
+ return path if vf
537
+ return nil if path.root? || !File.exist?(path)
538
+ root_finder.call(path.parent)
539
+ end
540
+
541
+ @root_path = root_finder.call(cwd)
542
+ end
543
+
544
+ # This returns the path which Vagrant uses to determine the location
545
+ # of the file lock. This is specific to each operating system.
546
+ def lock_path
547
+ @lock_path || tmp_path.join("vagrant.lock")
548
+ end
549
+
550
+ # This locks Vagrant for the duration of the block passed to this
551
+ # method. During this time, any other environment which attempts
552
+ # to lock which points to the same lock file will fail.
553
+ def lock
554
+ # This allows multiple locks in the same process to be nested
555
+ return yield if @lock_acquired
556
+
557
+ File.open(lock_path, "w+") do |f|
558
+ # The file locking fails only if it returns "false." If it
559
+ # succeeds it returns a 0, so we must explicitly check for
560
+ # the proper error case.
561
+ raise Errors::EnvironmentLockedError if f.flock(File::LOCK_EX | File::LOCK_NB) === false
562
+
563
+ begin
564
+ # Mark that we have a lock
565
+ @lock_acquired = true
566
+
567
+ yield
568
+ ensure
569
+ # We need to make sure that no matter what this is always
570
+ # reset to false so we don't think we have a lock when we
571
+ # actually don't.
572
+ @lock_acquired = false
573
+ end
574
+ end
575
+ end
576
+
577
+ #---------------------------------------------------------------
578
+ # Load Methods
579
+ #---------------------------------------------------------------
580
+
581
+ # This sets the `@home_path` variable properly.
582
+ #
583
+ # @return [Pathname]
584
+ def setup_home_path
585
+ @home_path = Pathname.new(File.expand_path(@home_path ||
586
+ ENV["VAGRANT_HOME"] ||
587
+ default_home_path))
588
+ @logger.info("Home path: #{@home_path}")
589
+
590
+ # Setup the list of child directories that need to be created if they
591
+ # don't already exist.
592
+ dirs = [@home_path]
593
+ subdirs = ["boxes", "data", "gems", "rgloader", "tmp"]
594
+ dirs += subdirs.collect { |subdir| @home_path.join(subdir) }
595
+
596
+ # Go through each required directory, creating it if it doesn't exist
597
+ dirs.each do |dir|
598
+ next if File.directory?(dir)
599
+
600
+ begin
601
+ @logger.info("Creating: #{dir}")
602
+ FileUtils.mkdir_p(dir)
603
+ rescue Errno::EACCES
604
+ raise Errors::HomeDirectoryNotAccessible, :home_path => @home_path.to_s
605
+ end
606
+ end
607
+
608
+ # Create the version file to mark the version of the home directory
609
+ # we're using.
610
+ version_file = @home_path.join("setup_version")
611
+ if !version_file.file?
612
+ @logger.debug("Setting up the version file.")
613
+ version_file.open("w") do |f|
614
+ f.write("1.1")
615
+ end
616
+ end
617
+
618
+ # Create the rgloader/loader file so we can use encoded files.
619
+ loader_file = @home_path.join("rgloader", "loader.rb")
620
+ if !loader_file.file?
621
+ source_loader = Vagrant.source_root.join("templates/rgloader.rb")
622
+ FileUtils.cp(source_loader.to_s, loader_file.to_s)
623
+ end
624
+ end
625
+
626
+ # This creates the local data directory and show an error if it
627
+ # couldn't properly be created.
628
+ def setup_local_data_path
629
+ if @local_data_path.nil?
630
+ @logger.warn("No local data path is set. Local data cannot be stored.")
631
+ return
632
+ end
633
+
634
+ @logger.info("Local data path: #{@local_data_path}")
635
+
636
+ # If the local data path is a file, then we are probably seeing an
637
+ # old (V1) "dotfile." In this case, we upgrade it. The upgrade process
638
+ # will remove the old data file if it is successful.
639
+ if @local_data_path.file?
640
+ upgrade_v1_dotfile(@local_data_path)
641
+ end
642
+
643
+ begin
644
+ @logger.debug("Creating: #{@local_data_path}")
645
+ FileUtils.mkdir_p(@local_data_path)
646
+ rescue Errno::EACCES
647
+ raise Errors::LocalDataDirectoryNotAccessible,
648
+ :local_data_path => @local_data_path.to_s
649
+ end
650
+ end
651
+
652
+ protected
653
+
654
+ # This method copies the private key into the home directory if it
655
+ # doesn't already exist.
656
+ #
657
+ # This must be done because `ssh` requires that the key is chmod
658
+ # 0600, but if Vagrant is installed as a separate user, then the
659
+ # effective uid won't be able to read the key. So the key is copied
660
+ # to the home directory and chmod 0600.
661
+ def copy_insecure_private_key
662
+ if !@default_private_key_path.exist?
663
+ @logger.info("Copying private key to home directory")
664
+
665
+ source = File.expand_path("keys/vagrant", Vagrant.source_root)
666
+ destination = @default_private_key_path
667
+
668
+ begin
669
+ FileUtils.cp(source, destination)
670
+ rescue Errno::EACCES
671
+ raise Errors::CopyPrivateKeyFailed,
672
+ :source => source,
673
+ :destination => destination
674
+ end
675
+ end
676
+
677
+ if !Util::Platform.windows?
678
+ # On Windows, permissions don't matter as much, so don't worry
679
+ # about doing chmod.
680
+ if Util::FileMode.from_octal(@default_private_key_path.stat.mode) != "600"
681
+ @logger.info("Changing permissions on private key to 0600")
682
+ @default_private_key_path.chmod(0600)
683
+ end
684
+ end
685
+ end
686
+
687
+ # This returns the default home directory path for Vagrant, which
688
+ # can differ depending on the system.
689
+ #
690
+ # @return [Pathname]
691
+ def default_home_path
692
+ path = "~/.vagrant.d"
693
+
694
+ # On Windows, we default ot the USERPROFILE directory if it
695
+ # is available. This is more compatible with Cygwin and sharing
696
+ # the home directory across shells.
697
+ if Util::Platform.windows? && ENV["USERPROFILE"]
698
+ path = "#{ENV["USERPROFILE"]}/.vagrant.d"
699
+ end
700
+
701
+ Pathname.new(path)
702
+ end
703
+
704
+ # Finds the Vagrantfile in the given directory.
705
+ #
706
+ # @param [Pathname] path Path to search in.
707
+ # @return [Pathname]
708
+ def find_vagrantfile(search_path, filenames=nil)
709
+ filenames ||= ["Vagrantfile", "vagrantfile"]
710
+ filenames.each do |vagrantfile|
711
+ current_path = search_path.join(vagrantfile)
712
+ return current_path if current_path.file?
713
+ end
714
+
715
+ nil
716
+ end
717
+
718
+ # Loads the Vagrant plugins by properly setting up RubyGems so that
719
+ # our private gem repository is on the path.
720
+ def load_plugins
721
+ # Add our private gem path to the gem path and reset the paths
722
+ # that Rubygems knows about.
723
+ ENV["GEM_PATH"] = "#{@gems_path}#{::File::PATH_SEPARATOR}#{ENV["GEM_PATH"]}"
724
+ ::Gem.clear_paths
725
+
726
+ # If we're in a Bundler environment, don't load plugins. This only
727
+ # happens in plugin development environments.
728
+ if defined?(Bundler)
729
+ require 'bundler/shared_helpers'
730
+ if Bundler::SharedHelpers.in_bundle?
731
+ @logger.warn("In a bundler environment, not loading environment plugins!")
732
+ return
733
+ end
734
+ end
735
+
736
+ # Load the plugins
737
+ plugins_json_file = @home_path.join("plugins.json")
738
+ @logger.debug("Loading plugins from: #{plugins_json_file}")
739
+ if plugins_json_file.file?
740
+ data = JSON.parse(plugins_json_file.read)
741
+ data["installed"].each do |plugin|
742
+ @logger.info("Loading plugin from JSON: #{plugin}")
743
+ begin
744
+ Vagrant.require_plugin(plugin)
745
+ rescue Errors::PluginLoadError => e
746
+ @ui.error(e.message + "\n")
747
+ rescue Errors::PluginLoadFailed => e
748
+ @ui.error(e.message + "\n")
749
+ end
750
+ end
751
+ end
752
+ end
753
+
754
+ # This upgrades a Vagrant 1.0.x "dotfile" to the new V2 format.
755
+ #
756
+ # This is a destructive process. Once the upgrade is complete, the
757
+ # old dotfile is removed, and the environment becomes incompatible for
758
+ # Vagrant 1.0 environments.
759
+ #
760
+ # @param [Pathname] path The path to the dotfile
761
+ def upgrade_v1_dotfile(path)
762
+ @logger.info("Upgrading V1 dotfile to V2 directory structure...")
763
+
764
+ # First, verify the file isn't empty. If it is an empty file, we
765
+ # just delete it and go on with life.
766
+ contents = path.read.strip
767
+ if contents.strip == ""
768
+ @logger.info("V1 dotfile was empty. Removing and moving on.")
769
+ path.delete
770
+ return
771
+ end
772
+
773
+ # Otherwise, verify there is valid JSON in here since a Vagrant
774
+ # environment would always ensure valid JSON. This is a sanity check
775
+ # to make sure we don't nuke a dotfile that is not ours...
776
+ @logger.debug("Attempting to parse JSON of V1 file")
777
+ json_data = nil
778
+ begin
779
+ json_data = JSON.parse(contents)
780
+ @logger.debug("JSON parsed successfully. Things are okay.")
781
+ rescue JSON::ParserError
782
+ # The file could've been tampered with since Vagrant 1.0.x is
783
+ # supposed to ensure that the contents are valid JSON. Show an error.
784
+ raise Errors::DotfileUpgradeJSONError,
785
+ :state_file => path.to_s
786
+ end
787
+
788
+ # Alright, let's upgrade this guy to the new structure. Start by
789
+ # backing up the old dotfile.
790
+ backup_file = path.dirname.join(".vagrant.v1.#{Time.now.to_i}")
791
+ @logger.info("Renaming old dotfile to: #{backup_file}")
792
+ path.rename(backup_file)
793
+
794
+ # Now, we create the actual local data directory. This should succeed
795
+ # this time since we renamed the old conflicting V1.
796
+ setup_local_data_path
797
+
798
+ if json_data["active"]
799
+ @logger.debug("Upgrading to V2 style for each active VM")
800
+ json_data["active"].each do |name, id|
801
+ @logger.info("Upgrading dotfile: #{name} (#{id})")
802
+
803
+ # Create the machine configuration directory
804
+ directory = @local_data_path.join("machines/#{name}/virtualbox")
805
+ FileUtils.mkdir_p(directory)
806
+
807
+ # Write the ID file
808
+ directory.join("id").open("w+") do |f|
809
+ f.write(id)
810
+ end
811
+ end
812
+ end
813
+
814
+ # Upgrade complete! Let the user know
815
+ @ui.info(I18n.t("vagrant.general.upgraded_v1_dotfile",
816
+ :backup_path => backup_file.to_s))
817
+ end
818
+ end
819
+ end