Skip to content

Commit

Permalink
Compiler: use Colorize.on_tty_only!
Browse files Browse the repository at this point in the history
To support TTY detection

Fixed specs and wrapper script
  • Loading branch information
makenowjust committed Nov 30, 2017
1 parent 6ee1bf9 commit d6c6e18
Show file tree
Hide file tree
Showing 15 changed files with 78 additions and 99 deletions.
9 changes: 7 additions & 2 deletions bin/crystal
Original file line number Diff line number Diff line change
Expand Up @@ -115,16 +115,21 @@ __has_colors() {
return 1
fi
}
if __has_colors && [ -t 1 ] && [ -t 2 ]; then
SUPPORTS_COLORS=true
else
SUPPORTS_COLORS=false
fi
__error_msg() {
if __has_colors; then
if $SUPPORTS_COLORS; then
# bold red coloring
printf '%b\n' "\033[31;1m$@\033[0m" 1>&2
else
printf '%b\n' "$@" 1>&2
fi
}
__warning_msg() {
if __has_colors; then
if $SUPPORTS_COLORS; then
# brown coloring
printf '%b\n' "\033[33m$@\033[0m" 1>&2
else
Expand Down
4 changes: 1 addition & 3 deletions spec/compiler/semantic/did_you_mean_spec.cr
Original file line number Diff line number Diff line change
Expand Up @@ -237,8 +237,6 @@ describe "Semantic: did you mean" do
end

it "suggests a better alternative to logical operators (#2715)" do
message = "undefined method 'and'"
message = " (did you mean '&&'?)".colorize.yellow.bold.to_s
assert_error %(
def rand(x : Int32)
end
Expand All @@ -252,7 +250,7 @@ describe "Semantic: did you mean" do
if "a".bytes and 1
1
end
), message
), "did you mean '&&'"
end

it "says did you mean in instance var declaration" do
Expand Down
2 changes: 2 additions & 0 deletions spec/compiler/semantic/generic_class_spec.cr
Original file line number Diff line number Diff line change
Expand Up @@ -636,7 +636,9 @@ describe "Semantic: generic class" do
begin
semantic(nodes)
rescue ex : TypeException
old_enabled, Colorize.enabled = Colorize.enabled?, false
msg = ex.to_s.lines.map(&.strip)
Colorize.enabled = old_enabled
msg.count("- Foo(T).foo(x : Int32)").should eq(1)
end
end
Expand Down
31 changes: 23 additions & 8 deletions src/compiler/crystal/command.cr
Original file line number Diff line number Diff line change
Expand Up @@ -52,11 +52,12 @@ class Crystal::Command
private getter options

def initialize(@options : Array(String))
@color = true
@progress_tracker = ProgressTracker.new
end

def run
Colorize.on_tty_only!

command = options.first?
case
when !command
Expand Down Expand Up @@ -106,7 +107,6 @@ class Crystal::Command
rescue ex : Crystal::LocationlessException
error ex.message
rescue ex : Crystal::Exception
ex.color = @color
if @config.try(&.output_format) == "json"
STDERR.puts ex.to_json
else
Expand Down Expand Up @@ -359,9 +359,12 @@ class Crystal::Command
end
end

opts.on("--color", "Enable colored output") do
Colorize.enabled = true
end

opts.on("--no-color", "Disable colored output") do
@color = false
compiler.color = false
Colorize.enabled = false
end

unless no_codegen
Expand Down Expand Up @@ -509,9 +512,11 @@ class Crystal::Command
puts opts
exit
end
opts.on("--color", "Enable colored output") do
Colorize.enabled = true
end
opts.on("--no-color", "Disable colored output") do
@color = false
compiler.color = false
Colorize.enabled = false
end
opts.invalid_option { }
end
Expand All @@ -527,7 +532,17 @@ class Crystal::Command

private def error(msg, exit_code = 1)
# This is for the case where the main command is wrong
@color = false if ARGV.includes?("--no-color")
Crystal.error msg, @color, exit_code: exit_code
ARGV.each do |arg|
case arg
when "--"
break
when "--color"
Colorize.enabled = true
when "--no-color"
Colorize.enabled = false
end
end

Crystal.error msg, exit_code: exit_code
end
end
24 changes: 14 additions & 10 deletions src/compiler/crystal/command/format.cr
Original file line number Diff line number Diff line change
Expand Up @@ -34,8 +34,12 @@ class Crystal::Command
exit
end

opts.on("--color", "Enable colored output") do
Colorize.enabled = true
end

opts.on("--no-color", "Disable colored output") do
@color = false
Colorize.enabled = false
end
end

Expand Down Expand Up @@ -85,8 +89,8 @@ class Crystal::Command

print result
rescue ex : InvalidByteSequenceError
STDERR.print "Error: ".colorize.toggle(@color).red.bold
STDERR.print "source is not a valid Crystal source file: ".colorize.toggle(@color).bold
STDERR.print "Error: ".colorize.red.bold
STDERR.print "source is not a valid Crystal source file: ".colorize.bold
STDERR.puts ex.message
exit 1
rescue ex : Crystal::SyntaxException
Expand All @@ -112,8 +116,8 @@ class Crystal::Command

