Crypt-OpenSSL-AES

 view release on metacpan or  search on metacpan

lib/Crypt/OpenSSL/AES.pm  view on Meta::CPAN

package Crypt::OpenSSL::AES;

# Copyright (C) 2006 - 2024 DelTel, Inc.
#
# This library is free software; you can redistribute it and/or modify
# it under the same terms as Perl itself, either Perl version 5.8.5 or,
# at your option, any later version of Perl 5 you may have available.

use 5.008000;
use strict;
use warnings;

our $VERSION = '0.23';

require Exporter;

our @ISA = qw(Exporter);

require XSLoader;
XSLoader::load('Crypt::OpenSSL::AES', $VERSION);

sub CLONE_SKIP { 1 }

# Preloaded methods go here.

1;
__END__
=head1 NAME

Crypt::OpenSSL::AES - A Perl wrapper around OpenSSL's AES library

=head1 SYNOPSIS

     use Crypt::OpenSSL::AES;
     use Crypt::URandom qw( urandom );  # Always use a strong random source

     # Basic usage (defaults to AES-ECB based on key length; ECB is not recommended)
     my $key    = urandom(32);
     my $cipher = Crypt::OpenSSL::AES->new($key);

     # Recommended usage: AES-256-CBC with proper Initialization Vector and Padding
     my $secure_key = urandom(32); # 32 bytes (256 bits) for AES-256
     my $iv         = urandom(16); # 16 bytes (128 bits) block size for AES

     my $secure_cipher = Crypt::OpenSSL::AES->new(
         $secure_key,
         {
             cipher  => 'AES-256-CBC',
             iv      => $iv,
             padding => 1, # 1 for standard block padding, 0 for no padding
         }
     );

     my $plaintext = "Confidential data to be encrypted.";
     my $encrypted = $secure_cipher->encrypt($plaintext);
     my $decrypted = $secure_cipher->decrypt($encrypted);

=head1 DESCRIPTION

This module implements a wrapper around OpenSSL.  Specifically, it
wraps the methods related to the US Government's Advanced
Encryption Standard (the Rijndael algorithm).  The original version
supports only AES ECB (electronic codebook mode encryption).

This module is compatible with Crypt::CBC (and likely other modules
that utilize a block cipher to make a stream cipher).

This module is an alternative to the implementation provided by
Crypt::Rijndael which implements AES itself. In contrast, this module
is simply a wrapper around the OpenSSL library.

As of version 0.09 additional AES ciphers are supported.  Those are:

=over 4

=item Block Ciphers

The blocksize is 16 bytes and must be padded if not a multiple of the
blocksize.

=over 4

=item AES-128-ECB, AES-192-ECB and AES-256-ECB (no IV)

Supports padding

=item AES-128-CBC, AES-192-CBC and AES-256-CBC

Supports padding and iv

=back

=item Stream Ciphers

The blocksize is 1 byte. OpenSSL does not pad even if padding
is set (the default).

=over 4

=item AES-128-CFB, AES-192-CFB and AES-256-CFB

Supports iv

=item AES-128-CTR, AES-192-CTR and AES-256-CTR

Supports iv

=item AES-128-OFB, AES-192-OFB and AES-256-OFB

Supports iv

=back

=back

=head1 FIPS COMPLIANCE

When using OpenSSL 3.0+ built with FIPS support, pass C<provider_props => 'fips=yes'>
to the constructor to ensure only FIPS-validated algorithm implementations are used.

B<AES-ECB is not approved for general data encryption under FIPS 140-3.>
Use AES-CBC or AES-CTR with a random IV instead.

    my $cipher = Crypt::OpenSSL::AES->new($key, {
        cipher         => 'AES-256-CBC',
        iv             => $iv,
        padding        => 1,
        provider_props => 'fips=yes',
    });

    # Check at runtime:
    warn "FIPS mode active\n" if Crypt::OpenSSL::AES::fips_mode();

=head1 mod_perl / THREADED ENVIRONMENTS

B<Never store a Crypt::OpenSSL::AES object in a package variable under
mod_perl with the worker or event MPM.> Each request handler must
construct its own object. The underlying C<EVP_CIPHER_CTX> is not
thread-safe.

Under prefork MPM this restriction does not apply, but you should still
avoid constructing cipher objects at C<use> time (i.e., at server startup
before the fork), because OpenSSL's PRNG state is not safely shared
across C<fork()>.

