--- loncom/auth/lonlogin.pm 2022/05/25 18:05:56 1.196 +++ loncom/auth/lonlogin.pm 2025/02/17 18:56:10 1.212 @@ -1,7 +1,7 @@ # The LearningOnline Network # Login Screen # -# $Id: lonlogin.pm,v 1.196 2022/05/25 18:05:56 raeburn Exp $ +# $Id: lonlogin.pm,v 1.212 2025/02/17 18:56:10 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','linkprotpbid','linkprotpburl') { + 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','linkprotpbid','linkprotpburl') { + if ($env{'form.'.$item} ne '') { + $link_info{$item} = $env{'form.'.$item}; + } + } } elsif ($env{'form.linkkey'} ne '') { $link_info{'linkkey'} = $env{'form.linkkey'}; } @@ -218,8 +241,16 @@ 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'}); + } } + (undef,undef,undef,my $clientmathml,my $clientunicode,undef,my $clientmobile) = + &Apache::loncommon::decode_user_agent($r); + # # If browser sent an old cookie for which the session file had been removed # check if configuration for user's domain has a portal URL set. If so @@ -240,8 +271,12 @@ sub handler { # -------------------------------- Prevent users from attempting to login twice if ($handle ne '') { &Apache::lonnet::transfer_profile_to_env($lonidsdir,$handle); + my $args = {}; + if ($clientunicode && !$clientmathml) { + $args->{'browser.unicode'} = 1; + } my $start_page = - &Apache::loncommon::start_page('Already logged in'); + &Apache::loncommon::start_page('Already logged in','',$args); my $end_page = &Apache::loncommon::end_page(); my $dest = '/adm/roles'; @@ -249,16 +284,48 @@ sub handler { $dest = &HTML::Entities::encode($env{'form.firsturl'},'\'"<>&'); } if (($env{'form.ltoken'}) || ($env{'form.linkprot'})) { - my $linkprot; + my ($linkprot,$linkprotuser,$linkprotexit,$linkprotpbid,$linkprotpburl); 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'}; + } + if ($info{'linkprotpbid'} ne '') { + $linkprotpbid = $info{'linkprotpbid'}; + } + if ($info{'linkprotpburl'} ne '') { + $linkprotpburl = $info{'linkprotpburl'}; + } } else { $linkprot = $env{'form.linkprot'}; + $linkprotuser = $env{'form.linkprotuser'}; + $linkprotexit = $env{'form.linkprotexit'}; + $linkprotpbid = $env{'form.linkprotpbid'}; + $linkprotpburl = $env{'form.linkprotpburl'}; } 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, + linkprotpbid => $linkprotpbid, + linkprotpburl => $linkprotpburl, + ); + 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,11 +371,27 @@ 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!').'

' .'

'.&mt('Please either [_1]continue the current session[_2] or [_3]log out[_4].', '','','','').'

