#!/usr/bin/perl ## Copyright (c) GEANT ## This software was developed by RENATER. The research leading to these results has received funding ## from the European Community¹s Seventh Framework Programme (FP7/2007-2013) under grant agreement nº 238875 (GÉANT). ## 15/09/2014, Olivier Salaün ## Web interface for the eduGAIN Access Check Account Manager use strict; use warnings; use utf8; use lib qw(lib conf); use CGI; use CGI::Cookie; use CGI::Util; use Template; use Template::Constants qw( :debug ); use IdPAccountManager::TestAccount; use IdPAccountManager::SAMLMetadata; use IdPAccountManager::ServiceProvider; use IdPAccountManager::AuthenticationToken; ## Defining parameters format my $urn_or_url_regex = '(http(s?):\/\/|urn:)[^\\\$\*\"\'\`\^\|\<\>\n\s]+' ; ## Format de type URL HTTP ou URN my $url_regex = 'http(s?):\/\/[^\\\$\*\"\'\`\^\|\<\>\n\s]+'; my $email_regex = '([\w\-\_\.\/\+\=\'\&]+|\".*\")\@[\w\-]+(\.[\w\-]+)+'; my $domains_regex = '[\w\.\-]+(,[\w\.\-]+)*'; my %format = ( ## URL #'attributeauthority' => $url_regex, 'sp_entityid' => $urn_or_url_regex, ); my %actions = ( 'select_sp' => { 'title_en' => 'Select your Service Provider' }, 'account_wizard' => { 'title_en' => 'Select your Service Provider' }, 'generate_token' => { 'title_en' => 'Generate an authentication token' }, 'validate_token' => { 'title_en' => 'Complete Email Challenge' }, 'home' => { 'title_en' => $Conf::global{'app_name'} }, ); ## Gives writes for the group umask 0002; chdir $Conf::global{'root_manager_dir'}; my $request = new WebRequest; if (defined $request) { $request->execute(); } $request->respond(); package WebRequest; ## New web request sub new { my $pkg = shift; my $request = {}; IdPAccountManager::Tools::do_log('info', ""); my $http_query = new CGI; ## Input parameters my %in_vars = $http_query->Vars; $request->{'param_in'} = \%in_vars; ## Check if admin acts as another user $request->{'cookies'} = CGI::Cookie->fetch; #if (defined $request->{'cookies'}{'as_user'} && $request->{'is_admin'}) { # $request->{'utilisateur'} = $request->{'as_user'} = $request->{'cookies'}{'as_user'}->value; # $request->{'is_admin'} = 0; #} ## Usefull data for output (web pages or mail notices) $request->{'param_out'}{'url_cgi'} = $ENV{'SCRIPT_NAME'}; $request->{'param_out'}{'env'} = \%ENV; $request->{'param_out'}{'actions'} = \%actions; $request->{'param_out'}{'conf'} = \%Conf::global; ## Dumping input data #open TMP, ">/tmp/account_manager.in"; IdPAccountManager::Tools::dump_var($request->{'param_in'}, 0, \*TMP); close TMP; ## Clean input vars foreach my $key (keys %{ $request->{'param_in'} }) { #IdPAccountManager::Tools::do_log('trace', "PARAM_ENTREE: %s=%s", $key, $request->{'param_in'}{$key}); ## Removing all ^M (0D) $request->{'param_in'}{$key} =~ s/\r//g; $request->{'param_in'}{$key} =~ s/\s+$//; ## Remove trailing spaces $request->{'param_in'}{$key} =~ s/^\s+//; ## Remove leading spaces #if ($request->{'param_in'}{$key} =~ /\0/) { # my @valeurs = split /\0/, $request->{'param_in'}{$key}; # $request->{'param_in'}{$key} = $valeurs[0]; ## Only keep first value of multi-valued parameters #} ## If action_xx param is set, then action=xx ## Usefull to have sementicless values in submit forms if ($key =~ /^action_(\w+)$/) { #IdPAccountManager::Tools::do_log('trace', "ACTION $key"); $request->{'param_in'}{'action'} = $1; } } ## Check the requested action if ($request->{'param_in'}{'action'}) { $request->{'action'} = $request->{'param_in'}{'action'}; } else { ## Default action IdPAccountManager::Tools::do_log('info', "Default action"); $request->{'action'} = 'home'; } bless $request, $pkg; return $request; } ## Execute a web request sub execute { my $self = shift; IdPAccountManager::Tools::do_log('debug', ""); my $status; ## Check input parameters format foreach my $key (keys %{ $self->{'param_in'} }) { if ( $self->{'param_in'}{$key} !~ /^\s*$/ && defined $format{$key} && !ref($format{$key})) { unless ($self->{'param_in'}{$key} =~ /^$format{$key}$/) { push @{ $self->{'param_out'}{'errors'} }, "format_$key"; IdPAccountManager::Tools::do_log('error', "Incorrect parameter format : $key"); return undef; } } } do { ## Actions can be chained $self->{'action'} = $self->{'next_action'} if ($self->{'next_action'}); delete $self->{'next_action'}; ## Prevent loops if (defined $actions{ $self->{'action'} }) { ## Execute the target subroutine named req_actionName no strict 'refs'; my $sub = 'req_' . $self->{'action'}; $status = &{$sub}($self); } else { ## Inknown action push @{ $self->{'param_out'}{'errors'} }, "unknown_action"; IdPAccountManager::Tools::do_log('error', "Unknown action '%s'", $self->{'action'}); } } while ($self->{'next_action'}); #return undef if (!defined $status); return 1; } ## Return HTML content sub respond { my $self = shift; IdPAccountManager::Tools::do_log('debug', ""); ## Dump output data #open TMP, ">/tmp/account_registry.out"; IdPAccountManager::Tools::dump_var($self->{'param_out'}, 0, \*TMP); close TMP; ## Enable dumping off all variables in web pages #$self->{'param_out'}{'dump'} = $self->{'param_out'}; ## Automatic pass object entries to the output hash foreach my $key (keys %{$self}) { #IdPAccountManager::Tools::do_log('trace', "Passing $key"); $self->{'param_out'}{$key} ||= $self->{$key} unless ($key eq 'param_out'); } ## An action may redirect to an external URL if ($self->{'url_redirection'}) { #IdPAccountManager::Tools::do_log('trace', "URL Redirect : $self->{'url_redirection'}"); printf "Location: %s\n\n", $self->{'url_redirection'}; } else { #$self->{'param_out'}{'cookie'} = CGI::Cookie->new(-name=>'as_user',-value=>$self->{'as_user'},-expires=>'-1M'); ## Parse template my $tt2 = Template->new( { ENCODING => 'iso-8859-1', ## le défaut apparemment FILTERS => { 'encode_utf8', => [ \&IdPAccountManager::Tools::encode_utf8, 0 ], 'escape_quotes' => [ \&IdPAccountManager::Tools::escape_quotes, 0 ] }, INCLUDE_PATH => $Conf::global{'root_manager_dir'} . ':' . $Conf::global{'root_manager_dir'} . '/templates/accountProfiles', #DEBUG => 'all', #DEBUG => 'caller', #DEBUG => 'parser' } ); my $template; ## nobanner is used to do AJAX to get only pieces of HTML to load in the web client if ($self->{'param_in'}{'style'} eq 'nobanner') { $template = 'templates/web/index-nobanner.tt2.html'; } else { $template = 'templates/web/index.tt2.html'; } unless ($tt2->process($template, $self->{'param_out'}, \*STDOUT)) { printf "Content-type: text/plain\n\n Error: %s", $tt2->error(); IdPAccountManager::Tools::do_log('error', "Web parser error : %s", $tt2->error()); } } ## Ignore some type of errors my @errors_admin; foreach my $id_error (@{ $self->{'param_out'}{'errors'} }) { unless ($id_error =~ /^(error_x)$/) { push @errors_admin, $id_error; } } ## Mail notification of admins about the error if (@errors_admin) { $self->{'param_out'}{'subject'} = 'Error notification - web interface'; IdPAccountManager::Tools::mail_notice( 'template' => 'templates/mail/notification_generic_error.tt2.eml', 'data' => $self->{'param_out'} ); } } ## Return the list of known SPs first sub req_account_wizard { my $self = shift; IdPAccountManager::Tools::do_log('info', ""); my $federation_metadata = new IdPAccountManager::SAMLMetadata; unless ( $federation_metadata->load( federation_metadata_file_path => $Conf::global{'federation_metadata_file_path'} ) ) { push @{ $self->{'param_out'}{'errors'} }, "internal"; IdPAccountManager::Tools::do_log('error', "Failed to load federation metadata : $!"); return undef; } unless ($federation_metadata->parse()) { push @{ $self->{'param_out'}{'errors'} }, "internal"; IdPAccountManager::Tools::do_log('error', "Failed to parse federation metadata : $!"); return undef; } $self->{'param_out'}{'federation_metadata_as_hashref'} = $federation_metadata->{'federation_metadata_as_hashref'}; return 1; } ## Select a Service Provider and return metadata sctucture for the SP ## Sample URL : https://dev-edugain.renater.fr/accountmanager?action=select_sp&sp_entityid=http%3A%2F%2Fsp.lat.csc.fi sub req_select_sp { my $self = shift; IdPAccountManager::Tools::do_log('info', ""); unless ($self->{'param_in'}{'sp_entityid'}) { push @{ $self->{'param_out'}{'errors'} }, "missing_sp_entityid"; IdPAccountManager::Tools::do_log('error', "Missing parameter sp_entityid"); return undef; } my $federation_metadata = new IdPAccountManager::SAMLMetadata; unless ( $federation_metadata->load( federation_metadata_file_path => $Conf::global{'federation_metadata_file_path'} ) ) { push @{ $self->{'param_out'}{'errors'} }, "internal"; IdPAccountManager::Tools::do_log('error', "Failed to load federation metadata : $!"); return undef; } unless ( $federation_metadata->parse( filter_entity_id => $self->{'param_in'}{'sp_entityid'} ) ) { push @{ $self->{'param_out'}{'errors'} }, "internal"; IdPAccountManager::Tools::do_log('error', "Failed to parse federation metadata : $!"); return undef; } ## Create a serviceprovider object to store major parameters for this SP in DB my $service_provider = new IdPAccountManager::ServiceProvider( entityid => $self->{'param_in'}{'sp_entityid'}); ## Prepare data #open TMP, ">/tmp/account_manager_metadata.dump"; IdPAccountManager::Tools::dump_var($federation_metadata->{'federation_metadata_as_hashref'}[0], 0, \*TMP); close TMP; my $sp_metadata_as_hashref = $federation_metadata->{'federation_metadata_as_hashref'}[0]; my @contacts; if (defined $sp_metadata_as_hashref->{'contacts'}) { foreach my $contact (@{ $sp_metadata_as_hashref->{'contacts'} }) { my $email = $contact->{'EmailAddress'}; $email =~ s/^(mailto:)//; ## Remove 'mailto:' prefixes if any push @contacts, $email; } } my $display_name; if (defined $sp_metadata_as_hashref->{'display_name'}) { ## Use English version of displayName if available if ($sp_metadata_as_hashref->{'display_name'}{'en'}) { $display_name = $sp_metadata_as_hashref->{'display_name'}{'en'}; ## Else any language } else { foreach my $lang (keys %{ $sp_metadata_as_hashref->{'display_name'} }) { #IdPAccountManager::Tools::do_log('TRACE', "Display name(%s): %s", $lang, $sp_metadata_as_hashref->{'display_name'}{$lang}); $display_name = $sp_metadata_as_hashref->{'display_name'}{$lang}; last; } } } ## Try loading DB object first if ($service_provider->load(speculative => 1)) { $service_provider->contacts(join(',', @contacts)); $service_provider->displayname($display_name); } else { $service_provider = new IdPAccountManager::ServiceProvider( entityid => $self->{'param_in'}{'sp_entityid'}, contacts => join(',', @contacts), displayname => $display_name ); unless (defined $service_provider) { push @{ $self->{'param_out'}{'errors'} }, "internal"; IdPAccountManager::Tools::do_log('error', "Failed to create serviceprovider object"); return undef; } } unless ($service_provider->save()) { push @{ $self->{'param_out'}{'errors'} }, "internal"; IdPAccountManager::Tools::do_log('error', "Failed to save serviceprovider object"); return undef; } $self->{'param_out'}{'sp_metadata_as_hashref'} = $federation_metadata->{'federation_metadata_as_hashref'}[0]; $self->{'param_out'}{'serviceprovider'} = $service_provider; return 1; } ## Generate an authentication token to validate an email address ## Sample call : dev-edugain.renater.fr/accountmanager?action=generate_token&style=nobanner&sp_entityid=https%3A%2F%2Fsourcesup.cru.fr%2Fshibboleth&email_address=support%40renater.fr sub req_generate_token { my $self = shift; IdPAccountManager::Tools::do_log('info', ""); unless ($self->{'param_in'}{'sp_entityid'}) { push @{ $self->{'param_out'}{'errors'} }, "missing_sp_entityid"; IdPAccountManager::Tools::do_log('error', "Missing parameter sp_entityid"); return undef; } unless ($self->{'param_in'}{'email_address'}) { push @{ $self->{'param_out'}{'errors'} }, "email_address"; IdPAccountManager::Tools::do_log('error', "Missing parameter email_address"); return undef; } ## Create a serviceprovider object to load parameters for this SP from DB my $service_provider = new IdPAccountManager::ServiceProvider( entityid => $self->{'param_in'}{'sp_entityid'}); # Try loading DB object first unless ($service_provider->load(speculative => 1)) { push @{ $self->{'param_out'}{'errors'} }, "internal"; IdPAccountManager::Tools::do_log( 'error', "Failed to load SP with entityid '%s'", $self->{'param_in'}{'sp_entityid'} ); return undef; } ## Check that email_address is a known contact for this SP unless ($service_provider->is_contact($self->{'param_in'}{'email_address'})) { push @{ $self->{'param_out'}{'errors'} }, "internal"; IdPAccountManager::Tools::do_log( 'error', "Requested a token for %s for an unautorized address '%s'", $self->{'param_in'}{'sp_entityid'}, $self->{'param_in'}{'email_address'} ); return undef; } my $authentication_token = new IdPAccountManager::AuthenticationToken( 'email_address' => $self->{'param_in'}{'email_address'}, 'sp_entityid' => $self->{'param_in'}{'sp_entityid'} ); unless (defined $authentication_token) { push @{ $self->{'param_out'}{'errors'} }, "internal"; IdPAccountManager::Tools::do_log('error', "Failed to create authentication token"); return undef; } ## First remove token if one exist for this email+SP if ($authentication_token->load()) { unless ($authentication_token->delete()) { push @{ $self->{'param_out'}{'errors'} }, "internal"; IdPAccountManager::Tools::do_log( 'error', "Failed to delete previous authentication token with ID %s", $authentication_token->get('id') ); return undef; } $authentication_token = new IdPAccountManager::AuthenticationToken( 'email_address' => $self->{'param_in'}{'email_address'}, 'sp_entityid' => $self->{'param_in'}{'sp_entityid'} ); unless (defined $authentication_token) { push @{ $self->{'param_out'}{'errors'} }, "internal"; IdPAccountManager::Tools::do_log('error', "Failed to create authentication token"); return undef; } } unless ($authentication_token->save()) { push @{ $self->{'param_out'}{'errors'} }, "internal"; IdPAccountManager::Tools::do_log('error', "Failed to save authentication token"); return undef; } $self->{'param_out'}{'authentication_token'} = $authentication_token->get('token'); $self->{'param_out'}{'email_address'} = $self->{'param_in'}{'email_address'}; $self->{'param_out'}{'sp_entityid'} = $self->{'param_in'}{'sp_entityid'}; $self->{'param_out'}{'to'} = $self->{'param_in'}{'email_address'}; ## Send the challenge email with the token IdPAccountManager::Tools::mail_notice( 'template' => 'templates/mail/send_authentication_token.tt2.eml', 'to' => $self->{'param_in'}{'email_address'}, 'data' => $self->{'param_out'} ); IdPAccountManager::Tools::do_log( 'info', "Token send to %s for sp_entityid=%s;token=%s", $self->{'param_in'}{'email_address'}, $self->{'param_in'}{'sp_entityid'}, $self->{'param_out'}{'authentication_token'} ); return 1; } ## Validate an authentication token ## Test accounts get created ## Sample call : dev-edugain.renater.fr/accountmanager?action=validate_token&style=nobanner&sp_entityid=https%3A%2F%2Fsourcesup.cru.fr%2Fshibboleth&authentication_token=c1cfecb51ea40d39a695 sub req_validate_token { my $self = shift; IdPAccountManager::Tools::do_log('info', ""); unless ($self->{'param_in'}{'sp_entityid'}) { push @{ $self->{'param_out'}{'errors'} }, "missing_sp_entityid"; IdPAccountManager::Tools::do_log('error', "Missing parameter sp_entityid"); return undef; } unless ($self->{'param_in'}{'authentication_token'}) { push @{ $self->{'param_out'}{'errors'} }, "missing_authentication_token"; IdPAccountManager::Tools::do_log('error', "Missing parameter authentication_token"); return undef; } my $authentication_token = new IdPAccountManager::AuthenticationToken( token => $self->{'param_in'}{'authentication_token'}); unless ($authentication_token->load()) { push @{ $self->{'param_out'}{'errors'} }, "wrong_token"; IdPAccountManager::Tools::do_log( 'error', "Failed to validate authentication token %s for sp_entityid %s", $self->{'param_in'}{'authentication_token'}, $self->{'param_in'}{'sp_entityid'} ); return undef; } unless ($authentication_token->get('sp_entityid') eq $self->{'param_in'}{'sp_entityid'}) { push @{ $self->{'param_out'}{'errors'} }, "wrong_token_for_sp"; IdPAccountManager::Tools::do_log( 'error', "Authentication token %s cannot be used for SP with entityid %s", $self->{'param_in'}{'authentication_token'}, $self->{'param_in'}{'sp_entityid'} ); return undef; } ## delete the token unless ($authentication_token->delete()) { IdPAccountManager::Tools::do_log( 'error', "Failed to delete authentication token %s", $self->{'param_in'}{'authentication_token'} ); } ## create test accounts my @test_accounts = IdPAccountManager::TestAccount::create_test_accounts_for_sp( sp_entityid => $self->{'param_in'}{'sp_entityid'}); unless (@test_accounts) { push @{ $self->{'param_out'}{'errors'} }, "accounts_creation_failed"; IdPAccountManager::Tools::do_log( 'error', "Failed to create test accounts for SP with entityid %s", $self->{'param_in'}{'sp_entityid'} ); return undef; } ## Update simpleSAMLphp configuration to enable test accounts unless (IdPAccountManager::Tools::update_ssp_authsources()) { push @{ $self->{'param_out'}{'errors'} }, "accounts_creation_failed"; IdPAccountManager::Tools::do_log('error', "Failed to create simpleSAMLphp configuration file"); return undef; } IdPAccountManager::Tools::do_log( 'info', "Token validated for sp_entityid=%s;token=%s", $self->{'param_in'}{'sp_entityid'}, $self->{'param_in'}{'authentication_token'} ); $self->{'param_out'}{'sp_entityid'} = $self->{'param_in'}{'sp_entityid'}; $self->{'param_out'}{'test_accounts'} = \@test_accounts; return 1; } ## Return the homepage of the service sub req_home { my $self = shift; IdPAccountManager::Tools::do_log('info', ""); return 1; }