--- loncom/auth/lonlogin.pm 2022/06/01 12:12:32 1.198 +++ loncom/auth/lonlogin.pm 2023/02/03 23:09:30 1.205 @@ -1,7 +1,7 @@ # The LearningOnline Network # Login Screen # -# $Id: lonlogin.pm,v 1.198 2022/06/01 12:12:32 raeburn Exp $ +# $Id: lonlogin.pm,v 1.205 2023/02/03 23:09:30 raeburn Exp $ # # Copyright Michigan State University Board of Trustees # @@ -50,7 +50,7 @@ sub handler { $ENV{'REDIRECT_QUERY_STRING'}), ['interface','username','domain','firsturl','localpath','localres', 'token','role','symb','iptoken','btoken','ltoken','ttoken','linkkey', - 'saml','sso','retry']); + 'saml','sso','retry','display']); # -- check if they are a migrating user if (defined($env{'form.token'})) { @@ -68,6 +68,11 @@ sub handler { $env{'form.ltoken'} = $info{'ltoken'}; } elsif ($info{'linkprot'}) { $env{'form.linkprot'} = $info{'linkprot'}; + foreach my $item ('linkprotuser','linkprotexit') { + if ($info{$item} ne '') { + $env{'form.'.$item} = $info{$item}; + } + } } elsif ($info{'linkkey'} ne '') { $env{'form.linkkey'} = $info{'linkkey'}; } @@ -170,6 +175,19 @@ sub handler { if ($env{'form.symb'}) { $info{'symb'} = $env{'form.symb'}; } + if (($env{'form.firsturl'} eq '/adm/email') && ($env{'form.display'} ne '')) { + if ($env{'form.sso'}) { + if ($env{'form.mailrecip'}) { + $info{'display'} = &escape($env{'form.display'}); + $info{'mailrecip'} = &escape($env{'form.mailrecip'}); + } + } else { + if (($env{'form.username'}) && ($env{'form.domain'})) { + $info{'display'} = &escape($env{'form.display'}); + $info{'mailrecip'} = &escape($env{'form.username'}.':'.$env{'form.domain'}); + } + } + } my $balancer_token = &Apache::lonnet::tmpput(\%info,$found_server); unless (($balancer_token eq 'con_lost') || ($balancer_token eq 'refused') || ($balancer_token eq 'unknown_cmd') || ($balancer_token eq 'no_such_host')) { @@ -181,6 +199,11 @@ sub handler { $link_info{'ltoken'} = $env{'form.ltoken'}; } elsif ($env{'form.linkprot'}) { $link_info{'linkprot'} = $env{'form.linkprot'}; + foreach my $item ('linkprotuser','linkprotexit') { + if ($env{'form.'.$item} ne '') { + $link_info{$item} = $env{'form.'.$item}; + } + } } elsif ($env{'form.linkkey'} ne '') { $link_info{'linkkey'} = $env{'form.linkkey'}; } @@ -218,6 +241,11 @@ sub handler { $balcookie = $info{'balcookie'}; &Apache::lonnet::tmpdel($env{'form.btoken'}); delete($env{'form.btoken'}); + if (($env{'form.firsturl'} eq '/adm/email') && + (exists($info{'display'})) && (exists($info{'mailrecip'}))) { + $env{'form.display'} = &unescape($info{'display'}); + $env{'form.mailrecip'} = &unescape($info{'mailrecip'}); + } } # @@ -249,16 +277,38 @@ sub handler { $dest = &HTML::Entities::encode($env{'form.firsturl'},'\'"<>&'); } if (($env{'form.ltoken'}) || ($env{'form.linkprot'})) { - my $linkprot; + my ($linkprot,$linkprotuser,$linkprotexit); if ($env{'form.ltoken'}) { my %info = &Apache::lonnet::tmpget($env{'form.ltoken'}); $linkprot = $info{'linkprot'}; - my $delete = &Apache::lonnet::tmpdel($env{'form.ltoken'}); + if ($info{'linkprotuser'} ne '') { + $linkprotuser = $info{'linkprotuser'}; + } + if ($info{'linkprotexit'} ne '') { + $linkprotexit = $info{'linkprotexit'}; + } } else { $linkprot = $env{'form.linkprot'}; + $linkprotuser = $env{'form.linkprotuser'}; + $linkprotexit = $env{'form.linkprotexit'}; } if ($linkprot) { my ($linkprotector,$deeplink) = split(/:/,$linkprot,2); + if (($deeplink =~ m{^/tiny/$match_domain/\w+$}) && + ($linkprotuser ne '') && ($linkprotuser ne $env{'user.name'}.':'.$env{'user.domain'})) { + my $ip = &Apache::lonnet::get_requestor_ip(); + my %linkprotinfo = ( + origurl => $deeplink, + linkprot => $linkprot, + linkprotuser => $linkprotuser, + linkprotexit => $linkprotexit, + ); + if ($env{'form.ltoken'}) { + my $delete = &Apache::lonnet::tmpdel($env{'form.ltoken'}); + } + &Apache::migrateuser::logout($r,$ip,$handle,undef,undef,\%linkprotinfo); + return OK; + } if ($env{'user.linkprotector'}) { my @protectors = split(/,/,$env{'user.linkprotector'}); unless (grep(/^\Q$linkprotector\E$/,@protectors)) { @@ -304,6 +354,20 @@ sub handler { } } } + if ($env{'form.ltoken'}) { + my $delete = &Apache::lonnet::tmpdel($env{'form.ltoken'}); + } + if (($env{'form.firsturl'} eq '/adm/email') && ($env{'form.display'})) { + if ($env{'form.mailrecip'}) { + if ($env{'form.mailrecip'} eq "$env{'user.name'}:$env{'user.domain'}") { + $dest .= (($dest=~/\?/)?'&':'?') . 'display='.&escape($env{'form.display'}). + '&mailrecip='.&escape($env{'form.mailrecip'}); + } + } elsif (($env{'form.username'} eq $env{'user.name'}) && ($env{'form.domain'} eq $env{'user.domain'})) { + $dest .= (($dest=~/\?/)?'&':'?') . 'display='.&escape($env{'form.display'}). + '&mailrecip='.&escape("$env{'user.name'}:$env{'form.domain'}"); + } + } $r->print( $start_page .'

