Alien-Base-ModuleBuild
view release on metacpan or search on metacpan
lib/Alien/Base/ModuleBuild/FAQ.pod view on Meta::CPAN
# ABSTRACT: Frequently Asked Questions about Alien::Base::ModuleBuild
# PODNAME: Alien::Base::ModuleBuild::FAQ
# VERSION
__END__
=pod
=encoding UTF-8
=head1 NAME
Alien::Base::ModuleBuild::FAQ - Frequently Asked Questions about Alien::Base::ModuleBuild
=head1 VERSION
version 1.17
=head1 SYNOPSIS
perldoc Alien::Base::FAQ
=head1 DESCRIPTION
B<NOTE>: Please consider for new development of L<Alien>s that you use
L<Alien::Build> and L<alienfile> instead. Like L<Alien::Base::ModuleBuild> they work
with L<Alien::Base>. Unlike L<Alien::Base::ModuleBuild> they are more easily customized
and handle a number of corner cases better. For a good place to start,
please see L<Alien::Base::ModuleBuild::API>. Although the
Alien-Base / Alien-Build team will continue to maintain this module,
(we will continue to fix bugs where appropriate), we aren't adding any
new features to this module.
This document serves to answer the most frequently asked questions made by L<Alien::Base> authors.
=head2 What is Alien and Alien::Base?
Alien is a Perl namespace for defining dependencies in CPAN for libraries and tools which are not "native"
to CPAN. For a manifesto style description of the Why, and How see L<Alien>. L<Alien::Base> is a base
class and framework for creating Alien distributions. The idea is to address as many of the common challenges
to developing Alien modules in the base class to simplify the process.
=head2 How do I specify a minimum or exact version requirement for packages that use pkg-config?
The C<alien_version_check> attribute to L<Alien::Base::ModuleBuild> will be executed to determine if
the library is provided by the operating system. The default for this is C<%{pkg_config} --modversion %n>
which simply checks to see if any version of that package is available, and prints the version
number. You can use the C<--atleast-version>, C<--exact-version> options to require a specific range of versions,
but these flags do not work with the C<--modversion> flag, so be sure to invoke separately.
use Alien::Base::ModuleBuild;
Alien::Base::ModuleBuild->new(
dist_name => 'Alien::Foo',
alien_name => 'foo',
configure_requires => { 'Alien::Base::ModuleBuild' => '0.022' }, # required for %{pkg_config}
alien_version_check => '%{pkg_config} --atleast-version 1.2.3 %n && %{pkg_config} --modversion %n',
...
)->create_build_script;
It is better to use the built in C<%{pkg_config}> helper as it will use the system provided pkg-config
if it is available and fallback on the pure perl L<PkgConfig> if not.
You can also use C<--exact-version> to specify an exact version.
=head2 How to create an Alien module for packages that do not support pkg-config?
Although L<Alien::Base> and L<Alien::Base::ModuleBuild> assume packages come with a C<pkg-config>
C<.pc> file to determine compiler and linker flags by default, you can implement an Alien module
for packages that do use C<pkg-config> by following these tasks:
=over 4
=item subclass L<Alien::Base::ModuleBuild> and implement C<alien_check_installed_version>
Create a subclass of L<Alien::Base::ModuleBuild> and put it in the C<inc> directory of your distribution so
that it can be used during install but won't I<be installed>.
# inc/My/ModuleBuild.pm
package My::ModuleBuild;
use parent 'Alien::Base::ModuleBuild';
sub alien_check_installed_version {
my($class) = @_;
# determine if your library is already provided by the system
my $version = ...;
# return false if the library is NOT provided by the system
return unless defined $version;
# otherwise return the version detected
# (if you cannot determine the version it
# is usually sufficient to return a true value)
return $version;
}
There are number of methods you can use to determine if the system provides your library. From Perl
methods include L<Devel::CheckLib>, L<ExtUtils::CBuilder>, L<ExtUtils::CChecker>, L<Config::AutoConf>,
L<FFI::CheckLib> among others. It is also frequently possible to determine if a library is installed
using a C<-config> suffixed program. For example C<libxml2> comes with xml2-config which provides the
lib/Alien/Base/ModuleBuild/FAQ.pod view on Meta::CPAN
"cmake -G 'Unix Makefiles' -DCMAKE_MAKE_PROGRAM=$Config{make} -DCMAKE_INSTALL_PREFIX:PATH=%s",
"$Config{make}",
],
alien_install_commands => [
"$Config{make} install",
],
...
)->create_build_script;
=head3 vanilla Makefiles?
If you want to use the same C<make> as Perl, you can use L<Config>:
# Build.PL
use Alien::Base::ModuleBuild;
use Config;
Alien::Base::ModuleBuild->new(
...
alien_build_commands => [
"$Config{make}",
],
alien_install_commands => [
"$Config{make} install",
],
...
)->create_build_script;
=head3 GNU Makefiles?
Some packages require GNU Make's unique syntax. Perl's L<Config> provides an entry for C<gmake>, but it is
frequently wrong. Do not depend on it. Instead you can use L<Alien::gmake> to provide a real GNU Make (either
from the operating system, or built from source):
# Build.PL
use Alien::Base::ModuleBuild;
Alien::Base::ModuleBuild->new(
...
alien_bin_requires => {
'Alien::gmake' => 0.11, # needed for %{gmake} helper
},
alien_build_commands => [
"%{gmake}",
],
alien_install_commands => [
"%{gmake} install",
],
...
)->create_build_script;
=head2 When debugging my package build, I get different results!
If you get results from running the commands in your shell different to what happens when your C<Alien::>
distribution attempts to build, it may be because your environment is different than the one that your
distribution is using. For example, if you use L<Alien::CMake> or L<Alien::gmake> to build with specific tools
that are provided by your operating system, L<Alien::Base::ModuleBuild> will adjust the path before executing
build and install commands.
In the alien build directory (usually C<_alien>) you will find environment files that you can source
into your shell (C<env.csh> for tcsh and C<env.sh> for bourne based shells), which should provide the
identical environment used by the build process in order to troubleshoot the build manually.
% source _alien/env.sh
=head2 Can/Should I write a tool oriented Alien module using C<Alien::Base> that provides executables instead of a library?
Certainly. The original intent was to provide libraries, but tools are also quite doable using the
C<Alien::Base> tool set. A simple minded example of this which is fairly easy to replicate is L<Alien::m4>.
In general, this means specifying a subclass in your C<Build.PL> and bundling it in your distribution C<inc> directory.
C<Build.PL>:
...
use lib 'inc';
use My::ModuleBuild;
My::ModuleBuild->new(
...
)->create_build_script;
C<inc/My/ModuleBuild.pm>:
package My::ModuleBuild;
use strict;
use warnings;
use parent 'Alien::Base::ModuleBuild';
use Capture::Tiny qw( capture );
sub alien_check_installed_version
{
# see Alien::Base::ModuleBuild#alien_check_installed_version for details
my($self) = @_;
my($stdout, $stderr) = capture { system 'mytool', '--version' };
# return empty list if the tool is unavailable on the system,
# or unacceptable.
return if $! || ...;
# parse from stdout or stderr
my $version = ...;
return $version;
}
sub alien_check_built_version
{
# see Alien::Base::ModuleBuild#alien_check_built_version for details
# return empty list if the tool version cannot be found, or if it
# is unacceptable. Note that this will cause a failure, so "unknown"
# may be reasonable if the tool version cannot be determined.
return if ...;
# determine from the tool itself, or the current directory.
my $version = ...;
return $version;
}
1;
( run in 2.081 seconds using v1.01-cache-2.11-cpan-df04353d9ac )