version 1.1, 2004/08/11 18:37:23
|
version 1.14, 2005/03/31 22:09:40
|
Line 26
|
Line 26
|
# |
# |
# http://www.lon-capa.org/ |
# http://www.lon-capa.org/ |
# |
# |
############################################################################### |
#-------------------------------------------------------------------- |
# |
|
# Expects |
|
# |
|
# ../key/$class.key - key file $username:$keynumber |
|
# ../rawdata/$class.log - log file |
|
# ../rawdata/$class.seq - sequence file |
|
# ../data writable |
|
# ------------------------------------------------------------------ Course log |
|
|
|
# |
# |
# Exit codes |
# Exit codes |
# 0 Everything is okay |
# 0 Everything is okay |
Line 43
|
Line 34
|
# 2 Activity log does not exist |
# 2 Activity log does not exist |
# 3 Unable to connect to database |
# 3 Unable to connect to database |
# 4 Unable to create database tables |
# 4 Unable to create database tables |
# 5 Unspecified error? |
# 5 Unable to open log file |
|
# 6 Unable to get lock on activity log |
|
# |
|
|
|
# |
|
# Notes: |
|
# |
|
# Logging is done via the $logthis variable, which may be the result of |
|
# overcleverness. log via $logthis->('logtext'); Those are parentheses, |
|
# not curly braces. If the -log command line parameter is set, the $logthis |
|
# routine is set to a routine which writes to a file. If the command line |
|
# parameter is not set $logthis is set to ¬hing, which does what you |
|
# would expect. |
# |
# |
|
|
use strict; |
use strict; |
use DBI; |
use DBI; |
use lib '/home/httpd/lib/perl/Apache'; |
use lib '/home/httpd/lib/perl/Apache'; |
|
use lib '/home/httpd/lib/perl/'; |
|
use LONCAPA::Configuration(); |
|
use Apache::lonmysql(); |
use lonmysql(); |
use lonmysql(); |
use Time::HiRes(); |
use Time::HiRes(); |
use Getopt::Long(); |
use Getopt::Long(); |
|
use IO::File; |
|
use File::Copy; |
|
use Fcntl qw(:flock); |
|
use HTML::TokeParser; |
|
|
# |
# |
# Determine parameters |
# Determine parameters |
my ($help,$course,$domain,$drop,$file,$time_run,$nocleanup); |
my ($help,$course,$domain,$drop_when_done,$srcfile,$logfile,$time_run,$nocleanup,$log,$backup); |
&Getopt::Long::GetOptions( "course=s" => \$course, |
&Getopt::Long::GetOptions( "course=s" => \$course, |
"domain=s" => \$domain, |
"domain=s" => \$domain, |
|
"backup" => \$backup, |
"help" => \$help, |
"help" => \$help, |
"logfile=s" => \$file, |
"logfile=s" => \$logfile, |
|
"srcfile=s" => \$srcfile, |
"timerun" => \$time_run, |
"timerun" => \$time_run, |
"nocleanup" => \$nocleanup, |
"nocleanup" => \$nocleanup, |
"drop" => \$drop); |
"dropwhendone" => \$drop_when_done, |
|
"log" => \$log); |
if (! defined($course) || $help) { |
if (! defined($course) || $help) { |
print<<USAGE; |
print<<USAGE; |
parse_activity_log.pl |
parse_activity_log.pl |
Line 70 parse_activity_log.pl
|
Line 83 parse_activity_log.pl
|
Process a lon-capa activity log into a database. |
Process a lon-capa activity log into a database. |
Parameters: |
Parameters: |
course Required |
course Required |
domain Optional |
domain optional |
drop optional if present, drop all course |
backup optional if present, backup the activity log file |
specific activity log tables. |
before processing it |
file optional Specify the file to parse, including path |
dropwhendone optional if present, drop all course |
|
specific activity log tables after processing. |
|
srcfile optional Specify the file to parse, including path |
time optional if present, print out timing data |
time optional if present, print out timing data |
nocleanup optional if present, do not remove old files |
nocleanup optional if present, do not remove old files |
|
log optional if present, prepare log file of activity |
|
logfile optional specifies the logfile to use |
Examples: |
Examples: |
$0 -course=123456abcdef -domain=msu |
$0 -course=123456abcdef -domain=msu |
$0 -course=123456abcdef -file=activity.log |
$0 -course=123456abcdef -srcfile=activity.log |
|
$0 -course-123456abcdef -log -logfile=/tmp/logfile -dropwhendone |
USAGE |
USAGE |
exit; |
exit; |
} |
} |
|
|
## |
## |
## Set up timing code |
## Set up timing code |
## |
|
my $time_this = \¬hing; |
my $time_this = \¬hing; |
if ($time_run) { |
if ($time_run) { |
$time_this = \&time_action; |
$time_this = \&time_action; |
Line 95 my $initial_time = Time::HiRes::time;
|
Line 112 my $initial_time = Time::HiRes::time;
|
## |
## |
## Read in configuration parameters |
## Read in configuration parameters |
## |
## |
my %perlvar; |
my %perlvar = %{&LONCAPA::Configuration::read_conf('loncapa.conf')}; |
&initialize_configuration(); |
|
if (! defined($domain) || $domain eq '') { |
if (! defined($domain) || $domain eq '') { |
$domain = $perlvar{'lonDefDomain'}; |
$domain = $perlvar{'lonDefDomain'}; |
} |
} |
&update_process_name($course.'@'.$domain); |
&update_process_name($course.'@'.$domain); |
|
|
## |
## |
|
## Set up logging code |
|
my $logthis = \¬hing; |
|
|
|
if ($log) { |
|
if (! $logfile) { |
|
$logfile = $perlvar{'lonDaemons'}.'/tmp/parse_activity_log.log.'.time; |
|
} |
|
print STDERR "$0: logging to $logfile".$/; |
|
if (! open(LOGFILE,">$logfile")) { |
|
warn("Unable to open $logfile for writing. Run aborted."); |
|
exit 5; |
|
} else { |
|
$logthis = \&log_to_file; |
|
} |
|
} |
|
|
|
|
|
## |
## Determine filenames |
## Determine filenames |
## |
## |
my $sourcefilename; # activity log data |
my $sourcefilename; # activity log data |
my $newfilename; # $sourcefilename will be renamed to this |
my $newfilename; # $sourcefilename will be renamed to this |
my $sql_filename; # the mysql backup data file name. |
my $error_filename; # Errors in parsing the activity log will be written here |
if ($file) { |
if ($srcfile) { |
$sourcefilename = $file; |
$sourcefilename = $srcfile; |
} else { |
} else { |
$sourcefilename = &get_filename($course,$domain); |
$sourcefilename = &get_filename($course,$domain); |
} |
} |
$sql_filename = $sourcefilename; |
my $sql_filename = $sourcefilename; |
$sql_filename =~ s|[^/]*$|activitylog.sql|; |
$sql_filename =~ s|[^/]*$|activity.log.sql|; |
|
my $gz_sql_filename = $sql_filename.'.gz'; |
|
# |
|
my $xml_filename = $sourcefilename; |
|
$xml_filename =~ s|[^/]*$|activity.log.xml|; |
|
my $gz_xml_filename = $xml_filename.'.gz'; |
|
# |
|
$error_filename = $sourcefilename; |
|
$error_filename =~ s|[^/]*$|activity.log.errors|; |
|
$logthis->('Beginning logging '.time); |
|
|
|
# |
|
# Wait for a lock on the lockfile to avoid collisions |
|
my $lockfilename = $sourcefilename.'.lock'; |
|
open(LOCKFILE,'>'.$lockfilename); |
|
if (!flock(LOCKFILE,LOCK_EX)) { |
|
warn("Unable to lock $lockfilename. Aborting".$/); |
|
exit 6; |
|
} |
|
|
## |
## |
## There will only be a $newfilename file if a copy of this program is already |
## There will only be a $newfilename file if a copy of this program is already |
Line 122 $sql_filename =~ s|[^/]*$|activitylog.sq
|
Line 175 $sql_filename =~ s|[^/]*$|activitylog.sq
|
my $newfilename = $sourcefilename.'.processing'; |
my $newfilename = $sourcefilename.'.processing'; |
if (-e $newfilename) { |
if (-e $newfilename) { |
warn "$newfilename exists"; |
warn "$newfilename exists"; |
|
$logthis->($newfilename.' exists, so I cannot work on it.'); |
exit 2; |
exit 2; |
} |
} |
|
|
if (-e $sourcefilename) { |
if (-e $sourcefilename) { |
|
$logthis->('renaming '.$sourcefilename.' to '.$newfilename); |
rename($sourcefilename,$newfilename); |
rename($sourcefilename,$newfilename); |
|
Copy($newfilename,$newfilename.'.'.time) if ($backup); |
|
$logthis->("renamed $sourcefilename to $newfilename"); |
|
} else { |
|
my $command = 'touch '.$newfilename; |
|
$logthis->($command); |
|
system($command); |
|
$logthis->('touch was completed'); |
} |
} |
|
|
|
close(LOCKFILE); |
|
|
## |
## |
## Table definitions |
## Table definitions |
## |
## |
my $prefix = $course.'_'.$domain.'_'; |
my %tables = &table_names($course,$domain); |
my $student_table = $prefix.'students'; |
|
my $student_table_def = |
my $student_table_def = |
{ id => $student_table, |
{ id => $tables{'student'}, |
permanent => 'no', |
permanent => 'no', |
columns => [ |
columns => [ |
{ name => 'student_id', |
{ name => 'student_id', |
Line 149 my $student_table_def =
|
Line 212 my $student_table_def =
|
'PRIMARY KEY' => ['student_id',], |
'PRIMARY KEY' => ['student_id',], |
}; |
}; |
|
|
my $res_table = $prefix.'resource'; |
|
my $res_table_def = |
my $res_table_def = |
{ id => $res_table, |
{ id => $tables{'res'}, |
permanent => 'no', |
permanent => 'no', |
columns => [{ name => 'res_id', |
columns => [{ name => 'res_id', |
type => 'MEDIUMINT UNSIGNED', |
type => 'MEDIUMINT UNSIGNED', |
Line 164 my $res_table_def =
|
Line 226 my $res_table_def =
|
'PRIMARY KEY' => ['res_id'], |
'PRIMARY KEY' => ['res_id'], |
}; |
}; |
|
|
my $action_table = $prefix.'actions'; |
#my $action_table_def = |
my $action_table_def = |
#{ id => $action_table, |
{ id => $action_table, |
# permanent => 'no', |
permanent => 'no', |
# columns => [{ name => 'action_id', |
columns => [{ name => 'action_id', |
# type => 'MEDIUMINT UNSIGNED', |
type => 'MEDIUMINT UNSIGNED', |
# restrictions => 'NOT NULL', |
restrictions => 'NOT NULL', |
# auto_inc => 'yes', }, |
auto_inc => 'yes', }, |
# { name => 'action', |
{ name => 'action', |
# type => 'VARCHAR(100)', |
type => 'VARCHAR(100)', |
# restrictions => 'NOT NULL'}, |
restrictions => 'NOT NULL'}, |
# ], |
], |
# 'PRIMARY KEY' => ['action_id',], |
'PRIMARY KEY' => ['action_id',], |
#}; |
}; |
|
|
|
my $machine_table = $prefix.'machine_table'; |
|
my $machine_table_def = |
my $machine_table_def = |
{ id => $machine_table, |
{ id => $tables{'machine'}, |
permanent => 'no', |
permanent => 'no', |
columns => [{ name => 'machine_id', |
columns => [{ name => 'machine_id', |
type => 'MEDIUMINT UNSIGNED', |
type => 'MEDIUMINT UNSIGNED', |
Line 194 my $machine_table_def =
|
Line 254 my $machine_table_def =
|
'PRIMARY KEY' => ['machine_id',], |
'PRIMARY KEY' => ['machine_id',], |
}; |
}; |
|
|
my $activity_table = $prefix.'activity'; |
|
my $activity_table_def = |
my $activity_table_def = |
{ id => $activity_table, |
{ id => $tables{'activity'}, |
permanent => 'no', |
permanent => 'no', |
columns => [ |
columns => [ |
{ name => 'res_id', |
{ name => 'res_id', |
Line 206 my $activity_table_def =
|
Line 265 my $activity_table_def =
|
type => 'DATETIME', |
type => 'DATETIME', |
restrictions => 'NOT NULL',}, |
restrictions => 'NOT NULL',}, |
{ name => 'student_id', |
{ name => 'student_id', |
type => 'VARCHAR(100) BINARY', |
type => 'MEDIUMINT UNSIGNED', |
restrictions => 'NOT NULL',}, |
restrictions => 'NOT NULL',}, |
{ name => 'action_id', |
{ name => 'action', |
type => 'VARCHAR(100) BINARY', |
type => 'VARCHAR(10)', |
restrictions => 'NOT NULL',}, |
restrictions => 'NOT NULL',}, |
{ name => 'idx', # This is here in case a student |
{ name => 'idx', # This is here in case a student |
type => 'MEDIUMINT UNSIGNED', # has multiple submissions during |
type => 'MEDIUMINT UNSIGNED', # has multiple submissions during |
restrictions => 'NOT NULL', # one second. It happens, trust |
restrictions => 'NOT NULL', # one second. It happens, trust |
auto_inc => 'yes', }, # me. |
auto_inc => 'yes', }, # me. |
{ name => 'machine_id', |
{ name => 'machine_id', |
type => 'VARCHAR(100) BINARY', |
type => 'MEDIUMINT UNSIGNED', |
restrictions => 'NOT NULL',}, |
restrictions => 'NOT NULL',}, |
{ name => 'action_values', |
{ name => 'action_values', |
type => 'MEDIUMTEXT', }, |
type => 'MEDIUMTEXT', }, |
], |
], |
'PRIMARY KEY' => ['res_id','time','student_id','action_id','idx'], |
'PRIMARY KEY' => ['time','student_id','res_id','idx'], |
|
'KEY' => [{columns => ['student_id']}, |
|
{columns => ['time']},], |
}; |
}; |
my @Activity_Tables = ($student_table_def,$res_table_def, |
|
$action_table_def,$machine_table_def, |
|
$activity_table_def); |
|
|
|
|
|
|
my @Activity_Table = ($activity_table_def); |
|
my @ID_Tables = ($student_table_def,$res_table_def,$machine_table_def); |
|
|
## |
## |
## End of table definitions |
## End of table definitions |
## |
## |
|
$logthis->('tables = '.join(',',keys(%tables))); |
|
|
# |
$logthis->('Connectiong to mysql'); |
&Apache::lonmysql::set_mysql_user_and_password($perlvar{'lonSqlUser'}, |
&Apache::lonmysql::set_mysql_user_and_password('www', |
$perlvar{'lonSqlAccess'}); |
$perlvar{'lonSqlAccess'}); |
if (!&Apache::lonmysql::verify_sql_connection()) { |
if (!&Apache::lonmysql::verify_sql_connection()) { |
warn "Unable to connect to MySQL database."; |
warn "Unable to connect to MySQL database."; |
|
$logthis->("Unable to connect to MySQL database."); |
exit 3; |
exit 3; |
} |
} |
|
$logthis->('SQL connection is up'); |
|
|
if ($drop) { &drop_tables(); } |
my $missing_table = &check_for_missing_tables(values(%tables)); |
if (-e $sql_filename) { |
if (-s $gz_sql_filename && ! -s $gz_xml_filename) { |
# if ANY one of the tables does not exist, load the tables from the |
my $backup_modification_time = (stat($gz_sql_filename))[9]; |
# backup. |
$logthis->($gz_sql_filename.' was last modified '. |
my @Current_Tables = &Apache::lonmysql::tables_in_db(); |
localtime($backup_modification_time). |
my %Found; |
'('.$backup_modification_time.')'); |
foreach my $tablename (@Current_Tables) { |
if ($missing_table) { |
foreach my $table (@Activity_Tables) { |
# If the backup happened prior to the last table modification, |
if ($tablename eq $table->{'id'}) { |
# we need to save the tables. |
$Found{$tablename}++; |
if (&latest_table_modification_time() > $backup_modification_time) { |
} |
# Save the current tables in case we need them another time. |
|
$logthis->('Backing existing tables up'); |
|
&backup_tables_as_xml($gz_xml_filename.'.save_'.time,\%tables); |
} |
} |
} |
$time_this->(); |
foreach my $table (@Activity_Tables) { |
&load_backup_sql_tables($gz_sql_filename); |
if (! $Found{$table->{'id'}}) { |
&backup_tables_as_xml($gz_xml_filename,\%tables); |
$time_this->(); |
$time_this->('load backup tables'); |
&load_backup_tables($sql_filename); |
} |
$time_this->('load backup tables'); |
} elsif (-s $gz_xml_filename) { |
last; |
my $backup_modification_time = (stat($gz_xml_filename))[9]; |
|
$logthis->($gz_xml_filename.' was last modified '. |
|
localtime($backup_modification_time). |
|
'('.$backup_modification_time.')'); |
|
if ($missing_table) { |
|
my $table_modification_time = $backup_modification_time; |
|
# If the backup happened prior to the last table modification, |
|
# we need to save the tables. |
|
if (&latest_table_modification_time() > $backup_modification_time) { |
|
# Save the current tables in case we need them another time. |
|
$logthis->('Backing existing tables up'); |
|
&backup_tables_as_xml($gz_xml_filename.'.save_'.time,\%tables); |
} |
} |
} |
$time_this->(); |
|
# We have to make our own tables for the xml format |
|
&drop_tables(); |
|
&create_tables(); |
|
&load_backup_xml_tables($gz_xml_filename,\%tables); |
|
$time_this->('load backup tables'); |
|
} |
} |
} |
|
|
|
## |
|
## Ensure the tables we need exist |
# create_tables does not complain if the tables already exist |
# create_tables does not complain if the tables already exist |
|
$logthis->('creating tables'); |
if (! &create_tables()) { |
if (! &create_tables()) { |
warn "Unable to create tables"; |
warn "Unable to create tables"; |
|
$logthis->('Unable to create tables'); |
exit 4; |
exit 4; |
} |
} |
|
|
|
## |
|
## Read the ids used for various tables |
|
$logthis->('reading id tables'); |
&read_id_tables(); |
&read_id_tables(); |
|
$logthis->('finished reading id tables'); |
|
|
## |
## |
## Do the main bit of work |
## Set up the errors file |
if (-e $newfilename) { |
my $error_fh = IO::File->new(">>$error_filename"); |
my $result = &process_courselog($newfilename); |
|
|
## |
|
## Parse the course log |
|
$logthis->('processing course log'); |
|
if (-s $newfilename) { |
|
my $result = &process_courselog($newfilename,$error_fh,\%tables); |
if (! defined($result)) { |
if (! defined($result)) { |
# Something went wrong along the way... |
# Something went wrong along the way... |
|
$logthis->('process_courselog returned undef'); |
exit 5; |
exit 5; |
} elsif ($result > 0) { |
} elsif ($result > 0) { |
$time_this->(); |
$time_this->(); |
&backup_tables($sql_filename); |
$logthis->('process_courselog returned '.$result.' backing up tables'); |
|
&backup_tables_as_xml($gz_xml_filename,\%tables); |
$time_this->('write backup tables'); |
$time_this->('write backup tables'); |
} |
} |
|
if ($drop_when_done) { &drop_tables(); $logthis->('dropped tables'); } |
} |
} |
|
close($error_fh); |
|
|
## |
## |
## Clean up the filesystem |
## Clean up the filesystem |
## |
|
&Apache::lonmysql::disconnect_from_db(); |
&Apache::lonmysql::disconnect_from_db(); |
unlink($newfilename) if (! $nocleanup); |
unlink($newfilename) if (-e $newfilename && ! $nocleanup); |
|
|
|
## |
|
## Print timing data |
|
$logthis->('printing timing data'); |
if ($time_run) { |
if ($time_run) { |
print "Overall time: ".(Time::HiRes::time - $initial_time).$/; |
my $elapsed_time = Time::HiRes::time - $initial_time; |
|
print "Overall time: ".$elapsed_time.$/; |
print &outputtimes(); |
print &outputtimes(); |
|
$logthis->("Overall time: ".$elapsed_time); |
|
$logthis->(&outputtimes()); |
|
} |
|
|
|
if ($log) { |
|
close LOGFILE; |
|
} |
|
|
|
foreach my $file ($lockfilename, $error_filename,$logfile) { |
|
if (-z $file) { |
|
unlink($file); |
|
} |
} |
} |
|
|
exit 0; # Everything is okay, so end here before it gets worse. |
exit 0; # Everything is okay, so end here before it gets worse. |
|
|
######################################################## |
######################################################## |
######################################################## |
######################################################## |
|
sub table_names { |
|
my ($course,$domain) = @_; |
|
my $prefix = $course.'_'.$domain.'_'; |
|
# |
|
my %tables = |
|
( student =>&Apache::lonmysql::fix_table_name($prefix.'students'), |
|
res =>&Apache::lonmysql::fix_table_name($prefix.'resource'), |
|
machine =>&Apache::lonmysql::fix_table_name($prefix.'machine_table'), |
|
activity=>&Apache::lonmysql::fix_table_name($prefix.'activity'), |
|
); |
|
return %tables; |
|
} |
|
|
|
######################################################## |
|
######################################################## |
## |
## |
## Process Course Log |
## Process Course Log |
## |
## |
Line 308 exit 0; # Everything is okay, so end h
|
Line 437 exit 0; # Everything is okay, so end h
|
# |
# |
# Returns the number of lines in the activity.log file that were processed. |
# Returns the number of lines in the activity.log file that were processed. |
sub process_courselog { |
sub process_courselog { |
my ($inputfile) = @_; |
my ($inputfile,$error_fh,$tables) = @_; |
if (! open(IN,$inputfile)) { |
if (! open(IN,$inputfile)) { |
warn "Unable to open '$inputfile' for reading"; |
warn "Unable to open '$inputfile' for reading"; |
|
$logthis->("Unable to open '$inputfile' for reading"); |
return undef; |
return undef; |
} |
} |
my ($linecount,$insertcount); |
my ($linecount,$insertcount); |
my $dbh = &Apache::lonmysql::get_dbh(); |
my $dbh = &Apache::lonmysql::get_dbh(); |
# |
# |
# Timing variables |
&store_entry(); |
my @RowData; |
|
while (my $line=<IN>){ |
while (my $line=<IN>){ |
# last if ($linecount > 1000); |
# last if ($linecount > 1000); |
# |
# |
Line 327 sub process_courselog {
|
Line 456 sub process_courselog {
|
$linecount++; |
$linecount++; |
# print $linecount++.$/; |
# print $linecount++.$/; |
my ($timestamp,$host,$log)=split(/\:/,$line,3); |
my ($timestamp,$host,$log)=split(/\:/,$line,3); |
$time_this->('splitline'); |
|
# |
# |
# $log has the actual log entries; currently still escaped, and |
# $log has the actual log entries; currently still escaped, and |
# %26(timestamp)%3a(url)%3a(user)%3a(domain) |
# %26(timestamp)%3a(url)%3a(user)%3a(domain) |
Line 339 sub process_courselog {
|
Line 467 sub process_courselog {
|
# get delimiter between timestamped entries to be &&& |
# get delimiter between timestamped entries to be &&& |
$log=~s/\%26(\d{9,10})\%3a/\&\&\&$1\%3a/g; |
$log=~s/\%26(\d{9,10})\%3a/\&\&\&$1\%3a/g; |
$log = &unescape($log); |
$log = &unescape($log); |
$time_this->('translate_and_unescape'); |
|
# now go over all log entries |
# now go over all log entries |
my $machine_id = &get_id($machine_table,'machine',$host); |
if (! defined($host)) { $host = 'unknown'; } |
foreach (split(/\&\&\&/,$log)) { |
my $prevchunk = 'none'; |
$time_this->(); |
foreach my $chunk (split(/\&\&\&/,$log)) { |
my ($time,$res,$uname,$udom,$action,@values)= split(/:/,$_); |
my $warningflag = ''; |
|
my ($time,$res,$uname,$udom,$action,@values)= split(/:/,$chunk); |
if (! defined($res) || $res =~ /^\s*$/) { |
if (! defined($res) || $res =~ /^\s*$/) { |
$res = '/adm/roles'; |
$res = '/adm/roles'; |
$action = 'log in'; |
$action = 'LOGIN'; |
} |
} |
if ($res =~ m|^/prtspool/|) { |
if ($res =~ m|^/prtspool/|) { |
$res = '/prtspool/'; |
$res = '/prtspool/'; |
} |
} |
if (! defined($action) || $action eq '') { |
if (! defined($action) || $action eq '') { |
$action = 'view'; |
$action = 'VIEW'; |
|
} |
|
if ($action !~ /^(LOGIN|VIEW|POST|CSTORE|STORE)$/) { |
|
$warningflag .= 'action'; |
|
print $error_fh 'full log entry:'.$log.$/; |
|
print $error_fh 'error on chunk:'.$chunk.$/; |
|
$logthis->('(action) Unable to parse '.$/.$chunk.$/. |
|
'got '. |
|
'time = '.$time.$/. |
|
'res = '.$res.$/. |
|
'uname= '.$uname.$/. |
|
'udom = '.$udom.$/. |
|
'action='.$action.$/. |
|
'@values = '.join('&',@values)); |
|
next; #skip it if we cannot understand what is happening. |
} |
} |
my $student = $uname.':'.$udom; |
|
$time_this->('split_and_error_check'); |
|
my $student_id = &get_id($student_table,'student',$student); |
|
my $res_id = &get_id($res_table,'resource',$res); |
|
my $action_id = &get_id($action_table,'action',$action); |
|
my $sql_time = &Apache::lonmysql::sqltime($time); |
|
my $values = $dbh->quote(join('',@values)); |
|
$time_this->('get_ids'); |
|
# |
|
my $row = [$res_id, |
|
qq{'$sql_time'}, |
|
$student_id, |
|
$action_id, |
|
qq{''}, # idx |
|
$machine_id, |
|
$values]; |
|
push(@RowData,$row); |
|
$time_this->('push_row'); |
|
# |
# |
} |
my %data = (student => $uname.':'.$udom, |
$time_this->(); |
resource => $res, |
if ($linecount % 100 == 0) { |
machine => $host, |
my $result = &Apache::lonmysql::bulk_store_rows($activity_table, |
action => $action, |
undef, |
time => &Apache::lonmysql::sqltime($time)); |
\@RowData); |
if ($action eq 'POST') { |
$time_this->('bulk_store_rows'); |
$data{'action_values'} = |
if (! defined($result)) { |
$dbh->quote(join('&',map { &escape($_); } @values)); |
warn "Error occured during insert.". |
} else { |
&Apache::lonmysql::get_error(); |
$data{'action_values'} = $dbh->quote(join('&',@values)); |
} |
} |
undef(@RowData); |
my $error = &store_entry($dbh,$tables,\%data); |
|
if ($error) { |
|
$logthis->('error store_entry:'.$error." on %data"); |
|
} |
|
$prevchunk = $chunk; |
} |
} |
} |
} |
if (@RowData) { |
my $result = &store_entry($dbh,$tables); |
$time_this->(); |
if (! defined($result)) { |
my $result = &Apache::lonmysql::bulk_store_rows($activity_table, |
my $error = &Apache::lonmysql::get_error(); |
undef, |
warn "Error occured during insert.".$error; |
\@RowData); |
$logthis->('error = '.$error); |
$time_this->('bulk_store_rows'); |
|
if (! defined($result)) { |
|
warn "Error occured during insert.". |
|
&Apache::lonmysql::get_error(); |
|
} |
|
undef(@RowData); |
|
} |
} |
close IN; |
close IN; |
# print "Number of lines: ".$linecount.$/; |
|
# print "Number of inserts: ".$insertcount.$/; |
|
return $linecount; |
return $linecount; |
} |
} |
|
|
|
|
|
## |
|
## default value for $logthis and $time_this |
|
sub nothing { |
|
return; |
|
} |
|
|
|
## |
|
## Logging routine (look for $log) |
|
## |
|
sub log_to_file { |
|
my ($input)=@_; |
|
print LOGFILE $input.$/; |
|
} |
|
|
## |
## |
## Timing routines |
## Timing routines |
## |
## |
Line 413 sub process_courselog {
|
Line 548 sub process_courselog {
|
my %Timing; |
my %Timing; |
my $starttime; |
my $starttime; |
|
|
sub nothing { |
|
return; |
|
} |
|
|
|
sub time_action { |
sub time_action { |
my ($key) = @_; |
my ($key) = @_; |
if (defined($key)) { |
if (defined($key)) { |
Line 444 sub outputtimes {
|
Line 575 sub outputtimes {
|
|
|
} |
} |
|
|
|
sub latest_table_modification_time { |
|
my $latest_time; |
|
foreach my $table (@Activity_Table,@ID_Tables) { |
|
my %tabledata = &Apache::lonmysql::table_information($table->{'id'}); |
|
next if (! scalar(keys(%tabledata))); # table does not exist |
|
if (! defined($latest_time) || |
|
$latest_time < $tabledata{'Update_time'}) { |
|
$latest_time = $tabledata{'Update_time'}; |
|
} |
|
} |
|
return $latest_time; |
|
} |
|
|
|
sub check_for_missing_tables { |
|
my @wanted_tables = @_; |
|
# Check for missing tables |
|
my @Current_Tables = &Apache::lonmysql::tables_in_db(); |
|
my %Found; |
|
foreach my $tablename (@Current_Tables) { |
|
foreach my $table (@wanted_tables) { |
|
if ($tablename eq $table) { |
|
$Found{$tablename}++; |
|
} |
|
} |
|
} |
|
$logthis->('Found tables '.join(',',keys(%Found))); |
|
my $missing_a_table = 0; |
|
foreach my $table (@wanted_tables) { |
|
if (! $Found{$table}) { |
|
$logthis->('Missing table '.$table); |
|
$missing_a_table = 1; |
|
last; |
|
} |
|
} |
|
return $missing_a_table; |
|
} |
|
|
## |
## |
## Use mysqldump to store backups of the tables |
## Use mysqldump to store backups of the tables |
## |
## |
sub backup_tables { |
sub backup_tables_as_sql { |
my ($sql_filename) = @_; |
my ($gz_sql_filename) = @_; |
my $command = qq{mysqldump --opt loncapa }; |
my $command = qq{mysqldump --quote-names --opt loncapa }; |
|
foreach my $table (@ID_Tables,@Activity_Table) { |
foreach my $table (@Activity_Tables) { |
|
my $tablename = $table->{'id'}; |
my $tablename = $table->{'id'}; |
|
$tablename =~ s/\`//g; |
$command .= $tablename.' '; |
$command .= $tablename.' '; |
} |
} |
$command .= '>'.$sql_filename; |
$command .= '| gzip >'.$gz_sql_filename; |
warn $command.$/; |
$logthis->($command); |
system($command); |
system($command); |
} |
} |
|
|
## |
## |
## Load in mysqldumped files |
## Load in mysqldumped files |
## |
## |
sub load_backup_tables { |
sub load_backup_sql_tables { |
my ($sql_filename) = @_; |
my ($gz_sql_filename) = @_; |
return undef if (! -e $sql_filename); |
if (-s $gz_sql_filename) { |
# Check for .my.cnf |
$logthis->('loading data from gzipped sql file'); |
my $command = 'mysql -e "SOURCE '.$sql_filename.'" loncapa'; |
my $command='gzip -dc '.$gz_sql_filename.' | mysql --database=loncapa'; |
warn $command.$/; |
system($command); |
system($command); |
$logthis->('finished loading gzipped data');; |
|
} else { |
|
return undef; |
|
} |
} |
} |
|
|
## |
## |
## |
## |
## |
## |
sub initialize_configuration { |
|
# Fake it for now: |
|
$perlvar{'lonSqlUser'} = 'www'; |
|
$perlvar{'lonSqlAccess'} = 'localhostkey'; |
|
$perlvar{'lonUsersDir'} = '/home/httpd/lonUsers'; |
|
$perlvar{'lonDefDomain'} = '103'; |
|
} |
|
|
|
sub update_process_name { |
sub update_process_name { |
my ($text) = @_; |
my ($text) = @_; |
$0 = 'parse_activity_log.pl: '.$text; |
$0 = 'parse_activity_log.pl: '.$text; |
Line 496 sub get_filename {
|
Line 658 sub get_filename {
|
} |
} |
|
|
sub create_tables { |
sub create_tables { |
foreach my $table (@Activity_Tables) { |
foreach my $table (@ID_Tables,@Activity_Table) { |
my $table_id = &Apache::lonmysql::create_table($table); |
my $table_id = &Apache::lonmysql::create_table($table); |
if (! defined($table_id)) { |
if (! defined($table_id)) { |
warn "Unable to create table ".$table->{'id'}.$/; |
warn "Unable to create table ".$table->{'id'}.$/; |
warn &Apache::lonmysql::build_table_creation_request($table).$/; |
$logthis->('Unable to create table '.$table->{'id'}); |
|
$logthis->(join($/,&Apache::lonmysql::build_table_creation_request($table))); |
return 0; |
return 0; |
} |
} |
} |
} |
Line 508 sub create_tables {
|
Line 671 sub create_tables {
|
} |
} |
|
|
sub drop_tables { |
sub drop_tables { |
foreach my $table (@Activity_Tables) { |
foreach my $table (@ID_Tables,@Activity_Table) { |
my $table_id = $table->{'id'}; |
my $table_id = $table->{'id'}; |
&Apache::lonmysql::drop_table($table_id); |
&Apache::lonmysql::drop_table($table_id); |
} |
} |
Line 525 sub drop_tables {
|
Line 688 sub drop_tables {
|
my %IDs; |
my %IDs; |
|
|
sub read_id_tables { |
sub read_id_tables { |
foreach my $table (@Activity_Tables) { |
foreach my $table (@ID_Tables) { |
my @Data = &Apache::lonmysql::get_rows($table->{'id'}); |
my @Data = &Apache::lonmysql::get_rows($table->{'id'}); |
|
my $count = 0; |
foreach my $row (@Data) { |
foreach my $row (@Data) { |
$IDs{$table->{'id'}}->{$row->[1]} = $row->[0]; |
$IDs{$table->{'id'}}->{$row->[1]} = $row->[0]; |
} |
} |
} |
} |
|
return; |
} |
} |
|
|
sub get_id { |
sub get_id { |
Line 551 sub get_id {
|
Line 716 sub get_id {
|
$IDs{$table}->{$value}=$Data[0]->[0]; |
$IDs{$table}->{$value}=$Data[0]->[0]; |
return $IDs{$table}->{$value}; |
return $IDs{$table}->{$value}; |
} else { |
} else { |
warn "Unable to retrieve id for $table $fieldname $value".$/; |
$logthis->("Unable to retrieve id for $table $fieldname $value"); |
return undef; |
return undef; |
} |
} |
} |
} |
Line 559 sub get_id {
|
Line 724 sub get_id {
|
|
|
} # End of ID scoping |
} # End of ID scoping |
|
|
|
############################################################### |
|
############################################################### |
|
## |
|
## Save as XML |
|
## |
|
############################################################### |
|
############################################################### |
|
sub backup_tables_as_xml { |
|
my ($filename,$tables) = @_; |
|
open(XMLFILE,"|gzip - > $filename") || return ('error:unable to write '.$filename); |
|
my $query = qq{ |
|
SELECT B.resource, |
|
A.time, |
|
A.idx, |
|
C.student, |
|
A.action, |
|
E.machine, |
|
A.action_values |
|
FROM $tables->{'activity'} AS A |
|
LEFT JOIN $tables->{'res'} AS B ON B.res_id=A.res_id |
|
LEFT JOIN $tables->{'student'} AS C ON C.student_id=A.student_id |
|
LEFT JOIN $tables->{'machine'} AS E ON E.machine_id=A.machine_id |
|
ORDER BY A.time DESC |
|
}; |
|
$query =~ s/\s+/ /g; |
|
my $dbh = &Apache::lonmysql::get_dbh(); |
|
my $sth = $dbh->prepare($query); |
|
if (! $sth->execute()) { |
|
$logthis->('<font color="blue">'. |
|
'WARNING: Could not retrieve from database:'. |
|
$sth->errstr().'</font>'); |
|
return undef; |
|
} else { |
|
my ($res,$sqltime,$idx,$student,$action,$machine,$action_values); |
|
if ($sth->bind_columns(\$res,\$sqltime,\$idx,\$student,\$action, |
|
\$machine,\$action_values)) { |
|
|
|
while ($sth->fetch) { |
|
print XMLFILE '<row>'. |
|
qq{<resource>$res</resource>}. |
|
qq{<time>$sqltime</time>}. |
|
qq{<idx>$idx</idx>}. |
|
qq{<student>$student</student>}. |
|
qq{<action>$action</action>}. |
|
qq{<machine>$machine</machine>}. |
|
qq{<action_values>$action_values</action_values>}. |
|
'</row>'.$/; |
|
} |
|
} else { |
|
warn "Unable to bind to columns.\n"; |
|
return undef; |
|
} |
|
} |
|
close XMLFILE; |
|
return; |
|
} |
|
|
|
############################################################### |
|
############################################################### |
|
## |
|
## load as xml |
|
## |
|
############################################################### |
|
############################################################### |
|
sub load_backup_xml_tables { |
|
my ($filename,$tables) = @_; |
|
my $xmlfh; |
|
open($xmlfh,"cat $filename | gzip -d - |"); |
|
if (! defined($xmlfh)) { |
|
return ('error:unable to read '.$filename); |
|
} |
|
my $dbh = &Apache::lonmysql::get_dbh(); |
|
my $parser = HTML::TokeParser->new($xmlfh); |
|
$parser->xml_mode('1'); |
|
&store_entry(); |
|
my %data; |
|
while (my $token = $parser->get_token()) { |
|
if ($token->[0] eq 'S' && $token->[1] eq 'row') { |
|
undef(%data); |
|
} |
|
foreach my $tag ('resource','time','idx', |
|
'student','action','machine','action_values') { |
|
if ($token->[0] eq 'S' && $token->[1] eq $tag) { |
|
my $text = $parser->get_text("/$tag"); |
|
$data{$tag} = $text; |
|
} |
|
} |
|
if ($token->[0] eq 'E' && $token->[1] eq 'row') { |
|
$data{'action_values'} =qq{'$data{'action_values'}'}; |
|
my $error = &store_entry($dbh,$tables,\%data); |
|
} |
|
} |
|
&store_entry($dbh,$tables); |
|
return; |
|
} |
|
|
|
|
|
####################################################################### |
|
####################################################################### |
|
## |
|
## store_entry - accumulate data to be inserted into the database |
|
## |
|
## Pass no values in to clear accumulator |
|
## Pass ($dbh,\%tables) to initiate storage of values |
|
## Pass ($dbh,\%tables,\%data) to use normally |
|
## |
|
####################################################################### |
|
####################################################################### |
|
|
|
{ |
|
my @rows; |
|
my $max_row_count = 100; |
|
|
|
sub store_entry { |
|
if (! @_) { |
|
undef(@rows); |
|
return ''; |
|
} |
|
my ($dbh,$tables,$data) = @_; |
|
return if (! defined($tables)); |
|
if (defined($data)) { |
|
my $error; |
|
foreach my $field ('student','resource','action','time') { |
|
if (! defined($data->{$field}) || $data->{$field} eq ':' || |
|
$data->{$field}=~ /^\s*$/) { |
|
$error.=$field.','; |
|
} |
|
} |
|
if ($error) { $error=~s/,$//; return $error; } |
|
# |
|
my $student_id = &get_id($tables->{'student'},'student', |
|
$data->{'student'}); |
|
my $res_id = &get_id($tables->{'res'}, |
|
'resource',$data->{'resource'}); |
|
my $machine_id = &get_id($tables->{'machine'}, |
|
'machine',$data->{'machine'}); |
|
my $idx = $data->{'idx'}; if (! $idx) { $idx = "''"; } |
|
# |
|
push(@rows,[$res_id, |
|
qq{'$data->{'time'}'}, |
|
$student_id, |
|
qq{'$data->{'action'}'}, |
|
$idx, |
|
$machine_id, |
|
$data->{'action_values'}]); |
|
} |
|
if (defined($tables) && |
|
( (! defined($data) && scalar(@rows)) || scalar(@rows)>$max_row_count) |
|
){ |
|
# Store the rows |
|
my $result = |
|
&Apache::lonmysql::bulk_store_rows($tables->{'activity'}, |
|
undef, |
|
\@rows); |
|
if (! defined($result)) { |
|
my $error = &Apache::lonmysql::get_error(); |
|
warn "Error occured during insert.".$error; |
|
return $error; |
|
} |
|
undef(@rows); |
|
return $result if (! defined($data)); |
|
} |
|
|
|
return ''; |
|
} |
|
|
|
} # end of scope for &store_entry |
|
|
############################################################### |
############################################################### |
############################################################### |
############################################################### |
Line 578 sub unescape {
|
Line 910 sub unescape {
|
$str =~ s/%([a-fA-F0-9][a-fA-F0-9])/pack("C",hex($1))/eg; |
$str =~ s/%([a-fA-F0-9][a-fA-F0-9])/pack("C",hex($1))/eg; |
return $str; |
return $str; |
} |
} |
|
|