| line | stmt | bran | cond | sub | pod | time | code | 
| 1 |  |  |  |  |  |  | #======================================================================== | 
| 2 |  |  |  |  |  |  | # | 
| 3 |  |  |  |  |  |  | # Badger::Factory | 
| 4 |  |  |  |  |  |  | # | 
| 5 |  |  |  |  |  |  | # DESCRIPTION | 
| 6 |  |  |  |  |  |  | #   Factory module for loading and instantiating other modules. | 
| 7 |  |  |  |  |  |  | # | 
| 8 |  |  |  |  |  |  | # NOTE | 
| 9 |  |  |  |  |  |  | #   This module has grown organically to fit a number of (possibly | 
| 10 |  |  |  |  |  |  | #   conflicting) needs.  It needs to be completely refactored, and | 
| 11 |  |  |  |  |  |  | #   probably split into a number of different factory modules.  The | 
| 12 |  |  |  |  |  |  | #   TT3 code on which this was originally based had separate base class | 
| 13 |  |  |  |  |  |  | #   factory modules for modules (that just got loaded), objects (that | 
| 14 |  |  |  |  |  |  | #   got loaded and instantiated) and single objects (that got loaded, | 
| 15 |  |  |  |  |  |  | #   created and cached).  With hindsight, it was a mistake to try and | 
| 16 |  |  |  |  |  |  | #   cram all that functionality into one module.  It should be separated | 
| 17 |  |  |  |  |  |  | #   into a base class module/API and a number of specialised subclasses. | 
| 18 |  |  |  |  |  |  | # | 
| 19 |  |  |  |  |  |  | # AUTHOR | 
| 20 |  |  |  |  |  |  | #   Andy Wardley | 
| 21 |  |  |  |  |  |  | # | 
| 22 |  |  |  |  |  |  | #======================================================================== | 
| 23 |  |  |  |  |  |  |  | 
| 24 |  |  |  |  |  |  | package Badger::Factory; | 
| 25 |  |  |  |  |  |  |  | 
| 26 | 24 |  |  | 24 |  | 635 | use Badger::Debug ':dump'; | 
|  | 24 |  |  |  |  | 48 |  | 
|  | 24 |  |  |  |  | 150 |  | 
| 27 |  |  |  |  |  |  | use Badger::Class | 
| 28 | 24 |  |  |  |  | 373 | version   => 0.01, | 
| 29 |  |  |  |  |  |  | debug     => 0, | 
| 30 |  |  |  |  |  |  | base      => 'Badger::Prototype Badger::Exporter', | 
| 31 |  |  |  |  |  |  | import    => 'class', | 
| 32 |  |  |  |  |  |  | utils     => 'plural blessed textlike dotid camel_case', | 
| 33 |  |  |  |  |  |  | words     => 'ITEM ITEMS ISA', | 
| 34 |  |  |  |  |  |  | constants => 'PKG ARRAY HASH REFS ONCE DEFAULT LOADED', | 
| 35 |  |  |  |  |  |  | constant  => { | 
| 36 |  |  |  |  |  |  | OBJECT         => 'object', | 
| 37 |  |  |  |  |  |  | FOUND          => 'found', | 
| 38 |  |  |  |  |  |  | FOUND_REF      => 'found_ref', | 
| 39 |  |  |  |  |  |  | PATH_SUFFIX    => '_PATH', | 
| 40 |  |  |  |  |  |  | NAMES_SUFFIX   => '_NAMES', | 
| 41 |  |  |  |  |  |  | DEFAULT_SUFFIX => '_DEFAULT', | 
| 42 |  |  |  |  |  |  | }, | 
| 43 |  |  |  |  |  |  | messages  => { | 
| 44 |  |  |  |  |  |  | no_item    => 'No item(s) specified for factory to manage', | 
| 45 |  |  |  |  |  |  | no_default => 'No default defined for %s factory', | 
| 46 |  |  |  |  |  |  | bad_ref    => 'Invalid reference for %s factory item %s: %s', | 
| 47 |  |  |  |  |  |  | bad_method => qq{Can't locate object method "%s" via package "%s" at %s line %s}, | 
| 48 | 24 |  |  | 24 |  | 176 | }; | 
|  | 24 |  |  |  |  | 65 |  | 
| 49 |  |  |  |  |  |  |  | 
| 50 |  |  |  |  |  |  | our $RUNAWAY = 0; | 
| 51 |  |  |  |  |  |  | our $AUTOLOAD; | 
| 52 |  |  |  |  |  |  |  | 
| 53 |  |  |  |  |  |  | *init = \&init_factory; | 
| 54 |  |  |  |  |  |  |  | 
| 55 |  |  |  |  |  |  |  | 
| 56 |  |  |  |  |  |  | sub init_factory { | 
| 57 | 25 |  |  | 25 | 0 | 72 | my ($self, $config) = @_; | 
| 58 | 25 |  |  |  |  | 108 | my $class = $self->class; | 
| 59 | 25 |  |  |  |  | 73 | my ($item, $items, $path, $map, $default); | 
| 60 |  |  |  |  |  |  |  | 
| 61 |  |  |  |  |  |  | # 'item' and 'items' can be specified as config params or we look for | 
| 62 |  |  |  |  |  |  | # $ITEM and $ITEMS variables in the current package or those of any | 
| 63 |  |  |  |  |  |  | # base classes.  NOTE: $ITEM and $ITEMS must be in the same package | 
| 64 | 25 | 100 |  |  |  | 100 | unless ($item = $config->{ item }) { | 
| 65 | 24 |  |  |  |  | 196 | foreach my $pkg ($class->heritage) { | 
| 66 | 24 |  |  | 24 |  | 181 | no strict   REFS; | 
|  | 24 |  |  |  |  | 58 |  | 
|  | 24 |  |  |  |  | 999 |  | 
| 67 | 24 |  |  | 24 |  | 138 | no warnings ONCE; | 
|  | 24 |  |  |  |  | 50 |  | 
|  | 24 |  |  |  |  | 26400 |  | 
| 68 |  |  |  |  |  |  |  | 
| 69 | 24 | 50 |  |  |  | 56 | if (defined ($item = ${ $pkg.PKG.ITEM })) { | 
|  | 24 |  |  |  |  | 79 |  | 
| 70 | 24 |  |  |  |  | 66 | $items = ${ $pkg.PKG.ITEMS }; | 
|  | 24 |  |  |  |  | 63 |  | 
| 71 | 24 |  |  |  |  | 69 | last; | 
| 72 |  |  |  |  |  |  | } | 
| 73 |  |  |  |  |  |  | } | 
| 74 |  |  |  |  |  |  | } | 
| 75 | 25 | 50 |  |  |  | 76 | return $self->error_msg('no_item') | 
| 76 |  |  |  |  |  |  | unless $item; | 
| 77 |  |  |  |  |  |  |  | 
| 78 |  |  |  |  |  |  | # use 'items' in config, or grokked from $ITEMS, or guess plural | 
| 79 | 25 |  | 33 |  |  | 197 | $items = $config->{ items } || $items || plural($item); | 
| 80 |  |  |  |  |  |  |  | 
| 81 | 25 |  |  |  |  | 62 | my $ipath    = $item.PATH_SUFFIX; | 
| 82 | 25 |  |  |  |  | 57 | my $inames   = $item.NAMES_SUFFIX; | 
| 83 | 25 |  |  |  |  | 64 | my $idefault = $item.DEFAULT_SUFFIX; | 
| 84 |  |  |  |  |  |  |  | 
| 85 |  |  |  |  |  |  | # Merge all XXXX_PATH package vars with any 'xxxx_path' or 'path' config | 
| 86 |  |  |  |  |  |  | # items.  Ditto for XXXX_NAME / 'xxxx_name' / 'aka' and  XXXXS/ 'xxxxs' | 
| 87 |  |  |  |  |  |  |  | 
| 88 | 25 |  |  |  |  | 140 | my @path  = @$config{ path  => lc $ipath  }; | 
| 89 | 25 |  |  |  |  | 72 | my @names = @$config{ names => lc $inames }; | 
| 90 |  |  |  |  |  |  |  | 
| 91 | 25 |  |  |  |  | 175 | $self->{ path     } = $class->list_vars(uc $ipath, @path); | 
| 92 | 25 |  |  |  |  | 187 | $self->{ names    } = $class->hash_vars(uc $inames, @names); | 
| 93 | 25 |  |  |  |  | 137 | $self->{ $items   } = $class->hash_vars(uc $items, $config->{ $items }); | 
| 94 | 25 |  |  |  |  | 76 | $self->{ items    } = $items; | 
| 95 | 25 |  |  |  |  | 61 | $self->{ item     } = $item; | 
| 96 | 25 |  |  |  |  | 72 | $self->{ loaded   } = { }; | 
| 97 |  |  |  |  |  |  | $self->{ no_cache } = defined $config->{ no_cache }     # quick hack | 
| 98 |  |  |  |  |  |  | ? $config->{ no_cache } | 
| 99 | 25 | 50 | 50 |  |  | 205 | : $class->any_var('NO_CACHE') || 0; | 
| 100 |  |  |  |  |  |  |  | 
| 101 |  |  |  |  |  |  | # see if a 'xxxx_default' or 'default' configuration option is specified | 
| 102 |  |  |  |  |  |  | # or look for the first XXXX_DEFAULT or DEFAULT package variable. | 
| 103 |  |  |  |  |  |  | $default = $config->{ $idefault } | 
| 104 |  |  |  |  |  |  | || $config->{ default } | 
| 105 | 25 |  | 33 |  |  | 395 | || $class->any_var_in( uc $idefault, uc DEFAULT ); | 
| 106 | 25 | 50 |  |  |  | 84 | if ($default) { | 
| 107 | 0 |  |  |  |  | 0 | $self->debug("Setting default to $default") if DEBUG; | 
| 108 | 0 |  |  |  |  | 0 | $self->{ default } = $self->{ names }->{ default } = $default; | 
| 109 |  |  |  |  |  |  | } | 
| 110 |  |  |  |  |  |  |  | 
| 111 |  |  |  |  |  |  | $self->debug( | 
| 112 |  |  |  |  |  |  | "Initialised $item/$items factory\n", | 
| 113 |  |  |  |  |  |  | " Path: ", $self->dump_data($self->{ path }), "\n", | 
| 114 |  |  |  |  |  |  | "Names: ", $self->dump_data($self->{ names }) | 
| 115 | 25 |  |  |  |  | 40 | ) if DEBUG; | 
| 116 |  |  |  |  |  |  |  | 
| 117 | 25 |  |  |  |  | 100 | return $self; | 
| 118 |  |  |  |  |  |  | } | 
| 119 |  |  |  |  |  |  |  | 
| 120 |  |  |  |  |  |  | sub path { | 
| 121 | 46 |  |  | 46 | 1 | 119 | my $self = shift->prototype; | 
| 122 |  |  |  |  |  |  | return @_ | 
| 123 |  |  |  |  |  |  | ? ($self->{ path } = ref $_[0] eq ARRAY ? shift : [ @_ ]) | 
| 124 | 46 | 0 |  |  |  | 138 | :  $self->{ path }; | 
|  |  | 50 |  |  |  |  |  | 
| 125 |  |  |  |  |  |  | } | 
| 126 |  |  |  |  |  |  |  | 
| 127 |  |  |  |  |  |  | sub default { | 
| 128 | 1 |  |  | 1 | 1 | 3 | my $self = shift->prototype; | 
| 129 |  |  |  |  |  |  | return @_ | 
| 130 |  |  |  |  |  |  | ? ($self->{ default } = $self->{ names }->{ default } = shift) | 
| 131 | 1 | 50 |  |  |  | 7 | :  $self->{ default }; | 
| 132 |  |  |  |  |  |  | } | 
| 133 |  |  |  |  |  |  |  | 
| 134 |  |  |  |  |  |  | sub items { | 
| 135 | 0 |  |  | 0 | 1 | 0 | my $self  = shift->prototype; | 
| 136 | 0 |  |  |  |  | 0 | my $items = $self->{ $self->{ items } }; | 
| 137 | 0 | 0 |  |  |  | 0 | if (@_) { | 
| 138 | 0 | 0 |  |  |  | 0 | my $args = ref $_[0] eq HASH ? shift : { @_ }; | 
| 139 | 0 |  |  |  |  | 0 | @$items{ keys %$args } = values %$args; | 
| 140 |  |  |  |  |  |  | } | 
| 141 | 0 |  |  |  |  | 0 | return $items; | 
| 142 |  |  |  |  |  |  | } | 
| 143 |  |  |  |  |  |  |  | 
| 144 |  |  |  |  |  |  | sub item { | 
| 145 | 85 | 100 |  | 85 | 1 | 196 | my $self = shift; $self = $self->prototype unless ref $self; | 
|  | 85 |  |  |  |  | 240 |  | 
| 146 | 85 |  |  |  |  | 243 | my ($type, @args) = $self->type_args(@_); | 
| 147 |  |  |  |  |  |  |  | 
| 148 |  |  |  |  |  |  | # In most cases we're expecting $type to be a name (e.g. Table) which we | 
| 149 |  |  |  |  |  |  | # lookup in the items hash, or tack onto one of the module bases in the | 
| 150 |  |  |  |  |  |  | # path (e.g. Template::Plugin) to create a full module name which we load | 
| 151 |  |  |  |  |  |  | # and instantiate (e.g. Template::Plugin::Table).  However, the name might | 
| 152 |  |  |  |  |  |  | # be explicitly mapped to a  reference of some kind, or the $type passed | 
| 153 |  |  |  |  |  |  | # in could already be a reference (e.g. Template::TT2::Filters allow the | 
| 154 |  |  |  |  |  |  | # first argument to be a code ref or object which implements the required | 
| 155 |  |  |  |  |  |  | # filtering behaviour).  In which case, we bypass any name-based lookup | 
| 156 |  |  |  |  |  |  | # and skip straight onto the "look what I found!" phase | 
| 157 |  |  |  |  |  |  |  | 
| 158 | 85 | 100 |  |  |  | 264 | return $self->found($type, $type, \@args) | 
| 159 |  |  |  |  |  |  | unless textlike $type; | 
| 160 |  |  |  |  |  |  |  | 
| 161 | 83 |  |  |  |  | 149 | $type = $type . '';     # auto-stringify any textlike objects | 
| 162 |  |  |  |  |  |  |  | 
| 163 |  |  |  |  |  |  | # OK, so $type is a string.  We'll also create a canonical version of the | 
| 164 |  |  |  |  |  |  | # name (lower case dotted) to provide a case/syntax insensitve fallback | 
| 165 |  |  |  |  |  |  | # (e.g. so "foo.bar" can match against "Foo.Bar", "Foo::Bar" and so on) | 
| 166 |  |  |  |  |  |  |  | 
| 167 | 83 |  |  |  |  | 199 | my $items = $self->{ $self->{ items } }; | 
| 168 | 83 |  |  |  |  | 204 | my $canon = dotid $type; | 
| 169 |  |  |  |  |  |  |  | 
| 170 | 83 |  |  |  |  | 109 | $self->debug("Looking for '$type' or '$canon' in $self->{ items }") if DEBUG; | 
| 171 |  |  |  |  |  |  | #   $self->debug("types: ", $self->dump_data($self->{ types })) if DEBUG; | 
| 172 |  |  |  |  |  |  |  | 
| 173 |  |  |  |  |  |  | # false but defined entry indicates the item is not found | 
| 174 |  |  |  |  |  |  | return $self->not_found($type, \@args) | 
| 175 |  |  |  |  |  |  | if exists $items->{ $type } | 
| 176 | 83 | 50 | 66 |  |  | 345 | && not $items->{ $type }; | 
| 177 |  |  |  |  |  |  |  | 
| 178 |  |  |  |  |  |  | my $item = $items->{ $type  } | 
| 179 | 83 |  | 100 |  |  | 433 | || $items->{ $canon } | 
| 180 |  |  |  |  |  |  | # TODO: this needs to be defined-or, like // | 
| 181 |  |  |  |  |  |  | # Plugins can return an empty string to indicate that they | 
| 182 |  |  |  |  |  |  | # do nothing. | 
| 183 |  |  |  |  |  |  | # HMMM.... or does it? | 
| 184 |  |  |  |  |  |  | ||  $self->find($type, \@args) | 
| 185 |  |  |  |  |  |  | #            ||  $self->default($type, \@args) | 
| 186 |  |  |  |  |  |  | ||  return $self->not_found($type, \@args); | 
| 187 |  |  |  |  |  |  |  | 
| 188 |  |  |  |  |  |  | $items->{ $type } = $item | 
| 189 | 81 | 50 |  |  |  | 271 | unless $self->{ no_cache }; | 
| 190 |  |  |  |  |  |  |  | 
| 191 | 81 |  |  |  |  | 276 | return $self->found($type, $item, \@args); | 
| 192 |  |  |  |  |  |  | } | 
| 193 |  |  |  |  |  |  |  | 
| 194 |  |  |  |  |  |  | sub type_args { | 
| 195 |  |  |  |  |  |  | # Simple method to grok $type and @args from argument list.  The only | 
| 196 |  |  |  |  |  |  | # processing it does is to set $type to 'default' if it is undefined or | 
| 197 |  |  |  |  |  |  | # false. Subclasses can re-define this to insert their own type mapping or | 
| 198 |  |  |  |  |  |  | # argument munging, e.g. to inject values into the configuration params | 
| 199 |  |  |  |  |  |  | # for an object | 
| 200 | 65 |  |  | 65 | 1 | 89 | shift; | 
| 201 | 65 |  | 100 |  |  | 153 | my $type = shift || DEFAULT; | 
| 202 | 65 |  |  |  |  | 87 | my @args; | 
| 203 |  |  |  |  |  |  |  | 
| 204 | 65 | 50 |  |  |  | 122 | if (ref $type eq HASH) { | 
| 205 | 0 |  |  |  |  | 0 | @args = ($type, @_); | 
| 206 | 0 |  | 0 |  |  | 0 | $type = $type->{ type } || DEFAULT; | 
| 207 |  |  |  |  |  |  | } | 
| 208 |  |  |  |  |  |  | else { | 
| 209 | 65 |  |  |  |  | 131 | @args = @_; | 
| 210 |  |  |  |  |  |  | } | 
| 211 |  |  |  |  |  |  |  | 
| 212 | 65 |  |  |  |  | 145 | return ($type, @args); | 
| 213 |  |  |  |  |  |  | } | 
| 214 |  |  |  |  |  |  |  | 
| 215 |  |  |  |  |  |  | sub find { | 
| 216 | 46 |  |  | 46 | 1 | 80 | my $self   = shift; | 
| 217 | 46 |  |  |  |  | 74 | my $type   = shift; | 
| 218 | 46 |  |  |  |  | 145 | my $bases  = $self->path; | 
| 219 | 46 |  |  |  |  | 65 | my $module; | 
| 220 |  |  |  |  |  |  |  | 
| 221 |  |  |  |  |  |  | # run the type through the type map to handle any unusual capitalisation, | 
| 222 |  |  |  |  |  |  | # spelling, aliases, etc. | 
| 223 | 46 |  | 66 |  |  | 175 | $type = $self->{ names }->{ $type } || $type; | 
| 224 |  |  |  |  |  |  |  | 
| 225 | 46 |  |  |  |  | 86 | foreach my $base (@$bases) { | 
| 226 | 54 | 100 |  |  |  | 157 | return $module | 
| 227 |  |  |  |  |  |  | if $module = $self->load( $self->module_names($base, $type) ); | 
| 228 |  |  |  |  |  |  | } | 
| 229 |  |  |  |  |  |  |  | 
| 230 | 2 |  |  |  |  | 36 | return undef; | 
| 231 |  |  |  |  |  |  | } | 
| 232 |  |  |  |  |  |  |  | 
| 233 |  |  |  |  |  |  | sub load { | 
| 234 | 54 |  |  | 54 | 1 | 89 | my $self   = shift; | 
| 235 | 54 |  |  |  |  | 92 | my $loaded = $self->{ loaded }; | 
| 236 |  |  |  |  |  |  |  | 
| 237 | 54 |  |  |  |  | 99 | foreach my $module (@_) { | 
| 238 |  |  |  |  |  |  | # see if we've previously loaded a module with this name (true | 
| 239 |  |  |  |  |  |  | # value) or failed to load a module (defined but false value) | 
| 240 |  |  |  |  |  |  |  | 
| 241 | 64 | 100 |  |  |  | 217 | if ($loaded->{ $module }) { | 
|  |  | 50 |  |  |  |  |  | 
| 242 | 12 |  |  |  |  | 15 | $self->debug("$module has been previously loaded") if DEBUG; | 
| 243 | 12 |  |  |  |  | 59 | return $module; | 
| 244 |  |  |  |  |  |  | } | 
| 245 |  |  |  |  |  |  | elsif (defined $loaded->{ $module }) { | 
| 246 | 0 |  |  |  |  | 0 | next; | 
| 247 |  |  |  |  |  |  | } | 
| 248 |  |  |  |  |  |  |  | 
| 249 | 24 |  |  | 24 |  | 186 | no strict REFS; | 
|  | 24 |  |  |  |  | 62 |  | 
|  | 24 |  |  |  |  | 27140 |  | 
| 250 | 52 |  |  |  |  | 73 | $self->debug("attempting to load $module") if DEBUG; | 
| 251 |  |  |  |  |  |  |  | 
| 252 |  |  |  |  |  |  | # Some filesystems are case-insensitive (like Apple's HFS), so an | 
| 253 |  |  |  |  |  |  | # attempt to load Badger::Example::foo may succeed, when the correct | 
| 254 |  |  |  |  |  |  | # package name is actually Badger::Example::Foo.  We double-check | 
| 255 |  |  |  |  |  |  | # by looking for $VERSION or @ISA.  This is a bit dodgy because we might be | 
| 256 |  |  |  |  |  |  | # loading something that has no ISA.  Need to cross-check with | 
| 257 |  |  |  |  |  |  | # what's going on in Badger::Class _autoload() | 
| 258 |  |  |  |  |  |  |  | 
| 259 | 52 |  |  |  |  | 63 | my $loadname; | 
| 260 | 52 | 100 |  |  |  | 139 | if ( ($loadname = class($module)->maybe_load) ) { | 
| 261 | 32 |  |  |  |  | 48 | $self->debug("loaded $module") if DEBUG; | 
| 262 | 32 |  |  |  |  | 77 | $loaded->{ $module } = $loadname; | 
| 263 | 32 |  |  |  |  | 250 | return $module | 
| 264 |  |  |  |  |  |  | } | 
| 265 |  |  |  |  |  |  |  | 
| 266 | 20 |  |  |  |  | 41 | $self->debug("failed to load $module") if DEBUG; | 
| 267 |  |  |  |  |  |  | } | 
| 268 |  |  |  |  |  |  |  | 
| 269 | 10 |  |  |  |  | 32 | return undef; | 
| 270 |  |  |  |  |  |  | } | 
| 271 |  |  |  |  |  |  |  | 
| 272 |  |  |  |  |  |  |  | 
| 273 |  |  |  |  |  |  | sub found { | 
| 274 | 83 |  |  | 83 | 1 | 1503 | my ($self, $type, $item, $args) = @_; | 
| 275 |  |  |  |  |  |  |  | 
| 276 | 83 | 100 |  |  |  | 229 | if (ref $item) { | 
| 277 |  |  |  |  |  |  | # if it's a reference we found then forward it onto the appropriate | 
| 278 |  |  |  |  |  |  | # method, e.g found_array(), found_hash(), found_code().  Fall back | 
| 279 |  |  |  |  |  |  | # on found_ref() | 
| 280 | 29 | 100 |  |  |  | 135 | my $iref = blessed($item) | 
| 281 |  |  |  |  |  |  | ? OBJECT | 
| 282 |  |  |  |  |  |  | : lc ref $item; | 
| 283 |  |  |  |  |  |  |  | 
| 284 | 29 |  |  |  |  | 49 | $self->debug( | 
| 285 |  |  |  |  |  |  | "Looking for handler methods: ", | 
| 286 |  |  |  |  |  |  | FOUND,'_'.$iref, "() or ", | 
| 287 |  |  |  |  |  |  | FOUND_REF, "()" | 
| 288 |  |  |  |  |  |  | ) if DEBUG; | 
| 289 |  |  |  |  |  |  |  | 
| 290 |  |  |  |  |  |  | my $method | 
| 291 |  |  |  |  |  |  | = $self->can(FOUND . '_' . $iref) | 
| 292 |  |  |  |  |  |  | || $self->can(FOUND_REF) | 
| 293 | 29 |  | 50 |  |  | 104 | || return $self->error_msg( bad_ref => $self->{ item }, $type, $iref ); | 
| 294 |  |  |  |  |  |  |  | 
| 295 | 29 |  |  |  |  | 95 | $item = $method->($self, $type, $item, $args); | 
| 296 |  |  |  |  |  |  | } | 
| 297 |  |  |  |  |  |  | else { | 
| 298 |  |  |  |  |  |  | # otherwise it's the name of a module | 
| 299 | 54 |  |  |  |  | 222 | $item = $self->found_module($type, $item, $args); | 
| 300 |  |  |  |  |  |  | } | 
| 301 |  |  |  |  |  |  |  | 
| 302 |  |  |  |  |  |  | # NOTE: an item can be defined but false, e.g. a Template::Plugin which | 
| 303 |  |  |  |  |  |  | # return '' from its new() method to indicate it does nothing objecty | 
| 304 | 83 | 50 |  |  |  | 194 | return unless defined $item; | 
| 305 |  |  |  |  |  |  |  | 
| 306 | 83 |  |  |  |  | 97 | $self->debug("Found result: $type => $item") if DEBUG; | 
| 307 |  |  |  |  |  |  |  | 
| 308 |  |  |  |  |  |  | # TODO: what about caching result?  Do we always leave that to subclasses? | 
| 309 | 83 |  |  |  |  | 219 | return $self->result($type, $item, $args); | 
| 310 |  |  |  |  |  |  | } | 
| 311 |  |  |  |  |  |  |  | 
| 312 |  |  |  |  |  |  | sub found_module { | 
| 313 |  |  |  |  |  |  | # This method is called when a module name is found, either by being | 
| 314 |  |  |  |  |  |  | # predefined in the factory entry table, or loaded on demand from disk. | 
| 315 |  |  |  |  |  |  | # It ensures the module is loaded and and instantiates an object from the | 
| 316 |  |  |  |  |  |  | # class name | 
| 317 | 54 |  |  | 54 | 1 | 118 | my ($self, $type, $module, $args) = @_; | 
| 318 | 54 |  |  |  |  | 63 | $self->debug("Found module: $type => $module") if DEBUG; | 
| 319 | 54 |  | 66 |  |  | 235 | $self->{ loaded }->{ $module } ||= class($module)->load; | 
| 320 | 54 |  |  |  |  | 188 | return $self->construct($type, $module, $args); | 
| 321 |  |  |  |  |  |  | } | 
| 322 |  |  |  |  |  |  |  | 
| 323 |  |  |  |  |  |  | sub found_array { | 
| 324 |  |  |  |  |  |  | # This method is called when an ARRAY reference is found.  We assume that | 
| 325 |  |  |  |  |  |  | # the first item is the module name (which needs to be loaded) and the | 
| 326 |  |  |  |  |  |  | # second item is the class name (which needs to be instantiated). | 
| 327 | 4 |  |  | 4 | 1 | 13 | my ($self, $type, $item, $args) = @_; | 
| 328 | 4 |  |  |  |  | 11 | my ($module, $class) = @$item; | 
| 329 | 4 |  | 66 |  |  | 34 | $self->{ loaded }->{ $module } ||= class($module)->load; | 
| 330 | 4 |  |  |  |  | 30 | return $self->construct($type, $class, $args); | 
| 331 |  |  |  |  |  |  | } | 
| 332 |  |  |  |  |  |  |  | 
| 333 |  |  |  |  |  |  | sub not_found { | 
| 334 | 2 |  |  | 2 | 1 | 7 | my ($self, $type, @args) = @_; | 
| 335 |  |  |  |  |  |  |  | 
| 336 |  |  |  |  |  |  | return $type eq DEFAULT | 
| 337 |  |  |  |  |  |  | ? $self->error_msg( no_default => $self->{ item } ) | 
| 338 | 2 | 100 |  |  |  | 21 | : $self->error_msg( not_found => $self->{ item }, $type ); | 
| 339 |  |  |  |  |  |  | } | 
| 340 |  |  |  |  |  |  |  | 
| 341 |  |  |  |  |  |  | sub construct { | 
| 342 | 59 |  |  | 59 | 1 | 140 | my ($self, $type, $class, $args) = @_; | 
| 343 | 59 |  |  |  |  | 71 | $self->debug("constructing class: $type => $class") if DEBUG; | 
| 344 | 59 |  |  |  |  | 276 | return $class->new(@$args); | 
| 345 |  |  |  |  |  |  | } | 
| 346 |  |  |  |  |  |  |  | 
| 347 |  |  |  |  |  |  | sub module_names { | 
| 348 | 54 |  |  | 54 | 1 | 78 | my $self = shift; | 
| 349 |  |  |  |  |  |  | my @bits = | 
| 350 | 119 |  |  |  |  | 263 | map { camel_case($_) } | 
| 351 | 54 |  |  |  |  | 99 | map { split /[\.]+/ } @_; | 
|  | 108 |  |  |  |  | 305 |  | 
| 352 |  |  |  |  |  |  |  | 
| 353 |  |  |  |  |  |  | return ( | 
| 354 | 54 |  |  |  |  | 117 | join( PKG, map { ucfirst $_ } @bits ), | 
|  | 119 |  |  |  |  | 407 |  | 
| 355 |  |  |  |  |  |  | join( PKG, @bits ) | 
| 356 |  |  |  |  |  |  | ); | 
| 357 |  |  |  |  |  |  | } | 
| 358 |  |  |  |  |  |  |  | 
| 359 |  |  |  |  |  |  |  | 
| 360 |  |  |  |  |  |  | sub can { | 
| 361 | 35 |  |  | 35 | 1 | 78 | my ($self, $name) = @_; | 
| 362 |  |  |  |  |  |  |  | 
| 363 |  |  |  |  |  |  | # upgrade class methods to calls on prototype | 
| 364 | 35 | 50 |  |  |  | 79 | $self = $self->prototype unless ref $self; | 
| 365 |  |  |  |  |  |  |  | 
| 366 |  |  |  |  |  |  | # NOTE: this method can get called before we've called init_factory() | 
| 367 |  |  |  |  |  |  | # to define the item/items members, so we tread carefully. | 
| 368 | 35 | 50 | 33 |  |  | 257 | if ($self->{ item } && $self->{ item } eq $name) { | 
|  |  | 50 | 33 |  |  |  |  | 
| 369 | 0 |  |  |  |  | 0 | return $self->SUPER::can('item'); | 
| 370 |  |  |  |  |  |  | } | 
| 371 |  |  |  |  |  |  | elsif ($self->{ items } && $self->{ items } eq $name) { | 
| 372 | 0 |  |  |  |  | 0 | return $self->SUPER::can('items'); | 
| 373 |  |  |  |  |  |  | } | 
| 374 |  |  |  |  |  |  | else { | 
| 375 | 35 |  |  |  |  | 285 | return $self->SUPER::can($name); | 
| 376 |  |  |  |  |  |  | } | 
| 377 |  |  |  |  |  |  | } | 
| 378 |  |  |  |  |  |  |  | 
| 379 |  |  |  |  |  |  | sub result { | 
| 380 | 33 |  |  | 33 | 1 | 182 | $_[2]; | 
| 381 |  |  |  |  |  |  | } | 
| 382 |  |  |  |  |  |  |  | 
| 383 |  |  |  |  |  |  | sub AUTOLOAD { | 
| 384 | 12 |  |  | 12 |  | 108 | my ($self, @args) = @_; | 
| 385 | 12 |  |  |  |  | 101 | my ($name) = ($AUTOLOAD =~ /([^:]+)$/ ); | 
| 386 | 12 | 100 |  |  |  | 346 | return if $name eq 'DESTROY'; | 
| 387 |  |  |  |  |  |  |  | 
| 388 | 7 |  |  |  |  | 14 | $self->debug("AUTOLOAD $name") if DEBUG; | 
| 389 |  |  |  |  |  |  |  | 
| 390 | 7 |  |  |  |  | 17 | local $RUNAWAY = $RUNAWAY; | 
| 391 | 7 | 50 |  |  |  | 21 | $self->error("AUTOLOAD went runaway on $name") | 
| 392 |  |  |  |  |  |  | if ++$RUNAWAY > 10; | 
| 393 |  |  |  |  |  |  |  | 
| 394 |  |  |  |  |  |  | # upgrade class methods to calls on prototype | 
| 395 | 7 | 100 |  |  |  | 74 | $self = $self->prototype unless ref $self; | 
| 396 |  |  |  |  |  |  |  | 
| 397 | 7 |  |  |  |  | 13 | $self->debug("factory item: $self->{ item }") if DEBUG; | 
| 398 |  |  |  |  |  |  |  | 
| 399 | 7 | 100 |  |  |  | 57 | if ($name eq $self->{ item }) { | 
|  |  | 50 |  |  |  |  |  | 
|  |  | 50 |  |  |  |  |  | 
| 400 | 6 |  |  |  |  | 27 | $self->class->method( $name => $self->can('item') ); | 
| 401 |  |  |  |  |  |  | } | 
| 402 |  |  |  |  |  |  | elsif ($name eq $self->{ items }) { | 
| 403 | 0 |  |  |  |  | 0 | $self->class->method( $name => $self->can('items') ) | 
| 404 |  |  |  |  |  |  | } | 
| 405 |  |  |  |  |  |  | elsif (my $item = $self->try( item => $name, @args )) { | 
| 406 | 0 |  |  |  |  | 0 | return $item; | 
| 407 |  |  |  |  |  |  | } | 
| 408 |  |  |  |  |  |  | else { | 
| 409 | 1 |  |  |  |  | 5 | my ($pkg, $file, $line) = caller; | 
| 410 | 1 |  | 33 |  |  | 10 | my $class = ref $self || $self; | 
| 411 | 1 |  |  |  |  | 8 | die $self->message( bad_method => $name, $class, $file, $line ), "\n"; | 
| 412 |  |  |  |  |  |  | } | 
| 413 |  |  |  |  |  |  |  | 
| 414 |  |  |  |  |  |  | # should be installed now | 
| 415 | 6 |  |  |  |  | 24 | $self->$name(@args); | 
| 416 |  |  |  |  |  |  | } | 
| 417 |  |  |  |  |  |  |  | 
| 418 |  |  |  |  |  |  |  | 
| 419 |  |  |  |  |  |  | 1; | 
| 420 |  |  |  |  |  |  |  | 
| 421 |  |  |  |  |  |  | __END__ |