Rand Stats

Hash::Util

zef:lizmat

Actions Status

NAME

Raku port of Perl's Hash::Util module 0.23

SYNOPSIS

use Hash::Util <
  lock_hash unlock_hash lock_hash_recurse unlock_hash_recurse
  lock_keys lock_keys_plus unlock_keys
  lock_value unlock_value
  hash_locked hash_unlocked
  hidden_keys legal_keys all_keys
>;

my %hash = foo => 42, bar => 23;

# Ways to restrict a hash
lock_keys(%hash);
lock_keys(%hash, @keyset);
lock_keys_plus(%hash, @additional_keys);

# Ways to inspect the properties of a restricted hash
my @legal = legal_keys(%hash);
my @hidden = hidden_keys(%hash);
all_keys(%hash,@keys,@hidden);
my $is_locked = hash_locked(%hash);

# Remove restrictions on the hash
unlock_keys(%hash);

# Lock individual values in a hash
lock_value(  %hash, 'foo');
unlock_value(%hash, 'foo');

# Ways to change the restrictions on both keys and values
lock_hash  (%hash);
unlock_hash(%hash);

DESCRIPTION

This module tries to mimic the behaviour of Perl's Hash::Util module as closely as possible in the Raku Programming Language.

Hash::Util contains a set of functions that support restricted hashes. It introduces the ability to restrict a hash to a certain set of keys. No keys outside of this set can be added. It also introduces the ability to lock an individual key so it cannot be deleted and the ability to ensure that an individual value cannot be changed.

By default Hash::Util does not export anything.

MAYBE MAP IS ALL YOU NEED

If you want to use this module for the sole purpose of only once locking a hash into an immutable state (calling only lock_hash once on a hash), then it is much better to turn your hash into a Map upon initialization by adding the is Map trait:

my %hash is Map = foo => 42, bar => 23;

This will have exactly the same effect as:

my %hash = foo => 42, bar => 23;
lock_hash(%hash);

but won't need to load the Hash::Util module and will be much better performant because it won't need any additional run-time checks, because Map is the immutable version of Hash in Raku.

PORTING CAVEATS

Functions that pertain to the unique implementation of Perl hashes, have not been ported. These include:

hash_seed hash_value hv_store bucket_stats
bucket_info bucket_array hash_traversal_mask

Also field hashes (the tools to create inside-out objects) have not been ported is these were deemed rather useless in the Raku environment where everything is a true object. This pertains to the functions:

fieldhash fieldhashes

Since the concept of references does not exist as such in Raku, it didn't make sense to separately port the "_ref" versions of the subroutines. They are however available as aliases to the non "_ref" versions::

lock_hashref unlock_hashref lock_hashref_recurse unlock_hashref_recurse
lock_ref_keys lock_ref_keys_plus unlock_ref_keys
lock_ref_value unlock_ref_value
hashref_locked hashref_unlocked

FUNCTIONS

lock_keys HASH, [KEYS]

lock_keys(%hash);
lock_keys(%hash, @keys);

Restricts the given %hash's set of keys to @keys. If @keys is not given it restricts it to its current keyset. No more keys can be added. :delete and :exists will still work, but will not alter the set of allowed keys. Returns the hash it worked on.

unlock_keys HASH

unlock_keys(%hash);

Removes the restriction on the %hash's keyset.

Note that if any of the values of the hash have been locked they will not be unlocked after this sub executes. Returns the hash it worked on.

lock_keys_plus HASH, KEYS

lock_keys_plus(%hash,@additional_keys)

Similar to lock_keys, with the difference being that the optional key list specifies keys that may or may not be already in the hash. Essentially this is an easier way to say

lock_keys(%hash,@additional_keys,%hash.keys);

lock_value HASH, KEY

lock_value(%hash, $key);

Locks the value for an individual key of a hash. The value of a locked key cannot be changed. Unless %hash has already been locked the key/value could be deleted regardless of this setting. Returns the hash on which it operated.

unlock_value HASH, KEY

unlock_value(%hash, $key);

Unlocks the value for an individual key of a hash. Returns the hash on which it operated.

lock_hash HASH

lock_hash(%hash);

Locks an entire hash, making all keys and values read-only. No value can be changed, no keys can be added or deleted. Returns the hash it operated on. If you only want to lock a hash only once after it has been initialized, it's better to make it a Map:

my %hash is Map = foo => 42, bar => 23;

This will have the same effect as lock_hash, but will be much more performant as no extra overhead is needed for checking access at runtime.

unlock_hash HASH

unlock_hash(%hash);

Does the opposite of lock_hash. All keys and values are made writable. All values can be changed and keys can be added and deleted. Returns the hash it operated on.

lock_hash_recurse HASH

lock_hash_recurse(%hash);

Locks an entire hash and any hashes it references recursively, making all keys and values read-only. No value can be changed, no keys can be added or deleted. Returns the hash it originally operated on.

This method only recurses into hashes that are referenced by another hash. Thus a Hash of Hashes (HoH) will all be restricted, but a Hash of Arrays of Hashes (HoAoH) will only have the top hash restricted.

unlock_hash_recurse HASH

unlock_hash_recurse(%hash);

Does the opposite of lock_hash_recurse(). All keys and values are made writable. All values can be changed and keys can be added and deleted. Returns the hash it originally operated on.

Identical recursion restrictions apply as to lock_hash_recurse.

hash_locked HASH

say "Hash is locked!" if hash_locked(%hash);

Returns true if the hash and/or its keys are locked.

hash_unlocked HASH

say "Hash is unlocked!" if hash_unlocked(%hash);

Returns true if the hash and/or its keys are not locked.

my @legal = legal_keys(%hash);

Returns the list of the keys that are legal in a restricted hash. In the case of an unrestricted hash this is identical to calling %hash.keys.

hidden_keys HASH

my @hidden = hidden_keys(%hash);

Returns the list of the keys that are legal in a restricted hash but do not have a value associated to them. Thus if 'foo' is a "hidden" key of the %hash it will return False for both defined and :exists tests.

In the case of an unrestricted hash this will return an empty list.

all_keys HASH, VISIBLE, HIDDEN

all_keys(%hash,@visible,@hidden);

Populates the arrays @visible with the all the keys that would pass an exists tests, and populates @hidden with the remaining legal keys that have not been utilized. Returns the hash it operated on.

SEE ALSO

Scalar::Util, List::Util

AUTHOR

Elizabeth Mattijsen liz@raku.rocks

Source can be located at: https://github.com/lizmat/Hash-Util . Comments and Pull Requests are welcome.

COPYRIGHT AND LICENSE

Copyright 2018, 2019, 2020, 2021 Elizabeth Mattijsen

This library is free software; you can redistribute it and/or modify it under the Artistic License 2.0.

Re-imagined from the Perl version as part of the CPAN Butterfly Plan. Perl version originally developed by the Perl 5 Porters, subsequently maintained by Steve Hay.