describe Appsignal::Extension do describe ".agent_config" do subject { Appsignal::Extension.agent_config } it { is_expected.to have_key("version") } it { is_expected.to have_key("triples") } end describe ".agent_version" do subject { Appsignal::Extension.agent_version } it { is_expected.to be_kind_of(String) } end context "when the extension library can be loaded" do subject { Appsignal::Extension } it "should indicate that the extension is loaded" do expect(Appsignal.extension_loaded?).to be_truthy end context "without valid config" do let(:out_stream) { std_stream } let(:output) { out_stream.read } describe ".start" do it "outputs a warning about not starting the extension" do capture_std_streams(out_stream, out_stream) do subject.start end expect(output).to match \ %r{ WARNING:\sError\swhen\sreading\sappsignal\sconfig,\s appsignal\s\(as\s(\d{2,4})/(\d{2,4})\)\snot\sstarting }x end end describe ".stop" do it "does nothing" do capture_std_streams(out_stream, out_stream) do subject.stop end expect(output).to be_empty end end end context "with a valid config" do before do build_config.write_to_environment end it "should have a start and stop method" do subject.start subject.stop end context "with a transaction" do subject { Appsignal::Extension.start_transaction("request_id", "http_request", 0) } it "should have a start_event method" do subject.start_event(0) end it "should have a finish_event method" do subject.finish_event("name", "title", "body", 0, 0) end it "should have a record_event method" do subject.record_event("name", "title", "body", 0, 1000, 1000) end it "should have a set_error method" do subject.set_error("name", "message", Appsignal::Extension.data_map_new) end it "should have a set_sample_data method" do subject.set_sample_data("params", Appsignal::Extension.data_map_new) end it "should have a set_action method" do subject.set_action("value") end it "should have a set_namespace method" do subject.set_namespace("value") end it "should have a set_queue_start method" do subject.set_queue_start(10) end it "should have a set_metadata method" do subject.set_metadata("key", "value") end it "should have a finish method" do subject.finish(0) end it "should have a duplicate method" do subject.duplicate("request_id") end it "should have a complete method" do subject.complete end end it "should have a set_gauge method" do subject.set_gauge("key", 1.0, Appsignal::Extension.data_map_new) end it "should have a increment_counter method" do subject.increment_counter("key", 1.0, Appsignal::Extension.data_map_new) end it "should have a add_distribution_value method" do subject.add_distribution_value("key", 1.0, Appsignal::Extension.data_map_new) end end end context "when the extension library cannot be loaded", :extension_installation_failure do let(:ext) { Appsignal::Extension } around do |example| Appsignal::Testing.without_testing { example.run } end it "should indicate that the extension is not loaded" do expect(Appsignal.extension_loaded?).to be_falsy end it "does not raise errors when methods are called" do ext.appsignal_start ext.something end describe Appsignal::Extension::MockData do it "does not error on missing data_map_new extension method calls" do map = ext.data_map_new expect(map).to be_kind_of(Appsignal::Extension::MockData) # Does not raise errors any arbitrary method call that does not exist map.set_string("key", "value") map.set_int("key", 123) map.something end it "does not error on missing data_array_new extension method calls" do array = ext.data_array_new expect(array).to be_kind_of(Appsignal::Extension::MockData) # Does not raise errors any arbitrary method call that does not exist array.append_string("value") array.append_int(123) array.something end end describe Appsignal::Extension::MockTransaction do it "does not error on missing transaction extension method calls" do transaction = described_class.new transaction.start_event(0) transaction.finish_event( "name", "title", "body", Appsignal::EventFormatter::DEFAULT, 0 ) transaction.something end end end end