<?php /* * This is the backup process CLI application. * * Rather than use 2 threads, I've decided to poll. This is because the threading in PHP is no longer supported. * Instead we should use parallel: * https://www.php.net/manual/en/parallel.run.php * which I don't like. Basically, it doesn't look like you can share code (by way of a reference)?!! * * So we are polling the RPC layer every 0.1secs. In practice, this will be plenty for the backup app. * */ include __DIR__."/../lib/sysVcom.php"; //create the comms file touch('BU-commander'); //remove all queues at startup clear_IPC( '../svc/BU-commander' ); rpc_setGlobals( 'procName', '../svc/BU-commander' ); rpc_setGlobals( 'max_chan', 8182 ); // problem is that the paths are not correct wrt this location, so we need to adjust them include __DIR__."/../inc/paths.php"; $g_database_path = '../' . $g_database_path; include_once "../inc/bu_list_content.php"; // get the latest data and cache it $g_bu_table = array(); $g_bu_colIdxs = array(); getbu_list_all( $g_bu_table, $g_bu_colIdxs ); //print_r($g_bu_table); //print_r( $g_bu_table[18][ $g_bu_colIdxs['Dir_Dest'] ]); while(true){ $bDidWork = false; $bDidWork |= check_rpc_clients(); $bDidWork |= check_bu_sched(); if(!$bDidWork){ time_nanosleep( 0, 100000000); } } /** * Return true if the backup was kicked off */ function check_bu_sched(){ global $g_bu_table; global $g_bu_colIdxs; //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'] ]; } if($idfound==0) return; // is it time...? $str_bu_dt = $g_bu_table[$idfound][ $g_bu_colIdxs['BU_DATE']] . ' ' . $g_bu_table[$idfound][ $g_bu_colIdxs['BU_TIME']] . ':00'; $schedTime = new DateTime( $str_bu_dt, new DateTimeZone("UTC") ); if( $schedTime > new DateTime('now', new DateTimeZone("UTC") ) ){ //...no it's not return; } kickoff_backup( $idfound ); print( "check after forked: running = ". $g_bu_table[$idfound][ $g_bu_colIdxs['BuRunning'] ] . "\n"); return true; } /** * 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 ){ global $g_bu_table; global $g_bu_colIdxs; // we should be able to use pcntl_fork() $pid = pcntl_fork(); if ( $pid == -1 ) { print("fork failed"); return; } $bu_test = false; $bu_type = $g_bu_table[$buid][ $g_bu_colIdxs['BuType'] ]; $bu_src = $g_bu_table[$buid][ $g_bu_colIdxs['Dir_Src'] ]; $bu_dest = $g_bu_table[$buid][ $g_bu_colIdxs['Dir_Dest'] ]; $exFiles = $g_bu_table[$buid][ $g_bu_colIdxs['Files_Ex'] ]; if ( $pid ) { // parent process print( "parentpid = ". getmypid() . "\n" ); $g_bu_table[$buid][ $g_bu_colIdxs['BuRunning'] ] = 1; return; } else { // child process $pid = getmypid(); print( "childpid = $pid\n" ); print( "kickoff_backup for $buid\n" ); include_once __DIR__."/../inc/bu-common.php"; $bu_result = ''; global $g_data_dir; $g_data_dir = __DIR__."/../data"; if($bu_type==1 ){ $bu_result = run_backup_DB( $bu_test, $buid, $bu_src, $bu_dest ); }else{ $bu_result = run_backup_F( $bu_test, $buid, $bu_src, $bu_dest, $exFiles ); } print( "backup ended for $buid, result = $bu_result\n" ); $obj = rpc( 'm:fork-complete', $pid, "backup-complete", $buid, $scheduled ); //we use our rpc mechanism to notify parent //print_r( $obj ); //print( "\n" ); exit(0); } } $msgArr = null; /** * Return true if a request was found and handled */ function check_rpc_clients(){ $caller_id = 0; $obj = rpc_listen( $caller_id, $msgArr, false ); if($caller_id==0){ //nothing to do return false; } $obj = handle_rpc_req( $obj ); //send msg back rpc_reply( $caller_id, $obj ); //done with the message unset( $msgArr[$caller_id] ); if( $obj[0] == 'm:fork-complete' ){ // this is required to clean up the child process $status; pcntl_waitpid( $obj[1], $status ); if( $obj[2]="backup-complete"){ backupComplete( $obj[3], $obj[4] ); } } return true; } /** * Called by the parent after a client notified that a backup completed. */ function backupComplete( $buid, $scheduled ){ //reload the table data - client updates the error field global $g_bu_table; global $g_bu_colIdxs; $g_bu_table = array(); getbu_list_all( $g_bu_table ); include_once __DIR__."/../inc/bu-common.php"; if($scheduled){ //if DB error, do nothing $err = $g_bu_table[$buid][ $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'] ] ); $dt_new = null; switch( $g_bu_table[$buid][ $g_bu_colIdxs['BU_REPEAT'] ] ){ case 'M': //monthly date_add( $date, new DateInterval('P1M') ); $dt_new = date_format( $date, "Y-m-d"); break; case 'W': //weekly date_add( $date, new DateInterval('P1W') ); $dt_new = date_format( $date, "Y-m-d"); break; case 'D': //daily date_add( $date, new DateInterval('P1D') ); $dt_new = date_format( $date, "Y-m-d"); break; } $g_bu_table[$buid][ $g_bu_colIdxs['BU_DATE'] ] = $dt_new; set_sched_date( $buid, $dt_new ); } }//if manually kicked off - nothing to do //TODO: remove the outputfiles $dt = gmdate("Y-m-d H:i:s"); set_last_run_date( $buid, $dt ); $g_bu_table[$buid][ $g_bu_colIdxs['LastRunDt'] ] = $dt; } function handle_rpc_req( $obj ){ switch( $obj[0] ){ case 'm:fork-complete': //send the same message back to the caller return $obj; } //do something with the data print_r( $obj ); array_unshift( $obj, "Greetings" ); return $obj; } ?>