tnargav 1.2.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (452) hide show
  1. data/.gitignore +31 -0
  2. data/.gitsetup.yml +5 -0
  3. data/.travis.yml +7 -0
  4. data/.yardopts +1 -0
  5. data/CHANGELOG.md +1067 -0
  6. data/Gemfile +3 -0
  7. data/LICENSE +21 -0
  8. data/README.md +89 -0
  9. data/Rakefile +18 -0
  10. data/bin/vagrant +106 -0
  11. data/config/default.rb +36 -0
  12. data/contrib/README.md +12 -0
  13. data/contrib/bash/completion.sh +3 -0
  14. data/contrib/emacs/vagrant.el +8 -0
  15. data/contrib/vim/vagrantfile.vim +9 -0
  16. data/keys/README.md +17 -0
  17. data/keys/vagrant +27 -0
  18. data/keys/vagrant.pub +1 -0
  19. data/lib/vagrant/action/builder.rb +174 -0
  20. data/lib/vagrant/action/builtin/box_add.rb +81 -0
  21. data/lib/vagrant/action/builtin/call.rb +67 -0
  22. data/lib/vagrant/action/builtin/config_validate.rb +30 -0
  23. data/lib/vagrant/action/builtin/confirm.rb +38 -0
  24. data/lib/vagrant/action/builtin/destroy_confirm.rb +21 -0
  25. data/lib/vagrant/action/builtin/env_set.rb +24 -0
  26. data/lib/vagrant/action/builtin/graceful_halt.rb +73 -0
  27. data/lib/vagrant/action/builtin/handle_box_url.rb +77 -0
  28. data/lib/vagrant/action/builtin/handle_forwarded_port_collisions.rb +134 -0
  29. data/lib/vagrant/action/builtin/lock.rb +57 -0
  30. data/lib/vagrant/action/builtin/nfs.rb +118 -0
  31. data/lib/vagrant/action/builtin/provision.rb +70 -0
  32. data/lib/vagrant/action/builtin/set_hostname.rb +27 -0
  33. data/lib/vagrant/action/builtin/ssh_exec.rb +42 -0
  34. data/lib/vagrant/action/builtin/ssh_run.rb +43 -0
  35. data/lib/vagrant/action/general/package.rb +103 -0
  36. data/lib/vagrant/action/hook.rb +107 -0
  37. data/lib/vagrant/action/runner.rb +69 -0
  38. data/lib/vagrant/action/warden.rb +103 -0
  39. data/lib/vagrant/action.rb +43 -0
  40. data/lib/vagrant/batch_action.rb +113 -0
  41. data/lib/vagrant/box.rb +97 -0
  42. data/lib/vagrant/box_collection.rb +375 -0
  43. data/lib/vagrant/cli.rb +82 -0
  44. data/lib/vagrant/config/loader.rb +222 -0
  45. data/lib/vagrant/config/v1/dummy_config.rb +13 -0
  46. data/lib/vagrant/config/v1/loader.rb +105 -0
  47. data/lib/vagrant/config/v1/root.rb +60 -0
  48. data/lib/vagrant/config/v1.rb +9 -0
  49. data/lib/vagrant/config/v2/dummy_config.rb +13 -0
  50. data/lib/vagrant/config/v2/loader.rb +141 -0
  51. data/lib/vagrant/config/v2/root.rb +105 -0
  52. data/lib/vagrant/config/v2/util.rb +21 -0
  53. data/lib/vagrant/config/v2.rb +9 -0
  54. data/lib/vagrant/config/version_base.rb +80 -0
  55. data/lib/vagrant/config.rb +61 -0
  56. data/lib/vagrant/environment.rb +843 -0
  57. data/lib/vagrant/errors.rb +570 -0
  58. data/lib/vagrant/guest.rb +163 -0
  59. data/lib/vagrant/hosts.rb +28 -0
  60. data/lib/vagrant/machine.rb +325 -0
  61. data/lib/vagrant/machine_state.rb +45 -0
  62. data/lib/vagrant/plugin/v1/command.rb +169 -0
  63. data/lib/vagrant/plugin/v1/communicator.rb +98 -0
  64. data/lib/vagrant/plugin/v1/config.rb +112 -0
  65. data/lib/vagrant/plugin/v1/errors.rb +15 -0
  66. data/lib/vagrant/plugin/v1/guest.rb +92 -0
  67. data/lib/vagrant/plugin/v1/host.rb +66 -0
  68. data/lib/vagrant/plugin/v1/manager.rb +131 -0
  69. data/lib/vagrant/plugin/v1/plugin.rb +229 -0
  70. data/lib/vagrant/plugin/v1/provider.rb +68 -0
  71. data/lib/vagrant/plugin/v1/provisioner.rb +50 -0
  72. data/lib/vagrant/plugin/v1.rb +19 -0
  73. data/lib/vagrant/plugin/v2/command.rb +234 -0
  74. data/lib/vagrant/plugin/v2/communicator.rb +98 -0
  75. data/lib/vagrant/plugin/v2/components.rb +47 -0
  76. data/lib/vagrant/plugin/v2/config.rb +136 -0
  77. data/lib/vagrant/plugin/v2/errors.rb +15 -0
  78. data/lib/vagrant/plugin/v2/guest.rb +78 -0
  79. data/lib/vagrant/plugin/v2/host.rb +66 -0
  80. data/lib/vagrant/plugin/v2/manager.rb +173 -0
  81. data/lib/vagrant/plugin/v2/plugin.rb +226 -0
  82. data/lib/vagrant/plugin/v2/provider.rb +69 -0
  83. data/lib/vagrant/plugin/v2/provisioner.rb +47 -0
  84. data/lib/vagrant/plugin/v2.rb +22 -0
  85. data/lib/vagrant/plugin.rb +6 -0
  86. data/lib/vagrant/registry.rb +78 -0
  87. data/lib/vagrant/ui.rb +215 -0
  88. data/lib/vagrant/util/ansi_escape_code_remover.rb +34 -0
  89. data/lib/vagrant/util/busy.rb +59 -0
  90. data/lib/vagrant/util/counter.rb +24 -0
  91. data/lib/vagrant/util/downloader.rb +146 -0
  92. data/lib/vagrant/util/file_checksum.rb +38 -0
  93. data/lib/vagrant/util/file_mode.rb +12 -0
  94. data/lib/vagrant/util/hash_with_indifferent_access.rb +63 -0
  95. data/lib/vagrant/util/is_port_open.rb +38 -0
  96. data/lib/vagrant/util/line_ending_helpers.rb +14 -0
  97. data/lib/vagrant/util/network_ip.rb +28 -0
  98. data/lib/vagrant/util/platform.rb +58 -0
  99. data/lib/vagrant/util/retryable.rb +31 -0
  100. data/lib/vagrant/util/safe_chdir.rb +33 -0
  101. data/lib/vagrant/util/safe_exec.rb +36 -0
  102. data/lib/vagrant/util/safe_puts.rb +31 -0
  103. data/lib/vagrant/util/scoped_hash_override.rb +45 -0
  104. data/lib/vagrant/util/ssh.rb +150 -0
  105. data/lib/vagrant/util/stacked_proc_runner.rb +35 -0
  106. data/lib/vagrant/util/string_block_editor.rb +77 -0
  107. data/lib/vagrant/util/subprocess.rb +273 -0
  108. data/lib/vagrant/util/template_renderer.rb +83 -0
  109. data/lib/vagrant/util/which.rb +43 -0
  110. data/lib/vagrant/util.rb +12 -0
  111. data/lib/vagrant/version.rb +6 -0
  112. data/lib/vagrant.rb +258 -0
  113. data/plugins/README.md +5 -0
  114. data/plugins/commands/box/command/add.rb +51 -0
  115. data/plugins/commands/box/command/list.rb +41 -0
  116. data/plugins/commands/box/command/remove.rb +37 -0
  117. data/plugins/commands/box/command/repackage.rb +43 -0
  118. data/plugins/commands/box/command/root.rb +75 -0
  119. data/plugins/commands/box/plugin.rb +15 -0
  120. data/plugins/commands/destroy/command.rb +31 -0
  121. data/plugins/commands/destroy/plugin.rb +18 -0
  122. data/plugins/commands/halt/command.rb +33 -0
  123. data/plugins/commands/halt/plugin.rb +18 -0
  124. data/plugins/commands/init/command.rb +40 -0
  125. data/plugins/commands/init/plugin.rb +18 -0
  126. data/plugins/commands/package/command.rb +83 -0
  127. data/plugins/commands/package/plugin.rb +18 -0
  128. data/plugins/commands/plugin/action/bundler_check.rb +25 -0
  129. data/plugins/commands/plugin/action/install_gem.rb +81 -0
  130. data/plugins/commands/plugin/action/license_plugin.rb +54 -0
  131. data/plugins/commands/plugin/action/list_plugins.rb +54 -0
  132. data/plugins/commands/plugin/action/prune_gems.rb +149 -0
  133. data/plugins/commands/plugin/action/uninstall_plugin.rb +23 -0
  134. data/plugins/commands/plugin/action.rb +52 -0
  135. data/plugins/commands/plugin/command/base.rb +22 -0
  136. data/plugins/commands/plugin/command/install.rb +58 -0
  137. data/plugins/commands/plugin/command/license.rb +31 -0
  138. data/plugins/commands/plugin/command/list.rb +28 -0
  139. data/plugins/commands/plugin/command/root.rb +75 -0
  140. data/plugins/commands/plugin/command/uninstall.rb +28 -0
  141. data/plugins/commands/plugin/gem_helper.rb +74 -0
  142. data/plugins/commands/plugin/plugin.rb +22 -0
  143. data/plugins/commands/plugin/state_file.rb +57 -0
  144. data/plugins/commands/provision/command.rb +34 -0
  145. data/plugins/commands/provision/plugin.rb +18 -0
  146. data/plugins/commands/reload/command.rb +37 -0
  147. data/plugins/commands/reload/plugin.rb +18 -0
  148. data/plugins/commands/resume/command.rb +25 -0
  149. data/plugins/commands/resume/plugin.rb +17 -0
  150. data/plugins/commands/ssh/command.rb +63 -0
  151. data/plugins/commands/ssh/plugin.rb +17 -0
  152. data/plugins/commands/ssh_config/command.rb +49 -0
  153. data/plugins/commands/ssh_config/plugin.rb +18 -0
  154. data/plugins/commands/status/command.rb +39 -0
  155. data/plugins/commands/status/plugin.rb +18 -0
  156. data/plugins/commands/suspend/command.rb +25 -0
  157. data/plugins/commands/suspend/plugin.rb +18 -0
  158. data/plugins/commands/up/command.rb +57 -0
  159. data/plugins/commands/up/plugin.rb +17 -0
  160. data/plugins/commands/up/start_mixins.rb +26 -0
  161. data/plugins/communicators/ssh/communicator.rb +342 -0
  162. data/plugins/communicators/ssh/plugin.rb +19 -0
  163. data/plugins/guests/arch/cap/change_host_name.rb +18 -0
  164. data/plugins/guests/arch/cap/configure_networks.rb +23 -0
  165. data/plugins/guests/arch/guest.rb +11 -0
  166. data/plugins/guests/arch/plugin.rb +25 -0
  167. data/plugins/guests/debian/cap/change_host_name.rb +18 -0
  168. data/plugins/guests/debian/cap/configure_networks.rb +61 -0
  169. data/plugins/guests/debian/guest.rb +9 -0
  170. data/plugins/guests/debian/plugin.rb +25 -0
  171. data/plugins/guests/fedora/cap/configure_networks.rb +59 -0
  172. data/plugins/guests/fedora/cap/network_scripts_dir.rb +15 -0
  173. data/plugins/guests/fedora/guest.rb +11 -0
  174. data/plugins/guests/fedora/plugin.rb +25 -0
  175. data/plugins/guests/freebsd/cap/change_host_name.rb +14 -0
  176. data/plugins/guests/freebsd/cap/configure_networks.rb +39 -0
  177. data/plugins/guests/freebsd/cap/halt.rb +16 -0
  178. data/plugins/guests/freebsd/cap/mount_nfs_folder.rb +14 -0
  179. data/plugins/guests/freebsd/guest.rb +14 -0
  180. data/plugins/guests/freebsd/plugin.rb +35 -0
  181. data/plugins/guests/gentoo/cap/change_host_name.rb +17 -0
  182. data/plugins/guests/gentoo/cap/configure_networks.rb +43 -0
  183. data/plugins/guests/gentoo/guest.rb +9 -0
  184. data/plugins/guests/gentoo/plugin.rb +25 -0
  185. data/plugins/guests/linux/cap/halt.rb +16 -0
  186. data/plugins/guests/linux/cap/mount_nfs.rb +30 -0
  187. data/plugins/guests/linux/cap/mount_virtualbox_shared_folder.rb +40 -0
  188. data/plugins/guests/linux/cap/read_ip_address.rb +17 -0
  189. data/plugins/guests/linux/cap/shell_expand_guest_path.rb +26 -0
  190. data/plugins/guests/linux/guest.rb +11 -0
  191. data/plugins/guests/linux/plugin.rb +40 -0
  192. data/plugins/guests/openbsd/cap/halt.rb +16 -0
  193. data/plugins/guests/openbsd/guest.rb +11 -0
  194. data/plugins/guests/openbsd/plugin.rb +20 -0
  195. data/plugins/guests/pld/cap/network_scripts_dir.rb +11 -0
  196. data/plugins/guests/pld/guest.rb +11 -0
  197. data/plugins/guests/pld/plugin.rb +20 -0
  198. data/plugins/guests/redhat/cap/change_host_name.rb +18 -0
  199. data/plugins/guests/redhat/cap/configure_networks.rb +60 -0
  200. data/plugins/guests/redhat/cap/network_scripts_dir.rb +11 -0
  201. data/plugins/guests/redhat/guest.rb +11 -0
  202. data/plugins/guests/redhat/plugin.rb +30 -0
  203. data/plugins/guests/solaris/cap/change_host_name.rb +17 -0
  204. data/plugins/guests/solaris/cap/configure_networks.rb +25 -0
  205. data/plugins/guests/solaris/cap/halt.rb +21 -0
  206. data/plugins/guests/solaris/cap/mount_virtualbox_shared_folder.rb +28 -0
  207. data/plugins/guests/solaris/config.rb +18 -0
  208. data/plugins/guests/solaris/guest.rb +14 -0
  209. data/plugins/guests/solaris/plugin.rb +40 -0
  210. data/plugins/guests/suse/cap/change_host_name.rb +18 -0
  211. data/plugins/guests/suse/cap/network_scripts_dir.rb +11 -0
  212. data/plugins/guests/suse/guest.rb +11 -0
  213. data/plugins/guests/suse/plugin.rb +25 -0
  214. data/plugins/guests/ubuntu/cap/change_host_name.rb +23 -0
  215. data/plugins/guests/ubuntu/guest.rb +32 -0
  216. data/plugins/guests/ubuntu/plugin.rb +20 -0
  217. data/plugins/hosts/arch/host.rb +68 -0
  218. data/plugins/hosts/arch/plugin.rb +15 -0
  219. data/plugins/hosts/bsd/host.rb +102 -0
  220. data/plugins/hosts/bsd/plugin.rb +15 -0
  221. data/plugins/hosts/fedora/host.rb +51 -0
  222. data/plugins/hosts/fedora/plugin.rb +15 -0
  223. data/plugins/hosts/freebsd/host.rb +43 -0
  224. data/plugins/hosts/freebsd/plugin.rb +15 -0
  225. data/plugins/hosts/gentoo/host.rb +24 -0
  226. data/plugins/hosts/gentoo/plugin.rb +15 -0
  227. data/plugins/hosts/linux/host.rb +95 -0
  228. data/plugins/hosts/linux/plugin.rb +15 -0
  229. data/plugins/hosts/opensuse/host.rb +34 -0
  230. data/plugins/hosts/opensuse/plugin.rb +15 -0
  231. data/plugins/hosts/windows/host.rb +17 -0
  232. data/plugins/hosts/windows/plugin.rb +15 -0
  233. data/plugins/kernel_v1/config/nfs.rb +20 -0
  234. data/plugins/kernel_v1/config/package.rb +17 -0
  235. data/plugins/kernel_v1/config/ssh.rb +46 -0
  236. data/plugins/kernel_v1/config/vagrant.rb +31 -0
  237. data/plugins/kernel_v1/config/vm.rb +184 -0
  238. data/plugins/kernel_v1/plugin.rb +44 -0
  239. data/plugins/kernel_v2/config/nfs.rb +14 -0
  240. data/plugins/kernel_v2/config/package.rb +13 -0
  241. data/plugins/kernel_v2/config/ssh.rb +76 -0
  242. data/plugins/kernel_v2/config/ssh_connect.rb +40 -0
  243. data/plugins/kernel_v2/config/vagrant.rb +13 -0
  244. data/plugins/kernel_v2/config/vm.rb +416 -0
  245. data/plugins/kernel_v2/config/vm_provisioner.rb +55 -0
  246. data/plugins/kernel_v2/config/vm_subvm.rb +30 -0
  247. data/plugins/kernel_v2/plugin.rb +44 -0
  248. data/plugins/providers/virtualbox/action/boot.rb +49 -0
  249. data/plugins/providers/virtualbox/action/check_accessible.rb +23 -0
  250. data/plugins/providers/virtualbox/action/check_created.rb +21 -0
  251. data/plugins/providers/virtualbox/action/check_guest_additions.rb +45 -0
  252. data/plugins/providers/virtualbox/action/check_running.rb +21 -0
  253. data/plugins/providers/virtualbox/action/check_virtualbox.rb +22 -0
  254. data/plugins/providers/virtualbox/action/clean_machine_folder.rb +43 -0
  255. data/plugins/providers/virtualbox/action/clear_forwarded_ports.rb +18 -0
  256. data/plugins/providers/virtualbox/action/clear_network_interfaces.rb +31 -0
  257. data/plugins/providers/virtualbox/action/clear_shared_folders.rb +17 -0
  258. data/plugins/providers/virtualbox/action/created.rb +20 -0
  259. data/plugins/providers/virtualbox/action/customize.rb +36 -0
  260. data/plugins/providers/virtualbox/action/destroy.rb +19 -0
  261. data/plugins/providers/virtualbox/action/destroy_unused_network_interfaces.rb +16 -0
  262. data/plugins/providers/virtualbox/action/discard_state.rb +20 -0
  263. data/plugins/providers/virtualbox/action/export.rb +57 -0
  264. data/plugins/providers/virtualbox/action/forced_halt.rb +25 -0
  265. data/plugins/providers/virtualbox/action/forward_ports.rb +87 -0
  266. data/plugins/providers/virtualbox/action/import.rb +51 -0
  267. data/plugins/providers/virtualbox/action/is_paused.rb +20 -0
  268. data/plugins/providers/virtualbox/action/is_running.rb +20 -0
  269. data/plugins/providers/virtualbox/action/is_saved.rb +20 -0
  270. data/plugins/providers/virtualbox/action/match_mac_address.rb +21 -0
  271. data/plugins/providers/virtualbox/action/message_already_running.rb +16 -0
  272. data/plugins/providers/virtualbox/action/message_not_created.rb +16 -0
  273. data/plugins/providers/virtualbox/action/message_not_running.rb +16 -0
  274. data/plugins/providers/virtualbox/action/message_will_not_destroy.rb +17 -0
  275. data/plugins/providers/virtualbox/action/network.rb +410 -0
  276. data/plugins/providers/virtualbox/action/package.rb +20 -0
  277. data/plugins/providers/virtualbox/action/package_vagrantfile.rb +33 -0
  278. data/plugins/providers/virtualbox/action/prepare_forwarded_port_collision_params.rb +35 -0
  279. data/plugins/providers/virtualbox/action/prepare_nfs_settings.rb +64 -0
  280. data/plugins/providers/virtualbox/action/prune_nfs_exports.rb +20 -0
  281. data/plugins/providers/virtualbox/action/resume.rb +25 -0
  282. data/plugins/providers/virtualbox/action/sane_defaults.rb +91 -0
  283. data/plugins/providers/virtualbox/action/set_name.rb +40 -0
  284. data/plugins/providers/virtualbox/action/setup_package_files.rb +51 -0
  285. data/plugins/providers/virtualbox/action/share_folders.rb +128 -0
  286. data/plugins/providers/virtualbox/action/suspend.rb +20 -0
  287. data/plugins/providers/virtualbox/action.rb +311 -0
  288. data/plugins/providers/virtualbox/config.rb +86 -0
  289. data/plugins/providers/virtualbox/driver/base.rb +360 -0
  290. data/plugins/providers/virtualbox/driver/meta.rb +142 -0
  291. data/plugins/providers/virtualbox/driver/version_4_0.rb +485 -0
  292. data/plugins/providers/virtualbox/driver/version_4_1.rb +485 -0
  293. data/plugins/providers/virtualbox/driver/version_4_2.rb +482 -0
  294. data/plugins/providers/virtualbox/model/forwarded_port.rb +58 -0
  295. data/plugins/providers/virtualbox/plugin.rb +42 -0
  296. data/plugins/providers/virtualbox/provider.rb +92 -0
  297. data/plugins/providers/virtualbox/util/compile_forwarded_ports.rb +31 -0
  298. data/plugins/provisioners/ansible/config.rb +72 -0
  299. data/plugins/provisioners/ansible/plugin.rb +23 -0
  300. data/plugins/provisioners/ansible/provisioner.rb +45 -0
  301. data/plugins/provisioners/cfengine/cap/debian/cfengine_install.rb +19 -0
  302. data/plugins/provisioners/cfengine/cap/linux/cfengine_installed.rb +14 -0
  303. data/plugins/provisioners/cfengine/cap/linux/cfengine_needs_bootstrap.rb +34 -0
  304. data/plugins/provisioners/cfengine/cap/redhat/cfengine_install.rb +24 -0
  305. data/plugins/provisioners/cfengine/config.rb +120 -0
  306. data/plugins/provisioners/cfengine/plugin.rb +42 -0
  307. data/plugins/provisioners/cfengine/provisioner.rb +136 -0
  308. data/plugins/provisioners/chef/config/base.rb +85 -0
  309. data/plugins/provisioners/chef/config/chef_client.rb +57 -0
  310. data/plugins/provisioners/chef/config/chef_solo.rb +87 -0
  311. data/plugins/provisioners/chef/plugin.rb +33 -0
  312. data/plugins/provisioners/chef/provisioner/base.rb +93 -0
  313. data/plugins/provisioners/chef/provisioner/chef_client.rb +107 -0
  314. data/plugins/provisioners/chef/provisioner/chef_solo.rb +192 -0
  315. data/plugins/provisioners/puppet/config/puppet.rb +70 -0
  316. data/plugins/provisioners/puppet/config/puppet_server.rb +16 -0
  317. data/plugins/provisioners/puppet/plugin.rb +33 -0
  318. data/plugins/provisioners/puppet/provisioner/puppet.rb +117 -0
  319. data/plugins/provisioners/puppet/provisioner/puppet_server.rb +67 -0
  320. data/plugins/provisioners/shell/config.rb +56 -0
  321. data/plugins/provisioners/shell/plugin.rb +23 -0
  322. data/plugins/provisioners/shell/provisioner.rb +86 -0
  323. data/tasks/acceptance.rake +113 -0
  324. data/tasks/bundler.rake +3 -0
  325. data/tasks/test.rake +20 -0
  326. data/templates/commands/init/Vagrantfile.erb +111 -0
  327. data/templates/commands/ssh_config/config.erb +16 -0
  328. data/templates/config/messages.erb +14 -0
  329. data/templates/config/validation_failed.erb +7 -0
  330. data/templates/guests/arch/network_dhcp.erb +4 -0
  331. data/templates/guests/arch/network_static.erb +6 -0
  332. data/templates/guests/debian/network_dhcp.erb +11 -0
  333. data/templates/guests/debian/network_static.erb +7 -0
  334. data/templates/guests/fedora/network_dhcp.erb +6 -0
  335. data/templates/guests/fedora/network_static.erb +13 -0
  336. data/templates/guests/freebsd/network_dhcp.erb +3 -0
  337. data/templates/guests/freebsd/network_static.erb +3 -0
  338. data/templates/guests/gentoo/network_dhcp.erb +4 -0
  339. data/templates/guests/gentoo/network_static.erb +4 -0
  340. data/templates/guests/redhat/network_dhcp.erb +6 -0
  341. data/templates/guests/redhat/network_static.erb +8 -0
  342. data/templates/locales/en.yml +1126 -0
  343. data/templates/nfs/exports.erb +5 -0
  344. data/templates/nfs/exports_freebsd.erb +5 -0
  345. data/templates/nfs/exports_linux.erb +5 -0
  346. data/templates/package_Vagrantfile.erb +11 -0
  347. data/templates/provisioners/chef_client/client.erb +32 -0
  348. data/templates/provisioners/chef_solo/solo.erb +25 -0
  349. data/templates/rgloader.rb +9 -0
  350. data/test/acceptance/base.rb +48 -0
  351. data/test/acceptance/box_test.rb +99 -0
  352. data/test/acceptance/destroy_test.rb +37 -0
  353. data/test/acceptance/halt_test.rb +72 -0
  354. data/test/acceptance/init_test.rb +33 -0
  355. data/test/acceptance/networking/host_only_test.rb +37 -0
  356. data/test/acceptance/networking/port_forward_test.rb +125 -0
  357. data/test/acceptance/package_test.rb +46 -0
  358. data/test/acceptance/provisioning/basic_test.rb +61 -0
  359. data/test/acceptance/provisioning/chef_solo_test.rb +37 -0
  360. data/test/acceptance/provisioning/shell_test.rb +53 -0
  361. data/test/acceptance/resume_test.rb +17 -0
  362. data/test/acceptance/shared_folders_test.rb +84 -0
  363. data/test/acceptance/skeletons/chef_solo_basic/README.md +3 -0
  364. data/test/acceptance/skeletons/chef_solo_basic/cookbooks/basic/recipes/default.rb +5 -0
  365. data/test/acceptance/skeletons/chef_solo_json/README.md +3 -0
  366. data/test/acceptance/skeletons/chef_solo_json/cookbooks/basic/recipes/default.rb +6 -0
  367. data/test/acceptance/skeletons/provisioner_multi/README.md +3 -0
  368. data/test/acceptance/skeletons/provisioner_multi/cookbooks/basic/recipes/default.rb +5 -0
  369. data/test/acceptance/ssh_test.rb +46 -0
  370. data/test/acceptance/support/config.rb +42 -0
  371. data/test/acceptance/support/isolated_environment.rb +118 -0
  372. data/test/acceptance/support/matchers/have_color.rb +9 -0
  373. data/test/acceptance/support/matchers/match_output.rb +14 -0
  374. data/test/acceptance/support/matchers/succeed.rb +14 -0
  375. data/test/acceptance/support/network_tests.rb +29 -0
  376. data/test/acceptance/support/output.rb +95 -0
  377. data/test/acceptance/support/shared/base_context.rb +72 -0
  378. data/test/acceptance/support/shared/command_examples.rb +33 -0
  379. data/test/acceptance/support/virtualbox.rb +36 -0
  380. data/test/acceptance/suspend_test.rb +56 -0
  381. data/test/acceptance/up_basic_test.rb +33 -0
  382. data/test/acceptance/up_with_box_url.rb +40 -0
  383. data/test/acceptance/vagrant_test.rb +47 -0
  384. data/test/acceptance/version_test.rb +15 -0
  385. data/test/config/acceptance_boxes.yml +7 -0
  386. data/test/support/isolated_environment.rb +46 -0
  387. data/test/support/tempdir.rb +43 -0
  388. data/test/unit/base.rb +27 -0
  389. data/test/unit/support/dummy_provider.rb +16 -0
  390. data/test/unit/support/isolated_environment.rb +179 -0
  391. data/test/unit/support/shared/base_context.rb +104 -0
  392. data/test/unit/vagrant/action/builder_test.rb +242 -0
  393. data/test/unit/vagrant/action/builtin/call_test.rb +145 -0
  394. data/test/unit/vagrant/action/builtin/confirm_test.rb +36 -0
  395. data/test/unit/vagrant/action/builtin/env_set_test.rb +20 -0
  396. data/test/unit/vagrant/action/builtin/graceful_halt_test.rb +61 -0
  397. data/test/unit/vagrant/action/builtin/lock_test.rb +98 -0
  398. data/test/unit/vagrant/action/builtin/ssh_exec_test.rb +57 -0
  399. data/test/unit/vagrant/action/hook_test.rb +120 -0
  400. data/test/unit/vagrant/action/runner_test.rb +71 -0
  401. data/test/unit/vagrant/action/warden_test.rb +92 -0
  402. data/test/unit/vagrant/batch_action_test.rb +36 -0
  403. data/test/unit/vagrant/box_collection_test.rb +237 -0
  404. data/test/unit/vagrant/box_test.rb +122 -0
  405. data/test/unit/vagrant/cli_test.rb +27 -0
  406. data/test/unit/vagrant/config/loader_test.rb +184 -0
  407. data/test/unit/vagrant/config/v1/dummy_config_test.rb +24 -0
  408. data/test/unit/vagrant/config/v1/loader_test.rb +145 -0
  409. data/test/unit/vagrant/config/v1/root_test.rb +40 -0
  410. data/test/unit/vagrant/config/v2/dummy_config_test.rb +24 -0
  411. data/test/unit/vagrant/config/v2/loader_test.rb +151 -0
  412. data/test/unit/vagrant/config/v2/root_test.rb +97 -0
  413. data/test/unit/vagrant/config/v2/util_test.rb +21 -0
  414. data/test/unit/vagrant/config_test.rb +66 -0
  415. data/test/unit/vagrant/environment_test.rb +721 -0
  416. data/test/unit/vagrant/guest_test.rb +166 -0
  417. data/test/unit/vagrant/hosts_test.rb +37 -0
  418. data/test/unit/vagrant/machine_state_test.rb +26 -0
  419. data/test/unit/vagrant/machine_test.rb +384 -0
  420. data/test/unit/vagrant/plugin/v1/command_test.rb +143 -0
  421. data/test/unit/vagrant/plugin/v1/communicator_test.rb +9 -0
  422. data/test/unit/vagrant/plugin/v1/config_test.rb +50 -0
  423. data/test/unit/vagrant/plugin/v1/host_test.rb +5 -0
  424. data/test/unit/vagrant/plugin/v1/manager_test.rb +114 -0
  425. data/test/unit/vagrant/plugin/v1/plugin_test.rb +267 -0
  426. data/test/unit/vagrant/plugin/v1/provider_test.rb +18 -0
  427. data/test/unit/vagrant/plugin/v2/command_test.rb +238 -0
  428. data/test/unit/vagrant/plugin/v2/communicator_test.rb +9 -0
  429. data/test/unit/vagrant/plugin/v2/components_test.rb +17 -0
  430. data/test/unit/vagrant/plugin/v2/config_test.rb +60 -0
  431. data/test/unit/vagrant/plugin/v2/host_test.rb +5 -0
  432. data/test/unit/vagrant/plugin/v2/manager_test.rb +174 -0
  433. data/test/unit/vagrant/plugin/v2/plugin_test.rb +305 -0
  434. data/test/unit/vagrant/plugin/v2/provider_test.rb +18 -0
  435. data/test/unit/vagrant/registry_test.rb +128 -0
  436. data/test/unit/vagrant/util/ansi_escape_code_remover_test.rb +16 -0
  437. data/test/unit/vagrant/util/downloader_test.rb +57 -0
  438. data/test/unit/vagrant/util/file_checksum_test.rb +23 -0
  439. data/test/unit/vagrant/util/hash_with_indifferent_access_test.rb +38 -0
  440. data/test/unit/vagrant/util/is_port_open_test.rb +53 -0
  441. data/test/unit/vagrant/util/line_endings_helper_test.rb +16 -0
  442. data/test/unit/vagrant/util/network_ip_test.rb +17 -0
  443. data/test/unit/vagrant/util/retryable_test.rb +106 -0
  444. data/test/unit/vagrant/util/safe_chdir_test.rb +43 -0
  445. data/test/unit/vagrant/util/scoped_hash_override_test.rb +48 -0
  446. data/test/unit/vagrant/util/ssh_test.rb +30 -0
  447. data/test/unit/vagrant/util/string_block_editor_test.rb +106 -0
  448. data/test/unit/vagrant/util/which_test.rb +43 -0
  449. data/test/unit/vagrant_test.rb +56 -0
  450. data/tnargav.gemspec +69 -0
  451. data/vagrant.gemspec +69 -0
  452. metadata +730 -0
