view lib/IMPL/Config/Bag.pm @ 418:3f38dabaf5cc ref20150831

sync
author cin
date Mon, 28 Dec 2015 15:11:35 +0300
parents 3ed0c58e9da3
children b0481c071bea
line wrap: on
line source

package IMPL::Config::Bag;
use strict;

use IMPL::lang qw(:base);
use IMPL::declare {
	base => [
		'IMPL::Object' => undef
	],
	props => [
		_parents   => '*rw',    # array of parent bags
		_parentRev => '*rw',    # the timestamp of the parent
		_sealed    => '*rw',    # the bag has descendants
		_cache     => '*rw',    # own or cached entries
		_timestamp => '*rw',
		_entries   => '*rw',    # each entry is represented by hash
		  # { isa => linear_isa_hash, owner => owner_of_the_entry, value => value }
	]
};

sub CTOR {
	my ( $this, $p ) = @_;

	if ($p) {
		$p->_Seal();
		my @parents;
		push @parents, @{ $p->{$_parents} } if $p->{$_parents};
		push @parents, $p;
		$this->{$_parents}   = \@parents;
		$this->{$_parentRev} = $p->{$_timestamp};
	}

	$this->{$_timestamp} = 0;
	$this->{$_cache}     = {};
	$this->{$_entries}   = [];
}

sub GetParent {
	my ($this) = @_;

	$this->{$_parents} && $this->{$_parents}[ @{ $this->{$_parents} } - 1 ];
}

sub _Seal {
	unless ( $_[0]->{$_sealed} ) {
		$_[0]->{$_sealed}    = 1;
		$_[0]->{$_timestamp} = 0;    # from now the timestamp is important
	}
}

sub _Validate {
	my ($this) = @_;

	my $chain = $this->{$_parents}
	  or return 1;

	my $rev = 0;    # rev 0 means that parent was never modified
	                # this allows to made more efficient checks
	my $flush;

	foreach my $bag ( @$chain, $this ) {

		# we need to updated all bags after the first change was detected;
		if ( $flush ||= $rev and $bag->{$_parentRev} != $rev ) {
			$bag->{$_cache}     = {};
			$bag->{$_parentRev} = $rev;
		}
		$rev = $bag->{$_timestamp};
	}

	return $flush ? 0 : 1;
}

sub Resolve {
	my ( $this, $role ) = @_;

	die IMPL::InvalidArgumentException->new('role')
	  unless defined $role;

	if ( my $d = $this->_GetDescriptor($role) ) {
		return $d->{value};
	}
	else {
		return;
	}
}

sub _GetDescriptor {
	my ( $this, $role ) = @_;

	my $d = $this->{$_cache}{$role};

# return descriptor if this is own descriptor and its level is 1 (i.e. it can't be overriden by the parent cache)
# otherwise the cache must be validated
	return $d
	  if $d
	  and ( ( $d->{owner} == $this and $d->{isa}{$role} == 1 )
		or $this->_Validate() );

	# if there were no descriptor in cache we need to ensure that the cache
	# chain is valid before reolving starts
	$this->_Validate() unless $d;

	# the cache chain is valid
	# $d is not a valid descriptor

	$d = undef;
	my $prev;

	my $parents = $this->{$_parents};
	my @bags = $parents ? ( @$parents, $this ) : ($this);

	foreach my $bag (@bags) {

		# check the cache;
		unless ( my $t = $bag->{$_cache}{$role} ) {

	  # no cached entry this may be due cache flush
	  # go through own entries and find better entry than inherited from parents
			foreach my $entry ( @{ $bag->{$_entries} } ) {
				my $level = $entry->{isa}{$role};
				if ( $level and ( not($prev) or $level <= $prev ) ) {
					$d    = $entry;
					$prev = $level;
				}
			}

			#cache it
			$bag->{$_cache}{$role} = $d if $d;
		}
		else {
			$d    = $t;
			$prev = $d->{isa}{$role};
		}
	}

	return $d;
}

sub ResolveAll {
	my ( $this, $role ) = @_;

	return [
		map $_->{value},
		grep $_->{isa}{$role},
		map @{ $_->{$_entries} },
		@{ $this->{$_parents} || [] },
		$this
	];

}

sub Register {
	my ( $this, $isa, $value ) = @_;

	$isa = { $isa, 1 } unless ishash($isa);

	push @{ $this->{$_entries} },
	  { owner => $this, isa => $isa, value => $value };
	$this->{$_timestamp}++;

	delete $this->{$_cache}{$_} foreach keys %$isa;

	return $this;
}

1;