NAME

Acme::Util - Frequently Hacked Functions

DESCRIPTION

A cookbook of tasty functions for the terminally lazy, impatient and hubristic

TAGS

Rather than listing a group of functions by name, related subs can be imported using the tag syntax:

use Acme::Util qw(:tag);

The following tags have been defined:

:io

appendfile, atime, ctime, mtime, reader, readfile, writefile

:www

html, js, text, xml, xmlparse

:text

capitalize, commify, csv, ltrim, plural, rtrim, trim

:misc

any, apply, arrayref, coderef, find, hashref, scalarref, clone, id, respond, swap

:math

div, inf, infinity, isbig, isfloat, isinf, isnan, isneg, isnum, isuv

FUNCTIONS

any

usage

    # return 0 or 1
    my $flip = any();

    # return a random element of the ARRAY or a random HASH key
    my $el = any($hash_or_array_ref);

    # return a randomly selected list of ARRAY elements / HASH keys
    my @els = any($hash_or_array_ref, $number_of_items_to_pick);

    # as above, but return the elements as an ARRAY ref (if
    # $number_of_items_to_pick is greater than 1)
    my $els = any($hash_or_array_ref, $number_of_items_to_pick);

    # allow the same element to be picked multiple times
    my @els = any($hash_or_array_ref, $number_of_items_to_pick,
	$with_replacement);
    my $els = any($hash_or_array_ref, $number_of_items_to_pick,
	$with_replacement);

    # return a random number between 0 and $integer_max inclusive
    my $int = any($integer_max);

    # return a random number between $integer_min and $integer_max
    # inclusive
    my $int = any($integer_min, $integer_max);

description

Returns a randomly chosen member of the referenced ARRAY, or a random key from the referenced HASH.

If the second argument is supplied, that number of items is chosen and returned as a list - defaults to 1.

If the third argument is supplied, items can be repeatedly chosen - defaults to 0 (false).

If the first argument is an integer x, an integer between 0 and x (inclusive) is returned.

If no argument is supplied, 0 or 1 is returned randomly.

If the first and second arguments are integers x and y, then an integer between x and y (inclusive) is returned.

any($ref, ...) does not modify the original HASH/ARRAY.

any($ref, $reflength) returns a shuffled list e.g.

# perform a non-destructive shuffle of the elements of $array
my @shuffled = any($array, $#$array + 1)

When supplied with a HASH or ARRAY ref, and a second argument (specifying the number of items to pick) greater than 1, the selected values are returned as a list in list context and an ARRAY ref in scalar context.

appendfile

usage

appendfile($file, $data, %args);

description

This is a simple wrapper for writefile()'s APPEND option.

Appends $data to the file whose path is specified by $file.

See writefile() for %args options.

apply

usage

    my $result = apply $coderef, @optional_args;

	# or 

    my $result = apply sub { do_something() }, @optional_args;

	# or 

    my $result = apply \&Name::Space::sub, @optional_args;

description

Invokes the subroutine supplied as the first argument, passing any arguments in @optional_args as parameters.

Returns the result of the subroutine call.

arrayref

usage

do_something() if (arrayref $arg);

description

Returns a true value if $arg is a blessed or unblessed ARRAY reference.

Returns a false value otherwise.

atime

usage

my $atime = atime($file);

description

Returns the time (in seconds since the epoch) the specified file was last accessed.

capitalize

usage

my $capitalized = capitalize($string);

# or 

my $capitalized = capitalize($string, @do_not_capitalize_these_words);

description

Initial-capitalizes $string i.e. any words (defined as consecutive characters between word-boundaries (\b)) in $string that don't already contain capitals are lowercased and their initials are uppercased.

Note: apostrophes are treated as word characters, so:

"There's more than one way to do it"

becomes:

    "There's More Than One Way To Do It"
	

rather than:

"There'S More Than One Way To Do It"

Any arguments supplied after $string are treated as exceptions and left as is.

For non-trivial capitalization see the { case => 'highlight' } option of Damian Conway's Text::Autoformat.

In the absence of an explicit exception list, Acme::Util's capitalize() (and Text::Autoformat's { case => 'title' }) mechanically renders the text:

