Re: Setting delegated attributes in constructors

2018-02-13 Thread Buddy Burden

Merlyn,

I guess the problem here is that Moose doesn't really call the accessors in the constructor but just uses some shortcut to 
copy the constructor parameters to the object.


While I fully admit that I've not been following this discussion closely, whenever I've seen a statement such as this one in 
the past, the answer usually comes back "take a look at triggers."  No idea if that's helpful or not, but I thought I'd throw 
it out there in case it was. :-)


Luck!


-- Buddy


Re: Setting delegated attributes in constructors

2018-02-13 Thread Merlyn Kline
Thanks for quick and helpful replies. I'm not really trying to subvert
inheritance though I now see that my simplified example looks much like
that. I'd contemplated the BUILD(ARGS) route but couldn't see exactly how
to do it without losing some of the validation benefits (again, not really
visible in my simplified example). In my real code the composed in object
that I'm trying to delegate to is actually a lazily built DBIx result
object, FWIW.

I guess the problem here is that Moose doesn't really call the accessors in
the constructor but just uses some shortcut to copy the constructor
parameters to the object.

I think I'll look for a different solution. Or just stick with what I've
got now, where I call the accessors immediately after construction. Looks
ugly but there you go.

Thanks again,

Merlyn

On Tue, 13 Feb 2018 at 16:18 Harald Jörg  wrote:

> Merlyn Kline  writes:
>
> > I'd like to set delegated attributes in my object constructors
> > (e.g. as below) but this doesn't seem to work.  Am I missing
> > something?
>
> perldoc Moose::Manual::Delegation calls it "Attribute delegation", but
> actually you're not delegating attributes. Instead, you are delegating
> method calls to an attribute of your class.
>
> So this is what you are missing: Calling Class->new simply isn't calling
> $object->thing underneath.  If you add MooseX::StrictConstructor to your
> package Class, it will barf at you saying
>
>Found unknown attribute(s) init_arg passed to the constructor: thing
>
> BUILD or BUILDARGS can help to achieve what you're trying to do.
> --
> Cheers,
> haj
>
-- 

-- 
Merlyn


Re: Setting delegated attributes in constructors

2018-02-13 Thread Harald Jörg
Merlyn Kline  writes:

> I'd like to set delegated attributes in my object constructors
> (e.g. as below) but this doesn't seem to work.  Am I missing
> something?

perldoc Moose::Manual::Delegation calls it "Attribute delegation", but
actually you're not delegating attributes. Instead, you are delegating
method calls to an attribute of your class.

So this is what you are missing: Calling Class->new simply isn't calling
$object->thing underneath.  If you add MooseX::StrictConstructor to your
package Class, it will barf at you saying

   Found unknown attribute(s) init_arg passed to the constructor: thing

BUILD or BUILDARGS can help to achieve what you're trying to do.
--
Cheers,
haj


Re: Setting delegated attributes in constructors

2018-02-13 Thread Nick Perez
Delegated methods are typically used as a convenience technique to
class composition to avoid writing your own wrapper methods that call
the underlying methods the contained object has. It really wasn't
intended to subvert inheritance like you're trying to do.

HTH,
Nick




On Tue, 13 Feb 2018 10:17:19 +
Merlyn Kline  wrote:

> I'd like to set delegated attributes in my object constructors (e.g.
> as below)  but this doesn't seem to work. Am I missing something?
> 
> Merlyn
> 
> 
> #!/usr/bin/perl
> 
> package SubClass;
> 
> use Moose;
> 
> has thing => ( is => 'rw', isa => 'Int' );
> 
> before thing => sub {
> shift;
> print "subthing: ", @_ ? 'set to ' . (shift // '') :
> '(read)', "\n";
> return;
> };
> 
> 
> package Class;
> 
> use Moose;
> 
> has sub_object => ( is => 'rw', isa => 'SubClass', handles =>
> ['thing'], builder => '_build_sub_object' );
> #has sub_object => ( is => 'rw', isa => 'SubClass', handles =>
> ['thing'], default => sub { _build_sub_object(); } );
> 
> sub _build_sub_object {
> my $self = shift;
> 
> print "_build_sub_object\n";
> return SubClass->new;
> }
> 
> 
> package main;
> 
> use strict;
> use warnings;
> 
> print "Start\n";
> my $object = Class->new(thing => 1);
> print "Built\n";
> 
> print "Thing 1 = ", ($object->thing // ''), "\n";
> 
> $object->thing(2);
> 
> print "Thing 2 = ", ($object->thing // ''), "\n";
> 


Setting delegated attributes in constructors

2018-02-13 Thread Merlyn Kline
I'd like to set delegated attributes in my object constructors (e.g. as
below)  but this doesn't seem to work. Am I missing something?

Merlyn


#!/usr/bin/perl

package SubClass;

use Moose;

has thing => ( is => 'rw', isa => 'Int' );

before thing => sub {
shift;
print "subthing: ", @_ ? 'set to ' . (shift // '') : '(read)',
"\n";
return;
};


package Class;

use Moose;

has sub_object => ( is => 'rw', isa => 'SubClass', handles => ['thing'],
builder => '_build_sub_object' );
#has sub_object => ( is => 'rw', isa => 'SubClass', handles => ['thing'],
default => sub { _build_sub_object(); } );

sub _build_sub_object {
my $self = shift;

print "_build_sub_object\n";
return SubClass->new;
}


package main;

use strict;
use warnings;

print "Start\n";
my $object = Class->new(thing => 1);
print "Built\n";

print "Thing 1 = ", ($object->thing // ''), "\n";

$object->thing(2);

print "Thing 2 = ", ($object->thing // ''), "\n";

-- 

-- 
Merlyn