| 
									
										
										
										
											2016-08-18 22:11:42 +03:00
										 |  |  | require "pathname" | 
					
						
							| 
									
										
										
										
											2016-11-09 09:32:54 +01:00
										 |  |  | require "timeout" | 
					
						
							| 
									
										
										
										
											2016-08-18 22:11:42 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | require "hbc/artifact/base" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  | module Hbc | 
					
						
							|  |  |  |   module Artifact | 
					
						
							|  |  |  |     class UninstallBase < Base | 
					
						
							|  |  |  |       # TODO: 500 is also hardcoded in cask/pkg.rb, but much of | 
					
						
							|  |  |  |       #       that logic is probably in the wrong location | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       PATH_ARG_SLICE_SIZE = 500
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       ORDERED_DIRECTIVES = [ | 
					
						
							| 
									
										
										
										
											2016-10-14 20:33:16 +02:00
										 |  |  |         :early_script, | 
					
						
							|  |  |  |         :launchctl, | 
					
						
							|  |  |  |         :quit, | 
					
						
							|  |  |  |         :signal, | 
					
						
							|  |  |  |         :login_item, | 
					
						
							|  |  |  |         :kext, | 
					
						
							|  |  |  |         :script, | 
					
						
							|  |  |  |         :pkgutil, | 
					
						
							|  |  |  |         :delete, | 
					
						
							|  |  |  |         :trash, | 
					
						
							|  |  |  |         :rmdir, | 
					
						
							|  |  |  |       ].freeze | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |       # TODO: these methods were consolidated here from separate | 
					
						
							|  |  |  |       #       sources and should be refactored for consistency | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       def self.expand_path_strings(path_strings) | 
					
						
							| 
									
										
										
										
											2016-10-23 14:44:14 +02:00
										 |  |  |         path_strings.map do |path_string| | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |           path_string.start_with?("~") ? Pathname.new(path_string).expand_path : Pathname.new(path_string) | 
					
						
							| 
									
										
										
										
											2016-10-23 14:44:14 +02:00
										 |  |  |         end | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2016-08-18 22:11:42 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-19 07:57:41 +03:00
										 |  |  |       def self.expand_glob(path_strings) | 
					
						
							|  |  |  |         path_strings.flat_map(&Pathname.method(:glob)) | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |       def self.remove_relative_path_strings(action, path_strings) | 
					
						
							| 
									
										
										
										
											2016-10-23 14:44:14 +02:00
										 |  |  |         relative = path_strings.map do |path_string| | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |           path_string if %r{/\.\.(?:/|\Z)}.match(path_string) || !%r{\A/}.match(path_string) | 
					
						
							| 
									
										
										
										
											2016-10-23 14:44:14 +02:00
										 |  |  |         end.compact | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |         relative.each do |path_string| | 
					
						
							|  |  |  |           opoo "Skipping #{action} for relative path #{path_string}" | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |         path_strings - relative | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2016-08-18 22:11:42 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |       def self.remove_undeletable_path_strings(action, path_strings) | 
					
						
							| 
									
										
										
										
											2016-10-23 14:44:14 +02:00
										 |  |  |         undeletable = path_strings.map do |path_string| | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |           path_string if MacOS.undeletable?(Pathname.new(path_string)) | 
					
						
							| 
									
										
										
										
											2016-10-23 14:44:14 +02:00
										 |  |  |         end.compact | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |         undeletable.each do |path_string| | 
					
						
							|  |  |  |           opoo "Skipping #{action} for undeletable path #{path_string}" | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |         path_strings - undeletable | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2016-08-18 22:11:42 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-19 07:57:41 +03:00
										 |  |  |       def self.prepare_path_strings(action, path_strings, expand_tilde) | 
					
						
							|  |  |  |         path_strings = expand_path_strings(path_strings) if expand_tilde | 
					
						
							|  |  |  |         path_strings = remove_relative_path_strings(action, path_strings) | 
					
						
							|  |  |  |         path_strings = expand_glob(path_strings) | 
					
						
							|  |  |  |         remove_undeletable_path_strings(action, path_strings) | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-05 07:47:54 +01:00
										 |  |  |       def dispatch_uninstall_directives(expand_tilde: true) | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |         directives_set = @cask.artifacts[stanza] | 
					
						
							|  |  |  |         ohai "Running #{stanza} process for #{@cask}; your password may be necessary" | 
					
						
							| 
									
										
										
										
											2016-08-18 22:11:42 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |         directives_set.each do |directives| | 
					
						
							|  |  |  |           warn_for_unknown_directives(directives) | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2016-08-18 22:11:42 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |         ORDERED_DIRECTIVES.each do |directive_sym| | 
					
						
							|  |  |  |           directives_set.select { |h| h.key?(directive_sym) }.each do |directives| | 
					
						
							|  |  |  |             args = [directives] | 
					
						
							|  |  |  |             args << expand_tilde if [:delete, :trash, :rmdir].include?(directive_sym) | 
					
						
							|  |  |  |             send("uninstall_#{directive_sym}", *args) | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2016-08-18 22:11:42 +03:00
										 |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |       private | 
					
						
							| 
									
										
										
										
											2016-08-18 22:11:42 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |       def stanza | 
					
						
							|  |  |  |         self.class.artifact_dsl_key | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2016-08-18 22:11:42 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |       def warn_for_unknown_directives(directives) | 
					
						
							|  |  |  |         unknown_keys = directives.keys - ORDERED_DIRECTIVES | 
					
						
							|  |  |  |         return if unknown_keys.empty? | 
					
						
							| 
									
										
										
										
											2016-10-14 20:08:05 +02:00
										 |  |  |         opoo %Q(Unknown arguments to #{stanza} -- #{unknown_keys.inspect}. Running "brew update; brew cleanup; brew cask cleanup" will likely fix it.) | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2016-08-18 22:11:42 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |       # Preserve prior functionality of script which runs first. Should rarely be needed. | 
					
						
							|  |  |  |       # :early_script should not delete files, better defer that to :script. | 
					
						
							|  |  |  |       # If Cask writers never need :early_script it may be removed in the future. | 
					
						
							|  |  |  |       def uninstall_early_script(directives) | 
					
						
							|  |  |  |         uninstall_script(directives, directive_name: :early_script) | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2016-08-18 22:11:42 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |       # :launchctl must come before :quit/:signal for cases where app would instantly re-launch | 
					
						
							|  |  |  |       def uninstall_launchctl(directives) | 
					
						
							|  |  |  |         Array(directives[:launchctl]).each do |service| | 
					
						
							|  |  |  |           ohai "Removing launchctl service #{service}" | 
					
						
							|  |  |  |           [false, true].each do |with_sudo| | 
					
						
							|  |  |  |             plist_status = @command.run("/bin/launchctl", args: ["list", service], sudo: with_sudo, print_stderr: false).stdout | 
					
						
							| 
									
										
										
										
											2016-10-14 20:03:34 +02:00
										 |  |  |             if plist_status =~ /^\{/ | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |               @command.run!("/bin/launchctl", args: ["remove", service], sudo: with_sudo) | 
					
						
							|  |  |  |               sleep 1
 | 
					
						
							|  |  |  |             end | 
					
						
							|  |  |  |             paths = ["/Library/LaunchAgents/#{service}.plist", | 
					
						
							|  |  |  |                      "/Library/LaunchDaemons/#{service}.plist"] | 
					
						
							|  |  |  |             paths.each { |elt| elt.prepend(ENV["HOME"]) } unless with_sudo | 
					
						
							|  |  |  |             paths = paths.map { |elt| Pathname(elt) }.select(&:exist?) | 
					
						
							|  |  |  |             paths.each do |path| | 
					
						
							|  |  |  |               @command.run!("/bin/rm", args: ["-f", "--", path], sudo: with_sudo) | 
					
						
							|  |  |  |             end | 
					
						
							|  |  |  |             # undocumented and untested: pass a path to uninstall :launchctl | 
					
						
							|  |  |  |             next unless Pathname(service).exist? | 
					
						
							|  |  |  |             @command.run!("/bin/launchctl", args: ["unload", "-w", "--", service], sudo: with_sudo) | 
					
						
							|  |  |  |             @command.run!("/bin/rm",        args: ["-f", "--", service], sudo: with_sudo) | 
					
						
							|  |  |  |             sleep 1
 | 
					
						
							|  |  |  |           end | 
					
						
							| 
									
										
										
										
											2016-08-18 22:11:42 +03:00
										 |  |  |         end | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       # :quit/:signal must come before :kext so the kext will not be in use by a running process | 
					
						
							|  |  |  |       def uninstall_quit(directives) | 
					
						
							|  |  |  |         Array(directives[:quit]).each do |id| | 
					
						
							|  |  |  |           ohai "Quitting application ID #{id}" | 
					
						
							| 
									
										
										
										
											2016-11-09 09:32:54 +01:00
										 |  |  |           next if running_processes(id).empty? | 
					
						
							| 
									
										
										
										
											2016-10-14 20:08:05 +02:00
										 |  |  |           @command.run!("/usr/bin/osascript", args: ["-e", %Q(tell application id "#{id}" to quit)], sudo: true) | 
					
						
							| 
									
										
										
										
											2016-11-09 09:32:54 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |           begin | 
					
						
							|  |  |  |             Timeout.timeout(3) do | 
					
						
							|  |  |  |               Kernel.loop do | 
					
						
							|  |  |  |                 break if running_processes(id).empty? | 
					
						
							|  |  |  |               end | 
					
						
							|  |  |  |             end | 
					
						
							|  |  |  |           rescue Timeout::Error | 
					
						
							|  |  |  |             next | 
					
						
							|  |  |  |           end | 
					
						
							| 
									
										
										
										
											2016-08-18 22:11:42 +03:00
										 |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |       # :signal should come after :quit so it can be used as a backup when :quit fails | 
					
						
							|  |  |  |       def uninstall_signal(directives) | 
					
						
							|  |  |  |         Array(directives[:signal]).flatten.each_slice(2) do |pair| | 
					
						
							|  |  |  |           raise CaskInvalidError.new(@cask, "Each #{stanza} :signal must have 2 elements.") unless pair.length == 2
 | 
					
						
							| 
									
										
										
										
											2016-11-09 09:32:54 +01:00
										 |  |  |           signal, bundle_id = pair | 
					
						
							|  |  |  |           ohai "Signalling '#{signal}' to application ID '#{bundle_id}'" | 
					
						
							|  |  |  |           pids = running_processes(bundle_id).map(&:first) | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |           next unless pids.any? | 
					
						
							|  |  |  |           # Note that unlike :quit, signals are sent from the current user (not | 
					
						
							|  |  |  |           # upgraded to the superuser).  This is a todo item for the future, but | 
					
						
							|  |  |  |           # there should be some additional thought/safety checks about that, as a | 
					
						
							|  |  |  |           # misapplied "kill" by root could bring down the system. The fact that we | 
					
						
							|  |  |  |           # learned the pid from AppleScript is already some degree of protection, | 
					
						
							|  |  |  |           # though indirect. | 
					
						
							| 
									
										
										
										
											2016-11-09 09:32:54 +01:00
										 |  |  |           odebug "Unix ids are #{pids.inspect} for processes with bundle identifier #{bundle_id}" | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |           Process.kill(signal, *pids) | 
					
						
							|  |  |  |           sleep 3
 | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2016-08-18 22:11:42 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-09 09:32:54 +01:00
										 |  |  |       def running_processes(bundle_id) | 
					
						
							|  |  |  |         @command.run!("/bin/launchctl", args: ["list"]).stdout.lines | 
					
						
							|  |  |  |                 .map { |line| line.chomp.split("\t") } | 
					
						
							|  |  |  |                 .map { |pid, state, id| [pid.to_i, state.to_i, id] } | 
					
						
							|  |  |  |                 .select do |fields| | 
					
						
							|  |  |  |                   next if fields[0].zero? | 
					
						
							|  |  |  |                   fields[2] =~ /^#{Regexp.escape(bundle_id)}($|\.\d+)/ | 
					
						
							|  |  |  |                 end | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2016-08-18 22:11:42 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |       def uninstall_login_item(directives) | 
					
						
							|  |  |  |         Array(directives[:login_item]).each do |name| | 
					
						
							|  |  |  |           ohai "Removing login item #{name}" | 
					
						
							|  |  |  |           @command.run!("/usr/bin/osascript", | 
					
						
							| 
									
										
										
										
											2016-10-14 20:08:05 +02:00
										 |  |  |                         args: ["-e", %Q(tell application "System Events" to delete every login item whose name is "#{name}")], | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |                         sudo: false) | 
					
						
							|  |  |  |           sleep 1
 | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2016-08-18 22:11:42 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |       # :kext should be unloaded before attempting to delete the relevant file | 
					
						
							|  |  |  |       def uninstall_kext(directives) | 
					
						
							|  |  |  |         Array(directives[:kext]).each do |kext| | 
					
						
							|  |  |  |           ohai "Unloading kernel extension #{kext}" | 
					
						
							|  |  |  |           is_loaded = @command.run!("/usr/sbin/kextstat", args: ["-l", "-b", kext], sudo: true).stdout | 
					
						
							|  |  |  |           if is_loaded.length > 1
 | 
					
						
							|  |  |  |             @command.run!("/sbin/kextunload", args: ["-b", kext], sudo: true) | 
					
						
							|  |  |  |             sleep 1
 | 
					
						
							|  |  |  |           end | 
					
						
							| 
									
										
										
										
											2016-11-18 22:53:25 +01:00
										 |  |  |           @command.run!("/usr/sbin/kextfind", args: ["-b", kext], sudo: true).stdout.chomp.lines.each do |kext_path| | 
					
						
							|  |  |  |             ohai "Removing kernel extension #{kext_path}" | 
					
						
							|  |  |  |             @command.run!("/bin/rm", args: ["-rf", kext_path], sudo: true) | 
					
						
							|  |  |  |           end | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2016-08-18 22:11:42 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |       # :script must come before :pkgutil, :delete, or :trash so that the script file is not already deleted | 
					
						
							|  |  |  |       def uninstall_script(directives, directive_name: :script) | 
					
						
							|  |  |  |         executable, script_arguments = self.class.read_script_arguments(directives, | 
					
						
							|  |  |  |                                                                         "uninstall", | 
					
						
							|  |  |  |                                                                         { must_succeed: true, sudo: true }, | 
					
						
							|  |  |  |                                                                         { print_stdout: true }, | 
					
						
							|  |  |  |                                                                         directive_name) | 
					
						
							|  |  |  |         ohai "Running uninstall script #{executable}" | 
					
						
							|  |  |  |         raise CaskInvalidError.new(@cask, "#{stanza} :#{directive_name} without :executable.") if executable.nil? | 
					
						
							|  |  |  |         executable_path = @cask.staged_path.join(executable) | 
					
						
							| 
									
										
										
										
											2017-02-16 21:53:22 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         unless executable_path.exist? | 
					
						
							|  |  |  |           message = "uninstall script #{executable} does not exist" | 
					
						
							|  |  |  |           raise CaskError, "#{message}." unless force | 
					
						
							|  |  |  |           opoo "#{message}, skipping." | 
					
						
							|  |  |  |           return | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         @command.run("/bin/chmod", args: ["--", "+x", executable_path]) | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |         @command.run(executable_path, script_arguments) | 
					
						
							| 
									
										
										
										
											2016-08-18 22:11:42 +03:00
										 |  |  |         sleep 1
 | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |       def uninstall_pkgutil(directives) | 
					
						
							|  |  |  |         ohai "Removing files from pkgutil Bill-of-Materials" | 
					
						
							|  |  |  |         Array(directives[:pkgutil]).each do |regexp| | 
					
						
							|  |  |  |           pkgs = Hbc::Pkg.all_matching(regexp, @command) | 
					
						
							|  |  |  |           pkgs.each(&:uninstall) | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2016-08-18 22:11:42 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |       def uninstall_delete(directives, expand_tilde = true) | 
					
						
							|  |  |  |         Array(directives[:delete]).concat(Array(directives[:trash])).flatten.each_slice(PATH_ARG_SLICE_SIZE) do |path_slice| | 
					
						
							|  |  |  |           ohai "Removing files: #{path_slice.utf8_inspect}" | 
					
						
							| 
									
										
										
										
											2017-02-19 07:57:41 +03:00
										 |  |  |           path_slice = self.class.prepare_path_strings(:delete, path_slice, expand_tilde) | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |           @command.run!("/bin/rm", args: path_slice.unshift("-rf", "--"), sudo: true) | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2016-08-18 22:11:42 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |       # :trash functionality is stubbed as a synonym for :delete | 
					
						
							|  |  |  |       # TODO: make :trash work differently, moving files to the Trash | 
					
						
							|  |  |  |       def uninstall_trash(directives, expand_tilde = true) | 
					
						
							|  |  |  |         uninstall_delete(directives, expand_tilde) | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2016-08-18 22:11:42 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-19 07:57:41 +03:00
										 |  |  |       def uninstall_rmdir(directories, expand_tilde = true) | 
					
						
							|  |  |  |         action = :rmdir | 
					
						
							|  |  |  |         self.class.prepare_path_strings(action, Array(directories[action]).flatten, expand_tilde).each do |directory| | 
					
						
							| 
									
										
										
										
											2016-09-24 13:52:43 +02:00
										 |  |  |           next if directory.to_s.empty? | 
					
						
							|  |  |  |           ohai "Removing directory if empty: #{directory.to_s.utf8_inspect}" | 
					
						
							|  |  |  |           directory = Pathname.new(directory) | 
					
						
							|  |  |  |           next unless directory.exist? | 
					
						
							|  |  |  |           @command.run!("/bin/rm", | 
					
						
							|  |  |  |                         args:         ["-f", "--", directory.join(".DS_Store")], | 
					
						
							|  |  |  |                         sudo:         true, | 
					
						
							|  |  |  |                         print_stderr: false) | 
					
						
							|  |  |  |           @command.run("/bin/rmdir", | 
					
						
							|  |  |  |                        args:         ["--", directory], | 
					
						
							|  |  |  |                        sudo:         true, | 
					
						
							|  |  |  |                        print_stderr: false) | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2016-08-18 22:11:42 +03:00
										 |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | end |