Diff for /loncom/Lond.pm between versions 1.8.2.3.2.3 and 1.9

version 1.8.2.3.2.3, 2022/02/20 21:06:30 version 1.9, 2016/07/25 19:49:45
Line 37  use lib '/home/httpd/lib/perl/'; Line 37  use lib '/home/httpd/lib/perl/';
 use LONCAPA;  use LONCAPA;
 use Apache::lonnet;  use Apache::lonnet;
 use GDBM_File;  use GDBM_File;
 use Net::OAuth;  use Crypt::OpenSSL::X509;
 use Crypt::CBC;  
   
 sub dump_with_regexp {  sub dump_with_regexp {
     my ( $tail, $clientversion ) = @_;      my ( $tail, $clientversion ) = @_;
Line 240  sub check_homecourses { Line 240  sub check_homecourses {
                     }                      }
                 }                  }
                 unless (&untie_domain_hash($hashref)) {                  unless (&untie_domain_hash($hashref)) {
                     &Apache::lonnet::logthis("Failed to untie tied hash for nohist_courseids.db for $domain");                      &logthis("Failed to untie tied hash for nohist_courseids.db for $domain");
                 }                  }
             } else {              } else {
                 &Apache::lonnet::logthis("Failed to tie hash for nohist_courseids.db for $domain");                  &logthis("Failed to tie hash for nohist_courseids.db for $domain");
             }              }
         }          }
         foreach my $hashid (keys(%recent)) {          foreach my $hashid (keys(%recent)) {
Line 315  sub get_courseinfo_hash { Line 315  sub get_courseinfo_hash {
     };      };
     if ($@) {      if ($@) {
         if ($@ eq "timeout\n") {          if ($@ eq "timeout\n") {
             &Apache::lonnet::logthis("<font color='blue'>WARNING courseiddump for $cnum:$cdom from $home timedout</font>");              &logthis("<font color='blue'>WARNING courseiddump for $cnum:$cdom from $home timedout</font>");
         } else {          } else {
             &Apache::lonnet::logthis("<font color='yellow'>WARNING unexpected error during eval of call for courseiddump from $home</font>");              &logthis("<font color='yellow'>WARNING unexpected error during eval of call for courseiddump from $home</font>");
         }          }
     } else {      } else {
         if (ref($info{$cdom.'_'.$cnum}) eq 'HASH') {          if (ref($info{$cdom.'_'.$cnum}) eq 'HASH') {
Line 783  sub dump_profile_database { Line 783  sub dump_profile_database {
     return $qresult;      return $qresult;
 }  }
   
 sub is_course {  sub server_certs {
     my ($cdom,$cnum) = @_;      my ($perlvar) = @_;
       my %pemfiles = (
     return unless (($cdom =~ /^$LONCAPA::match_domain$/) &&                       key      => 'lonnetPrivateKey',
                    ($cnum =~ /^$LONCAPA::match_courseid$/));                       host     => 'lonnetCertificate',
     my $hashid = $cdom.':'.$cnum;                       hostname => 'lonnetHostnameCertificate',
     my ($iscourse,$cached) =                       ca       => 'lonnetCertificateAuthority',
         &Apache::lonnet::is_cached_new('iscourse',$hashid);                     );
     unless (defined($cached)) {      my (%md5hash,%info);
         my $hashref =      if (ref($perlvar) eq 'HASH') {
             &tie_domain_hash($cdom, "nohist_courseids", &GDBM_WRCREAT());          my $certsdir = $perlvar->{'lonCertificateDirectory'};
         if (ref($hashref) eq 'HASH') {          if (-d $certsdir) {
             my $esc_key = &escape($cdom.'_'.$cnum);              foreach my $key (keys(%pemfiles)) {
             if (exists($hashref->{$esc_key})) {                  if ($perlvar->{$pemfiles{$key}}) {
                 $iscourse = 1;                      my $file = $certsdir.'/'.$perlvar->{$pemfiles{$key}};
             } else {                      if (-e $file) {
                 $iscourse = 0;                          if ($key eq 'key') {
             }                              if (open(PIPE,"openssl rsa -noout -in $file -check |")) {
             &Apache::lonnet::do_cache_new('iscourse',$hashid,$iscourse,3600);                                  my $check = <PIPE>;
             unless (&untie_domain_hash($hashref)) {                                  close(PIPE);
                 &Apache::lonnet::logthis("Failed to untie tied hash for nohist_courseids.db for $cdom");                                  chomp($check);
             }                                  $info{$key}{'status'} = $check;
         } else {                              }
             &Apache::lonnet::logthis("Failed to tie hash for nohist_courseids.db for $cdom");                              if (open(PIPE,"openssl rsa -noout -modulus -in $file | openssl md5 |")) {
         }                                  $md5hash{$key} = <PIPE>;
     }                                  close(PIPE);
     return $iscourse;                              }
 }                          } else {
                               if ($key eq 'ca') {
 sub get_dom {                                  if (open(PIPE,"openssl verify -CAfile $file $file |")) {
     my ($userinput) = @_;                                      my $check = <PIPE>;
     my ($cmd,$udom,$namespace,$what) =split(/:/,$userinput,4);                                      close(PIPE);
     my $hashref = &tie_domain_hash($udom,$namespace,&GDBM_READER()) or                                      chomp($check);
         return "error: ".($!+0)." tie(GDBM) Failed while attempting $cmd";                                      if ($check eq "$file: OK") {
     my $qresult='';                                          $info{$key}{'status'} = 'ok';
     if (ref($hashref)) {                                      } else {
         chomp($what);                                          $check =~ s/^\Q$file\E\:?\s*//;
         my @queries=split(/\&/,$what);                                          $info{$key}{'status'} = $check;
         for (my $i=0;$i<=$#queries;$i++) {                                      }
             $qresult.="$hashref->{$queries[$i]}&";                                  }
         }                              } else {
         $qresult=~s/\&$//;                                  if (open(PIPE,"openssl x509 -noout -modulus -in $file | openssl md5 |")) {
     }                                      $md5hash{$key} = <PIPE>;
     &untie_user_hash($hashref) or                                      close(PIPE);
         return "error: ".($!+0)." untie(GDBM) Failed while attempting $cmd";                                  }
     return $qresult;                              }
 }                              my $x509 = Crypt::OpenSSL::X509->new_from_file($file);
                               my @items = split(/,\s+/,$x509->subject());
 sub store_dom {                              foreach my $item (@items) {
     my ($userinput) = @_;                                  my ($name,$value) = split(/=/,$item);
     my ($cmd,$dom,$namespace,$rid,$what) =split(/:/,$userinput);                                  if ($name eq 'CN') {
     my $hashref  = &tie_domain_hash($dom,$namespace,&GDBM_WRCREAT(),"S","$rid:$what") or                                      $info{$key}{'cn'} = $value;
         return "error: ".($!+0)." tie(GDBM) Failed while attempting $cmd";                                  }
     $hashref->{"version:$rid"}++;                              }
     my $version=$hashref->{"version:$rid"};                              $info{$key}{'start'} = $x509->notBefore();
     my $allkeys='';                              $info{$key}{'end'} = $x509->notAfter();
     my @pairs=split(/\&/,$what);                              $info{$key}{'alg'} = $x509->sig_alg_name();
     foreach my $pair (@pairs) {                              $info{$key}{'size'} = $x509->bit_length();
         my ($key,$value)=split(/=/,$pair);                              $info{$key}{'email'} = $x509->email();
         $allkeys.=$key.':';                          }
         $hashref->{"$version:$rid:$key"}=$value;  
     }  
     my $now = time;  
     $hashref->{"$version:$rid:timestamp"}=$now;  
     $allkeys.='timestamp';  
     $hashref->{"$version:keys:$rid"}=$allkeys;  
     &untie_user_hash($hashref) or  
         return "error: ".($!+0)." untie(GDBM) Failed while attempting $cmd";  
     return 'ok';  
 }  
   
 sub restore_dom {  
     my ($userinput) = @_;  
     my ($cmd,$dom,$namespace,$rid) = split(/:/,$userinput);  
     my $hashref = &tie_domain_hash($dom,$namespace,&GDBM_READER()) or  
         return "error: ".($!+0)." tie(GDBM) Failed while attempting $cmd";  
     my $qresult='';  
     if (ref($hashref)) {  
         chomp($rid);  
         my $version=$hashref->{"version:$rid"};  
         $qresult.="version=$version&";  
         my $scope;  
         for ($scope=1;$scope<=$version;$scope++) {  
             my $vkeys=$hashref->{"$scope:keys:$rid"};  
             my @keys=split(/:/,$vkeys);  
             my $key;  
             $qresult.="$scope:keys=$vkeys&";  
             foreach $key (@keys) {  
                 $qresult.="$scope:$key=".$hashref->{"$scope:$rid:$key"}."&";  
             }  
         }  
         $qresult=~s/\&$//;  
     }  
     &untie_user_hash($hashref) or  
         return "error: ".($!+0)." untie(GDBM) Failed while attempting $cmd";  
     return $qresult;  
 }  
   
 sub crslti_itemid {  
     my ($cdom,$cnum,$url,$method,$params,$loncaparev) = @_;  
     unless (ref($params) eq 'HASH') {  
         return;  
     }  
     if (($cdom eq '') || ($cnum eq '')) {  
         return;  
     }  
     my ($itemid,$consumer_key,$secret);  
   
     if (exists($params->{'oauth_callback'})) {  
         $Net::OAuth::PROTOCOL_VERSION = Net::OAuth::PROTOCOL_VERSION_1_0A;  
     } else {  
         $Net::OAuth::PROTOCOL_VERSION = Net::OAuth::PROTOCOL_VERSION_1_0;  
     }  
   
     my $consumer_key = $params->{'oauth_consumer_key'};  
     return if ($consumer_key eq '');  
   
     my (%crslti,%crslti_by_key);  
     my $hashid=$cdom.'_'.$cnum;  
     my ($result,$cached)=&Apache::lonnet::is_cached_new('courseltienc',$hashid);  
     if (defined($cached)) {  
         if (ref($result) eq 'HASH') {  
             %crslti = %{$result};  
         }  
     } else {  
         my $reply = &dump_with_regexp(join(":",($cdom,$cnum,'nohist_ltienc','','')),$loncaparev);  
         %crslti = %{&Apache::lonnet::unserialize($reply)};  
         my $cachetime = 24*60*60;  
         &Apache::lonnet::do_cache_new('courseltienc',$hashid,\%crslti,$cachetime);  
     }  
   
     return if (!keys(%crslti));  
   
     foreach my $id (keys(%crslti)) {  
         if (ref($crslti{$id}) eq 'HASH') {  
             my $key = $crslti{$id}{'key'};  
             if (($key ne '') && ($crslti{$id}{'secret'} ne '')) {  
                 push(@{$crslti_by_key{$key}},$id);  
             }  
         }  
     }  
   
     return if (!keys(%crslti_by_key));  
   
     my %courselti = &Apache::lonnet::get_course_lti($cnum,$cdom,'provider');  
   
     if (ref($crslti_by_key{$consumer_key}) eq 'ARRAY') {  
         foreach my $id (@{$crslti_by_key{$consumer_key}}) {  
             my $secret = $crslti{$id}{'secret'};  
             if (ref($courselti{$id}) eq 'HASH') {  
                 if ((exists($courselti{$id}{'cipher'})) &&  
                     ($courselti{$id}{'cipher'} =~ /^\d+$/)) {  
                     my $keynum = $courselti{$id}{'cipher'};  
                     my $privkey = &get_dom("getdom:$cdom:private:$keynum:lti:key");  
                     if ($privkey ne '') {  
                         my $cipher = new Crypt::CBC($privkey);  
                         $secret = $cipher->decrypt_hex($secret);  
                     }                      }
                 }                  }
             }              }
             my $request = Net::OAuth->request('request token')->from_hash($params,  
                                               request_url => $url,  
                                               request_method => $method,  
                                               consumer_secret => $secret,);  
             if ($request->verify()) {  
                 $itemid = $id;  
                 last;  
             }  
         }          }
     }      }
     return $itemid;      foreach my $key ('host','hostname') {
 }          if ($md5hash{$key}) {
               if ($md5hash{$key} eq $md5hash{'key'}) {
 sub domlti_itemid {                  $info{$key}{'status'} = 'ok';
     my ($dom,$context,$url,$method,$params,$loncaparev) = @_;  
     unless (ref($params) eq 'HASH') {  
         return;  
     }  
     if ($dom eq '') {  
         return;  
     }  
     my ($itemid,$consumer_key,$secret);  
   
     if (exists($params->{'oauth_callback'})) {  
         $Net::OAuth::PROTOCOL_VERSION = Net::OAuth::PROTOCOL_VERSION_1_0A;  
     } else {  
         $Net::OAuth::PROTOCOL_VERSION = Net::OAuth::PROTOCOL_VERSION_1_0;  
     }  
   
     my $consumer_key = $params->{'oauth_consumer_key'};  
     return if ($consumer_key eq '');  
   
     my ($name,$cachename);  
     if ($context eq 'linkprot') {  
         $name = $context;  
     } else {  
         $name = 'lti';  
     }  
     $cachename = $name.'enc';  
     my %ltienc;  
     my ($encresult,$enccached)=&Apache::lonnet::is_cached_new($cachename,$dom);  
     if (defined($enccached)) {  
         if (ref($encresult) eq 'HASH') {  
             %ltienc = %{$encresult};  
         }  
     } else {  
         my $reply = &get_dom("getdom:$dom:encconfig:$name");  
         my $ltiencref = &Apache::lonnet::thaw_unescape($reply);  
         if (ref($ltiencref) eq 'HASH') {  
             %ltienc = %{$ltiencref};  
         }  
         my $cachetime = 24*60*60;  
         &Apache::lonnet::do_cache_new($cachename,$dom,\%ltienc,$cachetime);  
     }  
   
     return if (!keys(%ltienc));  
   
     my %lti_by_key;  
     foreach my $id (keys(%ltienc)) {  
         if (ref($ltienc{$id}) eq 'HASH') {  
             my $key = $ltienc{$id}{'key'};  
             if (($key ne '') && ($ltienc{$id}{'secret'} ne '')) {  
                 push(@{$lti_by_key{$key}},$id);  
             }              }
         }          }
     }      }
     return if (!keys(%lti_by_key));      my $result;
       foreach my $key (keys(%info)) {
     my %lti = &Apache::lonnet::get_domain_lti($dom,$context);          $result .= &escape($key).'='.&Apache::lonnet::freeze_escape($info{$key}).'&';
   
     if (ref($lti_by_key{$consumer_key}) eq 'ARRAY') {  
         foreach my $id (@{$lti_by_key{$consumer_key}}) {  
             my $secret = $ltienc{$id}{'secret'};  
             if (ref($lti{$id}) eq 'HASH') {  
                 if ((exists($lti{$id}{'cipher'})) &&  
                     ($lti{$id}{'cipher'} =~ /^\d+$/)) {  
                     my $keynum = $lti{$id}{'cipher'};  
                     my $privkey = &get_dom("getdom:$dom:private:$keynum:lti:key");  
                     if ($privkey ne '') {  
                         my $cipher = new Crypt::CBC($privkey);  
                         $secret = $cipher->decrypt_hex($secret);  
                     }  
                 }  
             }  
             my $request = Net::OAuth->request('request token')->from_hash($params,  
                                               request_url => $url,  
                                               request_method => $method,  
                                               consumer_secret => $secret,);  
             if ($request->verify()) {  
                 $itemid = $id;  
                 last;  
             }  
         }  
     }      }
     return $itemid;      $result =~ s/\&$//;
       return $result;
 }  }
   
 1;  1;
Line 1162  courseID -- for the course for which the Line 985  courseID -- for the course for which the
 The contents of the inner hash, for that single item in the outer hash  The contents of the inner hash, for that single item in the outer hash
 are returned (and cached in memcache for 10 minutes).  are returned (and cached in memcache for 10 minutes).
   
 =item get_dom ( $userinput )  
   
 get_dom() will retrieve domain configuration information from a GDBM file  
 in /home/httpd/lonUsers/$dom on the primary library server in a domain.  
 The single argument passed is the string: $cmd:$udom:$namespace:$what  
 where $cmd is the command historically passed to lond - i.e., getdom  
 or egetdom, $udom is the domain, $namespace is the name of the GDBM file  
 (encconfig or configuration), and $what is a string containing names of  
 items to retrieve from the db file (each item name is escaped and separated  
 from the next item name with an ampersand). The return value is either:  
 error: followed by an error message, or a string containing the value (escaped)  
 for each item, again separated from the next item with an ampersand.  
   
 =back  =back
   

Removed from v.1.8.2.3.2.3  
changed lines
  Added in v.1.9


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