# -*- ruby -*- # vim: set nosta noet ts=4 sw=4: # frozen_string_literal: true require_relative 'helpers' require 'rspec' require 'loggability' require 'loggability/logger' describe Loggability do before( :each ) do Loggability.reset Loggability.configure end it "is itself a log host for the global logger" do expect( described_class.logger ).to be_a( Loggability::Logger ) expect( described_class.log_hosts ).to include( Loggability::GLOBAL_KEY => Loggability ) end describe "version methods" do it "returns a version string if asked" do expect( described_class.version_string ).to match( /\w+ [\d.]+/ ) end it "returns a version string with a build number if asked" do expect( described_class.version_string(true) ). to match(/\w+ [\d.]+ \(build [[:xdigit:]]+\)/) end end context "installed in a class as a log host" do before( :each ) do @class = Class.new do extend Loggability log_as :testing end end after( :each ) do Loggability.reset end it "is included in the list of log hosts" do expect( Loggability.log_hosts ).to include( :testing => @class ) end it "has an associated Loggability::Logger" do expect( @class.logger ).to be_a( Loggability::Logger ) end it "has an associated default Loggability::Logger" do expect( @class.default_logger ).to be( @class.logger ) end it "registers itself with the Loggability module" do expect( Loggability[@class] ).to be( @class.logger ) end it "registers its key with the Loggability module" do expect( Loggability[:testing] ).to be( @class.logger ) end it "has a proxy for its logger in its instances" do expect( @class.new.log.logger ).to be( @class.logger ) end it "wraps Logger instances assigned as its logger in a Loggability::Logger" do logger = ::Logger.new( $stderr ) @class.logger = logger expect( @class.logger ).to be_a( Loggability::Logger ) @class.log.debug "This shouldn't raise." end end context "installed in a class as a logging client" do before( :each ) do @loghost = Module.new do extend Loggability log_as :testing end @class = Class.new do extend Loggability log_to :testing end end after( :each ) do Loggability.reset end it "has a proxy for its log host's logger" do expect( @class.log.logger ).to be( @loghost.logger ) end it "is associated with its log host's logger through the Loggability module" do expect( Loggability[@class] ).to be( @loghost.logger ) end it "has a proxy for its log host's logger available from its instances" do obj = @class.new expect( obj.log.logger ).to be( @loghost.logger ) end it "creates a new log proxy when duped" do obj = @class.new expect( obj.log ).to_not be( obj.dup.log ) end it "is associated with its log host's logger via its instances through the Loggability module" do obj = @class.new expect( Loggability[obj] ).to be( @loghost.logger ) end it "propagates its log host key to subclasses" do subclass = Class.new( @class ) expect( subclass.log.logger ).to be( @loghost.logger ) expect( Loggability[subclass] ).to be( @loghost.logger ) end it "raises an exception if asked for a log host that hasn't yet been declared" do logged_class = Class.new { extend Loggability; log_to :the_void } expect { logged_class.log.logger }.to raise_error( ArgumentError, /no log host/i ) end end context "aggregate methods" do before( :each ) do Loggability.reset end let!( :loghost ) do Class.new do extend Loggability log_as :testing end end let!( :another_loghost ) do Class.new do extend Loggability log_as :testing_another end end it "can propagate a logging level to every loghost" do Loggability.level = :warn expect( Loggability[loghost].level ).to be( :warn ) end it "can propagate an outputter to every loghost" do Loggability.output_to( $stdout ) expect( Loggability[loghost].logdev.dev ).to be( $stdout ) end it "can propagate a formatter to every loghost" do Loggability.format_with( :color ) expect( Loggability[loghost].formatter ).to be_a( Loggability::Formatter::Color ) end describe "overrideable behaviors" do before( :each ) do @default_output = [] Loggability.level = :info Loggability.output_to( @default_output ) end it "can temporarily override where Loggability outputs to while executing a block" do tmp_output = [] Loggability[ loghost ].info "Before the override" Loggability.outputting_to( tmp_output ) do Loggability[ loghost ].info "During the override" end Loggability[ loghost ].info "After the override" expect( @default_output.size ).to eq( 2 ) expect( tmp_output.size ).to eq( 1 ) end it "can return an object that overrides where Loggability outputs to for any block" do tmp_output = [] with_tmp_logging = Loggability.outputting_to( tmp_output ) Loggability[ loghost ].info "Before the overrides" with_tmp_logging.call do Loggability[ loghost ].info "During the first override" end Loggability[ loghost ].info "Between overrides" with_tmp_logging.call do Loggability[ loghost ].info "During the second override" end Loggability[ loghost ].info "After the overrides" expect( @default_output.size ).to eq( 3 ) expect( tmp_output.size ).to eq( 2 ) end it "can temporarily override what level Loggability logs at while executing a block" do Loggability[ loghost ].debug "Before the override" Loggability.with_level( :debug ) do Loggability[ loghost ].debug "During the override" end Loggability[ loghost ].debug "After the override" expect( @default_output.size ).to eq( 1 ) end it "can return an object that overrides what level Loggability logs at for any block" do with_debug_logging = Loggability.with_level( :debug ) Loggability[ loghost ].debug "Before the overrides" with_debug_logging.call do Loggability[ loghost ].debug "During the first override" end Loggability[ loghost ].debug "Between overrides" with_debug_logging.call do Loggability[ loghost ].debug "During the second override" end Loggability[ loghost ].debug "After the overrides" expect( @default_output.size ).to eq( 2 ) end it "can temporarily override what formatter loggers use while executing a block" do Loggability[ loghost ].info "Before the override" Loggability.formatted_with( :html ) do Loggability[ loghost ].info "During the override" end Loggability[ loghost ].info "After the override" expect( @default_output.size ).to eq( 3 ) expect( @default_output.grep(/