Recommended pattern for mod_perl handlers:

    sub handler {
        my $r = shift;
        my $cipher = Crypt::OpenSSL::AES->new($key, { ... });
        # use $cipher only within this request
    }

    # httpd.conf or startup.pl
    PerlChildInitHandler sub {
        Crypt::OpenSSL::AES::post_fork_init();
    }

=over 4

=item new()

For compatibility with old versions you can simply pass the key to the
new constructor.

    # The default cipher is AES-ECB based on the key size
    my $cipher = Crypt::OpenSSL::AES->new($key);

    or

    # the keysize must match the cipher size
    # 16-bytes (128-bits) AES-128-xxx
    # 24-bytes (192-bits) AES-192-xxx
    # 32-bytes (256-bits) AES-256-xxx
    my $cipher = Crypt::OpenSSL::AES->new($key,
                    {
                        cipher  => 'AES-256-CBC',
                        iv      => $iv, # (16-bytes for supported ciphers)
                        padding => 1, (0 - no padding, 1 - padding)
                    });

    # cipher
    #   AES-128-ECB, AES-192-ECB and AES-256-ECB (no IV)
    #   AES-128-CBC, AES-192-CBC and AES-256-CBC
    #   AES-128-CFB, AES-192-CFB and AES-256-CFB
    #   AES-128-CTR, AES-192-CTR and AES-256-CTR
    #   AES-128-OFB, AES-192-OFB and AES-256-OFB
    #
    # iv - 16-byte random data
    #
    # padding
    #   0 - no padding
    #   1 - padding

=item $cipher->encrypt($data)

Encrypt data. For Block Ciphers (ECB and CBC) the size of C<$data>
must be exactly C<blocksize> in length (16 bytes) B<or> padding must be
enabled in the B<new> constructor, otherwise this function will croak.

For Stream ciphers (CFB, CTR or OFB) the block size is considered to
be 1 byte and no padding is required.

Crypt::CBC is no longer required to encrypt/decrypt data of arbitrary
lengths.

=item $cipher->decrypt($data)

Decrypts data. For Block Ciphers (ECB and CBC) the size of C<$data>
must be exactly C<blocksize> in length (16 bytes) B<or> padding must be
enabled in the B<new> constructor, otherwise this function will croak.

For Stream ciphers (CFB, CTR or OFB) the block size is considered to
be 1 byte and no padding is required.

Crypt::CBC is no longer required to encrypt/decrypt data of arbitrary
lengths.

=item $cipher->fips_mode()

Will return true (1) or false (0) depending whether the openssl 'fips=yes'
default property is set.

=item keysize

This method is used by Crypt::CBC to verify the key length.
This module actually supports key lengths of 16, 24, and 32 bytes,
but this method always returns 32 for Crypt::CBC's sake.

=item blocksize

This method is used by Crypt::CBC to check the block size.
The blocksize for AES is always 16 bytes.

=back

=head2 USE WITH CRYPT::CBC

As padding is now supported for the CBC cipher, Crypt::CBC is no
longer required but supported for backward compatibility.

	use Crypt::CBC;

	my $plaintext = "This is a test!!";
	my $password = "qwerty123";
	my $cipher = Crypt::CBC->new(
		-key    => $password,
		-cipher => "Crypt::OpenSSL::AES",
		-pbkdf  => 'pbkdf2',
	);

	my $encrypted = $cipher->encrypt($plaintext);
	my $decrypted = $cipher->decrypt($encrypted);

=head1 SEE ALSO

L<Crypt::CBC>

http://www.openssl.org/

http://en.wikipedia.org/wiki/Advanced_Encryption_Standard

http://www.csrc.nist.gov/encryption/aes/

=head1 BUGS

Need more (and better) test cases.

=head1 AUTHOR

Tolga Tarhan, E<lt>cpan at ttar dot orgE<gt>

The US Government's Advanced Encryption Standard is the Rijndael
Algorithm and was developed by Vincent Rijmen and Joan Daemen.

=head1 COPYRIGHT AND LICENSE

Copyright (C) 2006 - 2024 DelTel, Inc.

This library is free software; you can redistribute it and/or modify
it under the same terms as Perl itself, either Perl version 5.8.5 or,
at your option, any later version of Perl 5 you may have available.

=cut



( run in 0.943 second using v1.01-cache-2.11-cpan-e1769b4cff6 )