lib/terraform_landscape/output.rb



module TerraformLandscape
  # Encapsulates all communication to an output source.
  class Output
    # Whether colored output via ANSI escape sequences is enabled.
    # @return [true,false]
    attr_accessor :color_enabled

    # Creates a logger which outputs nothing.
    # @return [TerraformLandscape::Output]
    def self.silent
      new(File.open(File::NULL, 'w'))
    end

    # Creates a new {SlimLint::Logger} instance.
    #
    # @param out [IO] the output destination.
    def initialize(out)
      @out = out
      @color_enabled = tty?
    end

    # Print the specified output.
    #
    # @param output [String] the output to send
    # @param newline [true,false] whether to append a newline
    def puts(output, newline = true)
      @out.print(output)
      @out.print("\n") if newline
    end

    # Print the specified output without a newline.
    #
    # @param output [String] the output to send
    def print(output)
      puts(output, false)
    end

    # Print the specified output in bold face.
    # If output destination is not a TTY, behaves the same as {#log}.
    #
    # @param args [Array<String>]
    def bold(*args)
      color('1', *args)
    end

    # Print the specified output in a color indicative of error.
    # If output destination is not a TTY, behaves the same as {#log}.
    #
    # @param args [Array<String>]
    def error(*args)
      color(31, *args)
    end

    # Print the specified output in a bold face and color indicative of error.
    # If output destination is not a TTY, behaves the same as {#log}.
    #
    # @param args [Array<String>]
    def bold_error(*args)
      color('1;31', *args)
    end

    # Print the specified output in a color indicative of success.
    # If output destination is not a TTY, behaves the same as {#log}.
    #
    # @param args [Array<String>]
    def success(*args)
      color(32, *args)
    end

    # Print the specified output in a color indicative of a warning.
    # If output destination is not a TTY, behaves the same as {#log}.
    #
    # @param args [Array<String>]
    def warning(*args)
      color(33, *args)
    end

    # Print the specified output in a color indicating something worthy of
    # notice.
    # If output destination is not a TTY, behaves the same as {#log}.
    #
    # @param args [Array<String>]
    def notice(*args)
      color(35, *args)
    end

    # Print the specified output in a color indicating information.
    # If output destination is not a TTY, behaves the same as {#log}.
    #
    # @param args [Array<String>]
    def info(*args)
      color(36, *args)
    end

    # Print a blank line.
    def newline
      puts('')
    end

    # Whether this logger is outputting to a TTY.
    #
    # @return [true,false]
    def tty?
      @out.respond_to?(:tty?) && @out.tty?
    end

    # Connect directly to a readable stream
    def write_from(other_io)
      while (line = other_io.gets)
        print line
      end
    end

    private

    # Print output in the specified color.
    #
    # @param code [Integer,String] ANSI color code
    # @param output [String] output to print
    # @param newline [Boolean] whether to append a newline
    def color(code, output, newline = true)
      puts(color_enabled ? "\033[#{code}m#{output}\033[0m" : output, newline)
    end
  end
end