File.write(filename, result)
rescue ex : InvalidByteSequenceError
STDERR.print "Error: ".colorize.toggle(@color).red.bold
STDERR.print "file '#{Crystal.relative_filename(filename)}' is not a valid Crystal source file: ".colorize.toggle(@color).bold
STDERR.print "Error: ".colorize.red.bold
STDERR.print "file '#{Crystal.relative_filename(filename)}' is not a valid Crystal source file: ".colorize.bold
STDERR.puts ex.message
exit 1
rescue ex : Crystal::SyntaxException
Expand Down Expand Up @@ -159,21 +163,21 @@ class Crystal::Command
check_files << FormatResult.new(filename, FormatResult::Code::FORMAT)
else
File.write(filename, result)
STDOUT << "Format".colorize(:green).toggle(@color) << " " << filename << "\n"
STDOUT << "Format".colorize(:green) << " " << filename << "\n"
end
rescue ex : InvalidByteSequenceError
if check_files
check_files << FormatResult.new(filename, FormatResult::Code::INVALID_BYTE_SEQUENCE)
else
STDERR.print "Error: ".colorize.toggle(@color).red.bold
STDERR.print "file '#{Crystal.relative_filename(filename)}' is not a valid Crystal source file: ".colorize.toggle(@color).bold
STDERR.print "Error: ".colorize.red.bold
STDERR.print "file '#{Crystal.relative_filename(filename)}' is not a valid Crystal source file: ".colorize.bold
STDERR.puts ex.message
end
rescue ex : Crystal::SyntaxException
if check_files
check_files << FormatResult.new(filename, FormatResult::Code::SYNTAX)
else
STDERR << "Syntax Error:".colorize(:yellow).toggle(@color) << " " << ex.message << " at " << filename << ":" << ex.line_number << ":" << ex.column_number << "\n"
STDERR << "Syntax Error:".colorize(:yellow) << " " << ex.message << " at " << filename << ":" << ex.line_number << ":" << ex.column_number << "\n"
end
rescue ex
if check_files
Expand All @@ -186,7 +190,7 @@ class Crystal::Command
end

private def couldnt_format(file, ex = nil)
STDERR << "Error: ".colorize(:red).toggle(@color)
STDERR << "Error: ".colorize(:red)

if ex
STDERR.puts "couldn't format #{file}, please report a bug including the contents of it: https://github.com/crystal-lang/crystal/issues"
Expand Down
2 changes: 1 addition & 1 deletion src/compiler/crystal/command/spec.cr
Original file line number Diff line number Diff line change
Expand Up @@ -60,7 +60,7 @@ class Crystal::Command
end
end

unless @color
unless Colorize.enabled?
options << "--no-color"
end

Expand Down
16 changes: 4 additions & 12 deletions src/compiler/crystal/compiler.cr
Original file line number Diff line number Diff line change
Expand Up @@ -58,9 +58,6 @@ module Crystal
# Sets the mattr (features). Check LLVM docs to learn about this.
property mattr : String?

# If `false`, color won't be used in output messages.
property? color = true

# If `true`, skip cleanup process on semantic analysis.
property? no_cleanup = false

Expand Down Expand Up @@ -183,7 +180,6 @@ module Crystal
program.flags << "static" if static?
program.flags.concat @flags
program.wants_doc = wants_doc?
program.color = color?
program.stdout = stdout
program.show_error_trace = show_error_trace?
program.progress_tracker = @progress_tracker
Expand Down Expand Up @@ -215,8 +211,8 @@ module Crystal
parser.wants_doc = wants_doc?
parser.parse
rescue ex : InvalidByteSequenceError
stderr.print colorize("Error: ").red.bold
stderr.print colorize("file '#{Crystal.relative_filename(source.filename)}' is not a valid Crystal source file: ").bold
stderr.print "Error: ".colorize.red.bold
stderr.print "file '#{Crystal.relative_filename(source.filename)}' is not a valid Crystal source file: ".colorize.bold
stderr.puts ex.message
exit 1
end
Expand Down Expand Up @@ -468,7 +464,7 @@ module Crystal
TargetMachine.create(triple, @mcpu || "", @mattr || "", @release)
end
rescue ex : ArgumentError
stderr.print colorize("Error: ").red.bold
stderr.print "Error: ".colorize.red.bold
stderr.print "llc: "
stderr.puts ex.message
exit 1
Expand Down Expand Up @@ -518,11 +514,7 @@ module Crystal
end

private def error(msg, exit_code = 1)
Crystal.error msg, @color, exit_code, stderr: stderr
end

private def colorize(obj)
obj.colorize.toggle(@color)
Crystal.error msg, exit_code, stderr: stderr
end

# An LLVM::Module with information to compile it.
Expand Down
10 changes: 0 additions & 10 deletions src/compiler/crystal/exception.cr
Original file line number Diff line number Diff line change
Expand Up @@ -3,8 +3,6 @@ require "colorize"

