Net-SAML2
view release on metacpan or search on metacpan
lib/Net/SAML2/Binding/POST.pm view on Meta::CPAN
use Net::SAML2::XML::Sig;
use MIME::Base64 qw/ decode_base64 /;
use Crypt::OpenSSL::Verify;
use MIME::Base64;
use URI::Escape;
with 'Net::SAML2::Role::VerifyXML';
has 'cacert' => (isa => 'Maybe[Str]', is => 'ro');
has 'cert' => (isa => 'Str', is => 'ro', required => 0, predicate => 'has_cert');
has 'cert_text' => (isa => 'Str', is => 'ro');
has 'key' => (isa => 'Str', is => 'ro', required => 0, predicate => 'has_key');
sub handle_response {
my ($self, $response) = @_;
# unpack and check the signature
my $xml = decode_base64($response);
$self->verify_xml(
$xml,
no_xml_declaration => 1,
$self->cert_text ? (
cert_text => $self->cert_text
) : (),
$self->cacert ? (
cacert => $self->cacert
) : (),
);
return $xml;
}
sub sign_xml {
my ($self, $request) = @_;
croak("Need to have a cert specified") unless $self->has_cert;
croak("Need to have a key specified") unless $self->has_key;
my $signer = XML::Sig->new({
key => $self->key,
cert => $self->cert,
no_xml_declaration => 1,
}
);
my $signed_message = $signer->sign($request);
# saml-schema-protocol-2.0.xsd Schema hack
#
# The real fix here is to fix XML::Sig to accept a XPATH to
# place the signature in the correct location. Or use XML::LibXML
# here to do so
#
# The protocol schema defines a sequence which requires the order
# of the child elements in a Protocol based message:
#
# The dsig:Signature (should it exist) MUST follow the saml:Issuer
#
# 1: saml:Issuer
# 2: dsig:Signature
#
# Seems like an oversight in the SAML schema specifiation but...
$signed_message =~ s!(<dsig:Signature.*?</dsig:Signature>)!!s;
my $signature = $1;
$signed_message =~ s/(<\/saml\d*:Issuer>)/$1$signature/;
my $encoded_request = encode_base64($signed_message, "\n");
return $encoded_request;
}
__PACKAGE__->meta->make_immutable;
__END__
=pod
=encoding UTF-8
=head1 NAME
Net::SAML2::Binding::POST - HTTP POST binding for SAML
=head1 VERSION
version 0.85
=head1 SYNOPSIS
my $post = Net::SAML2::Binding::POST->new(
cacert => '/path/to/ca-cert.pem'
);
my $xml = $post->handle_response(
$saml_response
);
=head1 METHODS
=head2 new( )
Constructor. Returns an instance of the POST binding.
Arguments:
=over
=item B<cacert>
path to the CA certificate for verification
=item B<cert>
path to a certificate that is added to the signed XML. It needs to be the
certificate that includes the public key related to the B<key>
=item B<cert_text>
text form of the certificate in FORMAT_ASN1 or FORMAT_PEM that is used to
verify the signed XML.
=item B<key>
path to a key used to sign the XML.
( run in 0.882 second using v1.01-cache-2.11-cpan-ceb78f64989 )