diff --git a/html/inc/edit-sh-bu.php b/html/inc/edit-sh-bu.php
index c297476..9419d71 100644
--- a/html/inc/edit-sh-bu.php
+++ b/html/inc/edit-sh-bu.php
@@ -8,9 +8,9 @@
[$table, $colIdxs] = rpc( 'f:getbu_list_all', "where BUID=$db_id" );
- if(! array_key_exists( $db_id, $table) ) return "
Item not found $db_id
";
+ if( count($table)==0 ) return "Item not found $db_id
";
- $row = $table[$db_id];
+ $row = $table[0]; //we're expecting the first and only item
$bun = $row[ $colIdxs['BUName']];
$bus = $row[ $colIdxs['Dir_Src']];
diff --git a/html/inc/new-file-bu.php b/html/inc/new-file-bu.php
index 12df32a..004b9f6 100644
--- a/html/inc/new-file-bu.php
+++ b/html/inc/new-file-bu.php
@@ -16,8 +16,8 @@
[$table, $colIdxs] = rpc( 'f:getbu_list_all', "where BUID=$id" );
- if( !array_key_exists( $id, $table ) ) return null;
- $row = $table[$id];
+ if( count( $table )==0 ) return null;
+ $row = $table[0];
return array(
$row[ $colIdxs['BUName']],
diff --git a/html/svc/backup.php b/html/svc/backup.php
index 875dc83..6a0410b 100644
--- a/html/svc/backup.php
+++ b/html/svc/backup.php
@@ -1,9 +1,18 @@
0) return;
$html_rows='';
+ $cnt=0;
$results = $db->query('SELECT BUID, BUName, Dir_Src, Dir_Dest, Files_Ex, BuType, BuRunning, BU_DATE, BU_TIME, BU_REPEAT, BuError, LastRunDt FROM BULIST '.$orderby);
while ($row = $results->fetchArray()) {
- $allRows[ $row['BUID'] ] = array(
+
+ //debug_println("found buid = ({$row['BUID']})", DEBUG_MED );
+
+ $allRows[ $cnt++ ] = array(
$row['BUID'],
$row['BUName'],
$row['Dir_Src'],
@@ -53,6 +57,38 @@
}
/**
+ * Return a single row. Indexes are the same as the above function and must have been retrieved first.
+ */
+function getbu_row( $buid, $colIdxs ){
+
+ global $g_database_path;
+ $buid = (int)$buid;
+
+ $db = new SQLite3($g_database_path, SQLITE3_OPEN_READONLY );
+
+ $html_rows='';
+ $results = $db->query('SELECT BUID, BUName, Dir_Src, Dir_Dest, Files_Ex, BuType, BuRunning, BU_DATE, BU_TIME, BU_REPEAT, BuError, LastRunDt FROM BULIST where BUID='.$buid);
+ if($row = $results->fetchArray()) {
+ return array(
+ $row['BUID'],
+ $row['BUName'],
+ $row['Dir_Src'],
+ $row['Dir_Dest'],
+ $row['Files_Ex'],
+ $row['BuType'],
+ $row['BuRunning'],
+ $row['BU_DATE'],
+ $row['BU_TIME'],
+ $row['BU_REPEAT'],
+ $row['BuError'],
+ $row['LastRunDt']
+ );
+ }
+
+}
+
+
+/**
* Provides a simple json array of the BU state info for the UI to poll changes.
*/
function getbu_list_state(){
diff --git a/html/svc/main.php b/html/svc/main.php
index 2979813..aac1520 100644
--- a/html/svc/main.php
+++ b/html/svc/main.php
@@ -1,6 +1,23 @@
/dev/null 2>>/root/logs/sys-update.log
+$debug_level = 0;
+const DEBUG_LOW = 0;
+const DEBUG_MED = DEBUG_LOW+1;
+const DEBUG_HIGH = DEBUG_MED+1;
+const DEBUG_VHIGH = DEBUG_HIGH+1;
+
+
+function debug_print( $msg, $dl=0 ){
+ global $debug_level;
+ if($dl>$debug_level) return;
+ file_put_contents( __DIR__."/../data/rpc_op.log", $msg, FILE_APPEND);
+}
+
+function debug_println( $msg, $dl=0 ){
+ global $debug_level;
+ if($dl>$debug_level) return;
+ debug_print( date_format( date_create("now", new DateTimeZone("UTC")), 'Y-m-d H:i:s') . " $msg\n", $dl );
+}
//$argv[0] is always the name that was used to run the script.
if(count($argv)>1){
@@ -8,17 +25,31 @@
include __DIR__."/main-svc.php";
if($argv[1]==='stop'){
+ debug_println( "stopping" );
$rtn = run_main_command( 'f:cli-stop' );
print_r( $rtn );
exit(0);
}
if($argv[1]==='reload'){
+ debug_println( "reloading" );
$rtn = run_main_command( 'f:cli-reload' );
print_r( $rtn );
exit(0);
}
+ if($argv[1]==='debug_inc'){
+ debug_println( "debug_inc" );
+ run_main_command( 'f:debug_inc' );
+ exit(0);
+ }
+
+ if($argv[1]==='debug_dec'){
+ debug_println( "debug_dec" );
+ run_main_command( 'f:debug_dec' );
+ exit(0);
+ }
+
$rtn = run_main_command( $argv[1] );
print_r( $rtn );
exit(0);
@@ -41,11 +72,7 @@
include __DIR__."/../lib/sysVcom.php";
-function debug_print( $msg ){
- file_put_contents( __DIR__."/../data/rpc_op.log", $msg, FILE_APPEND);
-}
-
-debug_print( date_format( date_create("now", new DateTimeZone("UTC")), 'Y-m-d H:i:s') . " started\n");
+debug_println("started");
//create the comms file
touch('BU-commander');
@@ -57,7 +84,7 @@
rpc_setGlobals( 'max_chan', 8182 );
rpc_setGlobals( 'main_pid', getmypid() );
-print("rpc using ". __DIR__ . "/$g_procName with key = $g_k_caller\n" );
+print("rpc using ". __DIR__ . "/$g_procName\n" );
// problem is that the paths are not correct wrt this location, so we need to adjust them
include __DIR__."/../inc/paths.php";
@@ -68,13 +95,25 @@
include "db-reads.php";
include "db-writes.php";
include "file-line-reader.php";
+include "table-cache.php";
-// get the latest data and cache it
$g_bu_states = null;
-$g_bu_table = array();
-$g_bu_colIdxs = array();
-getbu_list_all( $g_bu_table, $g_bu_colIdxs, 'order by BU_DATE desc, BU_TIME desc' );
+
+
+// maximum number of forks allowed
+$g_fork_max = 4;
+// currently running fork count
$g_fork_cnt=0;
+//a queue of backup IDs waiting to commence
+$g_bu_queue = array();
+
+function isQueued( $buid ){
+ global $g_bu_queue;
+ foreach( $g_bu_queue as $q ){
+ if($q==$buid) return true;
+ }
+ return false;
+}
while(true){
@@ -87,6 +126,8 @@
$bDidWork |= check_rpc_clients();
$bDidWork |= check_bu_sched();
+ $bDidWork |= check_bu_queue();
+
if(!$bDidWork){
time_nanosleep( 0, 100000000);
@@ -95,107 +136,133 @@
}
/**
- * Clears all cached data. This is called after any DB updates. It can be called at any time since loading is lazy.
- */
-function clear_cached_data(){
-
- global $g_bu_states;
- global $g_bu_table;
-
- $g_bu_states = null;
- $g_bu_table = array();
-}
-
-/**
- * Make sure that the main table data is populated
- */
-function ensure_cached_table(){
- global $g_bu_table;
- if(count($g_bu_table)>0) return;
- $colIdxs = null; //we don't ever reload these
- getbu_list_all( $g_bu_table, $colIdxs, 'order by BU_DATE desc, BU_TIME desc' );
-}
-
-/**
* Return true if the backup was kicked off
*/
function check_bu_sched(){
global $g_bu_table;
global $g_bu_colIdxs;
- global $g_fork_cnt;
-
- if($g_fork_cnt>0){
- //we limit to a single fork for now
- return;
- }
-
- //get the first non null entry which is not running
- $keys = array_keys( $g_bu_table );
- $idfound = 0;
- foreach ($keys as $key) {
- if( $g_bu_table[$key][ $g_bu_colIdxs['BU_DATE'] ] == '' ) continue; //skip anything without a date
- if( $g_bu_table[$key][ $g_bu_colIdxs['BuRunning'] ] == 1 ) continue; //skip any already running
- if( $g_bu_table[$key][ $g_bu_colIdxs['BuError'] ] != null ) continue; //skip entries with problems
- $idfound = $g_bu_table[$key][ $g_bu_colIdxs['BUID'] ];
+ //get the first non null entry which is not running
+ $idfound = 0;
+ $row = null;
+ for( $i=0; $i new DateTime('now', new DateTimeZone("UTC") ) ){
//...no it's not
return;
}
-
- kickoff_backup( $idfound );
+
+ [ $res, $state ] = queue_backup( $idfound );
- //print( "check after forked: running = ". $g_bu_table[$idfound][ $g_bu_colIdxs['BuRunning'] ] . "\n");
+ return $state!=='queued'; //return true if something some work was started
+}
+function check_bu_queue(){
+ $res = dequeue_backup();
+ if($res==null) return false;
+ if($res[1]=='queued') return false;
return true;
}
/**
+ * Adds a backup ID to the queue. It is kicked off immediately if a slot is available.
+ */
+function queue_backup( $buid, $scheduled = true, $bu_test = false ){
+
+ global $g_bu_queue;
+
+ debug_println("(queue_backup) queue backup($buid)", DEBUG_MED );
+
+ array_push( $g_bu_queue, array($buid, $scheduled, $bu_test) );
+ $res = dequeue_backup( );
+ if($res!=null){
+ return $res; //either running or error.
+ }
+ debug_println("(queue_backup) backup($buid) in queue", DEBUG_MED );
+ return array("ok:", "queued" ); //will be run later
+}
+
+/**
+ * Removes the next back up item from the front of the queue and kicks it off if there are enough
+ * free forks. Returns the result of the kick-off or null if nothing to do.
+ */
+function dequeue_backup( ){
+
+ global $g_bu_queue;
+ global $g_fork_cnt;
+ global $g_fork_max;
+
+ if( count($g_bu_queue)==0){
+ return null; //nothing to do
+ }
+
+ if($g_fork_cnt>=$g_fork_max){
+ //There are no available forks
+ return null;
+ }
+ $bu_item = array_shift( $g_bu_queue );
+ debug_println("(dequeue_backup) dequeued backup($buid)", DEBUG_MED );
+ return kickoff_backup( $bu_item );
+}
+
+/**
* Kicks off a back up. Returns immediately. Deligates the work to a client fork.
*
* $scheduled:
* true if it is run according to the $schedule info in the db
* false if it has been manually initiated
*/
-function kickoff_backup( $buid, $scheduled = true, $bu_test = false ){
-
- ensure_cached_table();
+function kickoff_backup( $bu_item ){
+
+ [$buid, $scheduled, $bu_test] = $bu_item;
+
+ debug_println("(kickoff_backup) backup($buid) to run now", DEBUG_MED );
global $g_bu_table;
global $g_bu_colIdxs;
global $g_fork_cnt;
- $isRunning = $g_bu_table[$buid][ $g_bu_colIdxs['BuRunning'] ];
+ $row = &find_rec_by_id( $buid );
+
+ $isRunning = $row[ $g_bu_colIdxs['BuRunning'] ];
if( !$bu_test && $isRunning==1) return array("e:Error", "Backup already running" );
$i=0;
- $bu_row[$i++] = $g_bu_table[$buid][ $g_bu_colIdxs['BUName'] ];
- $bu_row[$i++] = $g_bu_table[$buid][ $g_bu_colIdxs['Dir_Src'] ];
- $bu_row[$i++] = $g_bu_table[$buid][ $g_bu_colIdxs['Dir_Dest'] ];
- $bu_row[$i++] = $g_bu_table[$buid][ $g_bu_colIdxs['Files_Ex'] ];
- $bu_row[$i++] = $g_bu_table[$buid][ $g_bu_colIdxs['BuType'] ];
+ $bu_row[$i++] = $row[ $g_bu_colIdxs['BUName'] ];
+ $bu_row[$i++] = $row[ $g_bu_colIdxs['Dir_Src'] ];
+ $bu_row[$i++] = $row[ $g_bu_colIdxs['Dir_Dest'] ];
+ $bu_row[$i++] = $row[ $g_bu_colIdxs['Files_Ex'] ];
+ $bu_row[$i++] = $row[ $g_bu_colIdxs['BuType'] ];
$bu_row[$i++] = $isRunning;
// we should be able to use pcntl_fork()
$pid = pcntl_fork();
- if ( $pid == -1 ) {
- debug_print("fork failed");
+ if ( $pid == -1 ) {
+ debug_println("fork failed");
return array("e:Error", "fork failed" );
}
if ( $pid ) {
// parent process
- debug_print( "parentpid = ". getmypid() . "\n" );
- $g_bu_table[$buid][ $g_bu_colIdxs['BuRunning'] ] = 1;
+ debug_println( "(kickoff_backup) bu($buid) running, parent(". getmypid().")", DEBUG_MED );
+ set_bu_running_db( $buid, true );
+
+ $row[ $g_bu_colIdxs['BuRunning'] ] = 1;
return array("ok:", "running" );
} else {
@@ -203,11 +270,10 @@
// child process
$g_fork_cnt++;
- $g_bu_table = array(); //data here is unique to the parent
+ $g_bu_table = array(); //for safety
$pid = getmypid();
- debug_print( "childpid = $pid\n" );
-
- debug_print( "kickoff_backup for $buid\n" );
+ debug_println( "child($pid) running backup($buid)" );
+
include_once __DIR__."/backup.php";
$bu_result = '';
global $g_data_dir;
@@ -215,7 +281,7 @@
$bu_result = run_backup_block( $buid, $bu_row, $bu_test );
- debug_print( "backup ended for $buid, result = $bu_result\n" );
+ debug_println( "child($pid) finished bu($buid)" );
$obj = rpc( 'm:fork-complete', $pid, "backup-complete", $buid, $scheduled ); //we use our rpc mechanism to notify parent
//print_r( $obj );
//print( "\n" );
@@ -244,25 +310,33 @@
if($caller_id==0) return false; //nothing to do
try{
+ if( !is_array($obj_in[0])) {
+ debug_println("req {$obj_in[0]}", DEBUG_MED );
+ }
$obj_out = handle_rpc_req( $obj_in );
+ if($obj_out==null) $obj_out = array('ok:'); //generic return value
}catch( Exception $e ){
- debug_print( "Exception: " . $e->getMessage() ."\n" );
+ debug_println( "Exception: " . $e->getMessage() );
$obj_out = array( 'e:Exception', $e );
}
-
+
+ if( !is_array($obj_out[0])) {
+ debug_println("reply {$obj_out[0]}", DEBUG_MED );
+ }
+
//send msg back
rpc_reply( $caller_id, $obj_out );
if( $obj_in[0] === 'm:fork-complete' ){
//clean up child process
- debug_print("clean up fork child pid = {$obj_in[1]} \n");
+ debug_println("wait for child({$obj_in[1]})", DEBUG_MED );
$status;
pcntl_waitpid( $obj_in[1], $status );
$g_fork_cnt--;
- debug_print("clean up done pid = {$obj_in[1]} \n");
+ debug_println("child({$obj_in[1]}) complete.", DEBUG_MED );
//finish any requirements for specific child processes (db updates etc)
if( count($obj_in)>2 && $obj_in[2]="backup-complete"){
@@ -274,28 +348,26 @@
}
+
/**
* Called by the parent after a client notified that a backup completed.
*/
function backupComplete( $buid, $scheduled ){
- global $g_bu_table;
global $g_bu_colIdxs;
-
- //reload the table data - client updates the error field (might move this)
- clear_cached_data();
- $nothanks = null;
- getbu_list_all( $g_bu_table, $nothanks, 'order by BU_DATE desc, BU_TIME desc' );
+
+ //reload the table data (in case child modified error field)
+ $row = reload_cached_rec( $buid );
if($scheduled){
//if DB error, do nothing
- $err = $g_bu_table[$buid][ $g_bu_colIdxs['BuError'] ];
+ $err = $row[ $g_bu_colIdxs['BuError'] ];
if($err==''){
//update date and time according to the repeat field.
- $date = date_create( $g_bu_table[$buid][ $g_bu_colIdxs['BU_DATE'] ] );
+ $date = date_create( $row[ $g_bu_colIdxs['BU_DATE'] ] );
$dt_new = null;
- switch( $g_bu_table[$buid][ $g_bu_colIdxs['BU_REPEAT'] ] ){
+ switch( $row[ $g_bu_colIdxs['BU_REPEAT'] ] ){
case 'M':
//monthly
date_add( $date, new DateInterval('P1M') );
@@ -315,15 +387,20 @@
break;
}
- $g_bu_table[$buid][ $g_bu_colIdxs['BU_DATE'] ] = $dt_new;
+ $row[ $g_bu_colIdxs['BU_DATE'] ] = $dt_new;
set_sched_date_db( $buid, $dt_new );
}
+
+ set_bu_running_db( $buid, false );
+
}//if manually kicked off - nothing to do
- //TODO: remove the outputfiles
-
+ $row = reload_cached_rec( $buid );
+
+ debug_println("(backupComplete) bu($buid) done", DEBUG_MED );
+
}
@@ -338,6 +415,7 @@
global $g_data_dir;
global $g_shutdown;
+ global $debug_level;
if( 'f:getbu_list_state' !== $obj[0] ){
//print_r( $obj );
@@ -365,7 +443,7 @@
//($table, $colIdxs) = rpc( 'f:getbu_list_all', "order by BUID" );
$table = array();
$colIdxs = array();
- getbu_list_all( $table, $colIdxs, $obj[1] );
+ getbu_list_all( $table, $colIdxs, $obj[1] );
return array( $table, $colIdxs );
case 'f:getbu_list_state':
@@ -377,40 +455,47 @@
return $g_bu_states;
case 'f:save_bu_schedule':
+ $buid = $obj[1]['id'];
$err = save_bu_schedule( $obj[1] );
- clear_cached_data();
+ reload_cached_rec( $buid );
return array($err);
case 'f:save_bu_item':
+ $buid = $obj[2];
//[$err] = rpc( 'f:save_bu_item', $postVars, $bu_id );
- $err = save_bu_item( $obj[1], $obj[2] );
- clear_cached_data();
+ $err = save_bu_item( $obj[1], $buid );
+ reload_cached_rec( $buid );
return array($err);
case 'f:set_bu_error':
- //print("re/setting backup error\n");
+ $buid = $obj[1];
$err = set_bu_error_db( $obj[1], $obj[2] );
- clear_cached_data();
+ reload_cached_rec( $buid );
return array($err);
case 'f:set_bu_running':
+ $buid = $obj[1];
$err = set_bu_running_db( $obj[1], $obj[2] );
- clear_cached_data();
+ reload_cached_rec( $buid );
return array($err);
case 'f:kickoff-backup':
$buid = $obj[1];
$isTest = $obj[2];
- return kickoff_backup( $buid, false, $isTest );
+ $res = queue_backup( $buid, false, $isTest );
+ if($res[1]==='queued') return array('ok:','running'); //ToDo: return queued and client to handle
+ return $res;
case 'f:set_last_run_date':
//[$err] = rpc( 'f:set_last_run_date', $buid, $str_dt );
- $err = set_last_run_date_db( $obj[1], $obj[2] );
- clear_cached_data();
+ $buid = $obj[1];
+ $err = set_last_run_date_db( $buid, $obj[2] );
+ reload_cached_rec( $buid );
return array($err);
case 'f:cli-reload':
clear_cached_data();
+ ensure_cached_table();
return array('ok:reload');
case 'f:cli-stop':
@@ -418,11 +503,21 @@
return array('ok:shutdown');
case 'f:delete_bu_item':
- //[$err] = rpc( 'f:delete_bu_item', $buid );
- $err = delete_bu_item_db( $obj[1] );
- clear_cached_data();
+ $buid = $obj[1];
+ $err = delete_bu_item_db( $buid );
+ delete_cached_rec( $buid );
return array($err);
+ case 'f:debug_inc': //increment the debug level
+ $debug_level++;
+ debug_println( "debug level $debug_level" );
+ return array('ok:');
+
+ case 'f:debug_dec': //decrement the debug level
+ $debug_level--;
+ if($debug_level<0) $debug_level=0;
+ debug_println( "debug level $debug_level" );
+ return array('ok:');
}
diff --git a/html/svc/table-cache.php b/html/svc/table-cache.php
new file mode 100644
index 0000000..54333ff
--- /dev/null
+++ b/html/svc/table-cache.php
@@ -0,0 +1,88 @@
+0) return;
+ getbu_list_all( $g_bu_table, $colIdxs, 'order by BU_DATE desc, BU_TIME desc' );
+ debug_println("cache loaded", DEBUG_MED );
+}
+
+
+/**
+ * Reload a single record by buid. The new row is returned by ref.
+ */
+function &reload_cached_rec( $buid ){
+
+ global $g_bu_table;
+ global $g_bu_colIdxs;
+
+ $row = getbu_row( $buid, $g_bu_colIdxs );
+
+ for( $i=0; $i
diff --git a/notes.txt b/notes.txt
index 4080ef5..c150247 100644
--- a/notes.txt
+++ b/notes.txt
@@ -27,9 +27,21 @@
reset the running flag:
-[root@server2 svc]# sqlite3 ../data/data.db
-update BULIST set BuRunning=false;
-.quit
+ [root@server2 svc]# sqlite3 ../data/data.db
+ update BULIST set BuRunning=false;
+ .quit
+Command line
+------------
+ Increment debug level
+ php main.php debug_inc
+ Decrement debug level
+ php main.php debug_dec
+ Debug levels are:
+ const DEBUG_LOW = 0;
+ const DEBUG_MED = DEBUG_LOW+1;
+ const DEBUG_HIGH = DEBUG_MED+1;
+ const DEBUG_VHIGH = DEBUG_HIGH+1;
+