'what i did on my summer vacation in monterey'

as:

'What I Did On My Summer Vacation In Monterey'

Whereas Text::Autoformat's { case => 'highlight' } option offers the much more titular:

'What I Did on my Summer Vacation in Monterey'

clone

usage

    use Acme::Util qw(clone);

    $a = Foo->new();
    $b = { alpha => 'beta', gamma => 'vlissides' };

    tie %c, 'Foo::Bar';

    $d = clone($a);
    $e = clone($b);
    $f = clone(\%c);

    # or

    my $node2 = {
	name	    => 'node2',
	children    => [ $node3, $node4 ],
	parent	    => weaken ($node1)	    # weaken() to avoid memory leak
    };

    # clone $node2 but preserve the original $node1 (rather than cloning
    # through it all the way to the root)

    my $clone = clone($node2, [ $node1 ]);

	# or, equivalently

    my $clone = clone($node2, [ $node2->{parent} ]);

description

clone() returns a recursive copy of its argument, which can be an arbitrary (scalar) type including nested HASH, ARRAY and reference types (including weak references), tied variables and objects.

To duplicate non-scalar types (e.g. lists, ARRAYs and HASHes), pass them to clone() by reference. e.g.

my $copy = clone (\@array);

# or

my %copy = %{ clone (\%hash) };

clone() takes an optional second argument: a reference to an ARRAY containing a list of exceptions i.e. values that should be 'passed-thru' verbatim. This is useful for, amongst other things, cloning nodes in a hierarchy without duplicating the structure all the way to the root.

For a slower, but more flexible solution see Storable's dclone().

coderef

usage

do_something() if (coderef $arg);

description

Returns a true value if $arg is a blessed or unblessed CODE reference.

Returns a false value otherwise.

commify

usage

my $pretty = commify($int);

description

Returns a reader-friendly representation of the supplied integer punctuated with commas at the customary thousands/millions (&c.) points.

Thus:

printf "%-9u\t%11s%s", $_, commify($_), $/
    for (map { $_ x $_ } 1 .. 9);

prints:

1                     1
22                   22
333                 333
4444              4,444
55555            55,555
666666          666,666
7777777       7,777,777
88888888     88,888,888
999999999   999,999,999

csv

usage

    my $csv = csv($fields, $arrayref);

	# or

    my $csv = csv [ 'alpha', 'beta', 'gamma', 'vlissides' ],
	[
	    { alpha => 'fee',   beta => 'fie',    gamma => 'foe',   vlissides => 'fum'    }, 
	    { alpha => 'foo',	beta => 'bar',	  gamma => 'baz',   vlissides => 'foobar'  }, 
	    { alpha => 'one',   beta => 'two',    gamma => 'three', vlissides => 'four' }
	];

	# or

     my $csv = csv [ 'alpha', 'beta', 'gamma', 'vlissides' ],
	[
	    [ qw(fee fie foe fum) ],
	    [ qw(foo bar baz foobar) ],
	    [ qw(one two three four) ],
	];

    # yields

    "alpha","beta","gamma","delta"
    "fee","fie","foe","fum"
    "foo","bar","baz","foobar"
    "one","two","three","four"

description

$fields:	an ARRAY ref representing a list of field names
$arrayref:	a reference to an ARRAY of HASH or ARRAY references

Returns either a string or (in list context) a list representing the rows of a CSV (Comma Separated Values) document generated from the supplied HASH or ARRAY refs and using the fields listed in $fields as the field names.

Field names and values are double quoted.

ctime

usage

my $ctime = ctime($file);

description

Returns the time (in seconds since the epoch) the specified file was created.

div

usage

my ($quotient, $remainder) = div ($numerator, $denominator);

# e.g.

my ($q, $r) = div (13, 3);

# $q = 4, $r = 1:
# 13 ($numerator) = 4 ($quotient) x 3 ($denominator) + 1 ($remainder) 

description

Integer division operator: in list context, returns the quotient and remainder when the first operand ($numerator) is divided by the second ($denominator).

i.e.

