NAME
Hash::Layout - hashes with predefined levels, composite keys and default values
SYNOPSIS
use Hash::Layout;
# Create new Hash::Layout object with 3 levels and unique delimiters:
my $HL = Hash::Layout->new({
levels => [
{ delimiter => ':' },
{ delimiter => '/' },
{}, # <-- last level never has a delimiter
]
});
# load using actual hash structure:
$HL->load({
'*' => {
'*' => {
foo_rule => 'always deny',
blah => 'thing'
},
NewYork => {
foo_rule => 'prompt'
}
}
});
# load using composite keys:
$HL->load({
'Office:NewYork/foo_rule' => 'allow',
'Store:*/foo_rule' => 'other',
'Store:London/blah' => 'purple'
});
# load composite keys w/o values (uses default_value):
$HL->load(qw/baz:bool_key flag01/);
# get a copy of the hash data:
my $hash = $HL->Data;
# $hash now contains:
#
# {
# "*" => {
# "*" => {
# blah => "thing",
# flag01 => 1,
# foo_rule => "always deny"
# },
# NewYork => {
# foo_rule => "prompt"
# }
# },
# Office => {
# NewYork => {
# foo_rule => "allow"
# }
# },
# Store => {
# "*" => {
# foo_rule => "other"
# },
# London => {
# blah => "purple"
# }
# },
# baz => {
# "*" => {
# bool_key => 1
# }
# }
# }
#
# lookup values by composite keys:
$HL->lookup('*:*/foo_rule') # 'always deny'
$HL->lookup('foo_rule') # 'always deny'
$HL->lookup('ABC:XYZ/foo_rule') # 'always deny' # (virtual/fallback)
$HL->lookup('Lima/foo_rule') # 'always deny' # (virtual/fallback)
$HL->lookup('NewYork/foo_rule') # 'prompt'
$HL->lookup('Office:NewYork/foo_rule') # 'allow'
$HL->lookup('Store:foo_rule') # 'other'
$HL->lookup('baz:Anything/bool_key') # 1 # (virtual/fallback)
# lookup values by full/absolute paths:
$HL->lookup_path(qw/ABC XYZ foo_rule/) # 'always deny' # (virtual/fallback)
$HL->lookup_path(qw/Store * foo_rule/) # 'other'
DESCRIPTION
Hash::Layout
provides deep hashes with a predefined number of levels which you can access using special "composite keys". These are essentially string paths that inflate into actual hash keys according to the defined levels and delimiter mappings, which can be the same or different for each level. This is useful both for shorter keys as well as merge/fallback to default values, such as when defining overlapping configs ranging from broad to narrowing scope (see example in SYNOPIS above).
This module is general-purpose, but was written specifically for the flexible filter() feature of DBIx::Class::Schema::Diff, so refer to its documentation as well for a real-world example application. There are also lots of examples and use scenarios in the unit tests under t/
.
METHODS
new
Create a new Hash::Layout instance. The following build options are supported:
- levels
-
Required. ArrayRef of level config definitions, or a numeric number of levels for default level configs. Each level can define its own
delimiter
(except the last level) and list ofregistered_keys
, both of which are optional and determine how ambiguous/partial composite keys are resolved.Level-specific delimiters provide a mechanism to supply partial paths in composite keys but resolve to a specific level. The word/string to the left of a delimiter character that is specific to a given level is resolved as the key of that level, however, the correct path order is required (keys are only tokenized in order from left to right).
Specific strings can also be declared to belong to a particular level with
registered_keys
. This also only effects how ambiguity is resolved with partial composite keys. See also theno_fill
andno_pad
options.See the unit tests for examples of exactly how this works.
Internally, the level configs are coerced into Hash::Layout::Level objects.
For Hash::Layouts that don't need/want level-specific delimiters, or level-specific registered_keys, a simple integer value can be supplied instead for default level configs all using
/
as the delimiter.So, this:
my $HL = Hash::Layout->new({ levels => 5 });
Is equivalent to:
$HL = Hash::Layout->new({ levels => [ { delimiter => '/' } { delimiter => '/' } { delimiter => '/' } { delimiter => '/' } {} #<-- last level never has a delimiter ] });
levels
is the only required parameter. - default_value
-
Value to assign keys when supplied to
load()
as simple strings instead of key/value pairs. Defaults to the standard bool/true value of1
. - default_key
-
Value to use for the key for levels which are not specified, as well as the key to use for default/fallback when looking up non-existant keys (see also
lookup_mode
). Defaults to a single asterisk(*)
. - no_fill
-
If true, partial composite keys are not expanded with the default_key (in the middle) to fill to the last level. Defaults to 0.
- no_pad
-
If true, partial composite keys are not expanded with the default_key (at the front or middle) to fill to the last level.
no_pad
impliesno_fill
. Again, see the tests for a more complete explanation. Defaults to 0. - allow_deep_values
-
If true, values at the bottom level are allowed to be hashes, too, for the purposes of addressing the deeper paths using composite keys (see
deep_delimiter
below). Defaults to 1. - deep_delimiter
-
When
allow_deep_values
is enabled, the deep_delimiter character is used to resolve composite key mappings into the deep hash values (i.e. beyond the predefined levels). Must be different from the delimiter used by any of the levels. Defaults to a single dot(.)
.For example:
$HL->lookup('something/foo.deeper.hash.path')
- lookup_mode
-
One of either
get
,fallback
ormerge
. Infallback
mode, when a non-existent composite key is looked up, the value of the first closest found key path using default keys is returned instead ofundef
as is the case withget
mode.merge
mode is likefallback
mode, except hashref values are merged with matching default key paths which are also hashrefs. Defaults tomerge
. - enable_globmatch
-
If true, key value lookup calls will automatically try to match wildcard (text glob) expressions. Defaults to false (0).
clone
Returns a new/cloned Hash::Layout
instance
coerce
Dynamic method coerces supplied value into a new Hash::Layout
instance with a new set of loaded data. See unit tests for more info.
coercer
CodeRef wrapper around coerce()
, suitable for use in a Moo-compatible attribute declaration
load
Loads new data into the hash.
Data can be supplied as hashrefs with normal/local keys or composite keys, or both. Composite keys can also be supplied as sub-keys and are resolved relative to the location in which they appear as one would expect.
Composite keys can also be supplied as simple strings w/o corresponding values in which case their value is set to whatever default_value
is set to (which defaults to 1).
See the unit tests for more details and lots of examples of using load()
.
set
Simpler alternative to load()
. Expects exactly two arguments as standard key/values.
resolve_key_path
Converts a composite key string into its full path and returns it as a list. Called internally wherever composite keys are resolved.
path_to_composite_key
Inverse of resolve_key_path
; takes a path as a list and returns a single composite key string (i.e. joins using the delimiters for each level). Obviously, it only returns fully-qualified, non-ambiguous (not partial) composite keys.
exists
Returns true if the supplied composite key exists and false if it doesn't. Does not consider default/fallback key paths.
exists_path
Like exists()
, but requires the key to be supplied as a resolved/fully-qualified path as a list of arguments. Used internally by exists()
.
get
Retrieves the real value of the supplied composite key, or undef if it does not exist. Use exists()
to distinguish undef values. Does not consider default/fallback key paths (that is what lookup()
is for).
get_path
Like get()
, but requires the key to be supplied as a resolved/fully-qualified path as a list of arguments. Used internally by get()
.
lookup
Returns the value of the supplied composite key, falling back to default key paths if it does not exist, depending on the value of lookup_mode
.
If the lookup_mode is set to 'get'
, lookup() behaves exactly the same as get().
If the lookup_mode is set to 'fallback'
and the supplied key does not exist, lookup() will search the hierarchy of matching default key paths, returning the first value that exists.
If the lookup_mode is set to 'merge'
, lookup() behaves the same as it does in 'fallback'
mode for all non-hashref values. For hashref values, the hierarchy of default key paths is searched and all matches (that are themselves hashrefs), including the exact/lowest value itself, are merged and returned.
lookup_path
Like lookup()
, but requires the key to be supplied as a resolved/fully-qualified path as a list of arguments. Used internally by lookup()
.
lookup_leaf_path
Like lookup_path()
, but only returns the value if it is a "leaf" (i.e. not a hashref with deeper sub-values). Empty hashrefs ({}
) are also considered leaf values.
lookup_path_globmatch
Like lookup_path()
, but matches/returns by comparing a full composite key string with wildcard (text glob) expressions.
delete
Deletes the supplied composite key and returns the deleted value, or undef if it does not exist. Does not consider default/fallback key paths, or delete multiple items at once (e.g. like the Linux rm
command does with shell globs).
delete_path
Like delete()
, but requires the key to be supplied as a resolved/fully-qualified path as a list of arguments. Used internally by delete()
.
Data
Returns a read-only (i.e. cloned) copy of the full loaded hash structure.
num_levels
Returns the number of levels defined for this Hash::Layout
instance.
level_keys
Returns a hashref of all the keys that have been loaded/exist for the supplied level index (the first level is at index 0
).
def_key_bitmask_strings
Debug method. Returns a list of all the default key paths as a list of bitmasks (in binary/string form). Any key path which has at least one default key at any level is considered a default path and is indexed as a bitmask, with '1' values representing the default key position(s). For instance, the key path {*}{*}{foo_rule}
from the 3-level example from the SYNOPSIS is indexed as the bitmask 110
(6
in decimal).
These bitmasks are used internally to efficiently search for and properly order default key values for quick fallback/merge lookups, even when there are a very large number of levels (and thus very, VERY large number of possible default paths). That is why they are tracked and indexed ahead of time.
This is a debug method which should not be needed to be used for any production code. I decided to leave it in just to help document some of the internal workings of this module.
reset
Clears and removes all loaded data and resets internal key indexes and counters.
EXAMPLES
For more examples, see the following:
The SYNOPSIS
The unit tests in
t/
AUTHOR
Henry Van Styn <vanstyn@cpan.org>
COPYRIGHT AND LICENSE
This software is copyright (c) 2014 by IntelliTree Solutions llc.
This is free software; you can redistribute it and/or modify it under the same terms as the Perl 5 programming language system itself.