App-Kit
view release on metacpan or search on metacpan
lib/App/Kit.pm view on Meta::CPAN
=head1 DESCRIPTION
A Lazy Façade to simplify your code/life. How?
Ever see this sort of thing in a growing code base:
package My::Thing;
use strict;
use warnings;
use My::Logger;
my $logger;
sub foo {
my ($x, $y, $z) = @_;
if ($x) {
$logger ||= MyLogger->new;
$logger->info("X is truly $x");
}
â¦
}
but if that module (or script) had access to your App::Kit object:
package My::Thing;
## no critic (RequireUseStrict) - MyApp does strict and warnings
use MyApp;
my $app = MyApp->multiton; # ok to do this here because it is really really cheap
sub foo {
my ($x, $y, $z) = @_;
if ($x) {
$app->log->info("X is truly $x");
}
â¦
}
Multiply that by hundreds of scripts and modules and vars and tests and wrappers. ick
Some specifics:
=head2 one class/one object to consistently and simply manage some of the complexities of a large code base
Donât pass around globals, a zillion args, have a bunch of was-this-loaded, do-we-have-that, etc.
Just create the object and use it everywhere. Done, it all works the same with no boiler plate required besides Foo->instance and use it when you need.
=head2 only what you need, when you need it
Only load and instantiate the things your code actually does, with no effort.
Reuse them throughout the code base, again, with no effort!
=head2 use default objects or set your own
The defaults will work and as your project expands you can customize if needed without needing to refactor your code.
For example, once you sprint the localization setup, you can change your classâs locale() to use your object.
=head2 easy mocking (for your tests!)
By default the lazy façade methods are readonly (technically 'rwp' so the class can fiddle with them internally if it needs to).
You can change make them readwrite via either of 2 mechanisms before the class is built via use().
Either:
use App::Kit::Util::RW; # must be loaded before App::Kit is use()d
use App::Kit;
or
BEGIN { $ENV{'App-Kit-Util-RW'} = 1; }; # must be set before App::Kit is use()d
use App::Kit;
then:
$app->log($test_logger); # e.g. say $test_logger stores what level and msg are called/given
⦠something that should call $app->log->info('This is a terrible info msg.') â¦
⦠test that $test_logger saw the expected levels and msgs â¦
The former might be desirable if you want to keep ENV clean, the latter for when you want to do/skip certain tests based on if it is true or not:
App-Kit-Util-RW=1 prove -w t
=head1 INTERFACE
=head2 auto imports
=head3 strict and warnings enabled automatically
=head3 try/catch/finally imported automatically (unless you say not to)
L<Try::Tiny> is enabled automatically unless you pass import() â-no-tryâ flag (Yoda was right: there *is* -no-try!):
use App::Kit '-no-try';
same goes for your App::Kit based object:
use My::App '-no-try';
=head2 constructors: multiton support
=head3 new()
Since the idea of this class is to share the objects it makes more sense to use multiton()/instance() in your code.
Returns a new object every time. Takes no arguments currently.
=head3 multiton()
Returns the same object on subsequent calls using the same arguments. Since, there are currently no arguments it is essentially a singleton.
See L<Role::Multiton> for more details.
=head3 instance()
Alias to multiton(). If you ever plan on modifying the constructor to another type (weirdo) you may want to use this in your code instead of multiton().
See L<Role::Multiton> for more details.
=head2 Lazy façade methods
Each method returns a lazy loaded/instantiated object that implements the actual functionality.
=head3 $app->log
Lazy façade to a logging object via L<App::Kit::Role::Log>.
=head3 $app->locale
Lazy façade to a maketext() object via L<App::Kit::Role::Locale>.
Has all the methods any L<Locale::Maketext::Utils> based object would have.
Localize your code now without needing an entire subsystem in place just yet!
=head3 $app->detect
Lazy façade to a context detection object via L<App::Kit::Role::Detect>.
=head3 $app->ctype
Lazy façade to a ctype utility object via L<App::Kit::Role::CType>.
=head3 $app->str
Lazy façade to a string utility object via L<App::Kit::Role::Str>.
=head3 $app->ns
Lazy façade to a name space utility object via L<App::Kit::Role::NS>.
=head3 $app->http
Lazy façade to an HTTP client object via L<App::Kit::Role::HTTP>.
=head3 $app->fs
Lazy façade to a file system utility object via L<App::Kit::Role::FS>.
=head3 $app->db
Lazy façade to a database utility object via L<App::Kit::Role::DB>.
=head3 $app->ex
Lazy façade to a system execution utility object via L<App::Kit::Role::Ex>.
=head1 DIAGNOSTICS
Throws no warnings or errors of its own.
All errors or warnings would come from perl, L<Moo>, or the façade object in question.
=head1 CONFIGURATION AND ENVIRONMENT
App::Kit requires no configuration files or environment variables.
If, however, the façade object in question does it will be documented specifically under it.
=head1 DEPENDENCIES
L<Moo> et al.
If you don't pass in -no-try: L<Try::Tiny> and L<Import::Into>
Other modules would be documented above under each façade object that brings them in.
=head1 INCOMPATIBILITIES
( run in 1.116 second using v1.01-cache-2.11-cpan-ceb78f64989 )