—package
Module::Load::Util;
use
Regexp::Pattern::Perl::Module ();
our
$AUTHORITY
=
'cpan:PERLANCAR'
;
# AUTHORITY
our
$DATE
=
'2024-05-13'
;
# DATE
our
$DIST
=
'Module-Load-Util'
;
# DIST
our
$VERSION
=
'0.012'
;
# VERSION
our
@EXPORT_OK
=
qw(
load_module_with_optional_args
instantiate_class_with_optional_args
call_module_function_with_optional_args
call_module_method_with_optional_args
)
;
sub
_normalize_module_with_optional_args {
my
$module_with_optional_args
=
shift
;
my
(
$module
,
$args
);
if
(
ref
$module_with_optional_args
eq
'ARRAY'
) {
die
"array form or module/class name must have 1 or 2 elements"
unless
@$module_with_optional_args
== 1 ||
@$module_with_optional_args
== 2;
$module
=
$module_with_optional_args
->[0];
$args
=
$module_with_optional_args
->[1] || [];
$args
= [
%$args
]
if
ref
$args
eq
'HASH'
;
die
"In array form of module/class name, the 2nd element must be "
.
"arrayref or hashref"
unless
ref
$args
eq
'ARRAY'
;
}
elsif
(
ref
$module_with_optional_args
) {
die
"module/class name must be string or 2-element array, not "
.
$module_with_optional_args
;
}
elsif
(
$module_with_optional_args
=~ /(.+?)[=,](.*)/) {
$module
= $1;
$args
= [
split
/,/, $2];
}
else
{
$module
=
$module_with_optional_args
;
$args
= [];
}
(
$module
,
$args
);
}
sub
_load_module {
my
$opts
=
ref
$_
[0] eq
'HASH'
?
shift
: {};
my
$module
=
shift
;
my
$do_load
=
defined
$opts
->{load} ?
$opts
->{load} : 1;
unless
(
$do_load
) {
if
(
$opts
->{ns_prefix}) {
$module
=
$opts
->{ns_prefix} . (
$opts
->{ns_prefix} =~ /::\z/ ?
''
:
'::'
) .
$module
;
}
elsif
(
$opts
->{ns_prefixes} && @{
$opts
->{ns_prefixes} }) {
$module
=
$opts
->{ns_prefixes}[0] . (
$opts
->{ns_prefixes}[0] =~ /::\z/ ?
''
:
'::'
) .
$module
;
}
return
$module
;
}
my
@ns_prefixes
=
$opts
->{ns_prefixes} ? @{
$opts
->{ns_prefixes}} :
defined
(
$opts
->{ns_prefix}) ? (
$opts
->{ns_prefix}) : (
''
);
my
$try_all
=
$opts
->{ns_prefixes} ? 1:0;
my
$module_with_prefix
;
for
my
$i
(0 ..
$#ns_prefixes
) {
my
$ns_prefix
=
$ns_prefixes
[
$i
];
if
(
length
$ns_prefix
) {
$module_with_prefix
=
$ns_prefix
. (
$ns_prefix
=~ /::\z/ ?
''
:
'::'
) .
$module
;
}
else
{
$module_with_prefix
=
$module
;
}
(
my
$module_with_prefix_pm
=
"$module_with_prefix.pm"
) =~ s!::!/!g;
if
(
$try_all
) {
eval
{
require
$module_with_prefix_pm
};
last
unless
$@;
warn
$@
if
$@ !~ /\ACan't locate/;
}
else
{
require
$module_with_prefix_pm
;
}
}
if
($@) {
die
"load_module_with_optional_args(): Failed to load module '$module' (all prefixes tried: "
.
join
(
", "
,
@ns_prefixes
).
")"
;
}
$module_with_prefix
;
}
sub
load_module_with_optional_args {
my
$opts
=
ref
(
$_
[0]) eq
'HASH'
?
shift
: {};
my
$module_with_optional_args
=
shift
;
my
$target_package
=
defined
$opts
->{target_package} ?
$opts
->{target_package} :
defined
$opts
->{
caller
} ?
$opts
->{
caller
} :
caller
(0);
# check because we will use eval ""
$target_package
=~
$Regexp::Pattern::Perl::Module::RE
{perl_modname}{pat}
or
die
"Invalid syntax in target package '$target_package'"
;
my
(
$module
,
$args
) = _normalize_module_with_optional_args(
$module_with_optional_args
);
$module
= _load_module(
$opts
,
$module
);
my
$do_import
=
defined
$opts
->{
import
} ?
$opts
->{
import
} : 1;
if
(
$do_import
) {
eval
"package $target_package; $module->import(\@{\$args});"
;
## no critic: BuiltinFunctions::ProhibitStringyEval
die
if
$@;
}
{
module
=>
$module
,
args
=>
$args
};
}
sub
instantiate_class_with_optional_args {
my
$opts
=
ref
(
$_
[0]) eq
'HASH'
? {%{
shift
()}} : {};
# shallow copy
my
$class_with_optional_args
=
shift
;
$opts
->{
import
} = 0;
$opts
->{target_package} =
caller
(0);
my
$res
= load_module_with_optional_args(
$opts
,
$class_with_optional_args
);
#use DD; print "Options: "; dd $opts; print "Result: "; dd $res;
my
$class
=
$res
->{module};
my
$args
=
$res
->{args};
my
$do_construct
=
defined
$opts
->{construct} ?
$opts
->{construct} : 1;
if
(
$do_construct
) {
my
$constructor
=
defined
$opts
->{constructor} ?
$opts
->{constructor} :
'new'
;
my
$obj
=
$class
->
$constructor
(
@$args
);
return
$obj
;
}
else
{
return
+{
class
=>
$class
,
args
=>
$args
};
}
}
sub
call_module_function_with_optional_args {
my
$opts
=
ref
(
$_
[0]) eq
'HASH'
?
shift
: {};
my
$module_with_optional_args
=
shift
;
my
(
$module
,
$args
) = _normalize_module_with_optional_args(
$module_with_optional_args
);
my
$function
;
if
(
defined
$opts
->{function}) {
$function
=
$opts
->{function};
}
else
{
$module
=~ s/\A(.+)::(\w+)\z/$1/ or
die
"Please specify MODULE::FUNCTION, not just module name '$module'"
;
$function
= $2;
}
$module
= _load_module(
$opts
,
$module
);
&{
"$module\::$function"
}(
@$args
);
}
sub
call_module_method_with_optional_args {
my
$opts
=
ref
(
$_
[0]) eq
'HASH'
?
shift
: {};
my
$module_with_optional_args
=
shift
;
my
(
$module
,
$args
) = _normalize_module_with_optional_args(
$module_with_optional_args
);
my
$method
;
if
(
defined
$opts
->{method}) {
$method
=
$opts
->{method};
}
else
{
$module
=~ s/\A(.+)::(\w+)\z/$1/ or
die
"Please specify MODULE::FUNCTION, not just module name '$module'"
;
$method
= $2;
}
$module
= _load_module(
$opts
,
$module
);
$module
->
$method
(
@$args
);
}
1;
# ABSTRACT: Some utility routines related to module loading
__END__
=pod
=encoding UTF-8
=head1 NAME
Module::Load::Util - Some utility routines related to module loading
=head1 VERSION
This document describes version 0.012 of Module::Load::Util (from Perl distribution Module-Load-Util), released on 2024-05-13.
=head1 SYNOPSIS
use Module::Load::Util qw(
load_module_with_optional_args
instantiate_class_with_optional_args
call_module_function_with_optional_args
call_module_method_with_optional_args
);
load_module_with_optional_args("Foo::Bar=import-arg1,import-arg2");
load_module_with_optional_args(["Foo::Bar", ["import-arg1", "import-arg2"]]);
my $obj = instantiate_class_with_optional_args("Some::Class=opt1,val1,opt2,val2");
my $obj = instantiate_class_with_optional_args(["Some::Class", {opt1=>"val1",opt2=>"val2"}]);
See more examples in each function's documentation in the L</FUNCTIONS> section.
=head1 DESCRIPTION
This module provides some utility routines related to module loading. Currently
what it offers now are the two functions L</load_module_with_optional_args> and
L</instantiate_class_with_optional_args>. These functions are designed for use
with command-line and/or plugin-based applications, because you can specify
module/class/plugin to load in a flexible format, as a string or 2-element
array. See L<wordlist> (from L<App::wordlist>), L<tabledata> (from
L<App::tabledata>), or L<ColorTheme> for some of the applications that use this
module.
Please see the functions' documentation for more details.
=head1 FUNCTIONS
=head2 load_module_with_optional_args
Usage:
load_module_with_optional_args( [ \%opts , ] $module_with_optional_args );
Examples:
load_module_with_optional_args("Color::RGB::Util"); # default imports, equivalent to runtime version of 'use Color::RGB::Util'
load_module_with_optional_args(["Color::RGB::Util", []]); # ditto
load_module_with_optional_args(["Color::RGB::Util", {}]); # ditto
load_module_with_optional_args("Color::RGB::Util=rgb2hsv"); # imports rgb2hsv. equivalent to runtime version of 'use Color::RGB::Util qw(rgb2hsv)'
load_module_with_optional_args(["Color::RGB::Util", ["rgb2hsv"]]); # ditto
load_module_with_optional_args(["Foo::Bar", {arg1=>1, arg2=>2}]); # equivalent to runtime version of 'use Foo::Bar qw(arg1 1 arg2 2)'. hashref will be list-ified
load_module_with_optional_args({import=>0}, "Color::RGB::Util"); # do not import, equivalent to runtime version of 'use Color::RGB::Util ()'
load_module_with_optional_args({ns_prefix=>"Color"}, "RGB::Util=rgb2hsv"); # equivalent to runtime version of 'use Color::RGB::Util qw(rgb2hsv)'
load_module_with_optional_args({ns_prefix=>"Color"}, ["RGB::Util", ["rgb2hsv"]]); # ditto
Load a module with C<require()> followed by calling the module's C<import()>
(unless instructed to skip importing). Main feature of this function is the
flexibility in the C<$module_with_optional_args> argument, as well as some
options like namespace prefix. Suitable to be used to load plugins for your
application, for example, where you can specify the plugin to load as simply a
string or a 2-element array.
C<$module_with_optional_args> can be a string containing module name (e.g.
C<"Foo::Bar">), or a string containing module name string followed by C<=>,
followed by comma-separated list of imports, a la perl's C<-M> (e.g.
C<"Foo::Bar=arg1,arg2">), or a 2-element array where the first element is the
module name and the second element is an arrayref or hashref containing import
arguments (e.g. C<< ["Foo::Bar", ["arg1","arg2"]] >> or C<< ["Foo::Bar",
{arg1=>"val",arg2=>"val"]] >>). Hashref list of arguments will still be passed
as a list to C<import()>.
Will die on require() or import() failure.
Will return a hashref containing module name and arguments, e.g. C<<
{module=>"Foo", args=>["arg1",1,"arg2",2]} >>.
Known options:
=over
=item * import
Bool. Defaults to true. Can be set to false to avoid import()-ing.
=item * ns_prefix
Str. Namespace to use. For example, if you set this to C<WordList> then with
C<$module_with_optional_args> set to C<ID::KBBI>, the module
L<WordList::ID::KBBI> will be loaded.
=item * ns_prefixes
Array of str. Like L</ns_prefix> but will attempt all prefixes and will fail if
all prefixes fail.
=item * target_package
Str. Target package to import() to. Default is caller(0).
=back
=head2 instantiate_class_with_optional_args
Usage:
instantiate_class_with_optional_args( [ \%opts , ] $class_with_optional_args );
Examples:
my $obj = instantiate_class_with_optional_args("WordList::Color::Any"); # equivalent to: require WordList::Color::Any; WordList::Color::Any->new;
my $obj = instantiate_class_with_optional_args(["WordList::Color::Any"], []]); # ditto
my $obj = instantiate_class_with_optional_args(["WordList::Color::Any"], {}]); # ditto
my $obj = instantiate_class_with_optional_args("WordList::Color::Any=theme,Foo"); # equivalent to: require WordList::Color::Any; WordList::Color::Any->new(theme=>"Foo");
my $obj = instantiate_class_with_optional_args(["WordList::Color::Any",{theme=>"Foo"}); # ditto
my $obj = instantiate_class_with_optional_args(["WordList::Color::Any",[theme=>"Foo"]); # ditto
my $obj = instantiate_class_with_optional_args(["Foo::Bar",[{arg1=>1, arg2=>2}]); # equivalent to: require Foo::Bar; Foo::Bar->new({arg1=>1, arg2=>2});
my $obj = instantiate_class_with_optional_args({ns_prefix=>"WordList"}, "Color::Any=theme,Foo"); # equivalent to: require WordList::Color::Any; WordList::Color::Any->new(theme=>"Foo");
This is like L</load_module_with_optional_args> but the constructor arguments
specified after C<=> will be passed to the class constructor instead of used as
import arguments.
When you use the 2-element array form of C<$class_with_optional_args>, the
hashref and arrayref constructor arguments will be converted to a list.
Known options:
=over
=item * construct
Bool. Default to true. If set to false, constructor will not be called and the
function will just return the hashref containing class name and arguments, e.g.
C<< {class=>"Foo", args=>["arg1",1,"args2",2]} >>.
=item * constructor
Str. Select constructor name. Defaults to C<new>.
=item * ns_prefix
Str. Like in L</load_module_with_optional_args>.
=item * ns_prefixes
Array of str. Like in L</load_module_with_optional_args>.
=item * load
Boolean. Default true. Whether to C<require> the class module. Sometimes you do
not want to C<require()>, e.g. when the class is already defined somewhere else.
=back
=head2 call_module_function_with_optional_args
Usage:
call_module_function_with_optional_args( [ \%opts , ] $function_with_optional_args );
Examples:
# function name will be stripped from module name
call_module_function_with_optional_args("App::ChromeUtils::chrome_is_running");
call_module_function_with_optional_args("App::ChromeUtils::start_chrome=quiet,1");
call_module_function_with_optional_args("Color::RGB::Util::int2rgb=100500");
call_module_function_with_optional_args(["App::ChromeUtils::start_chrome" => {quiet=>1}]);
call_module_function_with_optional_args(["Color::RGB::Util::int2rgb" => [100500]]);
call_module_function_with_optional_args({load=>0}, ["Color::RGB::Util::int2rgb" => [100500]]);
# if 'function' option is specified,
call_module_function_with_optional_args({function=>"chrome_is_running"}, "App::ChromeUtils");
call_module_function_with_optional_args({function=>"start_chrome"}, "App::ChromeUtils=quiet,1");
Load module then call module's function with optional arguments.
Known options:
=over
=item * load
=item * ns_prefix
=item * ns_prefixes
=item * function
=back
=head2 call_module_method_with_optional_args
Just like L</call_module_function_with_optional_args> except the subroutine call
is replaced with a method call instead.
Known options:
=over
=item * load
=item * ns_prefix
=item * ns_prefixes
=item * method
=back
=head1 HOMEPAGE
Please visit the project's homepage at L<https://metacpan.org/release/Module-Load-Util>.
=head1 SOURCE
Source repository is at L<https://github.com/perlancar/perl-Module-Load-Util>.
=head1 SEE ALSO
L<Module::Load>
L<Class::Load>
L<Sah::Schema::perl::modname_with_optional_args>
=head1 AUTHOR
perlancar <perlancar@cpan.org>
=head1 CONTRIBUTING
To contribute, you can send patches by email/via RT, or send pull requests on
GitHub.
Most of the time, you don't need to build the distribution yourself. You can
simply modify the code, then test via:
% prove -l
If you want to build the distribution (e.g. to try to install it locally on your
system), you can install L<Dist::Zilla>,
L<Dist::Zilla::PluginBundle::Author::PERLANCAR>,
L<Pod::Weaver::PluginBundle::Author::PERLANCAR>, and sometimes one or two other
Dist::Zilla- and/or Pod::Weaver plugins. Any additional steps required beyond
that are considered a bug and can be reported to me.
=head1 COPYRIGHT AND LICENSE
This software is copyright (c) 2024, 2023, 2022, 2021, 2020 by perlancar <perlancar@cpan.org>.
This is free software; you can redistribute it and/or modify it under
the same terms as the Perl 5 programming language system itself.
=head1 BUGS
Please report any bugs or feature requests on the bugtracker website L<https://rt.cpan.org/Public/Dist/Display.html?Name=Module-Load-Util>
When submitting a bug or request, please include a test-file or a
patch to an existing test-file that illustrates the bug or desired
feature.
=cut