| 
									
										
										
										
											2016-08-10 23:19:09 -07:00
										 |  |  | module Utils | 
					
						
							|  |  |  |   SHELL_PROFILE_MAP = { | 
					
						
							| 
									
										
										
										
											2016-09-17 15:32:44 +01:00
										 |  |  |     bash: "~/.bash_profile", | 
					
						
							|  |  |  |     csh: "~/.cshrc", | 
					
						
							|  |  |  |     fish: "~/.config/fish/config.fish", | 
					
						
							|  |  |  |     ksh: "~/.kshrc", | 
					
						
							|  |  |  |     sh: "~/.bash_profile", | 
					
						
							|  |  |  |     tcsh: "~/.tcshrc", | 
					
						
							|  |  |  |     zsh: "~/.zshrc", | 
					
						
							| 
									
										
										
										
											2016-08-10 23:19:09 -07:00
										 |  |  |   }.freeze | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   module Shell | 
					
						
							| 
									
										
										
										
											2016-09-11 19:12:53 +01:00
										 |  |  |     UNSAFE_SHELL_CHAR = %r{([^A-Za-z0-9_\-.,:/@\n])} | 
					
						
							| 
									
										
										
										
											2016-05-22 18:02:39 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-10 23:19:09 -07:00
										 |  |  |     # take a path and heuristically convert it | 
					
						
							| 
									
										
										
										
											2016-05-22 18:02:39 -07:00
										 |  |  |     # to a shell name, return nil if there's no match | 
					
						
							| 
									
										
										
										
											2016-09-05 14:45:29 +10:00
										 |  |  |     def path_to_shell(path) | 
					
						
							| 
									
										
										
										
											2016-08-10 23:19:09 -07:00
										 |  |  |       # we only care about the basename | 
					
						
							|  |  |  |       shell_name = File.basename(path) | 
					
						
							|  |  |  |       # handle possible version suffix like `zsh-5.2` | 
					
						
							|  |  |  |       shell_name.sub!(/-.*\z/m, "") | 
					
						
							|  |  |  |       shell_name.to_sym if %w[bash csh fish ksh sh tcsh zsh].include?(shell_name) | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2016-09-05 14:45:29 +10:00
										 |  |  |     module_function :path_to_shell | 
					
						
							| 
									
										
										
										
											2016-08-10 23:19:09 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-05 14:45:29 +10:00
										 |  |  |     def preferred_shell | 
					
						
							| 
									
										
										
										
											2016-08-10 23:19:09 -07:00
										 |  |  |       path_to_shell(ENV.fetch("SHELL", "")) | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2016-09-05 14:45:29 +10:00
										 |  |  |     module_function :preferred_shell | 
					
						
							| 
									
										
										
										
											2016-08-10 23:19:09 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-05 14:45:29 +10:00
										 |  |  |     def parent_shell | 
					
						
							| 
									
										
										
										
											2016-08-10 23:19:09 -07:00
										 |  |  |       path_to_shell(`ps -p #{Process.ppid} -o ucomm=`.strip) | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2016-09-05 14:45:29 +10:00
										 |  |  |     module_function :parent_shell | 
					
						
							| 
									
										
										
										
											2016-08-10 23:19:09 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-05 14:45:29 +10:00
										 |  |  |     def csh_quote(str) | 
					
						
							| 
									
										
										
										
											2016-08-10 23:19:09 -07:00
										 |  |  |       # ruby's implementation of shell_escape | 
					
						
							|  |  |  |       str = str.to_s | 
					
						
							|  |  |  |       return "''" if str.empty? | 
					
						
							|  |  |  |       str = str.dup | 
					
						
							|  |  |  |       # anything that isn't a known safe character is padded | 
					
						
							| 
									
										
										
										
											2016-05-22 18:02:39 -07:00
										 |  |  |       str.gsub!(UNSAFE_SHELL_CHAR, "\\\\" + "\\1") | 
					
						
							|  |  |  |       # newlines have to be specially quoted in csh | 
					
						
							| 
									
										
										
										
											2016-08-10 23:19:09 -07:00
										 |  |  |       str.gsub!(/\n/, "'\\\n'") | 
					
						
							|  |  |  |       str | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2016-09-05 14:45:29 +10:00
										 |  |  |     module_function :csh_quote | 
					
						
							| 
									
										
										
										
											2016-09-11 17:47:04 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-05 14:45:29 +10:00
										 |  |  |     def sh_quote(str) | 
					
						
							| 
									
										
										
										
											2016-08-10 23:19:09 -07:00
										 |  |  |       # ruby's implementation of shell_escape | 
					
						
							|  |  |  |       str = str.to_s | 
					
						
							|  |  |  |       return "''" if str.empty? | 
					
						
							|  |  |  |       str = str.dup | 
					
						
							|  |  |  |       # anything that isn't a known safe character is padded | 
					
						
							| 
									
										
										
										
											2016-05-22 18:02:39 -07:00
										 |  |  |       str.gsub!(UNSAFE_SHELL_CHAR, "\\\\" + "\\1") | 
					
						
							| 
									
										
										
										
											2016-08-10 23:19:09 -07:00
										 |  |  |       str.gsub!(/\n/, "'\n'") | 
					
						
							|  |  |  |       str | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2016-09-05 14:45:29 +10:00
										 |  |  |     module_function :sh_quote | 
					
						
							| 
									
										
										
										
											2016-08-10 23:19:09 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # quote values. quoting keys is overkill | 
					
						
							| 
									
										
										
										
											2016-09-05 14:45:29 +10:00
										 |  |  |     def export_value(shell, key, value) | 
					
						
							| 
									
										
										
										
											2016-08-10 23:19:09 -07:00
										 |  |  |       case shell | 
					
						
							|  |  |  |       when :bash, :ksh, :sh, :zsh | 
					
						
							|  |  |  |         "export #{key}=\"#{sh_quote(value)}\"" | 
					
						
							|  |  |  |       when :fish | 
					
						
							|  |  |  |         # fish quoting is mostly Bourne compatible except that | 
					
						
							|  |  |  |         # a single quote can be included in a single-quoted string via \' | 
					
						
							|  |  |  |         # and a literal \ can be included via \\ | 
					
						
							|  |  |  |         "set -gx #{key} \"#{sh_quote(value)}\"" | 
					
						
							|  |  |  |       when :csh, :tcsh | 
					
						
							| 
									
										
										
										
											2016-05-22 18:02:39 -07:00
										 |  |  |         "setenv #{key} #{csh_quote(value)};" | 
					
						
							| 
									
										
										
										
											2016-08-10 23:19:09 -07:00
										 |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2016-09-05 14:45:29 +10:00
										 |  |  |     module_function :export_value | 
					
						
							| 
									
										
										
										
											2016-08-10 23:19:09 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # return the shell profile file based on users' preferred shell | 
					
						
							| 
									
										
										
										
											2016-09-05 14:45:29 +10:00
										 |  |  |     def shell_profile | 
					
						
							| 
									
										
										
										
											2016-08-10 23:19:09 -07:00
										 |  |  |       SHELL_PROFILE_MAP.fetch(preferred_shell, "~/.bash_profile") | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2016-09-05 14:45:29 +10:00
										 |  |  |     module_function :shell_profile | 
					
						
							| 
									
										
										
										
											2016-05-22 16:03:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-05 14:45:29 +10:00
										 |  |  |     def prepend_path_in_shell_profile(path) | 
					
						
							| 
									
										
										
										
											2016-09-11 17:47:04 +01:00
										 |  |  |       case preferred_shell | 
					
						
							| 
									
										
										
										
											2016-05-22 18:02:39 -07:00
										 |  |  |       when :bash, :ksh, :sh, :zsh, nil | 
					
						
							| 
									
										
										
										
											2016-06-14 22:31:48 -07:00
										 |  |  |         "echo 'export PATH=\"#{sh_quote(path)}:$PATH\"' >> #{shell_profile}" | 
					
						
							| 
									
										
										
										
											2016-05-22 16:03:51 -07:00
										 |  |  |       when :csh, :tcsh | 
					
						
							|  |  |  |         "echo 'setenv PATH #{csh_quote(path)}:$PATH' >> #{shell_profile}" | 
					
						
							|  |  |  |       when :fish | 
					
						
							| 
									
										
										
										
											2016-05-22 18:02:39 -07:00
										 |  |  |         "echo 'set -g fish_user_paths \"#{sh_quote(path)}\" $fish_user_paths' >> #{shell_profile}" | 
					
						
							| 
									
										
										
										
											2016-05-22 16:03:51 -07:00
										 |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2016-09-05 14:45:29 +10:00
										 |  |  |     module_function :prepend_path_in_shell_profile | 
					
						
							| 
									
										
										
										
											2016-08-10 23:19:09 -07:00
										 |  |  |   end | 
					
						
							|  |  |  | end |