$numerator = $quotient * $denominator + $remainder

In scalar context, returns just the quotient. To return the remainder, use %.

find

usage

    my $index = find ($arrayref, $scalar)

	# or

    my $index = find ($arrayref, $scalar, $from)

description

Returns the offset of $scalar within $arrayref, with the first position denoted by 0.

$scalar can be a number, string or reference.

-1 is returned if the item is not found.

If the third argument is supplied, the search begins at that index (the offset is still calculated from the beginning of the ARRAY ref).

hashref

usage

do_something() if (hashref $arg);

description

Returns a true value if $arg is a blessed or unblessed HASH reference.

Returns a false value otherwise.

html

usage

    html ($text);

	# or

    html ();

description

Returns $text with the 'text/html' Content-type header prefixed; or just the header if no $text is provided.

Prints the prefixed page out directly if called in void context.

id

usage

id()

description

Returns a quick 'n' dirty Unique Identifier. Uses $$ (amongst other things), so not necessarily reliable under SpeedyCGI, mod_perl &c.

infinity inf

usage

my $inf = infinity()

    # or

my $inf = inf()

description

Perl 5.8 claims to support infinity natively, but falls short on many platforms. This utility function is a trivial wrapper for the Swiss-Army $Inf provided by Math::Complex, which jumps through every possible hoop to deliver a high-quality infinity product for your reckoning pleasure.

isnum

usage

isnum ($val)

description

Returns a nonzero value (indicating the numeric type) if $val is a number.

The numeric types are a conjunction of the following flags:

0x01  IS_NUMBER_IN_UV		(number within UV range - maybe not int)
0x02  IS_NUMBER_GREATER_THAN_UV_MAX (the pointed-to UV is undefined)
0x04  IS_NUMBER_NOT_INT		(saw . or E notation)
0x08  IS_NUMBER_NEG			(leading minus sign)
0x10  IS_NUMBER_INFINITY		(this is big)
0x20  IS_NUMBER_NAN			(this is not)

Rather than obliging the user to twiddle with bits, the following flavours of isnum (corresponding to the flags above) are also available:

isint
isuv
isbig
isfloat
isneg
isinf
isnan

isint returns -1 if its operand is a negative integer, 1 if it's 0 or a positive integer, and 0 otherwise.

The others always return 1 or 0.

js

usage

    js ($text);

	# or

    js ();

description

Returns the JavaScript in $text with the 'application/x-javascript' Content-type header prefixed; or just the header if no $text is provided.

Prints the prefixed page out directly if called in void context

ltrim

usage

my $trimmed = ltrim ($string);

description

Returns a copy of $string with whitespace removed from the beginning.

mtime

usage

my $mtime = mtime($file);

description

Returns the time (in seconds since the epoch) the specified file was last modified.

plural

usage

    my $plural = plural($stem, $count);

	# or

    my $plural = plural($stem, $count, $plural);

description

Plural() takes a singular word or word-stem as an argument; it evaluates $count to see if it is equal to 1; if it is, $stem is returned unchanged; otherwise $stem is pluralised by adding $plural, or 's' if $plural hasn't been supplied.

Thus:

my $plural = plural('error', $error);

will return:

    'errors' if $error == 0
    'error'  if $error == 1
    'errors' if $error >  1

This simple implementation does not support irregular plurals that modify the stem. For solutions to that, and indeed, most other non-trivial pluralization problems, the reader is referred to Damian Conway's Lingua::EN::Inflect.

reader

usage

    my $source = '';
    my $read = reader($path, IRS => '...', CHOMP => 1);

    while ($read->(\$source)) {
	do_something_with($source);
    }

	# or

    while ($read->()) { # implied target: $_
	do_something_with($_);
    }

description

This method implements a generator/continuation interface to the fine art of file slurpage. It provides a private (lexically scoped) filehandle and an associated file reader (in the form of a closure).

This closure should be called with a reference to the variable one wishes to be assigned the next line from the file.

If no argument is supplied then $_ is assumed to be the target.

The generator yields true while the file is slurping, and undef thereafter.

The file is automatically closed either when the file has been read, or the closure goes out of scope - whichever comes first.