' + .'
' .$end_page ); return OK; @@ -334,9 +417,6 @@ sub handler { # ----------------------------------------------------------- Process Interface $env{'form.interface'}=~s/\W//g; - (undef,undef,undef,undef,undef,undef,my $clientmobile) = - &Apache::loncommon::decode_user_agent($r); - my $iconpath= &Apache::loncommon::lonhttpdurl($r->dir_config('lonIconsURL')); @@ -429,7 +509,8 @@ 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','linkprotpbid','linkprotpburl'); foreach my $name (@names) { if ($env{'form.'.$name} ne '') { if ($name eq 'ltoken') { @@ -437,13 +518,29 @@ sub handler { if ($info{'linkprot'}) { $linkprot_for_login = $info{'linkprot'}; $tokenextras .= '&linkprot='.&escape($info{'linkprot'}); + foreach my $item ('linkprotuser','linkprotexit','linkprotpbid','linkprotpburl') { + 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')) { - if (($env{'form.retry'}) && (!$env{'form.ltoken'}) && ($name eq 'linkprot')) { + if ((($env{'form.retry'}) || ($env{'form.sso'})) && + (!$env{'form.ltoken'}) && ($name eq 'linkprot')) { $linkprot_for_login = $env{'form.linkprot'}; } $tokentype = 'link'; @@ -527,17 +624,39 @@ sub handler { .'' .'' .&mt('The LearningOnline Network with CAPA') - .'' + .'' + .'' .'' + .'' + .'
' + .'

'.&mt('This LON-CAPA server is temporarily not available for login.').'

'); if ($spares) { $r->print('

'.&mt('Please attempt to login to one of the following servers:') .'

' .$spares); } - $r->print('' + $r->print('
' .'' ); return OK; @@ -623,7 +742,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; @@ -644,6 +764,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 '') { @@ -719,10 +840,16 @@ ENDSAMLJS } } - $r->print(&Apache::loncommon::start_page('The LearningOnline Network with CAPA Login',$js, - { 'redirect' => [$expire,'/adm/roles'], - 'add_entries' => \%add_entries, - 'only_body' => 1,})); + my $args = { + 'redirect' => [$expire,'/adm/roles'], + 'add_entries' => \%add_entries, + 'only_body' => 1, + }; + if ($clientunicode && !$clientmathml) { + $args->{'browser.unicode'} = 1; + } + $r->print(&Apache::loncommon::start_page('The LearningOnline Network with CAPA Login', + $js,$args)); # ----------------------------------------------------------------------- Texts @@ -742,6 +869,7 @@ ENDSAMLJS 'forgotpw' => 'Forgot password?', 'newuser' => 'New User?', 'change' => 'Change?', + 'nojs' => 'Use of LON-CAPA requires Javascript to be enabled in your web browser.', ); # -------------------------------------------------- Change password field name @@ -764,8 +892,17 @@ ENDSAMLJS .' style="margin:0 auto; padding:10px; width:90%; height: auto; background-color:#FFFFFF;">' ); + my $target = '_top'; + if ($sessiondata{'linkprot'}) { + my ($linkprotector,$deeplink) = split(/:/,$sessiondata{'linkprot'},2); + if (($deeplink eq $sessiondata{'origurl'}) && + (($sessiondata{'linkprotuser'} eq $sessiondata{'username'}.':'.$sessiondata{'domain'}) || + ($sessiondata{'linkprotuser'} eq $sessiondata{'username'}))) { + $target = '_self'; + } + } $r->print(< +
@@ -790,9 +927,18 @@ ENDSERVERFORM .$lt{'log'} .''; - my $noscript_warning=''; + my $noscript_warning = <<"ENDNOJS"; + +
+ +
+ENDNOJS my $helpdeskscript; my $contactblock = &contactdisplay(\%lt,$servadm,$showadminmail, $authdomain,\$helpdeskscript, @@ -823,16 +969,22 @@ LFORM } $r->print(< -
+
+

$alttext +

HEADER + } else { + $r->print(''); } my $stdauthformstyle = 'inline-block'; my $ssoauthstyle = 'none'; + my $sso_onclick; my $logintype; - $r->print('
'); + $r->print('
'); if ($saml_landing) { $ssoauthstyle = 'inline-block'; $stdauthformstyle = 'none'; @@ -841,6 +993,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 '') { @@ -861,16 +1015,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 ''))) { @@ -907,7 +1089,7 @@ ENDSAML my ($linkprotector,$linkproturi) = split(/:/,$linkprot_for_login,2); if (($linkprotector =~ /^\d+(c|d)$/) && ($linkproturi =~ m{^/+tiny/+$LONCAPA::match_domain/+\w+$})) { my $set_target; - if ($env{'form.retry'}) { + if (($env{'form.retry'}) || ($env{'form.sso'})) { if ($linkproturi eq $env{'form.firsturl'}) { $set_target = " document.server.target = '_self';"; } @@ -925,11 +1107,22 @@ ENDTARG // ENDJS } + } elsif ($samlwindow) { + $in_frame_js = < +// + +ENDJS } $r->print(<
ENDTOP - my ($domainrow,$serverrow,$loadrow,$userloadrow,$versionrow); + my ($domainrow,$serverrow,$loadrow,$userloadrow,$versioninfo); $domainrow = <<"END"; - + $lt{'dom'}:  - - + +  $domain END $serverrow = <<"END"; - + $lt{'serv'}:  - +  $lonhost ($role) @@ -984,9 +1177,9 @@ END if ($loadlim) { $loadrow = <<"END"; - + $lt{'load'}:  - +  $loadpercent $lt{'perc'} @@ -996,9 +1189,9 @@ END if ($uloadlim) { $userloadrow = <<"END"; - + $lt{'userload'}:  - +  $userloadpercent $lt{'perc'} @@ -1006,31 +1199,28 @@ END END } if (($version ne '') && ($version ne '')) { - $versionrow = <<"END"; - - - $version - - -END + $versioninfo = "$version"; } $r->print(<
$domainrow $serverrow $loadrow $userloadrow -$versionrow
+ $versioninfo
$domainlogo
+

+$in_frame_js