'.&mt('You are already logged in!').'

' @@ -429,7 +493,7 @@ sub handler { # -------------------------------------------------------- Store away log token my ($tokenextras,$tokentype,$linkprot_for_login); - my @names = ('role','symb','iptoken','ltoken','linkprot','linkkey'); + my @names = ('role','symb','iptoken','ltoken','linkprotuser','linkprotexit','linkprot','linkkey','display'); foreach my $name (@names) { if ($env{'form.'.$name} ne '') { if ($name eq 'ltoken') { @@ -437,9 +501,24 @@ sub handler { if ($info{'linkprot'}) { $linkprot_for_login = $info{'linkprot'}; $tokenextras .= '&linkprot='.&escape($info{'linkprot'}); + foreach my $item ('linkprotuser','linkprotexit') { + if ($info{$item}) { + $tokenextras .= '&'.$item.'='.&escape($info{$item}); + } + } $tokentype = 'link'; last; } + } elsif ($env{'form.display'} && ($env{'form.firsturl'} eq '/adm/email')) { + if (($env{'form.mailrecip'}) || + ($env{'form.username'} =~ /^$match_username$/) && ($env{'form.domain'} =~ /^$match_domain$/)) { + $tokenextras .= '&'.$name.'='.&escape($env{'form.display'}); + if ($env{'form.mailrecip'}) { + $tokenextras .= '&mailrecip='.&escape($env{'form.mailrecip'}); + } else { + $tokenextras .= '&mailrecip='.&escape($env{'form.username'}.':'.$env{'form.domain'}); + } + } } else { $tokenextras .= '&'.$name.'='.&escape($env{'form.'.$name}); if (($name eq 'linkkey') || ($name eq 'linkprot')) { @@ -624,7 +703,8 @@ function enableInput() { ENDSCRIPT my ($lonhost_in_use,@hosts,%defaultdomconf,$saml_prefix,$saml_landing, - $samlssotext,$samlnonsso,$samlssoimg,$samlssoalt,$samlssourl,$samltooltip); + $samlssotext,$samlnonsso,$samlssoimg,$samlssoalt,$samlssourl,$samltooltip, + $samlwindow); %defaultdomconf = &Apache::loncommon::get_domainconf($defdom); @hosts = &Apache::lonnet::current_machine_ids(); $lonhost_in_use = $lonhost; @@ -645,6 +725,7 @@ ENDSCRIPT $samlssoalt = $defaultdomconf{$saml_prefix.'alt_'.$lonhost_in_use}; $samlssourl = $defaultdomconf{$saml_prefix.'url_'.$lonhost_in_use}; $samltooltip = $defaultdomconf{$saml_prefix.'title_'.$lonhost_in_use}; + $samlwindow = $defaultdomconf{$saml_prefix.'window_'.$lonhost_in_use}; } if ($saml_landing) { if ($samlssotext eq '') { @@ -832,6 +913,7 @@ HEADER my $stdauthformstyle = 'inline-block'; my $ssoauthstyle = 'none'; + my $sso_onclick; my $logintype; $r->print('
'); if ($saml_landing) { @@ -842,6 +924,8 @@ HEADER if ($samlssourl ne '') { $ssologin = $samlssourl; } + my $ssologin_for_js = &js_escape($ssologin); + my $querystr_for_js; if (($logtoken eq 'con_lost') || ($logtoken eq 'no_such_host')) { my $querystring; if ($env{'form.firsturl'} ne '') { @@ -862,16 +946,44 @@ HEADER } if ($querystring ne '') { $ssologin .= (($ssologin=~/\?/)?'&':'?') . $querystring; + $querystr_for_js = &js_escape($querystring); } } elsif ($logtoken ne '') { $ssologin .= (($ssologin=~/\?/)?'&':'?') . 'logtoken='.$logtoken; + $querystr_for_js = &js_escape('logtoken='.$logtoken); } my $ssohref; + if ($samlwindow) { + $sso_onclick = <<"ENDJS"; +if (document.getElementById('LC_sso_login_link')) { + var ssoelem = document.getElementById('LC_sso_login_link') + ssoelem.addEventListener('click',samlWinFunction,false); + var windows = {}; + function samlWinFunction(evt) { + evt.preventDefault(); + var url = '$ssologin_for_js'; + var name = 'lcssowin'; + var querystr = '$querystr_for_js'; + if (querystr) { + url += '?'+querystr+'&lcssowin=1'; + } else { + url += '?lcssowin=1'; + } + if ((typeof windows[name] !== 'undefined') && (!windows[name].closed)) { + windows[name].close(); + } + windows[name]=window.open(url,name,'width=350,height=600'); + windows[name].focus(); + return false; + } +} +ENDJS + } if ($samlssoimg ne '') { - $ssohref = ''. + $ssohref = ''. ''.$samlssoalt.''; } else { - $ssohref = ''.$samlssotext.''; + $ssohref = ''.$samlssotext.''; } if (($env{'form.saml'} eq 'no') || (($env{'form.username'} ne '') && ($env{'form.domain'} ne ''))) { @@ -926,11 +1038,22 @@ ENDTARG // ENDJS } + } elsif ($samlwindow) { + $in_frame_js = < +// + +ENDJS } $r->print(<{'only_body'} = 1; + } elsif ($env{'form.linkprot'}) { + $link_info{'linkprot'} = $env{'form.linkprot'}; + foreach my $item ('linkprotuser','linkprotexit') { + if ($env{'form.'.$item}) { + $link_info{$item} = $env{'form.'.$item}; + } + } + $args->{'only_body'} = 1; + } elsif ($env{'form.linkkey'} ne '') { + $link_info{'linkkey'} = $env{'form.linkkey'}; + } + my $token = &Apache::lonnet::tmpput(\%link_info,$desthost,'link'); + unless (($token eq 'con_lost') || ($token eq 'refused') || + ($token eq 'unknown_cmd') || ($token eq 'no_such_host')) { + $url .= '?ltoken='.$token; + } + } + } else { my $querystring; - if ($env{'form.firsturl'} =~ /[^\x00-\xFF]/) { - $querystring = &uri_escape_utf8($env{'form.firsturl'}); - } else { - $querystring = &uri_escape($env{'form.firsturl'}); + if ($env{'form.firsturl'} ne '') { + if ($env{'form.firsturl'} =~ /[^\x00-\xFF]/) { + $querystring = &uri_escape_utf8($env{'form.firsturl'}); + } else { + $querystring = &uri_escape($env{'form.firsturl'}); + } + $querystring = &HTML::Entities::encode($querystring,"'"); + $querystring = '?firsturl='.$querystring; } - $querystring = &HTML::Entities::encode($querystring,"'"); - $url .='?firsturl='.$querystring; - } - if (($env{'form.ltoken'}) || ($env{'form.linkkey'} ne '')) { - my %link_info; if ($env{'form.ltoken'}) { - $link_info{'ltoken'} = $env{'form.ltoken'}; - } elsif ($env{'form.linkkey'} ne '') { - $link_info{'linkkey'} = $env{'form.linkkey'}; - } - my $token = &Apache::lonnet::tmpput(\%link_info,$desthost,'link'); - unless (($token eq 'con_lost') || ($token eq 'refused') || - ($token eq 'unknown_cmd') || ($token eq 'no_such_host')) { - $url .= (($url=~/\?/)?'&':'?') . 'ttoken='.$token; + my %link_info = &Apache::lonnet::tmpget($env{'form.ltoken'}); + &Apache::lonnet::tmpdel($env{'form.ltoken'}); + my $token = &Apache::lonnet::tmpput(\%link_info,$desthost,'link'); + unless (($token eq 'con_lost') || ($token eq 'refused') || ($token =~ /^error:/) || + ($token eq 'unknown_cmd') || ($token eq 'no_such_host')) { + unless (($path eq '/adm/roles') || ($path eq '/adm/login')) { + $url = $protocol.'://'.$hostname.'/adm/roles'; + } + $querystring .= (($querystring =~/^\?/)?'&':'?') . 'ttoken='.$token; + } } + $url .= $querystring; } - my $start_page = &Apache::loncommon::start_page('Switching Server ...',undef, - {'redirect' => [0,$url],}); + $args->{'redirect'} = [0,$url]; + my $start_page = &Apache::loncommon::start_page('Switching Server ...',undef,$args); my $end_page = &Apache::loncommon::end_page(); return $start_page.$end_page; }