readfile

description

Swiss-Army Slurp

usage

    # vanilla
    readfile $path; # print the file
	# or
    my $file = readfile $path;
	
    # handle warnings/fatal errors
    my $file = readfile $path, WARN => \&my_warn, DIE => sub { die @_ };
	
    # lines
    my @file = readfile $path;
	
    # set Input Record Separator
    my @file = readfile $path, IRS => '...';
	
    # strip Input Record Separator from result
    my @file = readfile $path, IRS => '...', CHOMP => 1;

    # open a binary file (e.g. on Windows/Cygwin)
    my $file = readfile $path, BINARY => 1;
	
    # all together now...
    my $file = readfile (
	$path,
	WARN	=> $warn,
	DIE	=> $die,
	IRS	=> '...',
	BINARY  => 1,
	CHOMP	=> 1);

    # synonyms for IRS	    => DELIM, SPLIT, DELIMITER
    # synonym  for BINARY   => BINMODE

respond

usage

respond ($scalar)
respond (@list)

description

respond() performs a context-sensitive return:

    In void context the supplied arguments are printed.

    In scalar context it returns:
    
	the supplied argument if there's only one
	the concatenation of its arguments if there's more than one

    In list context it returns the arguments unchanged.

rtrim

usage

my $trimmed = rtrim ($string);

description

Returns a copy of $string with whitespace removed from the end.

scalarref

usage

do_something() if (scalarref $arg);

description

Returns a true value if $arg is a blessed or unblessed SCALAR reference.

Returns a false value otherwise.

squash

usage

my $text = squash ($string);

description

Returns a 'flat' copy of $string i.e. with initial and terminal whitespace removed and one or more internal whitespace characters (including carriage-returns and newlines) squashed into a single space.

swap

usage

swap($x, $y);

description

Sets the value of $x to $y and vice-versa.

swap() is prototyped to receive its arguments by reference, and modifies its arguments in place like chop, chomp &c.

No value is returned

text

usage

    text ($text);

	# or

    text ();

description

Returns $text with the 'text/plain' Content-type header prefixed; or just the header if no $text is provided.

Prints the prefixed page out directly if called in void context.

trim

usage

my $trimmed = trim ($string);

description

Returns a copy of $string with whitespace removed from the beginning and end; in addition, consecutive internal spaces are squashed to a single space.

urlize

usage:

my $url = urlize('Foo: BAR baz'); # returns 'foo_bar_baz'

# or

my $url = urlize('Foo - BAR - baz', 'html'); # returns 'foo_bar_baz.html'

description:

Makes its text argument URL-friendly.

Returns the first argument lowercased with any consecutive non-alphanumeric characters replaced by an underscore.

If the optional second argument is provided, this is appended as an extension prefixed by '.'

writefile

usage

writefile($file, $data, %args);

description

Write $data to filename $file. Additional herbs and spices are specified as a list of pairs, in the same manner as readfile (and reader).

Currently, the following options are supported:

APPEND => 1 for append (as opposed to truncate) MODE => $mode (to roll your own file access mode) BINARY/BINMODE => 1 to write a binary file (e.g. on Windows/Cygwin)

xml

usage

    xml ($text);

	# or

    xml ();

description

Returns (optional) $text with the 'text/xml' Content-type header prefixed; or just the header if no $text is provided.

Prints the prefixed page out directly if called in void context.

xmlparse

usage

xmlparse ($parser, $xml_path_or_data);

description

Convenience wrapper for XML::Parser (or XML::Parser::Expat - or indeed any parser that supports parse() and parsefile()) that is agnostic with regard to whether $xml is a file/filehandle or raw XML text.

The $parser should be prefabricated according to taste.

BUGS

clone() currently segfaults if it encounters a Regex object

SEE ALSO

Scalar::Util, List::Util, Clone, Storable, File::Butler, Toolbox

AUTHOR

chocolateboy: <chocolate.boy@email.com>

COPYRIGHT

Copyright (c) 2001-2003, chocolateboy.

This module is free software. It may be used, redistributed and/or modified under the same terms as Perl itself.