module Crystal
abstract class Exception < ::Exception
property? color = false

@filename : String | VirtualFile | Nil

def to_s(io)
Expand Down Expand Up @@ -46,14 +44,6 @@ module Crystal
Crystal.relative_filename(filename)
end

def colorize(obj)
obj.colorize.toggle(@color)
end

def with_color
::with_color.toggle(@color)
end

def replace_leading_tabs_with_spaces(line)
found_non_space = false
line.gsub do |char|
Expand Down
9 changes: 0 additions & 9 deletions src/compiler/crystal/program.cr
Original file line number Diff line number Diff line change
Expand Up @@ -67,9 +67,6 @@ module Crystal
# If `true`, doc comments are attached to types and methods.
property? wants_doc = false

# If `true`, error messages can be colorized
property? color = true

# All required files. The set stores absolute files. This way
# files loaded by `require` nodes are only processed once.
getter requires = Set(String).new
Expand Down Expand Up @@ -501,12 +498,6 @@ module Crystal
"__temp_#{@temp_var_counter}"
end

# Colorizes the given object, depending on whether this program
# is configured to use colors.
def colorize(obj)
obj.colorize.toggle(@color)
end

private def abstract_value_type(type)
type.abstract = true
type.struct = true
Expand Down
24 changes: 10 additions & 14 deletions src/compiler/crystal/semantic/call_error.cr
Original file line number Diff line number Diff line change
Expand Up @@ -29,7 +29,7 @@ class Crystal::Path

similar_name = type.lookup_similar_path(self)
if similar_name
self.raise("undefined constant #{self} #{type.program.colorize("(did you mean '#{similar_name}')").yellow.bold}")
self.raise("undefined constant #{self} #{"(did you mean '#{similar_name}')".colorize.yellow.bold}")
else
self.raise("undefined constant #{self}")
end
Expand Down Expand Up @@ -101,15 +101,15 @@ class Crystal::Call
end

if obj && obj.type != owner
msg << colorize(" (compile-time type is #{obj.type})").yellow.bold
msg << " (compile-time type is #{obj.type})".colorize.yellow.bold
end

if similar_name
if similar_name == def_name
# This check is for the case `a if a = 1`
msg << colorize(" (If you declared '#{def_name}' in a suffix if, declare it in a regular if for this to work. If the variable was declared in a macro it's not visible outside it)").yellow.bold
msg << " (If you declared '#{def_name}' in a suffix if, declare it in a regular if for this to work. If the variable was declared in a macro it's not visible outside it)".colorize.yellow.bold
else
msg << colorize(" (did you mean '#{similar_name}'?)").yellow.bold
msg << " (did you mean '#{similar_name}'?)".colorize.yellow.bold
end
end

Expand All @@ -121,9 +121,9 @@ class Crystal::Call
if deps && deps.size == 1 && deps.first.same?(program.nil_var)
similar_name = scope.lookup_similar_instance_var_name(ivar.name)
if similar_name
msg << colorize(" (#{ivar.name} was never assigned a value, did you mean #{similar_name}?)").yellow.bold
msg << " (#{ivar.name} was never assigned a value, did you mean #{similar_name}?)".colorize.yellow.bold
else
msg << colorize(" (#{ivar.name} was never assigned a value)").yellow.bold
msg << " (#{ivar.name} was never assigned a value)".colorize.yellow.bold
end
end
end
Expand All @@ -135,7 +135,7 @@ class Crystal::Call

# If it's on an initialize method and there's a similar method name, it's probably a typo
if (def_name == "initialize" || def_name == "new") && (similar_def = owner.instance_type.lookup_similar_def("initialize", self.args.size, block))
inner_msg = colorize("do you maybe have a typo in this '#{similar_def.name}' method?").yellow.bold.to_s
inner_msg = "do you maybe have a typo in this '#{similar_def.name}' method?".colorize.yellow.bold.to_s
inner_exception = TypeException.for_node(similar_def, inner_msg)
end

Expand Down Expand Up @@ -387,10 +387,10 @@ class Crystal::Call
str << "\n - "
append_def_full_name a_def.owner, a_def, arg_types, str
if defs.size > 1 && a_def.same?(matched_def)
str << colorize(" (trying this one)").blue
str << " (trying this one)".colorize.blue
end
if a_def.args.any? { |arg| arg.default_value && arg.external_name == argument_name }
str << colorize(" (did you mean this one?)").yellow.bold
str << " (did you mean this one?)".colorize.yellow.bold
end
end
end
Expand Down Expand Up @@ -550,7 +550,7 @@ class Crystal::Call
str << named_arg.name
str << "'"
if similar_name
str << colorize(" (did you mean '#{similar_name}'?)").yellow.bold
str << " (did you mean '#{similar_name}'?)".colorize.yellow.bold
end

defs = owner.lookup_defs(a_def.name)
Expand Down Expand Up @@ -652,8 +652,4 @@ class Crystal::Call
"#{owner}##{method_name}"
end
end

private def colorize(obj)
program.colorize(obj)
end
end
Loading

0 comments on commit d6c6e18

Please sign in to comment.