tnargav 1.3.5 → 1.3.6
Sign up to get free protection for your applications and to get access to all the features.
- data/lib/vagrant/action.rb +45 -0
- data/lib/vagrant/action/builder.rb +174 -0
- data/lib/vagrant/action/builtin/box_add.rb +92 -0
- data/lib/vagrant/action/builtin/call.rb +67 -0
- data/lib/vagrant/action/builtin/config_validate.rb +30 -0
- data/lib/vagrant/action/builtin/confirm.rb +39 -0
- data/lib/vagrant/action/builtin/destroy_confirm.rb +21 -0
- data/lib/vagrant/action/builtin/env_set.rb +24 -0
- data/lib/vagrant/action/builtin/graceful_halt.rb +82 -0
- data/lib/vagrant/action/builtin/handle_box_url.rb +86 -0
- data/lib/vagrant/action/builtin/handle_forwarded_port_collisions.rb +134 -0
- data/lib/vagrant/action/builtin/lock.rb +57 -0
- data/lib/vagrant/action/builtin/mixin_provisioners.rb +43 -0
- data/lib/vagrant/action/builtin/nfs.rb +128 -0
- data/lib/vagrant/action/builtin/provision.rb +79 -0
- data/lib/vagrant/action/builtin/provisioner_cleanup.rb +35 -0
- data/lib/vagrant/action/builtin/set_hostname.rb +27 -0
- data/lib/vagrant/action/builtin/ssh_exec.rb +42 -0
- data/lib/vagrant/action/builtin/ssh_run.rb +43 -0
- data/lib/vagrant/action/builtin/wait_for_communicator.rb +84 -0
- data/lib/vagrant/action/general/package.rb +103 -0
- data/lib/vagrant/action/hook.rb +107 -0
- data/lib/vagrant/action/runner.rb +69 -0
- data/lib/vagrant/action/warden.rb +104 -0
- data/lib/vagrant/batch_action.rb +113 -0
- data/lib/vagrant/box.rb +97 -0
- data/lib/vagrant/box_collection.rb +393 -0
- data/lib/vagrant/cli.rb +74 -0
- data/lib/vagrant/config.rb +61 -0
- data/lib/vagrant/config/loader.rb +222 -0
- data/lib/vagrant/config/v1.rb +9 -0
- data/lib/vagrant/config/v1/dummy_config.rb +13 -0
- data/lib/vagrant/config/v1/loader.rb +105 -0
- data/lib/vagrant/config/v1/root.rb +60 -0
- data/lib/vagrant/config/v2.rb +9 -0
- data/lib/vagrant/config/v2/dummy_config.rb +13 -0
- data/lib/vagrant/config/v2/loader.rb +141 -0
- data/lib/vagrant/config/v2/root.rb +111 -0
- data/lib/vagrant/config/v2/util.rb +21 -0
- data/lib/vagrant/config/version_base.rb +80 -0
- data/lib/vagrant/environment.rb +819 -0
- data/lib/vagrant/errors.rb +614 -0
- data/lib/vagrant/guest.rb +172 -0
- data/lib/vagrant/hosts.rb +28 -0
- data/lib/vagrant/machine.rb +319 -0
- data/lib/vagrant/machine_state.rb +45 -0
- data/lib/vagrant/plugin.rb +6 -0
- data/lib/vagrant/plugin/v1.rb +19 -0
- data/lib/vagrant/plugin/v1/command.rb +169 -0
- data/lib/vagrant/plugin/v1/communicator.rb +98 -0
- data/lib/vagrant/plugin/v1/config.rb +112 -0
- data/lib/vagrant/plugin/v1/errors.rb +15 -0
- data/lib/vagrant/plugin/v1/guest.rb +92 -0
- data/lib/vagrant/plugin/v1/host.rb +66 -0
- data/lib/vagrant/plugin/v1/manager.rb +131 -0
- data/lib/vagrant/plugin/v1/plugin.rb +229 -0
- data/lib/vagrant/plugin/v1/provider.rb +68 -0
- data/lib/vagrant/plugin/v1/provisioner.rb +50 -0
- data/lib/vagrant/plugin/v2.rb +22 -0
- data/lib/vagrant/plugin/v2/command.rb +237 -0
- data/lib/vagrant/plugin/v2/communicator.rb +119 -0
- data/lib/vagrant/plugin/v2/components.rb +47 -0
- data/lib/vagrant/plugin/v2/config.rb +136 -0
- data/lib/vagrant/plugin/v2/errors.rb +15 -0
- data/lib/vagrant/plugin/v2/guest.rb +23 -0
- data/lib/vagrant/plugin/v2/host.rb +66 -0
- data/lib/vagrant/plugin/v2/manager.rb +173 -0
- data/lib/vagrant/plugin/v2/plugin.rb +226 -0
- data/lib/vagrant/plugin/v2/provider.rb +69 -0
- data/lib/vagrant/plugin/v2/provisioner.rb +53 -0
- data/lib/vagrant/registry.rb +78 -0
- data/lib/vagrant/ui.rb +215 -0
- data/lib/vagrant/util.rb +12 -0
- data/lib/vagrant/util/ansi_escape_code_remover.rb +34 -0
- data/lib/vagrant/util/busy.rb +59 -0
- data/lib/vagrant/util/counter.rb +24 -0
- data/lib/vagrant/util/downloader.rb +151 -0
- data/lib/vagrant/util/file_checksum.rb +38 -0
- data/lib/vagrant/util/file_mode.rb +12 -0
- data/lib/vagrant/util/hash_with_indifferent_access.rb +63 -0
- data/lib/vagrant/util/is_port_open.rb +38 -0
- data/lib/vagrant/util/line_ending_helpers.rb +14 -0
- data/lib/vagrant/util/network_ip.rb +28 -0
- data/lib/vagrant/util/platform.rb +101 -0
- data/lib/vagrant/util/retryable.rb +31 -0
- data/lib/vagrant/util/safe_chdir.rb +33 -0
- data/lib/vagrant/util/safe_exec.rb +36 -0
- data/lib/vagrant/util/safe_puts.rb +31 -0
- data/lib/vagrant/util/scoped_hash_override.rb +45 -0
- data/lib/vagrant/util/ssh.rb +155 -0
- data/lib/vagrant/util/stacked_proc_runner.rb +35 -0
- data/lib/vagrant/util/string_block_editor.rb +77 -0
- data/lib/vagrant/util/subprocess.rb +294 -0
- data/lib/vagrant/util/template_renderer.rb +83 -0
- data/lib/vagrant/util/which.rb +43 -0
- data/lib/vagrant/version.rb +6 -0
- data/plugins/commands/box/command/add.rb +51 -0
- data/plugins/commands/box/command/list.rb +41 -0
- data/plugins/commands/box/command/remove.rb +57 -0
- data/plugins/commands/box/command/repackage.rb +43 -0
- data/plugins/commands/box/command/root.rb +75 -0
- data/plugins/commands/box/plugin.rb +15 -0
- data/plugins/commands/destroy/command.rb +36 -0
- data/plugins/commands/destroy/plugin.rb +18 -0
- data/plugins/commands/halt/command.rb +33 -0
- data/plugins/commands/halt/plugin.rb +18 -0
- data/plugins/commands/help/command.rb +12 -0
- data/plugins/commands/help/plugin.rb +17 -0
- data/plugins/commands/init/command.rb +40 -0
- data/plugins/commands/init/plugin.rb +18 -0
- data/plugins/commands/package/command.rb +83 -0
- data/plugins/commands/package/plugin.rb +18 -0
- data/plugins/commands/plugin/action.rb +63 -0
- data/plugins/commands/plugin/action/bundler_check.rb +25 -0
- data/plugins/commands/plugin/action/install_gem.rb +106 -0
- data/plugins/commands/plugin/action/license_plugin.rb +54 -0
- data/plugins/commands/plugin/action/list_plugins.rb +54 -0
- data/plugins/commands/plugin/action/plugin_exists_check.rb +26 -0
- data/plugins/commands/plugin/action/prune_gems.rb +149 -0
- data/plugins/commands/plugin/action/uninstall_plugin.rb +23 -0
- data/plugins/commands/plugin/command/base.rb +22 -0
- data/plugins/commands/plugin/command/install.rb +41 -0
- data/plugins/commands/plugin/command/license.rb +31 -0
- data/plugins/commands/plugin/command/list.rb +28 -0
- data/plugins/commands/plugin/command/mixin_install_opts.rb +30 -0
- data/plugins/commands/plugin/command/root.rb +80 -0
- data/plugins/commands/plugin/command/uninstall.rb +28 -0
- data/plugins/commands/plugin/command/update.rb +41 -0
- data/plugins/commands/plugin/gem_helper.rb +74 -0
- data/plugins/commands/plugin/plugin.rb +22 -0
- data/plugins/commands/plugin/state_file.rb +57 -0
- data/plugins/commands/provision/command.rb +34 -0
- data/plugins/commands/provision/plugin.rb +18 -0
- data/plugins/commands/reload/command.rb +41 -0
- data/plugins/commands/reload/plugin.rb +18 -0
- data/plugins/commands/resume/command.rb +25 -0
- data/plugins/commands/resume/plugin.rb +17 -0
- data/plugins/commands/ssh/command.rb +63 -0
- data/plugins/commands/ssh/plugin.rb +17 -0
- data/plugins/commands/ssh_config/command.rb +49 -0
- data/plugins/commands/ssh_config/plugin.rb +18 -0
- data/plugins/commands/status/command.rb +44 -0
- data/plugins/commands/status/plugin.rb +18 -0
- data/plugins/commands/suspend/command.rb +25 -0
- data/plugins/commands/suspend/plugin.rb +18 -0
- data/plugins/commands/up/command.rb +67 -0
- data/plugins/commands/up/plugin.rb +17 -0
- data/plugins/commands/up/start_mixins.rb +37 -0
- data/plugins/communicators/ssh/communicator.rb +400 -0
- data/plugins/communicators/ssh/plugin.rb +19 -0
- data/plugins/guests/arch/cap/change_host_name.rb +17 -0
- data/plugins/guests/arch/cap/configure_networks.rb +30 -0
- data/plugins/guests/arch/guest.rb +11 -0
- data/plugins/guests/arch/plugin.rb +25 -0
- data/plugins/guests/coreos/cap/change_host_name.rb +15 -0
- data/plugins/guests/coreos/cap/configure_networks.rb +72 -0
- data/plugins/guests/coreos/guest.rb +9 -0
- data/plugins/guests/coreos/plugin.rb +25 -0
- data/plugins/guests/darwin/cap/change_host_name.rb +14 -0
- data/plugins/guests/darwin/cap/configure_networks.rb +52 -0
- data/plugins/guests/darwin/cap/halt.rb +16 -0
- data/plugins/guests/darwin/cap/mount_nfs_folder.rb +25 -0
- data/plugins/guests/darwin/cap/mount_vmware_shared_folder.rb +36 -0
- data/plugins/guests/darwin/cap/shell_expand_guest_path.rb +26 -0
- data/plugins/guests/darwin/cap/verify_vmware_hgfs.rb +12 -0
- data/plugins/guests/darwin/guest.rb +15 -0
- data/plugins/guests/darwin/plugin.rb +50 -0
- data/plugins/guests/debian/cap/change_host_name.rb +19 -0
- data/plugins/guests/debian/cap/configure_networks.rb +61 -0
- data/plugins/guests/debian/guest.rb +9 -0
- data/plugins/guests/debian/plugin.rb +25 -0
- data/plugins/guests/fedora/cap/configure_networks.rb +59 -0
- data/plugins/guests/fedora/cap/network_scripts_dir.rb +15 -0
- data/plugins/guests/fedora/guest.rb +11 -0
- data/plugins/guests/fedora/plugin.rb +25 -0
- data/plugins/guests/freebsd/cap/change_host_name.rb +14 -0
- data/plugins/guests/freebsd/cap/configure_networks.rb +39 -0
- data/plugins/guests/freebsd/cap/halt.rb +16 -0
- data/plugins/guests/freebsd/cap/mount_nfs_folder.rb +14 -0
- data/plugins/guests/freebsd/guest.rb +14 -0
- data/plugins/guests/freebsd/plugin.rb +35 -0
- data/plugins/guests/gentoo/cap/change_host_name.rb +17 -0
- data/plugins/guests/gentoo/cap/configure_networks.rb +43 -0
- data/plugins/guests/gentoo/guest.rb +9 -0
- data/plugins/guests/gentoo/plugin.rb +25 -0
- data/plugins/guests/linux/cap/halt.rb +16 -0
- data/plugins/guests/linux/cap/mount_nfs.rb +38 -0
- data/plugins/guests/linux/cap/mount_virtualbox_shared_folder.rb +59 -0
- data/plugins/guests/linux/cap/read_ip_address.rb +17 -0
- data/plugins/guests/linux/cap/shell_expand_guest_path.rb +29 -0
- data/plugins/guests/linux/guest.rb +11 -0
- data/plugins/guests/linux/plugin.rb +40 -0
- data/plugins/guests/omnios/cap/change_host_name.rb +17 -0
- data/plugins/guests/omnios/guest.rb +11 -0
- data/plugins/guests/omnios/plugin.rb +20 -0
- data/plugins/guests/openbsd/cap/change_host_name.rb +14 -0
- data/plugins/guests/openbsd/cap/configure_networks.rb +40 -0
- data/plugins/guests/openbsd/cap/halt.rb +16 -0
- data/plugins/guests/openbsd/cap/mount_nfs_folder.rb +14 -0
- data/plugins/guests/openbsd/guest.rb +11 -0
- data/plugins/guests/openbsd/plugin.rb +35 -0
- data/plugins/guests/pld/cap/network_scripts_dir.rb +11 -0
- data/plugins/guests/pld/guest.rb +11 -0
- data/plugins/guests/pld/plugin.rb +20 -0
- data/plugins/guests/redhat/cap/change_host_name.rb +18 -0
- data/plugins/guests/redhat/cap/configure_networks.rb +60 -0
- data/plugins/guests/redhat/cap/network_scripts_dir.rb +11 -0
- data/plugins/guests/redhat/guest.rb +11 -0
- data/plugins/guests/redhat/plugin.rb +30 -0
- data/plugins/guests/solaris/cap/change_host_name.rb +17 -0
- data/plugins/guests/solaris/cap/configure_networks.rb +25 -0
- data/plugins/guests/solaris/cap/halt.rb +22 -0
- data/plugins/guests/solaris/cap/mount_virtualbox_shared_folder.rb +31 -0
- data/plugins/guests/solaris/config.rb +18 -0
- data/plugins/guests/solaris/guest.rb +14 -0
- data/plugins/guests/solaris/plugin.rb +40 -0
- data/plugins/guests/solaris11/cap/change_host_name.rb +25 -0
- data/plugins/guests/solaris11/cap/configure_networks.rb +32 -0
- data/plugins/guests/solaris11/cap/halt.rb +26 -0
- data/plugins/guests/solaris11/cap/mount_virtualbox_shared_folder.rb +35 -0
- data/plugins/guests/solaris11/config.rb +22 -0
- data/plugins/guests/solaris11/guest.rb +15 -0
- data/plugins/guests/solaris11/plugin.rb +44 -0
- data/plugins/guests/suse/cap/change_host_name.rb +18 -0
- data/plugins/guests/suse/cap/configure_networks.rb +60 -0
- data/plugins/guests/suse/cap/network_scripts_dir.rb +11 -0
- data/plugins/guests/suse/guest.rb +11 -0
- data/plugins/guests/suse/plugin.rb +30 -0
- data/plugins/guests/ubuntu/cap/change_host_name.rb +24 -0
- data/plugins/guests/ubuntu/cap/mount_nfs.rb +19 -0
- data/plugins/guests/ubuntu/cap/mount_virtualbox_shared_folder.rb +14 -0
- data/plugins/guests/ubuntu/guest.rb +13 -0
- data/plugins/guests/ubuntu/plugin.rb +30 -0
- data/plugins/hosts/arch/host.rb +45 -0
- data/plugins/hosts/arch/plugin.rb +15 -0
- data/plugins/hosts/bsd/host.rb +178 -0
- data/plugins/hosts/bsd/plugin.rb +15 -0
- data/plugins/hosts/fedora/host.rb +57 -0
- data/plugins/hosts/fedora/plugin.rb +15 -0
- data/plugins/hosts/freebsd/host.rb +43 -0
- data/plugins/hosts/freebsd/plugin.rb +15 -0
- data/plugins/hosts/gentoo/host.rb +26 -0
- data/plugins/hosts/gentoo/plugin.rb +15 -0
- data/plugins/hosts/linux/host.rb +128 -0
- data/plugins/hosts/linux/plugin.rb +15 -0
- data/plugins/hosts/opensuse/host.rb +36 -0
- data/plugins/hosts/opensuse/plugin.rb +15 -0
- data/plugins/hosts/slackware/host.rb +26 -0
- data/plugins/hosts/slackware/plugin.rb +15 -0
- data/plugins/hosts/windows/host.rb +17 -0
- data/plugins/hosts/windows/plugin.rb +15 -0
- data/plugins/kernel_v1/config/nfs.rb +20 -0
- data/plugins/kernel_v1/config/package.rb +17 -0
- data/plugins/kernel_v1/config/ssh.rb +46 -0
- data/plugins/kernel_v1/config/vagrant.rb +31 -0
- data/plugins/kernel_v1/config/vm.rb +187 -0
- data/plugins/kernel_v1/plugin.rb +44 -0
- data/plugins/kernel_v2/config/nfs.rb +14 -0
- data/plugins/kernel_v2/config/package.rb +13 -0
- data/plugins/kernel_v2/config/ssh.rb +68 -0
- data/plugins/kernel_v2/config/ssh_connect.rb +40 -0
- data/plugins/kernel_v2/config/vagrant.rb +13 -0
- data/plugins/kernel_v2/config/vm.rb +486 -0
- data/plugins/kernel_v2/config/vm_provisioner.rb +55 -0
- data/plugins/kernel_v2/config/vm_subvm.rb +30 -0
- data/plugins/kernel_v2/plugin.rb +44 -0
- data/plugins/providers/virtualbox/action.rb +324 -0
- data/plugins/providers/virtualbox/action/boot.rb +23 -0
- data/plugins/providers/virtualbox/action/check_accessible.rb +23 -0
- data/plugins/providers/virtualbox/action/check_created.rb +21 -0
- data/plugins/providers/virtualbox/action/check_guest_additions.rb +45 -0
- data/plugins/providers/virtualbox/action/check_running.rb +21 -0
- data/plugins/providers/virtualbox/action/check_virtualbox.rb +22 -0
- data/plugins/providers/virtualbox/action/clean_machine_folder.rb +43 -0
- data/plugins/providers/virtualbox/action/clear_forwarded_ports.rb +18 -0
- data/plugins/providers/virtualbox/action/clear_network_interfaces.rb +31 -0
- data/plugins/providers/virtualbox/action/clear_shared_folders.rb +17 -0
- data/plugins/providers/virtualbox/action/created.rb +20 -0
- data/plugins/providers/virtualbox/action/customize.rb +43 -0
- data/plugins/providers/virtualbox/action/destroy.rb +19 -0
- data/plugins/providers/virtualbox/action/destroy_unused_network_interfaces.rb +23 -0
- data/plugins/providers/virtualbox/action/discard_state.rb +20 -0
- data/plugins/providers/virtualbox/action/export.rb +57 -0
- data/plugins/providers/virtualbox/action/forced_halt.rb +25 -0
- data/plugins/providers/virtualbox/action/forward_ports.rb +89 -0
- data/plugins/providers/virtualbox/action/import.rb +54 -0
- data/plugins/providers/virtualbox/action/is_paused.rb +20 -0
- data/plugins/providers/virtualbox/action/is_running.rb +20 -0
- data/plugins/providers/virtualbox/action/is_saved.rb +20 -0
- data/plugins/providers/virtualbox/action/match_mac_address.rb +21 -0
- data/plugins/providers/virtualbox/action/message_already_running.rb +16 -0
- data/plugins/providers/virtualbox/action/message_not_created.rb +16 -0
- data/plugins/providers/virtualbox/action/message_not_running.rb +16 -0
- data/plugins/providers/virtualbox/action/message_will_not_destroy.rb +17 -0
- data/plugins/providers/virtualbox/action/network.rb +424 -0
- data/plugins/providers/virtualbox/action/package.rb +20 -0
- data/plugins/providers/virtualbox/action/package_vagrantfile.rb +33 -0
- data/plugins/providers/virtualbox/action/prepare_forwarded_port_collision_params.rb +35 -0
- data/plugins/providers/virtualbox/action/prepare_nfs_settings.rb +69 -0
- data/plugins/providers/virtualbox/action/prune_nfs_exports.rb +20 -0
- data/plugins/providers/virtualbox/action/resume.rb +25 -0
- data/plugins/providers/virtualbox/action/sane_defaults.rb +91 -0
- data/plugins/providers/virtualbox/action/set_name.rb +52 -0
- data/plugins/providers/virtualbox/action/setup_package_files.rb +51 -0
- data/plugins/providers/virtualbox/action/share_folders.rb +128 -0
- data/plugins/providers/virtualbox/action/suspend.rb +20 -0
- data/plugins/providers/virtualbox/config.rb +123 -0
- data/plugins/providers/virtualbox/driver/base.rb +374 -0
- data/plugins/providers/virtualbox/driver/meta.rb +148 -0
- data/plugins/providers/virtualbox/driver/version_4_0.rb +485 -0
- data/plugins/providers/virtualbox/driver/version_4_1.rb +485 -0
- data/plugins/providers/virtualbox/driver/version_4_2.rb +499 -0
- data/plugins/providers/virtualbox/model/forwarded_port.rb +70 -0
- data/plugins/providers/virtualbox/plugin.rb +42 -0
- data/plugins/providers/virtualbox/provider.rb +92 -0
- data/plugins/providers/virtualbox/util/compile_forwarded_ports.rb +35 -0
- data/plugins/provisioners/ansible/config.rb +89 -0
- data/plugins/provisioners/ansible/plugin.rb +23 -0
- data/plugins/provisioners/ansible/provisioner.rb +100 -0
- data/plugins/provisioners/cfengine/cap/debian/cfengine_install.rb +19 -0
- data/plugins/provisioners/cfengine/cap/linux/cfengine_installed.rb +14 -0
- data/plugins/provisioners/cfengine/cap/linux/cfengine_needs_bootstrap.rb +34 -0
- data/plugins/provisioners/cfengine/cap/redhat/cfengine_install.rb +24 -0
- data/plugins/provisioners/cfengine/config.rb +126 -0
- data/plugins/provisioners/cfengine/plugin.rb +42 -0
- data/plugins/provisioners/cfengine/provisioner.rb +135 -0
- data/plugins/provisioners/chef/config/base.rb +116 -0
- data/plugins/provisioners/chef/config/chef_client.rb +66 -0
- data/plugins/provisioners/chef/config/chef_solo.rb +94 -0
- data/plugins/provisioners/chef/plugin.rb +33 -0
- data/plugins/provisioners/chef/provisioner/base.rb +113 -0
- data/plugins/provisioners/chef/provisioner/chef_client.rb +126 -0
- data/plugins/provisioners/chef/provisioner/chef_solo.rb +195 -0
- data/plugins/provisioners/file/config.rb +29 -0
- data/plugins/provisioners/file/plugin.rb +23 -0
- data/plugins/provisioners/file/provisioner.rb +16 -0
- data/plugins/provisioners/puppet/config/puppet.rb +93 -0
- data/plugins/provisioners/puppet/config/puppet_server.rb +16 -0
- data/plugins/provisioners/puppet/plugin.rb +33 -0
- data/plugins/provisioners/puppet/provisioner/puppet.rb +146 -0
- data/plugins/provisioners/puppet/provisioner/puppet_server.rb +74 -0
- data/plugins/provisioners/salt/bootstrap-salt.sh +6 -0
- data/plugins/provisioners/salt/config.rb +105 -0
- data/plugins/provisioners/salt/errors.rb +11 -0
- data/plugins/provisioners/salt/plugin.rb +22 -0
- data/plugins/provisioners/salt/provisioner.rb +249 -0
- data/plugins/provisioners/shell/config.rb +65 -0
- data/plugins/provisioners/shell/plugin.rb +23 -0
- data/plugins/provisioners/shell/provisioner.rb +98 -0
- data/templates/commands/init/Vagrantfile.erb +118 -0
- data/templates/commands/ssh_config/config.erb +20 -0
- data/templates/config/messages.erb +14 -0
- data/templates/config/validation_failed.erb +7 -0
- data/templates/guests/arch/network_dhcp.erb +4 -0
- data/templates/guests/arch/network_static.erb +5 -0
- data/templates/guests/coreos/etcd.service.erb +10 -0
- data/templates/guests/debian/network_dhcp.erb +11 -0
- data/templates/guests/debian/network_static.erb +7 -0
- data/templates/guests/fedora/network_dhcp.erb +6 -0
- data/templates/guests/fedora/network_static.erb +12 -0
- data/templates/guests/freebsd/network_dhcp.erb +3 -0
- data/templates/guests/freebsd/network_static.erb +3 -0
- data/templates/guests/gentoo/network_dhcp.erb +4 -0
- data/templates/guests/gentoo/network_static.erb +4 -0
- data/templates/guests/openbsd/network_dhcp.erb +1 -0
- data/templates/guests/openbsd/network_static.erb +1 -0
- data/templates/guests/redhat/network_dhcp.erb +6 -0
- data/templates/guests/redhat/network_static.erb +8 -0
- data/templates/guests/suse/network_dhcp.erb +6 -0
- data/templates/guests/suse/network_static.erb +10 -0
- data/templates/locales/en.yml +1248 -0
- data/templates/nfs/exports.erb +5 -0
- data/templates/nfs/exports_freebsd.erb +5 -0
- data/templates/nfs/exports_linux.erb +7 -0
- data/templates/provisioners/chef_client/client.erb +41 -0
- data/templates/provisioners/chef_solo/solo.erb +45 -0
- data/test/acceptance/base.rb +48 -0
- data/test/acceptance/box_test.rb +99 -0
- data/test/acceptance/destroy_test.rb +37 -0
- data/test/acceptance/halt_test.rb +72 -0
- data/test/acceptance/init_test.rb +33 -0
- data/test/acceptance/networking/host_only_test.rb +37 -0
- data/test/acceptance/networking/port_forward_test.rb +125 -0
- data/test/acceptance/package_test.rb +46 -0
- data/test/acceptance/provisioning/basic_test.rb +61 -0
- data/test/acceptance/provisioning/chef_solo_test.rb +37 -0
- data/test/acceptance/provisioning/shell_test.rb +53 -0
- data/test/acceptance/resume_test.rb +17 -0
- data/test/acceptance/shared_folders_test.rb +84 -0
- data/test/acceptance/skeletons/chef_solo_basic/README.md +3 -0
- data/test/acceptance/skeletons/chef_solo_basic/cookbooks/basic/recipes/default.rb +5 -0
- data/test/acceptance/skeletons/chef_solo_json/README.md +3 -0
- data/test/acceptance/skeletons/chef_solo_json/cookbooks/basic/recipes/default.rb +6 -0
- data/test/acceptance/skeletons/provisioner_multi/README.md +3 -0
- data/test/acceptance/skeletons/provisioner_multi/cookbooks/basic/recipes/default.rb +5 -0
- data/test/acceptance/ssh_test.rb +46 -0
- data/test/acceptance/support/config.rb +42 -0
- data/test/acceptance/support/isolated_environment.rb +118 -0
- data/test/acceptance/support/matchers/have_color.rb +9 -0
- data/test/acceptance/support/matchers/match_output.rb +14 -0
- data/test/acceptance/support/matchers/succeed.rb +14 -0
- data/test/acceptance/support/network_tests.rb +29 -0
- data/test/acceptance/support/output.rb +95 -0
- data/test/acceptance/support/shared/base_context.rb +72 -0
- data/test/acceptance/support/shared/command_examples.rb +33 -0
- data/test/acceptance/support/virtualbox.rb +36 -0
- data/test/acceptance/suspend_test.rb +56 -0
- data/test/acceptance/up_basic_test.rb +33 -0
- data/test/acceptance/up_with_box_url.rb +40 -0
- data/test/acceptance/vagrant_test.rb +47 -0
- data/test/acceptance/version_test.rb +15 -0
- data/test/config/acceptance_boxes.yml +7 -0
- data/test/support/isolated_environment.rb +46 -0
- data/test/support/tempdir.rb +43 -0
- data/test/unit/base.rb +27 -0
- data/test/unit/support/dummy_provider.rb +16 -0
- data/test/unit/support/isolated_environment.rb +179 -0
- data/test/unit/support/shared/base_context.rb +104 -0
- data/test/unit/vagrant/action/builder_test.rb +242 -0
- data/test/unit/vagrant/action/builtin/call_test.rb +145 -0
- data/test/unit/vagrant/action/builtin/confirm_test.rb +36 -0
- data/test/unit/vagrant/action/builtin/env_set_test.rb +20 -0
- data/test/unit/vagrant/action/builtin/graceful_halt_test.rb +60 -0
- data/test/unit/vagrant/action/builtin/lock_test.rb +98 -0
- data/test/unit/vagrant/action/builtin/ssh_exec_test.rb +57 -0
- data/test/unit/vagrant/action/hook_test.rb +120 -0
- data/test/unit/vagrant/action/runner_test.rb +71 -0
- data/test/unit/vagrant/action/warden_test.rb +92 -0
- data/test/unit/vagrant/batch_action_test.rb +36 -0
- data/test/unit/vagrant/box_collection_test.rb +237 -0
- data/test/unit/vagrant/box_test.rb +122 -0
- data/test/unit/vagrant/cli_test.rb +28 -0
- data/test/unit/vagrant/config/loader_test.rb +184 -0
- data/test/unit/vagrant/config/v1/dummy_config_test.rb +24 -0
- data/test/unit/vagrant/config/v1/loader_test.rb +145 -0
- data/test/unit/vagrant/config/v1/root_test.rb +40 -0
- data/test/unit/vagrant/config/v2/dummy_config_test.rb +24 -0
- data/test/unit/vagrant/config/v2/loader_test.rb +151 -0
- data/test/unit/vagrant/config/v2/root_test.rb +115 -0
- data/test/unit/vagrant/config/v2/util_test.rb +21 -0
- data/test/unit/vagrant/config_test.rb +66 -0
- data/test/unit/vagrant/environment_test.rb +746 -0
- data/test/unit/vagrant/guest_test.rb +179 -0
- data/test/unit/vagrant/hosts_test.rb +37 -0
- data/test/unit/vagrant/machine_state_test.rb +26 -0
- data/test/unit/vagrant/machine_test.rb +384 -0
- data/test/unit/vagrant/plugin/v1/command_test.rb +143 -0
- data/test/unit/vagrant/plugin/v1/communicator_test.rb +9 -0
- data/test/unit/vagrant/plugin/v1/config_test.rb +50 -0
- data/test/unit/vagrant/plugin/v1/host_test.rb +5 -0
- data/test/unit/vagrant/plugin/v1/manager_test.rb +114 -0
- data/test/unit/vagrant/plugin/v1/plugin_test.rb +267 -0
- data/test/unit/vagrant/plugin/v1/provider_test.rb +18 -0
- data/test/unit/vagrant/plugin/v2/command_test.rb +238 -0
- data/test/unit/vagrant/plugin/v2/communicator_test.rb +9 -0
- data/test/unit/vagrant/plugin/v2/components_test.rb +17 -0
- data/test/unit/vagrant/plugin/v2/config_test.rb +60 -0
- data/test/unit/vagrant/plugin/v2/host_test.rb +5 -0
- data/test/unit/vagrant/plugin/v2/manager_test.rb +174 -0
- data/test/unit/vagrant/plugin/v2/plugin_test.rb +305 -0
- data/test/unit/vagrant/plugin/v2/provider_test.rb +18 -0
- data/test/unit/vagrant/registry_test.rb +128 -0
- data/test/unit/vagrant/util/ansi_escape_code_remover_test.rb +16 -0
- data/test/unit/vagrant/util/downloader_test.rb +57 -0
- data/test/unit/vagrant/util/file_checksum_test.rb +23 -0
- data/test/unit/vagrant/util/hash_with_indifferent_access_test.rb +38 -0
- data/test/unit/vagrant/util/is_port_open_test.rb +53 -0
- data/test/unit/vagrant/util/line_endings_helper_test.rb +16 -0
- data/test/unit/vagrant/util/network_ip_test.rb +17 -0
- data/test/unit/vagrant/util/retryable_test.rb +106 -0
- data/test/unit/vagrant/util/safe_chdir_test.rb +43 -0
- data/test/unit/vagrant/util/scoped_hash_override_test.rb +48 -0
- data/test/unit/vagrant/util/ssh_test.rb +30 -0
- data/test/unit/vagrant/util/string_block_editor_test.rb +106 -0
- data/test/unit/vagrant/util/which_test.rb +43 -0
- data/test/unit/vagrant_test.rb +74 -0
- data/tnargav.gemspec +1 -1
- metadata +477 -1
@@ -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
|