@@ -0,0 +1,360 @@
1
+ require 'log4r'
2
+
3
+ require 'vagrant/util/busy'
4
+ require 'vagrant/util/platform'
5
+ require 'vagrant/util/retryable'
6
+ require 'vagrant/util/subprocess'
7
+
8
+ module VagrantPlugins
9
+ module ProviderVirtualBox
10
+ module Driver
11
+ # Base class for all VirtualBox drivers.
12
+ #
13
+ # This class provides useful tools for things such as executing
14
+ # VBoxManage and handling SIGINTs and so on.
15
+ class Base
16
+ # Include this so we can use `Subprocess` more easily.
17
+ include Vagrant::Util::Retryable
18
+
19
+ def initialize
20
+ @logger = Log4r::Logger.new("vagrant::provider::virtualbox::base")
21
+
22
+ # This flag is used to keep track of interrupted state (SIGINT)
23
+ @interrupted = false
24
+
25
+ # Set the path to VBoxManage
26
+ @vboxmanage_path = "VBoxManage"
27
+
28
+ if Vagrant::Util::Platform.windows? || Vagrant::Util::Platform.cygwin?
29
+ @logger.debug("Windows. Trying VBOX_INSTALL_PATH for VBoxManage")
30
+
31
+ # On Windows, we use the VBOX_INSTALL_PATH environmental
32
+ # variable to find VBoxManage.
33
+ if ENV.has_key?("VBOX_INSTALL_PATH")
34
+ # Get the path.
35
+ path = ENV["VBOX_INSTALL_PATH"]
36
+ @logger.debug("VBOX_INSTALL_PATH value: #{path}")
37
+
38
+ # There can actually be multiple paths in here, so we need to
39
+ # split by the separator ";" and see which is a good one.
40
+ path.split(";").each do |single|
41
+ # Make sure it ends with a \
42
+ single += "\\" if !single.end_with?("\\")
43
+
44
+ # If the executable exists, then set it as the main path
45
+ # and break out
46
+ vboxmanage = "#{path}VBoxManage.exe"
47
+ if File.file?(vboxmanage)
48
+ @vboxmanage_path = Vagrant::Util::Platform.cygwin_windows_path(vboxmanage)
49
+ break
50
+ end
51
+ end
52
+ end
53
+ end
54
+
55
+ @logger.info("VBoxManage path: #{@vboxmanage_path}")
56
+ end
57
+
58
+ # Clears the forwarded ports that have been set on the virtual machine.
59
+ def clear_forwarded_ports
60
+ end
61
+
62
+ # Clears the shared folders that have been set on the virtual machine.
63
+ def clear_shared_folders
64
+ end
65
+
66
+ # Creates a DHCP server for a host only network.
67
+ #
68
+ # @param [String] network Name of the host-only network.
69
+ # @param [Hash] options Options for the DHCP server.
70
+ def create_dhcp_server(network, options)
71
+ end
72
+
73
+ # Creates a host only network with the given options.
74
+ #
75
+ # @param [Hash] options Options to create the host only network.
76
+ # @return [Hash] The details of the host only network, including
77
+ # keys `:name`, `:ip`, and `:netmask`
78
+ def create_host_only_network(options)
79
+ end
80
+
81
+ # Deletes the virtual machine references by this driver.
82
+ def delete
83
+ end
84
+
85
+ # Deletes any host only networks that aren't being used for anything.
86
+ def delete_unused_host_only_networks
87
+ end
88
+
89
+ # Discards any saved state associated with this VM.
90
+ def discard_saved_state
91
+ end
92
+
93
+ # Enables network adapters on the VM.
94
+ #
95
+ # The format of each adapter specification should be like so:
96
+ #
97
+ # {
98
+ # :type => :hostonly,
99
+ # :hostonly => "vboxnet0",
100
+ # :mac_address => "tubes"
101
+ # }
102
+ #
103
+ # This must support setting up both host only and bridged networks.
104
+ #
105
+ # @param [Array<Hash>] adapters Array of adapters to enable.
106
+ def enable_adapters(adapters)
107
+ end
108
+
109
+ # Execute a raw command straight through to VBoxManage.
110
+ #
111
+ # @param [Array] command Command to execute.
112
+ def execute_command(command)
113
+ end
114
+
115
+ # Exports the virtual machine to the given path.
116
+ #
117
+ # @param [String] path Path to the OVF file.
118
+ # @yield [progress] Yields the block with the progress of the export.
119
+ def export(path)
120
+ end
121
+
122
+ # Forwards a set of ports for a VM.
123
+ #
124
+ # This will not affect any previously set forwarded ports,
125
+ # so be sure to delete those if you need to.
126
+ #
127
+ # The format of each port hash should be the following:
128
+ #
129
+ # {
130
+ # :name => "foo",
131
+ # :hostport => 8500,
132
+ # :guestport => 80,
133
+ # :adapter => 1,
134
+ # :protocol => "tcp"
135
+ # }
136
+ #
137
+ # Note that "adapter" and "protocol" are optional and will default
138
+ # to 1 and "tcp" respectively.
139
+ #
140
+ # @param [Array<Hash>] ports An array of ports to set. See documentation
141
+ # for more information on the format.
142
+ def forward_ports(ports)
143
+ end
144
+
145
+ # Halts the virtual machine (pulls the plug).
146
+ def halt
147
+ end
148
+
149
+ # Imports the VM from an OVF file.
150
+ #
151
+ # @param [String] ovf Path to the OVF file.
152
+ # @return [String] UUID of the imported VM.
153
+ def import(ovf)
154
+ end
155
+
156
+ # Returns a list of forwarded ports for a VM.
157
+ #
158
+ # @param [String] uuid UUID of the VM to read from, or `nil` if this
159
+ # VM.
160
+ # @param [Boolean] active_only If true, only VMs that are running will
161
+ # be checked.
162
+ # @return [Array<Array>]
163
+ def read_forwarded_ports(uuid=nil, active_only=false)
164
+ end
165
+
166
+ # Returns a list of bridged interfaces.
167
+ #
168
+ # @return [Hash]
169
+ def read_bridged_interfaces
170
+ end
171
+
172
+ # Returns the guest additions version that is installed on this VM.
173
+ #
174
+ # @return [String]
175
+ def read_guest_additions_version
176
+ end
177
+
178
+ # Returns a list of available host only interfaces.
179
+ #
180
+ # @return [Hash]
181
+ def read_host_only_interfaces
182
+ end
183
+
184
+ # Returns the MAC address of the first network interface.
185
+ #
186
+ # @return [String]
187
+ def read_mac_address
188
+ end
189
+
190
+ # Returns the folder where VirtualBox places it's VMs.
191
+ #
192
+ # @return [String]
193
+ def read_machine_folder
194
+ end
195
+
196
+ # Returns a list of network interfaces of the VM.
197
+ #
198
+ # @return [Hash]
199
+ def read_network_interfaces
200
+ end
201
+
202
+ # Returns the current state of this VM.
203
+ #
204
+ # @return [Symbol]
205
+ def read_state
206
+ end
207
+
208
+ # Returns a list of all forwarded ports in use by active
209
+ # virtual machines.
210
+ #
211
+ # @return [Array]
212
+ def read_used_ports
213
+ end
214
+
215
+ # Returns a list of all UUIDs of virtual machines currently
216
+ # known by VirtualBox.
217
+ #
218
+ # @return [Array<String>]
219
+ def read_vms
220
+ end
221
+
222
+ # Sets the MAC address of the first network adapter.
223
+ #
224
+ # @param [String] mac MAC address without any spaces/hyphens.
225
+ def set_mac_address(mac)
226
+ end
227
+
228
+ # Share a set of folders on this VM.
229
+ #
230
+ # @param [Array<Hash>] folders
231
+ def share_folders(folders)
232
+ end
233
+
234
+ # Reads the SSH port of this VM.
235
+ #
236
+ # @param [Integer] expected Expected guest port of SSH.
237
+ def ssh_port(expected)
238
+ end
239
+
240
+ # Starts the virtual machine.
241
+ #
242
+ # @param [String] mode Mode to boot the VM. Either "headless"
243
+ # or "gui"
244
+ def start(mode)
245
+ end
246
+
247
+ # Suspend the virtual machine.
248
+ def suspend
249
+ end
250
+
251
+ # Verifies that the driver is ready to accept work.
252
+ #
253
+ # This should raise a VagrantError if things are not ready.
254
+ def verify!
255
+ end
256
+
257
+ # Verifies that an image can be imported properly.
258
+ #
259
+ # @param [String] path Path to an OVF file.
260
+ # @return [Boolean]
261
+ def verify_image(path)
262
+ end
263
+
264
+ # Checks if a VM with the given UUID exists.
265
+ #
266
+ # @return [Boolean]
267
+ def vm_exists?(uuid)
268
+ end
269
+
270
+ # Execute the given subcommand for VBoxManage and return the output.
271
+ def execute(*command, &block)
272
+ # Get the options hash if it exists
273
+ opts = {}
274
+ opts = command.pop if command.last.is_a?(Hash)
275
+
276
+ tries = 0
277
+ tries = 3 if opts[:retryable]
278
+
279
+ # Variable to store our execution result
280
+ r = nil
281
+
282
+ # If there is an error with VBoxManage, this gets set to true
283
+ errored = false
284
+
285
+ retryable(:on => Vagrant::Errors::VBoxManageError, :tries => tries, :sleep => 1) do
286
+ # Execute the command
287
+ r = raw(*command, &block)
288
+
289
+ # If the command was a failure, then raise an exception that is
290
+ # nicely handled by Vagrant.
291
+ if r.exit_code != 0
292
+ if @interrupted
293
+ @logger.info("Exit code != 0, but interrupted. Ignoring.")
294
+ elsif r.exit_code == 126
295
+ # This exit code happens if VBoxManage is on the PATH,
296
+ # but another executable it tries to execute is missing.
297
+ # This is usually indicative of a corrupted VirtualBox install.
298
+ raise Vagrant::Errors::VBoxManageNotFoundError
299
+ else
300
+ errored = true
301
+ end
302
+ else
303
+ # Sometimes, VBoxManage fails but doesn't actual return a non-zero
304
+ # exit code. For this we inspect the output and determine if an error
305
+ # occurred.
306
+ if r.stderr =~ /VBoxManage: error:/
307
+ @logger.info("VBoxManage error text found, assuming error.")
308
+ errored = true
309
+ end
310
+ end
311
+ end
312
+
313
+ # If there was an error running VBoxManage, show the error and the
314
+ # output.
315
+ if errored
316
+ raise Vagrant::Errors::VBoxManageError,
317
+ :command => command.inspect,
318
+ :stderr => r.stderr
319
+ end
320
+
321
+ # Return the output, making sure to replace any Windows-style
322
+ # newlines with Unix-style.
323
+ r.stdout.gsub("\r\n", "\n")
324
+ end
325
+
326
+ # Executes a command and returns the raw result object.
327
+ def raw(*command, &block)
328
+ int_callback = lambda do
329
+ @interrupted = true
330
+ @logger.info("Interrupted.")
331
+ end
332
+
333
+ # Append in the options for subprocess
334
+ command << { :notify => [:stdout, :stderr] }
335
+
336
+ # The following is a workaround for a combined VirtualBox and
337
+ # Mac OS X 10.8 bug:
338
+ #
339
+ # Remove the DYLD_LIBRARY_PATH environmental variable on Mac. This
340
+ # is to fix a bug in Mac OS X 10.8 where a warning is printed to the
341
+ # console if this is set when executing certain programs, which
342
+ # can cause some VBoxManage commands to break because they work
343
+ # by just reading stdout and don't expect the OS to just inject
344
+ # garbage into it.
345
+ old_dyld_lib_path = ENV.delete("DYLD_LIBRARY_PATH")
346
+ old_ld_lib_path = ENV.delete("LD_LIBRARY_PATH")
347
+
348
+ Vagrant::Util::Busy.busy(int_callback) do
349
+ Vagrant::Util::Subprocess.execute(@vboxmanage_path, *command, &block)
350
+ end
351
+ ensure
352
+ # Reset the library path if it was set before. See above comments
353
+ # for more information on why this was unset in the first place.
354
+ ENV["DYLD_LIBRARY_PATH"] = old_dyld_lib_path if old_dyld_lib_path
355
+ ENV["LD_LIBRARY_PATH"] = old_ld_lib_path if old_ld_lib_path
356
+ end
357
+ end
358
+ end
359
+ end
360
+ end
@@ -0,0 +1,142 @@
1
+ require "forwardable"
2
+
3
+ require "log4r"
4
+
5
+ require File.expand_path("../base", __FILE__)
6
+
7
+ module VagrantPlugins
8
+ module ProviderVirtualBox
9
+ module Driver
10
+ class Meta < Base
11
+ # This is raised if the VM is not found when initializing a driver
12
+ # with a UUID.
13
+ class VMNotFound < StandardError; end
14
+
15
+ # We use forwardable to do all our driver forwarding
16
+ extend Forwardable
17
+
18
+ # The UUID of the virtual machine we represent
19
+ attr_reader :uuid
20
+
21
+ # The version of virtualbox that is running.
22
+ attr_reader :version
23
+
24
+ def initialize(uuid=nil)
25
+ # Setup the base
26
+ super()
27
+
28
+ @logger = Log4r::Logger.new("vagrant::provider::virtualbox::meta")
29
+ @uuid = uuid
30
+
31
+ # Read and assign the version of VirtualBox we know which
32
+ # specific driver to instantiate.
33
+ begin
34
+ @version = read_version || ""
35
+ rescue Vagrant::Util::Subprocess::LaunchError
36
+ # This means that VirtualBox was not found, so we raise this
37
+ # error here.
38
+ raise Vagrant::Errors::VirtualBoxNotDetected
39
+ end
40
+
41
+ # Instantiate the proper version driver for VirtualBox
42
+ @logger.debug("Finding driver for VirtualBox version: #{@version}")
43
+ driver_map = {
44
+ "4.0" => Version_4_0,
45
+ "4.1" => Version_4_1,
46
+ "4.2" => Version_4_2
47
+ }
48
+
49
+ driver_klass = nil
50
+ driver_map.each do |key, klass|
51
+ if @version.start_with?(key)
52
+ driver_klass = klass
53
+ break
54
+ end
55
+ end
56
+
57
+ if !driver_klass
58
+ supported_versions = driver_map.keys.sort.join(", ")
59
+ raise Vagrant::Errors::VirtualBoxInvalidVersion, :supported_versions => supported_versions
60
+ end
61
+
62
+ @logger.info("Using VirtualBox driver: #{driver_klass}")
63
+ @driver = driver_klass.new(@uuid)
64
+
65
+ if @uuid
66
+ # Verify the VM exists, and if it doesn't, then don't worry
67
+ # about it (mark the UUID as nil)
68
+ raise VMNotFound if !@driver.vm_exists?(@uuid)
69
+ end
70
+ end
71
+
72
+ def_delegators :@driver, :clear_forwarded_ports,
73
+ :clear_shared_folders,
74
+ :create_dhcp_server,
75
+ :create_host_only_network,
76
+ :delete,
77
+ :delete_unused_host_only_networks,
78
+ :discard_saved_state,
79
+ :enable_adapters,
80
+ :execute_command,
81
+ :export,
82
+ :forward_ports,
83
+ :halt,
84
+ :import,
85
+ :read_forwarded_ports,
86
+ :read_bridged_interfaces,
87
+ :read_guest_additions_version,
88
+ :read_host_only_interfaces,
89
+ :read_mac_address,
90
+ :read_mac_addresses,
91
+ :read_machine_folder,
92
+ :read_network_interfaces,
93
+ :read_state,
94
+ :read_used_ports,
95
+ :read_vms,
96
+ :resume,
97
+ :set_mac_address,
98
+ :set_name,
99
+ :share_folders,
100
+ :ssh_port,
101
+ :start,
102
+ :suspend,
103
+ :verify!,
104
+ :verify_image,
105
+ :vm_exists?
106
+
107
+ protected
108
+
109
+ # This returns the version of VirtualBox that is running.
110
+ #
111
+ # @return [String]
112
+ def read_version
113
+ # The version string is usually in one of the following formats:
114
+ #
115
+ # * 4.1.8r1234
116
+ # * 4.1.8r1234_OSE
117
+ # * 4.1.8_MacPortsr1234
118
+ #
119
+ # Below accounts for all of these.
120
+
121
+ # Note: We split this into multiple lines because apparently "".split("_")
122
+ # is [], so we have to check for an empty array in between.
123
+ output = execute("--version")
124
+ if output =~ /vboxdrv kernel module is not loaded/ ||
125
+ output =~ /VirtualBox kernel modules are not loaded/i
126
+ raise Vagrant::Errors::VirtualBoxKernelModuleNotLoaded
127
+ elsif output =~ /Please install/
128
+ # Check for installation incomplete warnings, for example:
129
+ # "WARNING: The character device /dev/vboxdrv does not
130
+ # exist. Please install the virtualbox-ose-dkms package and
131
+ # the appropriate headers, most likely linux-headers-generic."
132
+ raise Vagrant::Errors::VirtualBoxInstallIncomplete
133
+ end
134
+
135
+ parts = output.split("_")
136
+ return nil if parts.empty?
137
+ parts[0].split("r")[0]
138
+ end
139
+ end
140
+ end
141
+ end
142
+ end