# frozen_string_literal: true # Copyright, 2017, by Samuel G. D. Williams. # # Permission is hereby granted, free of charge, to any person obtaining a copy # of this software and associated documentation files (the "Software"), to deal # in the Software without restriction, including without limitation the rights # to use, copy, modify, merge, publish, distribute, sublicense, and/or sell # copies of the Software, and to permit persons to whom the Software is # furnished to do so, subject to the following conditions: # # The above copyright notice and this permission notice shall be included in # all copies or substantial portions of the Software. # # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR # IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, # FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE # AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER # LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, # OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN # THE SOFTWARE. require 'async/node' RSpec.describe Async::Node do describe '#parent=' do let(:child) {Async::Node.new(subject)} it "should construct nested tree" do expect(child.parent).to be subject expect(subject.children).to include(child) end it "should break nested tree" do child.parent = nil expect(child.parent).to be_nil expect(subject.children).to be_empty end it "can consume bottom to top" do child.consume expect(child.parent).to be_nil expect(subject.children).to be_empty end end describe '#print_hierarchy' do let(:buffer) {StringIO.new} let(:output) {buffer.string} let(:lines) {output.lines} let!(:child) {Async::Node.new(subject)} it "can print hierarchy to bufffer" do subject.print_hierarchy(buffer) expect(lines.size).to be 2 expect(lines[0]).to be =~ /#\n/ expect(lines[1]).to be =~ /\t#\n/ end end describe '#consume' do let(:middle) {Async::Node.new(subject)} let(:bottom) {Async::Node.new(middle)} it "can't consume middle node" do expect(bottom.parent).to be middle middle.consume expect(bottom.parent).to be middle end end describe '#annotate' do let(:annotation) {'reticulating splines'} it "should have no annotation by default" do expect(subject.annotation).to be_nil end it 'should output annotation when invoking #to_s' do subject.annotate(annotation) do expect(subject.to_s).to include(annotation) end end it 'can assign annotation' do subject.annotate(annotation) expect(subject.annotation).to be == annotation end end describe '#transient' do it 'can move transient child to parent' do # This example represents a persistent web connection (middle) with a background reader (child). We look at how when that connection goes out of scope, what happens to the child. # subject -> middle -> child (transient) middle = Async::Node.new(subject) child = Async::Node.new(middle, transient: true) expect(child).to be_transient expect(middle).to be_finished allow(child).to receive(:finished?).and_return(false) middle.consume # subject -> child (transient) expect(child.parent).to be subject expect(subject.children).to include(child) expect(subject.children).to_not include(middle) expect(child).to_not be_finished expect(subject).to be_finished expect(child).to receive(:stop) subject.stop end it 'can move transient sibling to parent' do # This example represents a server task (middle) which has a single task listening on incoming connections (child2), and a transient task which is monitoring those connections/some shared resource (child1). We look at what happens when the server listener finishes. # subject -> middle -> child1 (transient) # -> child2 middle = Async::Node.new(subject) child1 = Async::Node.new(middle, transient: true) child2 = Async::Node.new(middle) allow(child1).to receive(:finished?).and_return(false) middle.consume # subject -> middle -> child1 (transient) # -> child2 expect(child1.parent).to be middle expect(child2.parent).to be middle expect(middle.parent).to be subject expect(subject.children).to include(middle) expect(middle.children).to include(child1) expect(middle.children).to include(child2) child2.consume # subject -> child1 (transient) expect(child1.parent).to be subject expect(child2.parent).to be_nil expect(middle.parent).to be_nil expect(subject.children).to include(child1) expect(middle.children).to be_nil end it 'ignores non-transient children of transient parent' do # subject -> middle (transient) -> child middle = Async::Node.new(subject, transient: true) child = Async::Node.new(middle) allow(middle).to receive(:finished?).and_return(false) child.consume # subject -> middle (transient) expect(child.parent).to be_nil expect(middle.parent).to be subject expect(subject.children).to include(middle) expect(middle.children).to be_empty end end end