2011-08-16 12:27:19 +10:00
|
|
|
<?php defined('SYSPATH') or die('No direct access allowed.');
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This class provides OSB invoice task capabilities.
|
|
|
|
*
|
|
|
|
* @package OSB
|
|
|
|
* @subpackage Invoice
|
|
|
|
* @category Controllers/Task
|
|
|
|
* @author Deon George
|
|
|
|
* @copyright (c) 2010 Open Source Billing
|
|
|
|
* @license http://dev.osbill.net/license.html
|
|
|
|
*/
|
|
|
|
class Controller_Task_Invoice extends Controller_Task {
|
2011-10-11 19:52:31 +11:00
|
|
|
/**
|
|
|
|
* Email a list of invoice balances
|
|
|
|
*
|
|
|
|
* This function is typically used to list the overdue invoices to the admins
|
|
|
|
* @param string mode The callback method to use as the data list eg: overdue
|
|
|
|
*/
|
2011-09-26 20:12:54 +10:00
|
|
|
public function action_list() {
|
|
|
|
$mode = $this->request->param('id');
|
|
|
|
|
2011-10-14 16:44:12 +11:00
|
|
|
$i = ORM::factory('invoice');
|
2011-08-16 12:27:19 +10:00
|
|
|
$tm = 'list_'.$mode;
|
|
|
|
|
2011-10-14 16:44:12 +11:00
|
|
|
if (! method_exists($i,$tm))
|
2011-08-16 12:27:19 +10:00
|
|
|
throw new Kohana_Exception('Unknown Task List command :command',array(':command'=>$mode));
|
|
|
|
|
|
|
|
$total = $numinv = 0;
|
2011-08-27 16:33:46 +10:00
|
|
|
$duelist = View::factory('invoice/task/'.$tm.'_head');
|
2011-10-14 16:44:12 +11:00
|
|
|
foreach ($i->$tm() as $t) {
|
2011-08-16 12:27:19 +10:00
|
|
|
$duelist .= View::factory('invoice/task/'.$tm.'_body')
|
|
|
|
->set('io',$t);
|
|
|
|
|
|
|
|
$numinv++;
|
|
|
|
$total += $t->due();
|
|
|
|
}
|
2011-08-27 16:33:46 +10:00
|
|
|
$duelist .= View::factory('invoice/task/'.$tm.'_foot');
|
2011-08-16 12:27:19 +10:00
|
|
|
|
|
|
|
// Send our email
|
2011-09-27 21:22:13 +10:00
|
|
|
$et = Email_Template::instance('task_invoice_list_overdue');
|
2011-08-16 12:27:19 +10:00
|
|
|
|
|
|
|
// @todo Update this to be dynamic
|
|
|
|
$et->to = array('account'=>array(1,68));
|
|
|
|
$et->variables = array(
|
|
|
|
'TABLE'=>$duelist,
|
|
|
|
'NUM_INV'=>$numinv,
|
|
|
|
'TOTAL'=>$total,
|
|
|
|
);
|
|
|
|
$et->send();
|
|
|
|
|
|
|
|
$output = sprintf('List (%s) sent to: %s',$mode,implode(',',array_keys($et->to)));
|
|
|
|
$this->response->body($output);
|
|
|
|
}
|
2011-09-26 20:12:54 +10:00
|
|
|
|
2011-10-11 19:52:31 +11:00
|
|
|
/**
|
|
|
|
* Email a customers a reminder of their upcoming invoices that are due.
|
|
|
|
*/
|
2011-09-26 20:12:54 +10:00
|
|
|
public function action_remind_due() {
|
2011-10-11 19:52:31 +11:00
|
|
|
$action = array();
|
2011-09-26 20:12:54 +10:00
|
|
|
// @todo This should go in a config somewhere
|
|
|
|
$days = 5;
|
2011-09-27 21:22:13 +10:00
|
|
|
$key = 'remind_due';
|
2011-09-26 20:12:54 +10:00
|
|
|
|
2011-10-14 16:44:12 +11:00
|
|
|
foreach (ORM::factory('invoice')->list_due(time()+86400*$days) as $io) {
|
2011-09-26 20:12:54 +10:00
|
|
|
// If we have already sent a reminder, we'll skip to the next one.
|
2011-09-27 21:22:13 +10:00
|
|
|
if ($io->remind($key) AND (is_null($x=$this->request->param('id')) OR $x != 'again'))
|
2011-09-26 20:12:54 +10:00
|
|
|
continue;
|
|
|
|
|
|
|
|
// Send our email
|
2011-09-27 21:22:13 +10:00
|
|
|
$et = Email_Template::instance('task_invoice_'.$key);
|
2011-09-26 20:12:54 +10:00
|
|
|
|
|
|
|
$et->to = array('account'=>array($io->account_id));
|
|
|
|
$et->variables = array(
|
|
|
|
'DUE'=>$io->due(TRUE),
|
2011-09-27 21:22:13 +10:00
|
|
|
'DUE_DATE'=>$io->display('due_date'),
|
|
|
|
'FIRST_NAME'=>$io->account->first_name,
|
2011-09-26 20:12:54 +10:00
|
|
|
'INV_NUM'=>$io->refnum(),
|
|
|
|
'INV_URL'=>URL::site('user/invoice/view/'.$io->id,'http'),
|
2011-12-30 18:10:02 +11:00
|
|
|
'SITE_NAME'=>Company::name(),
|
2011-09-27 21:22:13 +10:00
|
|
|
);
|
|
|
|
|
|
|
|
// @todo Record email log id if possible.
|
2011-10-11 19:52:31 +11:00
|
|
|
if ($et->send()) {
|
2011-09-27 21:22:13 +10:00
|
|
|
$io->set_remind($key,time());
|
2011-10-11 19:52:31 +11:00
|
|
|
array_push($action,(string)$io);
|
|
|
|
}
|
2011-09-27 21:22:13 +10:00
|
|
|
}
|
|
|
|
|
2011-10-11 19:52:31 +11:00
|
|
|
$this->response->body(_('Due Reminders Sent: ').join('|',$action));
|
2011-09-27 21:22:13 +10:00
|
|
|
}
|
|
|
|
|
2011-10-11 19:52:31 +11:00
|
|
|
/**
|
|
|
|
* Email a customers when their invoices are now overdue.
|
|
|
|
*/
|
2011-09-27 21:22:13 +10:00
|
|
|
public function action_remind_overdue() {
|
2011-10-11 19:52:31 +11:00
|
|
|
$action = array();
|
2011-09-27 21:22:13 +10:00
|
|
|
$notice = $this->request->param('id');
|
|
|
|
$x = NULL;
|
|
|
|
|
|
|
|
if (preg_match('/:/',$notice))
|
|
|
|
list($notice,$x) = explode(':',$notice);
|
|
|
|
|
|
|
|
switch ($notice) {
|
|
|
|
case 1:
|
|
|
|
// @todo This should go in a config somewhere
|
|
|
|
$days = 2;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 2:
|
|
|
|
// @todo This should go in a config somewhere
|
|
|
|
$days = 7;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 3:
|
|
|
|
// @todo This should go in a config somewhere
|
2011-10-14 08:41:01 +11:00
|
|
|
$days = 13;
|
2011-09-27 21:22:13 +10:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
$this->response->body(_('Unknown Remind Period: ').$notice);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
$key = 'remind_overdue_'.$notice;
|
|
|
|
|
2011-10-14 16:44:12 +11:00
|
|
|
foreach (ORM::factory('invoice')->list_overdue_billing(time()-86400*$days,FALSE) as $io) {
|
2011-09-27 21:22:13 +10:00
|
|
|
// If we have already sent a reminder, we'll skip to the next one.
|
|
|
|
if ($io->remind($key) AND (is_null($x) OR $x != 'again'))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// Send our email
|
|
|
|
$et = Email_Template::instance('task_invoice_'.$key);
|
|
|
|
|
|
|
|
$et->to = array('account'=>array($io->account_id));
|
|
|
|
$et->variables = array(
|
|
|
|
'DUE'=>$io->due(TRUE),
|
2011-09-26 20:12:54 +10:00
|
|
|
'DUE_DATE'=>$io->display('due_date'),
|
2011-12-30 18:10:02 +11:00
|
|
|
'EMAIL'=>Company::email(),
|
2011-09-27 21:22:13 +10:00
|
|
|
'FIRST_NAME'=>$io->account->first_name,
|
|
|
|
'INV_NUM'=>$io->refnum(),
|
|
|
|
'INV_URL'=>URL::site('user/invoice/view/'.$io->id,'http'),
|
|
|
|
'LATE_FEE'=>'5.50', // @todo This should come from a config file.
|
|
|
|
'PAYMENTS_TABLE'=>$io->account->payment->list_recent_table(),
|
2011-12-30 18:10:02 +11:00
|
|
|
'SITE_NAME'=>Company::name(),
|
2011-09-26 20:12:54 +10:00
|
|
|
);
|
|
|
|
|
|
|
|
// @todo Record email log id if possible.
|
2011-10-11 19:52:31 +11:00
|
|
|
if ($et->send()) {
|
2011-09-27 21:22:13 +10:00
|
|
|
$io->set_remind($key,time());
|
2011-10-11 19:52:31 +11:00
|
|
|
array_push($action,(string)$io);
|
|
|
|
}
|
2011-09-26 20:12:54 +10:00
|
|
|
}
|
|
|
|
|
2011-10-11 19:52:31 +11:00
|
|
|
$this->response->body(_('Overdue Reminders Sent: ').join('|',$action));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Generate our services invoices, based on the service next invoice date
|
|
|
|
*
|
|
|
|
* @param int ID Service ID to generate invoice for (optional)
|
|
|
|
*/
|
2011-10-14 16:44:12 +11:00
|
|
|
public function action_services() {
|
|
|
|
// Used to only process X invoices in a row.
|
|
|
|
// @todo This should be a configuration item.
|
|
|
|
$max = 10;
|
2011-10-11 19:52:31 +11:00
|
|
|
$action = array();
|
|
|
|
$snd = array(); // Our service next billing dates that need to be updated if this is successful.
|
2011-10-14 16:44:12 +11:00
|
|
|
$sid = is_null($this->request->param('id')) ? NULL : explode(':',$this->request->param('id'));
|
2011-10-11 19:52:31 +11:00
|
|
|
|
|
|
|
// Sort our service by account_id, then we can generate 1 invoice.
|
2011-12-22 21:20:19 +11:00
|
|
|
$svs = ORM::factory('service')->list_invoicesoon()->as_array();
|
2011-10-11 19:52:31 +11:00
|
|
|
Sort::MAsort($svs,'account_id,date_next_invoice');
|
|
|
|
|
|
|
|
$aid = $due = $io = NULL;
|
2011-10-14 16:44:12 +11:00
|
|
|
$max_count = 0;
|
2011-10-11 19:52:31 +11:00
|
|
|
foreach ($svs as $so) {
|
2011-10-14 16:44:12 +11:00
|
|
|
// If we are generating an invoice for a service, skip to that service.
|
|
|
|
if (! is_null($sid) AND ! in_array($so->id,$sid))
|
2011-10-11 19:52:31 +11:00
|
|
|
continue;
|
|
|
|
|
|
|
|
// Close off invoice, and start a new one.
|
2011-10-14 16:44:12 +11:00
|
|
|
if (is_null($io) OR (! is_null($aid) AND $aid != $so->account_id) OR (! is_null($due) AND $due != $io->min_due($so->date_next_invoice))) {
|
2011-10-11 19:52:31 +11:00
|
|
|
// Close this invoice.
|
2011-10-14 16:44:12 +11:00
|
|
|
if (is_object($io)) {
|
2011-10-11 19:52:31 +11:00
|
|
|
// Save our invoice.
|
|
|
|
if (! $io->save())
|
|
|
|
throw new Kohana_Exception('Failed to save invoice :invoice for service :service',array(':invoice'=>$io->id,':service'=>$so->id));
|
|
|
|
}
|
|
|
|
|
2011-10-14 16:44:12 +11:00
|
|
|
// If we have issued the max number of invoices this round, finish.
|
|
|
|
if (++$max_count > $max)
|
|
|
|
break;
|
|
|
|
|
2011-10-11 19:52:31 +11:00
|
|
|
// Start a new invoice.
|
|
|
|
$io = ORM::factory('invoice');
|
|
|
|
$io->due_date = $due = $io->min_due($so->date_next_invoice);
|
|
|
|
$io->account_id = $aid = $so->account_id;
|
|
|
|
$io->status = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
$ppa = $so->product->get_price_array();
|
|
|
|
// @todo Need to check our recurr_weekday configuration for items that need to be pro-rated and items that are billed on absolute dates.
|
|
|
|
$pdata = Period::details($so->recur_schedule,$so->product->price_recurr_weekday,$so->date_next_invoice,TRUE);
|
|
|
|
$iio = $io->add_item();
|
|
|
|
|
|
|
|
$iio->service_id = $so->id;
|
|
|
|
$iio->product_id = $so->product_id;
|
2011-10-14 16:44:12 +11:00
|
|
|
$iio->quantity = $pdata['prorata'];
|
2011-10-11 19:52:31 +11:00
|
|
|
$iio->item_type = 0;
|
|
|
|
$iio->discount_amt = null; // @todo
|
2011-10-14 16:44:12 +11:00
|
|
|
$iio->price_type = $so->product->price_type; // @todo Do we need this?
|
2011-10-11 19:52:31 +11:00
|
|
|
// @todo Might be a better way to do this
|
2011-10-14 16:44:12 +11:00
|
|
|
$iio->price_base = $so->price ? $so->price : (isset($ppa[$so->recur_schedule]['price_base']) ? $ppa[$so->recur_schedule]['price_base'] : 0);
|
2011-10-11 19:52:31 +11:00
|
|
|
$iio->recurring_schedule = $so->recur_schedule;
|
|
|
|
$iio->date_start = $pdata['start_time']; // @todo
|
|
|
|
$iio->date_stop = $pdata['end_time']; // @todo
|
|
|
|
|
|
|
|
// Our service next billing date, if this invoice generation is successful.
|
|
|
|
$snd[$so->id] = $pdata['end_time']+86400;
|
|
|
|
|
2011-10-14 16:44:12 +11:00
|
|
|
// Check if there are any charges
|
|
|
|
$c = ORM::factory('charge')
|
|
|
|
->where('service_id','=',$so->id)
|
|
|
|
->where('status','=',0)
|
|
|
|
->where('sweep_type','=',6); // @todo This needs to be dynamic, not "6"
|
|
|
|
|
|
|
|
foreach ($c->find_all() as $co) {
|
|
|
|
$iio = $io->add_item();
|
|
|
|
|
|
|
|
$iio->service_id = $co->service_id;
|
|
|
|
$iio->product_id = $co->product_id;
|
|
|
|
$iio->charge_id = $co->id;
|
|
|
|
$iio->quantity = $co->quantity;
|
|
|
|
$iio->item_type = 5; // @todo This probably should not be hard coded as "5".
|
|
|
|
$iio->discount_amt = null; // @todo
|
|
|
|
$iio->price_base = $co->amount;
|
|
|
|
$iio->date_start = $co->date_orig;
|
|
|
|
$iio->date_stop = $co->date_orig; // @todo
|
|
|
|
|
|
|
|
// @todo Temp
|
|
|
|
$co->status=2;
|
|
|
|
$co->save();
|
|
|
|
}
|
|
|
|
|
2011-10-11 19:52:31 +11:00
|
|
|
array_push($action,(string)$so->id);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Save our invoice.
|
2011-10-14 16:44:12 +11:00
|
|
|
if (! $io->saved() AND ! $io->save()) {
|
|
|
|
print_r($io->items());
|
2011-10-11 19:52:31 +11:00
|
|
|
throw new Kohana_Exception('Failed to save invoice :invoice for service :service',array(':invoice'=>$io->id,':service'=>$so->id));
|
2011-10-14 16:44:12 +11:00
|
|
|
}
|
2011-10-11 19:52:31 +11:00
|
|
|
|
|
|
|
// Update our service next billing dates.
|
|
|
|
foreach ($snd as $sid=>$date) {
|
|
|
|
$so = ORM::factory('service',$sid);
|
|
|
|
$so->date_next_invoice = $date;
|
|
|
|
$so->save();
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->response->body(_('Services Invoiced: ').join('|',$action));
|
|
|
|
}
|
|
|
|
|
2011-10-14 16:44:12 +11:00
|
|
|
public function action_send() {
|
|
|
|
// Used to only process X invoices in a row.
|
|
|
|
// @todo This should be a configuration item.
|
2011-12-22 21:20:19 +11:00
|
|
|
$max = 10;
|
2011-10-13 09:20:08 +11:00
|
|
|
$action = array();
|
|
|
|
$iid = $this->request->param('id');
|
|
|
|
$x = NULL;
|
|
|
|
|
|
|
|
if (preg_match('/:/',$iid))
|
|
|
|
list($iid,$x) = explode(':',$iid);
|
|
|
|
|
2011-10-14 16:44:12 +11:00
|
|
|
// Get our list of invoices to send
|
|
|
|
$i = $iid ? ORM::factory('invoice')->where('id','=',$iid) : ORM::factory('invoice')->list_tosend();
|
2011-10-13 09:20:08 +11:00
|
|
|
|
|
|
|
$key = 'send';
|
|
|
|
|
2011-10-14 16:44:12 +11:00
|
|
|
$max_count = 0;
|
2011-10-13 09:20:08 +11:00
|
|
|
foreach ($i->find_all() as $io) {
|
|
|
|
// If we have already sent a reminder, we'll skip to the next one.
|
|
|
|
if ($io->remind($key) AND (is_null($x) OR $x != 'again'))
|
|
|
|
continue;
|
|
|
|
|
2011-10-14 16:44:12 +11:00
|
|
|
// If we have issued the max number of invoices this round, finish.
|
|
|
|
if (++$max_count > $max)
|
|
|
|
break;
|
|
|
|
|
2011-10-13 09:20:08 +11:00
|
|
|
// Send our email
|
|
|
|
$et = Email_Template::instance('task_invoice_'.$key);
|
|
|
|
$token = ORM::factory('module_method_token')
|
|
|
|
->method(array('invoice','user_download'))
|
|
|
|
->account($io->account)
|
|
|
|
->expire(time()+86400*21)
|
|
|
|
->uses(3)
|
|
|
|
->generate();
|
|
|
|
|
|
|
|
$et->to = array('account'=>array($io->account_id));
|
|
|
|
$et->variables = array(
|
|
|
|
'DUE'=>$io->due(TRUE),
|
|
|
|
'DUE_DATE'=>$io->display('due_date'),
|
2011-12-30 18:10:02 +11:00
|
|
|
'EMAIL'=>Company::email(),
|
2011-10-13 09:20:08 +11:00
|
|
|
'FIRST_NAME'=>$io->account->first_name,
|
|
|
|
'HTML_INVOICE'=>$io->html(),
|
|
|
|
'INV_NUM'=>$io->refnum(),
|
|
|
|
'INV_URL'=>URL::site('user/invoice/view/'.$io->id,'http'),
|
|
|
|
'INV_URL_DOWNLOAD'=>URL::site(sprintf('user/invoice/download/%s?token=%s',$io->id,$token),'http'),
|
2011-12-30 18:10:02 +11:00
|
|
|
'SITE_NAME'=>Company::name(),
|
2011-10-13 09:20:08 +11:00
|
|
|
);
|
|
|
|
|
|
|
|
// @todo Record email log id if possible.
|
|
|
|
if ($et->send()) {
|
2011-10-14 16:44:12 +11:00
|
|
|
$io->print_status = 1;
|
2011-10-13 09:20:08 +11:00
|
|
|
$io->set_remind($key,time());
|
|
|
|
array_push($action,(string)$io);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->response->body(_('Invoices Sent: ').join('|',$action));
|
|
|
|
}
|
|
|
|
|
2011-10-11 19:52:31 +11:00
|
|
|
/** END **/
|
|
|
|
|
|
|
|
public function action_audit_invoice_items() {
|
|
|
|
$output = '';
|
|
|
|
|
|
|
|
foreach (ORM::factory('invoice_item')->find_all() as $iio) {
|
|
|
|
if ($iio->product_name AND $iio->product_id) {
|
|
|
|
if (md5(strtoupper($iio->product_name)) == md5(strtoupper($iio->product->name()))) {
|
|
|
|
$iio->product_name = null;
|
|
|
|
$iio->save();
|
|
|
|
} else {
|
2011-10-13 09:20:08 +11:00
|
|
|
print_r(array("DIFF",'id'=>$iio->id,'pn'=>serialize($iio->product_name),'ppn'=>serialize($iio->product->name()),'pid'=>$iio->product_id,'test'=>strcasecmp($iio->product_name,$iio->product->name())));
|
2011-10-11 19:52:31 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->response->body($output);
|
2011-09-26 20:12:54 +10:00
|
|
|
}
|
2011-08-16 12:27:19 +10:00
|
|
|
}
|
|
|
|
?>
|