class Pman_Core_Notify extends Pman
{
- static $cli_desc = "Send out notification emails (usually from cron)";
+ static $cli_desc = "Runs the notification queue (usually from cron)
+ Normally used to sends out emails to anyone in the notification list.
+
+ /etc/cron.d/pman-core-notify
+ * * * * * www-data /usr/bin/php /home/gitlive/web.mtrack/admin.php Core/Notify > /dev/null
+
+";
static $cli_opts = array(
'debug' => array(
'old' => array(
'desc' => 'Show old messages..',
'default' => 0,
- 'short' => 'l',
+ 'short' => 'o',
'min' => 0,
'max' => 0,
- )
+ ),
+ 'force' => array(
+ 'desc' => 'Force redelivery, even if it has been sent before or not queued...',
+ 'default' => 0,
+ 'short' => 'f',
+ 'min' => 0,
+ 'max' => 0,
+ ),
+ 'generate' => array(
+ 'desc' => 'Generate notifications for a table, eg. cash_invoice',
+ 'default' => '',
+ 'short' => 'g',
+ 'min' => 0,
+ 'max' => 1,
+ ),
);
var $target = 'Core/NotifySend';
var $evtype = ''; // any notification...
// this script should only handle EMAIL notifications..
-
+ var $force = false;
function getAuth()
{
$ff = HTML_FlexyFramework::get();
}
//date_default_timezone_set('UTC');
// phpinfo();exit;
- $showold = $opts['old'];
+ $showold = !empty($opts['old']);
if (!empty($opts['old'])) {
$opts['list'] = 1; // force listing..
}
+ $this->force = empty($opts['force']) ? 0 : 1;
+
+ if (!empty($opts['send-to'])) {
+ $this->send_to = $opts['send-to'];
+ }
+
+ $w = DB_DataObject::factory('core_notify_recur');
+ if (is_a($w, 'DB_DataObject')) {
+ $w->generateNotifications();
+ }
+ if (!empty($opts['generate'])) {
+ $w = DB_DataObject::factory($opts['generate']);
+ if (is_a($w, 'DB_DataObject')) {
+ $w->generateNotifications();
+ }
+ exit;
+
+
+ }
+
+ //DB_DataObject::debugLevel(1);
$w = DB_DataObject::factory($this->table);
if (!$showold) {
- $w->whereAdd('act_when > sent'); // eg.. sent is not valid..
- $w->whereAdd('act_when < NOW()'); // eg.. not if future..
+
+ // standard
+
+ //$w->whereAdd('act_when > sent'); // eg.. sent is not valid..
+ $w->whereAdd("sent < '1970-01-01'"); // eg.. sent is not valid..
+
+ $w->whereAdd('act_start > NOW() - INTERVAL 14 DAY'); // ignore the ones stuck in the queue
+ if (!$this->force) {
+ $w->whereAdd('act_when < NOW()'); // eg.. not if future..
+ }
$w->orderBy('act_when ASC'); // oldest first.
$w->limit(1000); // we can run 1000 ...
$o = $w->object();
- echo "$e->id : $w->person_id_email email : ".
+ echo "$w->id : $w->person_id_email email : ".
$o->toEventString()." ". $w->status() . "\n";
}
exit;
}
-
+ $pushed = array();
while (true) {
if (empty($ar)) {
break;
}
+
$p = array_shift($ar);
if (!$this->poolfree()) {
array_unshift($ar,$p); /// put it back on..
continue;
}
if ($this->poolHasDomain($p->person_id_email)) {
+ if (in_array($p->person_id_email, $pushed)) {
+ // it's been pushed to the end, and nothing has
+ // been pushed since.
+ // give up, let the next run sort it out.
+ break;
+ }
+
$ar[] = $p; // push it on the end..
+
+ $pushed[] = $p->person_id_email;
+
echo "domain {$p->person_id_email} already on queue, pushing to end.\n";
sleep(3);
continue;
$this->run($p->id,$p->person_id_email);
+ $pushed = array();
+
}
+
+ // we should have a time limit here...
while(count($this->pool)) {
$this->poolfree();
sleep(3);
die("DONE\n");
}
- function run($id, $email)
+ function run($id, $email, $cmdOpts="")
{
// phpinfo();exit;
$tn = tempnam(ini_get('session.save_path'),'stdout') . '.stdout';
$php = $_SERVER["_"];
$sn = $_SERVER["SCRIPT_NAME"];
- $cwd = $sn[0] == '/' ? dirname($sn) : dirname(realpath(getcwd() . $sn)); // same as run on.. (so script should end up being same relatively..)
+ $cwd = $sn[0] == '/' ? dirname($sn) : dirname(realpath(getcwd() . '/'. $sn)); // same as run on.. (so script should end up being same relatively..)
$app = $cwd . '/' . basename($_SERVER["SCRIPT_NAME"]) . ' ' . $this->target . '/'. $id;
- $cmd = $php . ' ' . $app. ' &';
- echo $cmd . "\n";
+ if ($this->force) {
+ $app .= ' -f';
+ }
+ if (!empty($this->send_to)) {
+ $app .= ' --sent-to='.escapeshellarg($this->send_to);
+ }
+ $cmd = 'exec ' . $php . ' ' . $app . ' ' . $cmdOpts; //. ' &';
+
+
$pipe = array();
+ echo "call proc_open $cmd\n";
$p = proc_open($cmd, $descriptorspec, $pipes, $cwd );
+ $info = proc_get_status($p);
$this->pool[] = array(
'proc' => $p,
+ 'pid' => $info['pid'],
'out' => $tn,
'cmd' => $cmd,
- 'email' => $email
+ 'email' => $email,
+ 'pipes' => $pipes,
+ 'started' => time()
+
+
);
+ echo "RUN ({$info['pid']}) $cmd \n";
}
function poolfree()
{
$pool = array();
+ clearstatcache();
+ $maxruntime = 2 * 60; // 2 minutes.. ?? should be long enoguh
+
foreach($this->pool as $p) {
- $ar = proc_get_status($p['proc']);
- // print_r($p);
- //print_r($ar);
- if ($ar['running']) {
+
+ //echo "CHECK PID: " . $p['pid'] . "\n";
+ $info = proc_get_status($p['proc']);
+ //var_dump($info);
+
+ // update if necessday.
+ if ($info['pid']) {
+ echo "CHANING PID FROM " . $p['pid'] . " TO ". $info['pid']. "\n";
+ $p['pid'] = $info['pid'];
+ }
+
+ echo @file_get_contents('/proc/'. $p['pid'] .'/cmdline') . "\n";
+
+ if ($info['running']) {
+
+ //if (file_exists('/proc/'.$p['pid'])) {
+ $runtime = time() - $p['started'];
+ //echo "RUNTIME ({$p['pid']}): $runtime\n";
+ if ($runtime > $maxruntime) {
+
+ proc_terminate($p['proc'], 9);
+ //fclose($p['pipes'][1]);
+ fclose($p['pipes'][0]);
+ fclose($p['pipes'][2]);
+ echo "\nTERMINATING: ({$p['pid']}) " . $p['cmd'] . " : " . file_get_contents($p['out']) . "\n";
+ @unlink($p['out']);
+
+ continue;
+ }
+
$pool[] = $p;
continue;
}
- echo $p['cmd'] . " : " . file_get_contents($p['out']);
+ fclose($p['pipes'][0]);
+ fclose($p['pipes'][2]);
+ //echo "CLOSING: ({$p['pid']}) " . $p['cmd'] . " : " . file_get_contents($p['out']) . "\n";
+ //fclose($p['pipes'][1]);
+
+ proc_close($p['proc']);
+
+
+ //clearstatcache();
+ //if (file_exists('/proc/'.$p['pid'])) {
+ // $pool[] = $p;
+ // continue;
+ //}
+ echo "\nENDED: ({$p['pid']}) " . $p['cmd'] . " : " . file_get_contents($p['out']) . "\n";
+ @unlink($p['out']);
//unlink($p['out']);
}
+ echo "POOL SIZE: ". count($pool) ."\n";
$this->pool = $pool;
if (count($pool) < 10) {
return true;
}
+ function output()
+ {
+ die("Done\n");
+ }
}
\ No newline at end of file