Diff for /loncom/auth/lonacc.pm between versions 1.159.2.8.2.10 and 1.159.2.11

version 1.159.2.8.2.10, 2021/02/10 11:22:41 version 1.159.2.11, 2020/09/28 01:31:42
Line 355  sub sso_login { Line 355  sub sso_login {
     # login but immediately go to switch server to find us a new       # login but immediately go to switch server to find us a new 
     # machine      # machine
     &Apache::lonauth::success($r,$user,$domain,$home,'noredirect');      &Apache::lonauth::success($r,$user,$domain,$home,'noredirect');
             foreach my $item (keys(%form)) {  
                 $env{'form.'.$item} = $form{$item};  
             }  
             unless ($form{'symb'}) {  
                 unless (($r->uri eq '/adm/roles') || ($r->uri eq '/adm/sso')) {  
                     $env{'form.origurl'} = $r->uri;  
                 }  
             }  
             $env{'request.sso.login'} = 1;              $env{'request.sso.login'} = 1;
             if (defined($r->dir_config("lonSSOReloginServer"))) {              if (defined($r->dir_config("lonSSOReloginServer"))) {
                 $env{'request.sso.reloginserver'} =                  $env{'request.sso.reloginserver'} =
Line 377  sub sso_login { Line 369  sub sso_login {
  } else {   } else {
     # need to login them in, so generate the need data that      # need to login them in, so generate the need data that
     # migrate expects to do login      # migrate expects to do login
             my $ip = &Apache::lonnet::get_requestor_ip($r);              my $ip = $r->get_remote_host();
     my %info=('ip'        => $ip,      my %info=('ip'        => $ip,
       'domain'    => $domain,        'domain'    => $domain,
       'username'  => $user,        'username'  => $user,
Line 443  sub sso_login { Line 435  sub sso_login {
     return undef;      return undef;
 }  }
   
   sub needs_symb_check {
       my ($requrl) = @_;
       $requrl=~/\.(\w+)$/;
       if ((&Apache::loncommon::fileembstyle($1) eq 'ssi') ||
           ($requrl=~/^\/adm\/.*\/(aboutme|smppg|bulletinboard)(\?|$ )/x) ||
           ($requrl=~/^\/adm\/wrapper\//) ||
           ($requrl=~m|^/adm/coursedocs/showdoc/|) ||
           ($requrl=~m|\.problem/smpedit$|) ||
           ($requrl=~/^\/public\/.*\/syllabus$/) ||
           ($requrl=~/^\/adm\/(viewclasslist|navmaps)$/) ||
           ($requrl=~/^\/adm\/.*\/aboutme\/portfolio(\?|$)/)) {
           return 1;
       }
       return;
   }
   
 sub handler {  sub handler {
     my $r = shift;      my $r = shift;
     my $requrl=$r->uri;      my $requrl=$r->uri;
Line 537  sub handler { Line 545  sub handler {
                         if ($value =~ /^supplemental/) {                          if ($value =~ /^supplemental/) {
                             $suppext = 1;                              $suppext = 1;
                         }                          }
                         last;  
                     }                      }
                 }                  }
             }              }
Line 550  sub handler { Line 557  sub handler {
             my $lonhost = &Apache::lonnet::host_from_dns($hostname);              my $lonhost = &Apache::lonnet::host_from_dns($hostname);
             if ($lonhost) {              if ($lonhost) {
                 my $actual = &Apache::lonnet::absolute_url($hostname);                  my $actual = &Apache::lonnet::absolute_url($hostname);
                 my $exphostname = &Apache::lonnet::hostname($lonhost);  
                 my $expected = $Apache::lonnet::protocol{$lonhost}.'://'.$hostname;                  my $expected = $Apache::lonnet::protocol{$lonhost}.'://'.$hostname;
                 unless ($actual eq $expected) {                  unless ($actual eq $expected) {
                     $env{'request.use_absolute'} = $expected;                      $env{'request.use_absolute'} = $expected;
Line 572  sub handler { Line 578  sub handler {
         if ($env{'user.noloadbalance'} eq $r->dir_config('lonHostID')) {          if ($env{'user.noloadbalance'} eq $r->dir_config('lonHostID')) {
             $checkexempt = 1;              $checkexempt = 1;
         }          }
         unless (($checkexempt) || (($requrl eq '/adm/switchserver') && (!$r->is_initial_req()))) {          unless ($checkexempt) {
             ($is_balancer,$otherserver) =              ($is_balancer,$otherserver) =
                 &Apache::lonnet::check_loadbalancing($env{'user.name'},                  &Apache::lonnet::check_loadbalancing($env{'user.name'},
                                                      $env{'user.domain'});                                                       $env{'user.domain'});
             if ($is_balancer) {              if ($is_balancer) {
                 # Check if browser sent a LON-CAPA load balancer cookie (and this is a balancer)                  unless (($requrl eq '/adm/switchserver') && (!$r->is_initial_req())) {
                 my ($found_server,$balancer_cookie) = &Apache::lonnet::check_for_balancer_cookie($r);                      # Check if browser sent a LON-CAPA load balancer cookie (and this is a balancer)
                 if (($found_server) && ($balancer_cookie =~ /^\Q$env{'user.domain'}\E_\Q$env{'user.name'}\E_/)) {                      my ($found_server,$balancer_cookie) = &Apache::lonnet::check_for_balancer_cookie($r);
                     $otherserver = $found_server;                      if (($found_server) && ($balancer_cookie =~ /^\Q$env{'user.domain'}\E_\Q$env{'user.name'}\E_/)) {
                 }                          $otherserver = $found_server;
                 unless ($requrl eq '/adm/switchserver') {                      }
                     $r->set_handlers('PerlResponseHandler'=>  
                                      [\&Apache::switchserver::handler]);  
                 }                  }
               }
           }
           if ($is_balancer) {
               unless (($requrl eq '/adm/switchserver') && (!$r->is_initial_req())) {
                   $r->set_handlers('PerlResponseHandler'=>
                                    [\&Apache::switchserver::handler]);
                 if ($otherserver ne '') {                  if ($otherserver ne '') {
                     $env{'form.otherserver'} = $otherserver;                      $env{'form.otherserver'} = $otherserver;
                 }                  }
                 unless (($env{'form.origurl'}) || ($r->uri eq '/adm/roles') ||  
                         ($r->uri eq '/adm/switchserver') || ($r->uri eq '/adm/sso')) {  
                     $env{'form.origurl'} = $r->uri;  
                 }  
             }              }
         }              unless (($env{'form.origurl'}) || ($r->uri eq '/adm/roles') ||
         if ($requrl=~m{^/+tiny/+$match_domain/+\w+$}) {                      ($r->uri eq '/adm/switchserver') || ($r->uri eq '/adm/sso')) {
             if ($env{'user.name'} eq 'public' &&                  $env{'form.origurl'} = $r->uri;
                 $env{'user.domain'} eq 'public') {  
                 $env{'request.firsturl'}=$requrl;  
                 return FORBIDDEN;  
             } else {  
                 return OK;  
             }              }
         }          }
   
 # ---------------------------------------------------------------- Check access  # ---------------------------------------------------------------- Check access
  my $now = time;   my $now = time;
         my ($check_symb,$check_access,$check_block,$access,$poss_symb);  
  if ($requrl !~ m{^/(?:adm|public|(?:prt|zip)spool)/}   if ($requrl !~ m{^/(?:adm|public|(?:prt|zip)spool)/}
     || $requrl =~ /^\/adm\/.*\/(smppg|bulletinboard)(\?|$ )/x) {      || $requrl =~ /^\/adm\/.*\/(smppg|bulletinboard)(\?|$ )/x) {
             $check_access = 1;              my ($access,$poss_symb);
         }              if (($env{'request.course.id'}) && (!$suppext) && (&needs_symb_check($requrl))) {
         if ((!$check_access) && ($env{'request.course.id'})) {                  unless ($env{'form.symb'}) {
             if (($requrl eq '/adm/viewclasslist') ||                      if ($r->args) {
                 ($requrl =~ m{^(/adm/wrapper|)\Q/uploaded/$cdom/$cnum/docs/\E}) ||                          &Apache::loncommon::get_unprocessed_cgi($r->args,['symb']);
                 ($requrl =~ m{^/adm/.*/aboutme$}) ||                      }
                 ($requrl=~m{^/adm/coursedocs/showdoc/}) ||                  }
                 ($requrl=~m{^(/adm/wrapper|)/adm/$cdom/$cnum/\d+/ext\.tool$})) {  
                 $check_block = 1;  
             }  
         }  
         if (($env{'request.course.id'}) && (!$suppext)) {  
             $requrl=~/\.(\w+)$/;  
             if ((&Apache::loncommon::fileembstyle($1) eq 'ssi') ||  
                 ($requrl=~/^\/adm\/.*\/(aboutme|smppg|bulletinboard)(\?|$ )/x) ||  
                 ($requrl=~/^\/adm\/wrapper\//) ||  
                 ($requrl=~m|^/adm/coursedocs/showdoc/|) ||  
                 ($requrl=~m|\.problem/smpedit$|) ||  
                 ($requrl=~/^\/public\/.*\/syllabus$/) ||  
                 ($requrl=~/^\/adm\/(viewclasslist|navmaps)$/) ||  
                 ($requrl=~/^\/adm\/.*\/aboutme\/portfolio(\?|$)/) ||  
                 ($requrl=~m{^/adm/$cdom/$cnum/\d+/ext\.tool$})) {  
                 $check_symb = 1;  
             }  
         }  
         if (($check_access) || ($check_block)) {  
             if ($check_symb) {  
                 if ($env{'form.symb'}) {                  if ($env{'form.symb'}) {
                     $poss_symb=&Apache::lonnet::symbclean($env{'form.symb'});                      $poss_symb=&Apache::lonnet::symbclean($env{'form.symb'});
                 } elsif (($env{'request.course.id'}) && ($r->args ne '')) {  
                     my $query = $r->args;  
                     foreach my $pair (split(/&/,$query)) {  
                         my ($name, $value) = split(/=/,$pair);  
                         $name = &unescape($name);  
                         $value =~ tr/+/ /;  
                         $value =~ s/%([a-fA-F0-9][a-fA-F0-9])/pack("C",hex($1))/eg;  
                         if ($name eq 'symb') {  
                             $poss_symb = &Apache::lonnet::symbclean($value);  
                             last;  
                         }  
                     }  
                 }                  }
                 if ($poss_symb) {                  if ($poss_symb) {
                     my ($possmap,$resid,$url)=&Apache::lonnet::decode_symb($poss_symb);                      my ($possmap,$resid,$url)=&Apache::lonnet::decode_symb($poss_symb);
                     $url = &Apache::lonnet::clutter($url);                      $url = &Apache::lonnet::clutter($url);
                     my $toplevelmap = $env{'course.'.$env{'request.course.id'}.'.url'};                      unless (($url eq $requrl) && (&Apache::lonnet::is_on_map($possmap))) {
                     unless (($url eq $requrl) && (($possmap eq $toplevelmap) ||  
                                                   (&Apache::lonnet::is_on_map($possmap)))) {  
                         undef($poss_symb);                          undef($poss_symb);
                     }                      }
                     if ($poss_symb) {                      if ($poss_symb) {
Line 675  sub handler { Line 641  sub handler {
             } else {              } else {
                 $access=&Apache::lonnet::allowed('bre',$requrl);                  $access=&Apache::lonnet::allowed('bre',$requrl);
             }              }
         }  
         if ($check_block) {  
             if ($access eq 'B') {  
                 if ($poss_symb) {  
                     if (&Apache::lonnet::symbverify($poss_symb,$requrl)) {  
                         $env{'request.symb'} = $poss_symb;  
                     }  
                 }  
                 &Apache::blockedaccess::setup_handler($r);  
                 return OK;  
             }  
         } elsif ($check_access) {  
             if ($handle eq '') {              if ($handle eq '') {
                 unless ($access eq 'F') {                  unless ($access eq 'F') {
                     if ($requrl =~ m{^/res/$match_domain/$match_username/}) {                      if ($requrl =~ m{^/res/$match_domain/$match_username/}) {
Line 704  sub handler { Line 658  sub handler {
     }      }
             if ($access eq 'B') {              if ($access eq 'B') {
                 if ($poss_symb) {                  if ($poss_symb) {
                       if ($requrl=~m{^(/adm/.*/aboutme)/portfolio$}) {
                           $requrl = $1;
                       }
                     if (&Apache::lonnet::symbverify($poss_symb,$requrl)) {                      if (&Apache::lonnet::symbverify($poss_symb,$requrl)) {
                         $env{'request.symb'} = $poss_symb;                          $env{'request.symb'} = $poss_symb;
                     }                      }
Line 727  sub handler { Line 684  sub handler {
                         }                          }
                     }                      }
                 } elsif (($handle =~ /^publicuser_\d+$/) && (&Apache::lonnet::is_portfolio_url($requrl))) {                  } elsif (($handle =~ /^publicuser_\d+$/) && (&Apache::lonnet::is_portfolio_url($requrl))) {
                     my $clientip = &Apache::lonnet::get_requestor_ip($r);                      my $clientip = $r->get_remote_host();
                     if (&Apache::lonnet::allowed('bre',$requrl,undef,undef,$clientip) ne 'F') {                      if (&Apache::lonnet::allowed('bre',$requrl,undef,undef,$clientip) ne 'F') {
                         $env{'user.error.msg'}="$requrl:bre:1:1:Access Denied";                          $env{'user.error.msg'}="$requrl:bre:1:1:Access Denied";
                         return HTTP_NOT_ACCEPTABLE;                          return HTTP_NOT_ACCEPTABLE;
Line 758  sub handler { Line 715  sub handler {
     $env{'user.domain'} eq 'public' &&      $env{'user.domain'} eq 'public' &&
     $requrl !~ m{^/+(res|public|uploaded)/} &&      $requrl !~ m{^/+(res|public|uploaded)/} &&
     $requrl !~ m{^/adm/[^/]+/[^/]+/aboutme/portfolio$ }x &&      $requrl !~ m{^/adm/[^/]+/[^/]+/aboutme/portfolio$ }x &&
             $requrl !~ m{^/adm/blockingstatus/.*$} &&          $requrl !~ m{^/adm/blockingstatus/.*$} &&
     $requrl !~ m{^/+adm/(help|logout|restrictedaccess|randomlabel\.png)}) {      $requrl !~ m{^/+adm/(help|logout|restrictedaccess|randomlabel\.png)}) {
     $env{'request.querystring'}=$r->args;      $env{'request.querystring'}=$r->args;
     $env{'request.firsturl'}=$requrl;      $env{'request.firsturl'}=$requrl;
Line 768  sub handler { Line 725  sub handler {
  if ($env{'request.course.id'}) {   if ($env{'request.course.id'}) {
     &Apache::lonnet::countacc($requrl);      &Apache::lonnet::countacc($requrl);
             my $query=$r->args;              my $query=$r->args;
             if ($check_symb) {              if (&needs_symb_check($requrl)) {
 # ------------------------------------- This is serious stuff, get symb and log  # ------------------------------------- This is serious stuff, get symb and log
  my $symb;   my $symb;
  if ($query) {   if ($query) {
Line 776  sub handler { Line 733  sub handler {
  }   }
  if ($env{'form.symb'}) {   if ($env{'form.symb'}) {
     $symb=&Apache::lonnet::symbclean($env{'form.symb'});      $symb=&Apache::lonnet::symbclean($env{'form.symb'});
                     if (($requrl eq '/adm/navmaps') ||                      if ($requrl eq '/adm/navmaps') {
                         ($requrl =~ m{^/adm/wrapper/}) ||                          my ($map,$mid,$murl)=&Apache::lonnet::decode_symb($symb);
                         ($requrl =~ m{^/adm/coursedocs/showdoc/})) {                          &Apache::lonnet::symblist($map,$murl => [$murl,$mid]);
                         unless (&Apache::lonnet::symbverify($symb,$requrl)) {                      } elsif ($requrl =~ m|^/adm/wrapper/|
                             if (&Apache::lonnet::is_on_map($requrl)) {   || $requrl =~ m|^/adm/coursedocs/showdoc/|) {
                                 $symb = &Apache::lonnet::symbread($requrl);   my ($map,$mid,$murl)=&Apache::lonnet::decode_symb($symb);
                                 unless (&Apache::lonnet::symbverify($symb,$requrl)) {                          if ($map =~ /\.page$/) {
                                     undef($symb);                              my $mapsymb = &Apache::lonnet::symbread($map);
                                 }                              ($map,$mid,$murl)=&Apache::lonnet::decode_symb($mapsymb);
                             }  
                         }  
                         if ($symb) {  
                             if ($requrl eq '/adm/navmaps') {  
                                 my ($map,$mid,$murl)=&Apache::lonnet::decode_symb($symb);  
                                 &Apache::lonnet::symblist($map,$murl => [$murl,$mid]);  
                             } elsif (($requrl =~ m{^/adm/wrapper/}) ||  
                                      ($requrl =~ m{^/adm/coursedocs/showdoc/})) {  
                                 my ($map,$mid,$murl)=&Apache::lonnet::decode_symb($symb);  
                                 if ($map =~ /\.page$/) {  
                                     my $mapsymb = &Apache::lonnet::symbread($map);  
                                     ($map,$mid,$murl)=&Apache::lonnet::decode_symb($mapsymb);  
                                 }  
                                 &Apache::lonnet::symblist($map,$murl => [$murl,$mid],  
                                                           'last_known' =>[$murl,$mid]);  
                             }  
                         }                          }
    &Apache::lonnet::symblist($map,$murl => [$murl,$mid],
     'last_known' =>[$murl,$mid]);
     } elsif ((&Apache::lonnet::symbverify($symb,$requrl)) ||      } elsif ((&Apache::lonnet::symbverify($symb,$requrl)) ||
      (($requrl=~m|(.*)/smpedit$|) &&       (($requrl=~m|(.*)/smpedit$|) &&
       &Apache::lonnet::symbverify($symb,$1)) ||        &Apache::lonnet::symbverify($symb,$1)) ||
Line 825  sub handler { Line 768  sub handler {
                     if ($requrl=~m{^(/adm/.*/aboutme)/portfolio$}) {                      if ($requrl=~m{^(/adm/.*/aboutme)/portfolio$}) {
                         $requrl = $1;                          $requrl = $1;
                     }                      }
     $symb=&Apache::lonnet::symbread($requrl);                      unless ($suppext) {
                     if (&Apache::lonnet::is_on_map($requrl) && $symb) {          $symb=&Apache::lonnet::symbread($requrl);
                         my ($encstate,$invalidsymb);                          if (&Apache::lonnet::is_on_map($requrl) && $symb) {
                         unless (&Apache::lonnet::symbverify($symb,$requrl,\$encstate)) {                              my ($encstate,$invalidsymb);
                             $invalidsymb = 1;                              unless (&Apache::lonnet::symbverify($symb,$requrl,\$encstate)) {
                             #                                  $invalidsymb = 1;
                             # If $env{'request.enc'} inconsistent with encryption expected for $symb                                  #
                             # retrieved by lonnet::symbread(), call again to check for an instance of                                  # If $env{'request.enc'} is true, but no encryption for $symb retrieved
                             # $requrl in the course for which expected encryption matches request.enc.                                  # by original lonnet::symbread() call, call again to check for an instance
                             # If symb for different instance passes lonnet::symbverify(), use that as                                  # of $requrl in the course which has encryption, and set that as the symb.
                             # the symb for $requrl and call &Apache::lonnet::allowed() for that symb.                                  # If there is no such symb, or symbverify() fails for the new symb proceed
                             # Report invalid symb if there is no other symb. Redirect to /adm/ambiguous                                  # to report invalid symb.
                             # if multiple possible symbs consistent with request.enc available for $requrl.                                  #
                             #                                  if ($env{'request.enc'} && !$encstate) {
                             if (($env{'request.enc'} && !$encstate) || (!$env{'request.enc'} && $encstate)) {                                      my %possibles;
                                 my %possibles;                                      my $nocache = 1;
                                 my $nocache = 1;                                      $symb = &Apache::lonnet::symbread($requrl,'','','',\%possibles,$nocache);
                                 my $oldsymb = $symb;                                      if ($symb) {
                                 $symb = &Apache::lonnet::symbread($requrl,'','','',\%possibles,$nocache);                                          if (&Apache::lonnet::symbverify($symb,$requrl)) {
                                 if (($symb) && ($symb ne $oldsymb)) {  
                                     if (&Apache::lonnet::symbverify($symb,$requrl)) {  
                                         my $access=&Apache::lonnet::allowed('bre',$requrl,$symb);  
                                         if ($access eq 'B') {  
                                             $env{'request.symb'} = $symb;  
                                             &Apache::blockedaccess::setup_handler($r);  
                                             return OK;  
                                         } elsif (($access eq '2') || ($access eq 'F')) {  
                                             $invalidsymb = '';                                              $invalidsymb = '';
                                         }                                          }
                                       } elsif (keys(%possibles) > 1) {
                                           $r->internal_redirect('/adm/ambiguous');
                                           return OK;
                                     }                                      }
                                 } elsif (keys(%possibles) > 1) {  
                                     $r->internal_redirect('/adm/ambiguous');  
                                     return OK;  
                                 }                                  }
                             }                                  if ($invalidsymb) {
                             if ($invalidsymb) {  
                                 if ($requrl eq '/adm/navmaps') {  
                                     undef(symb);  
                                 } else {  
                                     $r->log_reason('Invalid symb for '.$requrl.': '.$symb);                                      $r->log_reason('Invalid symb for '.$requrl.': '.$symb);
                                     $env{'user.error.msg'}=                                      $env{'user.error.msg'}=
                                         "$requrl:bre:1:1:Invalid Access";                                          "$requrl:bre:1:1:Invalid Access";
Line 871  sub handler { Line 802  sub handler {
                                 }                                  }
                             }                              }
                         }                          }
                     }          if ($symb) {
     if ($symb) {      my ($map,$mid,$murl)=
  my ($map,$mid,$murl)=          &Apache::lonnet::decode_symb($symb);
     &Apache::lonnet::decode_symb($symb);                              if ($requrl eq '/adm/navmaps') {
                         if ($requrl eq '/adm/navmaps') {                                  &Apache::lonnet::symblist($map,$murl =>[$murl,$mid]);
                             &Apache::lonnet::symblist($map,$murl =>[$murl,$mid]);                              } else {
                         } else {                                  if (($map =~ /\.page$/) && ($requrl !~ /\.page$/)) {
                             if (($map =~ /\.page$/) && ($requrl !~ /\.page$/)) {                                      my $mapsymb = &Apache::lonnet::symbread($map);
                                 my $mapsymb = &Apache::lonnet::symbread($map);                                      ($map,$mid,$murl)=&Apache::lonnet::decode_symb($mapsymb);
                                 ($map,$mid,$murl)=&Apache::lonnet::decode_symb($mapsymb);                                  }
                                   &Apache::lonnet::symblist($map,$murl =>[$murl,$mid],
                                                             'last_known' =>[$murl,$mid]);
                             }                              }
                             &Apache::lonnet::symblist($map,$murl =>[$murl,$mid],          }
                                                       'last_known' =>[$murl,$mid]);  
                         }  
     }      }
  }   }
  $env{'request.symb'}=$symb;   $env{'request.symb'}=$symb;
Line 935  sub handler { Line 866  sub handler {
     }      }
 # ------------------------------------ See if this is a viewable portfolio file  # ------------------------------------ See if this is a viewable portfolio file
     if (&Apache::lonnet::is_portfolio_url($requrl)) {      if (&Apache::lonnet::is_portfolio_url($requrl)) {
         my $clientip = &Apache::lonnet::get_requestor_ip($r);          my $clientip = $r->get_remote_host();
  my $access=&Apache::lonnet::allowed('bre',$requrl,undef,undef,$clientip);   my $access=&Apache::lonnet::allowed('bre',$requrl,undef,undef,$clientip);
  if ($access eq 'A') {   if ($access eq 'A') {
     &Apache::restrictedaccess::setup_handler($r);      &Apache::restrictedaccess::setup_handler($r);

Removed from v.1.159.2.8.2.10  
changed lines
  Added in v.1.159.2.11


FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>