Land #4337 - Fix prompt coloring on Windows
commit
c670bb72df
|
@ -83,7 +83,7 @@ begin
|
||||||
Thread.current.priority = -20
|
Thread.current.priority = -20
|
||||||
|
|
||||||
output.prompting
|
output.prompting
|
||||||
line = ::Readline.readline(prompt, true)
|
line = readline_with_output(prompt, true)
|
||||||
::Readline::HISTORY.pop if (line and line.empty?)
|
::Readline::HISTORY.pop if (line and line.empty?)
|
||||||
ensure
|
ensure
|
||||||
Thread.current.priority = orig || 0
|
Thread.current.priority = orig || 0
|
||||||
|
@ -116,6 +116,37 @@ begin
|
||||||
#
|
#
|
||||||
attr_accessor :output
|
attr_accessor :output
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def readline_with_output(prompt, add_history=false)
|
||||||
|
# rb-readlines's Readline.readline hardcodes the input and output to $stdin and $stdout, which means setting
|
||||||
|
# `Readline.input` or `Readline.ouput` has no effect when running `Readline.readline` with rb-readline, so need
|
||||||
|
# to reimplement []`Readline.readline`](https://github.com/luislavena/rb-readline/blob/ce4908dae45dbcae90a6e42e3710b8c3a1f2cd64/lib/readline.rb#L36-L58)
|
||||||
|
# for rb-readline to support setting input and output. Output needs to be set so that colorization works for the
|
||||||
|
# prompt on Windows.
|
||||||
|
if defined? RbReadline
|
||||||
|
RbReadline.rl_instream = fd
|
||||||
|
RbReadline.rl_outstream = output
|
||||||
|
|
||||||
|
begin
|
||||||
|
line = RbReadline.readline(prompt)
|
||||||
|
rescue ::Exception => exception
|
||||||
|
RbReadline.rl_cleanup_after_signal()
|
||||||
|
RbReadline.rl_deprep_terminal()
|
||||||
|
|
||||||
|
raise exception
|
||||||
|
end
|
||||||
|
|
||||||
|
if add_history && line
|
||||||
|
RbReadline.add_history(line)
|
||||||
|
end
|
||||||
|
|
||||||
|
line.try(:dup)
|
||||||
|
else
|
||||||
|
::Readline.readline(prompt, true)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
end
|
end
|
||||||
rescue LoadError
|
rescue LoadError
|
||||||
end
|
end
|
||||||
|
|
|
@ -16,6 +16,76 @@ module Text
|
||||||
#
|
#
|
||||||
###
|
###
|
||||||
class Output::Stdio < Rex::Ui::Text::Output
|
class Output::Stdio < Rex::Ui::Text::Output
|
||||||
|
#
|
||||||
|
# Attributes
|
||||||
|
#
|
||||||
|
|
||||||
|
# @!attribute io
|
||||||
|
# The raw `IO` backing this Text output. Defaults to `$stdout`
|
||||||
|
#
|
||||||
|
# @return [#flush, #puts, #write]
|
||||||
|
attr_writer :io
|
||||||
|
|
||||||
|
#
|
||||||
|
# Constructor
|
||||||
|
#
|
||||||
|
|
||||||
|
# @param options [Hash{Symbol => IO}]
|
||||||
|
# @option options [IO]
|
||||||
|
def initialize(options={})
|
||||||
|
options.assert_valid_keys(:io)
|
||||||
|
|
||||||
|
super()
|
||||||
|
|
||||||
|
self.io = options[:io]
|
||||||
|
end
|
||||||
|
|
||||||
|
#
|
||||||
|
# Methods
|
||||||
|
#
|
||||||
|
|
||||||
|
def flush
|
||||||
|
io.flush
|
||||||
|
end
|
||||||
|
|
||||||
|
# IO to write to.
|
||||||
|
#
|
||||||
|
# @return [IO] Default to `$stdout`
|
||||||
|
def io
|
||||||
|
@io ||= $stdout
|
||||||
|
end
|
||||||
|
|
||||||
|
#
|
||||||
|
# Prints the supplied message to standard output.
|
||||||
|
#
|
||||||
|
def print_raw(msg = '')
|
||||||
|
if (Rex::Compat.is_windows and supports_color?)
|
||||||
|
WindowsConsoleColorSupport.new(io).write(msg)
|
||||||
|
else
|
||||||
|
io.print(msg)
|
||||||
|
end
|
||||||
|
|
||||||
|
io.flush
|
||||||
|
|
||||||
|
msg
|
||||||
|
end
|
||||||
|
alias_method :write, :print_raw
|
||||||
|
|
||||||
|
def puts(*args)
|
||||||
|
args.each do |argument|
|
||||||
|
line = argument.to_s
|
||||||
|
write(line)
|
||||||
|
|
||||||
|
unless line.ends_with? "\n"
|
||||||
|
# yes, this is output, but `IO#puts` uses `rb_default_rs`, which is
|
||||||
|
# [`$/`](https://github.com/ruby/ruby/blob/3af8e150aded9d162bfd41426aaaae0279e5a653/io.c#L12168-L12172),
|
||||||
|
# which is [`$INPUT_RECORD_SEPARATOR`](https://github.com/ruby/ruby/blob/3af8e150aded9d162bfd41426aaaae0279e5a653/lib/English.rb#L83)
|
||||||
|
write($INPUT_RECORD_SEPARATOR)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
nil
|
||||||
|
end
|
||||||
|
|
||||||
def supports_color?
|
def supports_color?
|
||||||
case config[:color]
|
case config[:color]
|
||||||
|
@ -31,20 +101,6 @@ class Output::Stdio < Rex::Ui::Text::Output
|
||||||
return (term and term.match(/(?:vt10[03]|xterm(?:-color)?|linux|screen|rxvt)/i) != nil)
|
return (term and term.match(/(?:vt10[03]|xterm(?:-color)?|linux|screen|rxvt)/i) != nil)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
#
|
|
||||||
# Prints the supplied message to standard output.
|
|
||||||
#
|
|
||||||
def print_raw(msg = '')
|
|
||||||
if (Rex::Compat.is_windows and supports_color?)
|
|
||||||
WindowsConsoleColorSupport.new($stdout).write(msg)
|
|
||||||
else
|
|
||||||
$stdout.print(msg)
|
|
||||||
end
|
|
||||||
$stdout.flush
|
|
||||||
|
|
||||||
msg
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
|
|
||||||
end
|
end
|
||||||
|
|
Loading…
Reference in New Issue