2010-11-30 09:41:08 +11:00
|
|
|
<?php defined('SYSPATH') or die('No direct access allowed.');
|
|
|
|
|
|
|
|
/**
|
2011-09-28 16:46:22 +10:00
|
|
|
* This class provides Admin Service functions
|
2010-11-30 09:41:08 +11:00
|
|
|
*
|
|
|
|
* @package OSB
|
|
|
|
* @subpackage Service
|
|
|
|
* @category Controllers/Admin
|
|
|
|
* @author Deon George
|
|
|
|
* @copyright (c) 2010 Open Source Billing
|
|
|
|
* @license http://dev.osbill.net/license.html
|
|
|
|
*/
|
2011-08-26 11:16:48 +10:00
|
|
|
class Controller_Admin_Service extends Controller_TemplateDefault_Admin {
|
2012-06-22 22:05:43 +10:00
|
|
|
// @todo This "module" menu items should belong in the module dir.
|
2011-08-26 11:16:48 +10:00
|
|
|
protected $secure_actions = array(
|
2011-10-14 16:44:12 +11:00
|
|
|
'autolist'=>FALSE, // @todo To Change
|
2012-05-13 09:29:44 +10:00
|
|
|
'adslstat'=>TRUE,
|
2011-08-27 16:33:46 +10:00
|
|
|
'list'=>TRUE,
|
2010-11-30 09:41:08 +11:00
|
|
|
'listbycheckout'=>TRUE,
|
2011-08-27 16:33:46 +10:00
|
|
|
'listadslbilling'=>TRUE,
|
2010-11-30 09:41:08 +11:00
|
|
|
'listadslservices'=>TRUE,
|
2011-09-28 16:46:22 +10:00
|
|
|
'listdomainservices'=>TRUE,
|
2011-10-14 16:44:12 +11:00
|
|
|
'listdomainservicesbysupplier'=>TRUE,
|
2012-06-22 22:05:43 +10:00
|
|
|
'listdomainservicesbydnshost'=>TRUE,
|
2011-09-28 16:46:22 +10:00
|
|
|
'listhostservices'=>TRUE,
|
2012-06-22 22:05:43 +10:00
|
|
|
'listhostservicesbysupplier'=>TRUE,
|
2010-11-30 09:41:08 +11:00
|
|
|
'listhspaservices'=>TRUE,
|
2011-12-27 00:52:46 +11:00
|
|
|
'listwebservices'=>TRUE,
|
2011-10-11 10:38:21 +11:00
|
|
|
'listinvoicesoon'=>TRUE,
|
2011-07-14 19:09:03 +10:00
|
|
|
'update'=>TRUE,
|
2012-01-09 12:35:24 +11:00
|
|
|
'view'=>TRUE,
|
2010-11-30 09:41:08 +11:00
|
|
|
);
|
|
|
|
|
2011-10-14 16:44:12 +11:00
|
|
|
public function action_autolist() {
|
|
|
|
$return = array();
|
|
|
|
|
2012-08-01 22:43:33 +10:00
|
|
|
$s = ORM::factory('service')->where_active();
|
2011-10-14 16:44:12 +11:00
|
|
|
if (isset($_REQUEST['aid']))
|
|
|
|
$s = $s->where('account_id','=',$_REQUEST['aid']);
|
|
|
|
|
|
|
|
// @todo This should limit the results so that users dont see other users services.
|
|
|
|
foreach ($s->find_all() as $so)
|
|
|
|
array_push($return,array(
|
|
|
|
'value'=>$so->id,
|
|
|
|
'text'=>sprintf('%s: %s',$so->id,$so->service_name()),
|
|
|
|
));
|
|
|
|
|
|
|
|
$this->auto_render = FALSE;
|
|
|
|
$this->response->headers('Content-Type','application/json');
|
|
|
|
$this->response->body(json_encode($return));
|
|
|
|
}
|
|
|
|
|
2011-08-27 16:33:46 +10:00
|
|
|
/**
|
|
|
|
* Show a list of services
|
|
|
|
*/
|
|
|
|
public function action_list() {
|
|
|
|
Block::add(array(
|
2011-09-28 16:46:22 +10:00
|
|
|
'title'=>_('Customer Services'),
|
2011-08-27 16:33:46 +10:00
|
|
|
'body'=>Table::display(
|
2011-09-28 16:46:22 +10:00
|
|
|
ORM::factory('service')->find_all(),
|
2011-08-27 16:33:46 +10:00
|
|
|
25,
|
|
|
|
array(
|
|
|
|
'id'=>array('label'=>'ID','url'=>'user/service/view/'),
|
2011-09-28 16:46:22 +10:00
|
|
|
'service_name()'=>array('label'=>'Details'),
|
2011-08-27 16:33:46 +10:00
|
|
|
'recur_schedule'=>array('label'=>'Billing'),
|
2012-03-30 15:13:01 +11:00
|
|
|
'price(TRUE,TRUE)'=>array('label'=>'Price','class'=>'right'),
|
2012-08-01 22:43:33 +10:00
|
|
|
'status'=>array('label'=>'Active'),
|
2011-08-27 16:33:46 +10:00
|
|
|
'account->accnum()'=>array('label'=>'Cust ID'),
|
|
|
|
'account->name()'=>array('label'=>'Customer'),
|
|
|
|
),
|
|
|
|
array(
|
|
|
|
'page'=>TRUE,
|
|
|
|
'type'=>'select',
|
2011-09-28 16:46:22 +10:00
|
|
|
'form'=>'user/service/view',
|
2011-08-27 16:33:46 +10:00
|
|
|
)),
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
2010-11-30 09:41:08 +11:00
|
|
|
/**
|
|
|
|
* List all services by their default checkout method
|
|
|
|
*/
|
|
|
|
public function action_listbycheckout() {
|
2011-10-14 16:44:12 +11:00
|
|
|
$svs = array();
|
|
|
|
// @todo This needs to be configurable
|
|
|
|
$go = ORM::factory('group',array('name'=>'Personal'));
|
|
|
|
|
|
|
|
foreach (ORM::factory('account')->where('status','=',1)->find_all() as $ao)
|
|
|
|
if ($ao->has_any('group',array($go)))
|
|
|
|
foreach ($ao->service->list_active() as $so)
|
|
|
|
if (! $so->service_billing->checkout_plugin_id)
|
|
|
|
array_push($svs,$so);
|
|
|
|
|
|
|
|
if ($svs)
|
|
|
|
Block::add(array(
|
|
|
|
'title'=>'Services that should be auto-billed',
|
|
|
|
'body'=>Table::display(
|
|
|
|
$svs,
|
|
|
|
25,
|
|
|
|
array(
|
|
|
|
'id'=>array('label'=>'ID','url'=>'user/service/view/'),
|
|
|
|
'service_name()'=>array('label'=>'Details'),
|
|
|
|
'recur_schedule'=>array('label'=>'Billing'),
|
2012-03-30 15:13:01 +11:00
|
|
|
'price(TRUE,TRUE)'=>array('label'=>'Price','class'=>'right'),
|
2012-08-01 22:43:33 +10:00
|
|
|
'status'=>array('label'=>'Active'),
|
2011-10-14 16:44:12 +11:00
|
|
|
'account->accnum()'=>array('label'=>'Cust ID'),
|
|
|
|
'account->name()'=>array('label'=>'Customer'),
|
|
|
|
'date_next_invoice'=>array('label'=>'Next Invoice'),
|
|
|
|
'account->invoices_due_total(NULL,TRUE)'=>array('label'=>'Due Invoices'),
|
|
|
|
),
|
|
|
|
array(
|
|
|
|
'page'=>TRUE,
|
|
|
|
'type'=>'select',
|
|
|
|
'form'=>'user/service/view',
|
|
|
|
)),
|
|
|
|
));
|
|
|
|
|
2012-08-01 22:43:33 +10:00
|
|
|
foreach (ORM::factory('checkout')->list_active() as $co) {
|
2011-10-14 16:44:12 +11:00
|
|
|
$svs = array();
|
|
|
|
|
|
|
|
foreach ($co->account->find_all() as $ao)
|
|
|
|
foreach ($ao->service->list_active() as $so)
|
|
|
|
if ($so->service_billing->checkout_plugin_id == $co->id)
|
|
|
|
array_push($svs,$so);
|
|
|
|
|
|
|
|
if ($svs)
|
|
|
|
Block::add(array(
|
|
|
|
'title'=>$co->name,
|
|
|
|
'body'=>Table::display(
|
|
|
|
$svs,
|
|
|
|
25,
|
|
|
|
array(
|
|
|
|
'id'=>array('label'=>'ID','url'=>'user/service/view/'),
|
|
|
|
'service_name()'=>array('label'=>'Details'),
|
|
|
|
'recur_schedule'=>array('label'=>'Billing'),
|
2012-03-30 15:13:01 +11:00
|
|
|
'price(TRUE,TRUE)'=>array('label'=>'Price','class'=>'right'),
|
2012-08-01 22:43:33 +10:00
|
|
|
'status'=>array('label'=>'Active'),
|
2011-10-14 16:44:12 +11:00
|
|
|
'account->accnum()'=>array('label'=>'Cust ID'),
|
|
|
|
'account->name()'=>array('label'=>'Customer'),
|
|
|
|
'date_next_invoice'=>array('label'=>'Next Invoice'),
|
|
|
|
'account->invoices_due_total(NULL,TRUE)'=>array('label'=>'Due Invoices'),
|
|
|
|
),
|
|
|
|
array(
|
|
|
|
'page'=>TRUE,
|
|
|
|
'type'=>'select',
|
|
|
|
'form'=>'user/service/view',
|
|
|
|
)),
|
|
|
|
));
|
2010-11-30 09:41:08 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-28 16:46:22 +10:00
|
|
|
private function consoltraffic($svs,$date) {
|
|
|
|
$data = array();
|
|
|
|
|
|
|
|
foreach ($svs as $so) {
|
|
|
|
$c = array();
|
|
|
|
foreach ($so->plugin()->get_traffic_data_monthly($date) as $metric => $ma) {
|
|
|
|
foreach ($ma as $month => $traffic) {
|
|
|
|
// Only count the service once, not for each metric.
|
|
|
|
if (! isset($c[$month])) {
|
|
|
|
if (isset($data['svs'][$month]))
|
|
|
|
$data['svs'][$month] += 1;
|
|
|
|
else
|
|
|
|
$data['svs'][$month] = 1;
|
|
|
|
|
|
|
|
$c[$month] = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isset($data['data'][$metric][$month]))
|
|
|
|
$data['data'][$metric][$month] += (int)$traffic;
|
|
|
|
else
|
|
|
|
$data['data'][$metric][$month] = (int)$traffic;
|
|
|
|
}
|
2010-11-30 09:41:08 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-28 16:46:22 +10:00
|
|
|
ksort($data['svs']);
|
|
|
|
foreach ($data['data'] as $metric => $details)
|
|
|
|
ksort($data['data'][$metric]);
|
2010-11-30 09:41:08 +11:00
|
|
|
|
2011-09-28 16:46:22 +10:00
|
|
|
return $data;
|
|
|
|
}
|
2010-11-30 09:41:08 +11:00
|
|
|
|
2011-09-28 16:46:22 +10:00
|
|
|
public function action_listadslservices() {
|
|
|
|
$svs = ORM::factory('service')->list_bylistgroup('ADSL');
|
|
|
|
$data = $this->consoltraffic($svs,time());
|
2010-11-30 09:41:08 +11:00
|
|
|
|
|
|
|
$google = GoogleChart::factory('vertical_bar');
|
|
|
|
$google->title = sprintf('ADSL traffic as at %s',date('Y-m-d',strtotime('yesterday')));
|
2011-09-28 16:46:22 +10:00
|
|
|
$google->series(array('title'=>array_keys($data['data']),'axis'=>'x','data'=>$data['data']));
|
|
|
|
$google->series(array('title'=>'Services','axis'=>'r','data'=>array('Services'=>$data['svs'])));
|
2010-11-30 09:41:08 +11:00
|
|
|
|
2011-09-28 16:46:22 +10:00
|
|
|
Block::add(array('body'=>$google));
|
2010-11-30 09:41:08 +11:00
|
|
|
|
|
|
|
Block::add(array(
|
2011-09-28 16:46:22 +10:00
|
|
|
'title'=>_('ADSL Services'),
|
|
|
|
'body'=>Table::display(
|
|
|
|
$svs,
|
|
|
|
NULL,
|
|
|
|
array(
|
|
|
|
'id'=>array('label'=>'ID','url'=>'user/service/view/'),
|
|
|
|
'name()'=>array('label'=>'Service'),
|
|
|
|
'plugin()->ipaddress()'=>array('label'=>'IP Address'),
|
2011-10-13 09:20:08 +11:00
|
|
|
'product->plugin()->speed'=>array('label'=>'Speed'),
|
2011-09-28 16:46:22 +10:00
|
|
|
'product->plugin()->allowance()'=>array('label'=>'Allowance'),
|
|
|
|
'plugin()->traffic_thismonth()'=>array('label'=>'This Month'),
|
|
|
|
'plugin()->traffic_lastmonth()'=>array('label'=>'Last Month'),
|
|
|
|
'recur_schedule'=>array('label'=>'Billing'),
|
2012-03-30 15:13:01 +11:00
|
|
|
'price(TRUE,TRUE)'=>array('label'=>'Price','class'=>'right'),
|
2011-09-28 16:46:22 +10:00
|
|
|
'account->accnum()'=>array('label'=>'Cust ID'),
|
|
|
|
'account->name()'=>array('label'=>'Customer'),
|
|
|
|
'date_next_invoice'=>array('label'=>'Next Invoice'),
|
2012-08-01 22:43:33 +10:00
|
|
|
'product->display("status")'=>array('label'=>'Current'),
|
2011-09-28 16:46:22 +10:00
|
|
|
),
|
|
|
|
array(
|
|
|
|
'type'=>'select',
|
|
|
|
'form'=>'user/service/view',
|
|
|
|
)),
|
2010-11-30 09:41:08 +11:00
|
|
|
));
|
|
|
|
}
|
|
|
|
|
|
|
|
public function action_listhspaservices() {
|
2011-09-28 16:46:22 +10:00
|
|
|
$svs = ORM::factory('service')->list_bylistgroup('HSPA');
|
|
|
|
$data = $this->consoltraffic($svs,time());
|
2010-11-30 09:41:08 +11:00
|
|
|
|
|
|
|
$google = GoogleChart::factory('vertical_bar');
|
|
|
|
$google->title = sprintf('HSPA traffic as at %s',date('Y-m-d',strtotime('yesterday')));
|
2011-09-28 16:46:22 +10:00
|
|
|
$google->series(array('title'=>array_keys($data['data']),'axis'=>'x','data'=>$data['data']));
|
|
|
|
$google->series(array('title'=>'Services','axis'=>'r','data'=>array('Services'=>$data['svs'])));
|
|
|
|
|
|
|
|
Block::add(array('body'=>$google));
|
2010-11-30 09:41:08 +11:00
|
|
|
|
|
|
|
Block::add(array(
|
2011-09-28 16:46:22 +10:00
|
|
|
'title'=>_('HSPA Services'),
|
|
|
|
'body'=>Table::display(
|
|
|
|
$svs,
|
|
|
|
NULL,
|
|
|
|
array(
|
|
|
|
'id'=>array('label'=>'ID','url'=>'user/service/view/'),
|
|
|
|
'name()'=>array('label'=>'Service'),
|
|
|
|
'plugin()->ipaddress()'=>array('label'=>'IP Address'),
|
2011-10-13 09:20:08 +11:00
|
|
|
'product->plugin()->speed'=>array('label'=>'Speed'),
|
2011-09-28 16:46:22 +10:00
|
|
|
'product->plugin()->allowance()'=>array('label'=>'Allowance'),
|
|
|
|
'plugin()->traffic_thismonth()'=>array('label'=>'This Month'),
|
|
|
|
'plugin()->traffic_lastmonth()'=>array('label'=>'Last Month'),
|
|
|
|
'recur_schedule'=>array('label'=>'Billing'),
|
2012-03-30 15:13:01 +11:00
|
|
|
'price(TRUE,TRUE)'=>array('label'=>'Price','class'=>'right'),
|
2011-09-28 16:46:22 +10:00
|
|
|
'account->accnum()'=>array('label'=>'Cust ID'),
|
|
|
|
'account->name()'=>array('label'=>'Customer'),
|
|
|
|
'date_next_invoice'=>array('label'=>'Next Invoice'),
|
|
|
|
),
|
|
|
|
array(
|
|
|
|
'type'=>'select',
|
|
|
|
'form'=>'user/service/view',
|
|
|
|
)),
|
2010-11-30 09:41:08 +11:00
|
|
|
));
|
2011-09-28 16:46:22 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
public function action_listdomainservices() {
|
|
|
|
$svs = ORM::factory('service')->list_bylistgroup('DOMAIN');
|
|
|
|
Sort::MAsort($svs,'name()');
|
2010-11-30 09:41:08 +11:00
|
|
|
|
|
|
|
Block::add(array(
|
2011-09-28 16:46:22 +10:00
|
|
|
'title'=>_('Domain Names'),
|
|
|
|
'body'=>Table::display(
|
|
|
|
$svs,
|
|
|
|
25,
|
|
|
|
array(
|
|
|
|
'id'=>array('label'=>'ID','url'=>'user/service/view/'),
|
|
|
|
'service_name()'=>array('label'=>'Details'),
|
|
|
|
'plugin()->display("domain_expire")'=>array('label'=>'Expire'),
|
|
|
|
'recur_schedule'=>array('label'=>'Billing'),
|
2012-03-30 15:13:01 +11:00
|
|
|
'price(TRUE,TRUE)'=>array('label'=>'Price','class'=>'right'),
|
2011-09-28 16:46:22 +10:00
|
|
|
'account->accnum()'=>array('label'=>'Cust ID'),
|
|
|
|
'account->name()'=>array('label'=>'Customer'),
|
|
|
|
'display("date_next_invoice")'=>array('label'=>'Next Invoice'),
|
|
|
|
),
|
|
|
|
array(
|
|
|
|
'page'=>TRUE,
|
|
|
|
'type'=>'select',
|
|
|
|
'form'=>'user/service/view',
|
|
|
|
)),
|
2010-11-30 09:41:08 +11:00
|
|
|
));
|
|
|
|
}
|
|
|
|
|
2011-10-14 16:44:12 +11:00
|
|
|
public function action_listdomainservicesbysupplier() {
|
|
|
|
$svs = ORM::factory('service')->list_bylistgroup('DOMAIN');
|
|
|
|
Sort::MAsort($svs,'plugin()->domain_registrar_id,name()');
|
|
|
|
|
|
|
|
$list = array();
|
|
|
|
|
|
|
|
foreach ($svs as $so)
|
|
|
|
$list[$so->plugin()->domain_registrar_id][] = $so;
|
|
|
|
|
|
|
|
foreach (array_keys($list) as $sid)
|
|
|
|
Block::add(array(
|
|
|
|
'title'=>sprintf(_('Domain Names by Supplier %s'),$sid),
|
|
|
|
'body'=>Table::display(
|
|
|
|
$list[$sid],
|
|
|
|
25,
|
|
|
|
array(
|
|
|
|
'id'=>array('label'=>'ID','url'=>'user/service/view/'),
|
|
|
|
'service_name()'=>array('label'=>'Details'),
|
|
|
|
'plugin()->display("domain_expire")'=>array('label'=>'Expire'),
|
|
|
|
'recur_schedule'=>array('label'=>'Billing'),
|
2012-03-30 15:13:01 +11:00
|
|
|
'price(TRUE,TRUE)'=>array('label'=>'Price','class'=>'right'),
|
2011-10-14 16:44:12 +11:00
|
|
|
'account->accnum()'=>array('label'=>'Cust ID'),
|
|
|
|
'account->name()'=>array('label'=>'Customer'),
|
|
|
|
'display("date_next_invoice")'=>array('label'=>'Next Invoice'),
|
|
|
|
),
|
|
|
|
array(
|
|
|
|
'page'=>TRUE,
|
|
|
|
'type'=>'select',
|
|
|
|
'form'=>'user/service/view',
|
|
|
|
)),
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
2012-06-22 22:05:43 +10:00
|
|
|
public function action_listdomainservicesbydnshost() {
|
|
|
|
$svs = ORM::factory('service')->list_bylistgroup('DOMAIN');
|
|
|
|
Sort::MAsort($svs,'plugin()->service_plugin_host,name()');
|
|
|
|
|
|
|
|
$list = array();
|
|
|
|
|
|
|
|
foreach ($svs as $so)
|
|
|
|
$list[$so->plugin()->service_plugin_host->host_server_id][] = $so;
|
|
|
|
|
|
|
|
foreach (array_keys($list) as $sid)
|
|
|
|
Block::add(array(
|
|
|
|
'title'=>sprintf(_('Domain Names by DNS Host [%s]'),$sid),
|
|
|
|
'body'=>Table::display(
|
|
|
|
$list[$sid],
|
|
|
|
25,
|
|
|
|
array(
|
|
|
|
'id'=>array('label'=>'ID','url'=>'user/service/view/'),
|
|
|
|
'service_name()'=>array('label'=>'Details'),
|
|
|
|
'plugin()->domain_registrar->id'=>array('label'=>'SID'),
|
|
|
|
'plugin()->domain_registrar->name'=>array('label'=>'Supplier'),
|
|
|
|
'display("date_next_invoice")'=>array('label'=>'Next Invoice'),
|
|
|
|
),
|
|
|
|
array(
|
|
|
|
'page'=>TRUE,
|
|
|
|
'type'=>'select',
|
|
|
|
'form'=>'user/service/view',
|
|
|
|
)),
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
2011-09-28 16:46:22 +10:00
|
|
|
public function action_listhostservices() {
|
|
|
|
$svs = ORM::factory('service')->list_bylistgroup('HOST');
|
|
|
|
Sort::MAsort($svs,'name()');
|
|
|
|
|
|
|
|
Block::add(array(
|
|
|
|
'title'=>_('Hosting Services'),
|
|
|
|
'body'=>Table::display(
|
|
|
|
$svs,
|
|
|
|
25,
|
|
|
|
array(
|
|
|
|
'id'=>array('label'=>'ID','url'=>'user/service/view/'),
|
|
|
|
'service_name()'=>array('label'=>'Details'),
|
|
|
|
'plugin()->display("host_expire")'=>array('label'=>'Expire'),
|
|
|
|
'recur_schedule'=>array('label'=>'Billing'),
|
2012-03-30 15:13:01 +11:00
|
|
|
'price(TRUE,TRUE)'=>array('label'=>'Price','class'=>'right'),
|
2011-09-28 16:46:22 +10:00
|
|
|
'account->accnum()'=>array('label'=>'Cust ID'),
|
|
|
|
'account->name()'=>array('label'=>'Customer'),
|
|
|
|
'display("date_next_invoice")'=>array('label'=>'Next Invoice'),
|
|
|
|
),
|
|
|
|
array(
|
|
|
|
'page'=>TRUE,
|
|
|
|
'type'=>'select',
|
|
|
|
'form'=>'user/service/view',
|
|
|
|
)),
|
|
|
|
));
|
|
|
|
}
|
2011-12-27 00:52:46 +11:00
|
|
|
|
2012-06-22 22:05:43 +10:00
|
|
|
public function action_listhostservicesbysupplier() {
|
|
|
|
$svs = ORM::factory('service')->list_bylistgroup('HOST');
|
|
|
|
Sort::MAsort($svs,'plugin()->host_server,name()');
|
|
|
|
|
|
|
|
$list = array();
|
|
|
|
|
|
|
|
foreach ($svs as $so)
|
|
|
|
$list[$so->plugin()->host_server_id][] = $so;
|
|
|
|
|
|
|
|
foreach (array_keys($list) as $sid)
|
|
|
|
Block::add(array(
|
|
|
|
'title'=>sprintf(_('Hosting by Supplier %s'),$sid),
|
|
|
|
'body'=>Table::display(
|
|
|
|
$list[$sid],
|
|
|
|
25,
|
|
|
|
array(
|
|
|
|
'id'=>array('label'=>'ID','url'=>'user/service/view/'),
|
|
|
|
'service_name()'=>array('label'=>'Details'),
|
|
|
|
'plugin()->display("host_expire")'=>array('label'=>'Expire'),
|
|
|
|
'recur_schedule'=>array('label'=>'Billing'),
|
|
|
|
'price(TRUE,TRUE)'=>array('label'=>'Price','class'=>'right'),
|
|
|
|
'account->accnum()'=>array('label'=>'Cust ID'),
|
|
|
|
'account->name()'=>array('label'=>'Customer'),
|
|
|
|
'display("date_next_invoice")'=>array('label'=>'Next Invoice'),
|
|
|
|
),
|
|
|
|
array(
|
|
|
|
'page'=>TRUE,
|
|
|
|
'type'=>'select',
|
|
|
|
'form'=>'user/service/view',
|
|
|
|
)),
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
2011-12-27 00:52:46 +11:00
|
|
|
public function action_listwebservices() {
|
|
|
|
$svs = ORM::factory('service')->list_bylistgroup('WEB');
|
|
|
|
Sort::MAsort($svs,'name()');
|
|
|
|
|
|
|
|
Block::add(array(
|
|
|
|
'title'=>_('SSL Services'),
|
|
|
|
'body'=>Table::display(
|
|
|
|
$svs,
|
|
|
|
25,
|
|
|
|
array(
|
|
|
|
'id'=>array('label'=>'ID','url'=>'user/service/view/'),
|
|
|
|
'service_name()'=>array('label'=>'Details'),
|
|
|
|
'recur_schedule'=>array('label'=>'Billing'),
|
2012-03-30 15:13:01 +11:00
|
|
|
'price(TRUE,TRUE)'=>array('label'=>'Price','class'=>'right'),
|
2011-12-27 00:52:46 +11:00
|
|
|
'account->accnum()'=>array('label'=>'Cust ID'),
|
|
|
|
'account->name()'=>array('label'=>'Customer'),
|
|
|
|
'display("date_next_invoice")'=>array('label'=>'Next Invoice'),
|
|
|
|
),
|
|
|
|
array(
|
|
|
|
'page'=>TRUE,
|
|
|
|
'type'=>'select',
|
|
|
|
'form'=>'user/service/view',
|
|
|
|
)),
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
2010-11-30 09:41:08 +11:00
|
|
|
/**
|
|
|
|
* Reconcile billing for an ADSL supplier
|
|
|
|
*
|
|
|
|
* @todo this should really be in a different class, since adsl wont be part of the main app
|
|
|
|
*/
|
|
|
|
public function action_listadslbilling($id) {
|
|
|
|
$aso = ORM::factory('adsl_supplier',$id);
|
|
|
|
|
|
|
|
// Process upload
|
|
|
|
// @todo This should be separated out by supplier in case each supplier has a different format
|
2011-07-14 08:59:32 +10:00
|
|
|
if ($_FILES) {
|
|
|
|
$files = Validation::factory($_FILES)
|
|
|
|
->rule('csv','Upload::valid')
|
|
|
|
->rule('csv','Upload::not_empty')
|
|
|
|
->rule('csv','Upload::type',array(':value',array('csv')))
|
|
|
|
->rule('csv','Upload::size',array(':value','10M'));
|
|
|
|
|
2011-10-14 16:44:12 +11:00
|
|
|
if ($files->check())
|
2011-07-14 08:59:32 +10:00
|
|
|
foreach ($files as $file)
|
|
|
|
$csv = $this->process($file);
|
2010-11-30 09:41:08 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
// @todo add a display if there are no items
|
|
|
|
$i = $j = 0;
|
|
|
|
$total = 0;
|
|
|
|
$summary = '';
|
2011-12-21 12:39:21 +11:00
|
|
|
$output = View::factory($this->viewpath().'/head');
|
2010-11-30 09:41:08 +11:00
|
|
|
$output .= '<table class="box-left">';
|
|
|
|
foreach ($aso->services(TRUE) as $so) {
|
2011-09-28 16:46:22 +10:00
|
|
|
$po = $so->plugin()->product();
|
|
|
|
|
2010-11-30 09:41:08 +11:00
|
|
|
// Reset our uploaded data
|
|
|
|
$uploaded = array();
|
2011-09-28 16:46:22 +10:00
|
|
|
$uploaded['excess'] = empty($csv[$so->plugin()->service_number]['excess']) ? 0 : $csv[$so->plugin()->service_number]['excess'];
|
|
|
|
|
2010-11-30 09:41:08 +11:00
|
|
|
// If our uploaded file has some cost data.
|
2011-09-28 16:46:22 +10:00
|
|
|
if (! empty($csv[$so->plugin()->service_number])) {
|
2011-10-14 16:44:12 +11:00
|
|
|
$uploaded['amount'] =
|
2011-09-28 16:46:22 +10:00
|
|
|
(empty($csv[$so->plugin()->service_number]['cost']) ? 0 : $csv[$so->plugin()->service_number]['cost']) +
|
|
|
|
(empty($csv[$so->plugin()->service_number]['credit']) ? 0 : $csv[$so->plugin()->service_number]['credit']);
|
2010-11-30 09:41:08 +11:00
|
|
|
|
|
|
|
// Record the the exception if the cost is not expected
|
2011-09-28 16:46:22 +10:00
|
|
|
if (round($po->adsl_supplier_plan->base_cost+$po->adsl_supplier_plan->tax(),2) != $uploaded['amount']) {
|
2011-12-21 12:39:21 +11:00
|
|
|
$summary .= View::factory($this->viewpath().'/summary')
|
2010-11-30 09:41:08 +11:00
|
|
|
->set('service',$so)
|
2011-09-28 16:46:22 +10:00
|
|
|
->set('plan',$po)
|
|
|
|
->set('planoverride',$so->plugin()->provided_adsl_plan_id ? TRUE : FALSE)
|
2010-11-30 09:41:08 +11:00
|
|
|
->set('amount',$uploaded['amount'])
|
|
|
|
->set('i',$j++%2);
|
|
|
|
|
|
|
|
$uploaded['checked'] = '';
|
|
|
|
} else {
|
|
|
|
$uploaded['checked'] = 'checked="checked"';
|
|
|
|
}
|
|
|
|
|
2011-09-28 16:46:22 +10:00
|
|
|
unset($csv[$so->plugin()->service_number]);
|
2010-11-30 09:41:08 +11:00
|
|
|
|
|
|
|
} else {
|
|
|
|
$uploaded['checked'] = '';
|
|
|
|
$uploaded['amount'] = 0;
|
|
|
|
}
|
2011-09-28 16:46:22 +10:00
|
|
|
|
2010-11-30 09:41:08 +11:00
|
|
|
$total += $uploaded['amount'];
|
|
|
|
|
2011-12-21 12:39:21 +11:00
|
|
|
$output .= View::factory($this->viewpath().'/body')
|
2010-11-30 09:41:08 +11:00
|
|
|
->set('service',$so)
|
2011-09-28 16:46:22 +10:00
|
|
|
->set('plan',$po)
|
|
|
|
->set('planoverride',$so->plugin()->provided_adsl_plan_id ? TRUE : FALSE)
|
2010-11-30 09:41:08 +11:00
|
|
|
->set('checked',$uploaded['checked'])
|
|
|
|
->set('amount',$uploaded['amount'])
|
2011-09-28 16:46:22 +10:00
|
|
|
->set('excess',$uploaded['excess'])
|
|
|
|
->set('adsl',$so->plugin())
|
2010-11-30 09:41:08 +11:00
|
|
|
->set('i',$i++%2);
|
|
|
|
}
|
|
|
|
|
2011-12-21 12:39:21 +11:00
|
|
|
$output .= View::factory($this->viewpath().'/foot')
|
2010-11-30 09:41:08 +11:00
|
|
|
->set('total',$total);
|
|
|
|
|
|
|
|
$output .= '</table>';
|
|
|
|
|
|
|
|
// Summary Report of remaining CSV items.
|
|
|
|
if (! empty($csv))
|
|
|
|
foreach ($csv as $service => $item) {
|
2011-12-21 12:39:21 +11:00
|
|
|
$summary .= View::factory($this->viewpath().'/summary_exception')
|
2010-11-30 09:41:08 +11:00
|
|
|
->set('service',$service)
|
|
|
|
->set('item',$item)
|
|
|
|
->set('i',$j++%2);
|
|
|
|
}
|
|
|
|
|
|
|
|
$output .= Form::open(NULL,array('enctype'=>'multipart/form-data'));
|
|
|
|
$output .= '<div>';
|
|
|
|
$output .= Form::file('csv');
|
2011-08-31 16:54:44 +10:00
|
|
|
$output .= Form::submit('submit','upload',array('class'=>'form_button'));
|
2010-11-30 09:41:08 +11:00
|
|
|
$output .= '</div>';
|
|
|
|
$output .= Form::close();
|
|
|
|
|
|
|
|
Block::add(array(
|
|
|
|
'title'=>_('ADSL Services'),
|
|
|
|
'body'=>$output,
|
2011-07-14 19:09:03 +10:00
|
|
|
));
|
2010-11-30 09:41:08 +11:00
|
|
|
|
|
|
|
if ($summary)
|
|
|
|
Block::add(array(
|
|
|
|
'title'=>_('Exception Charges'),
|
|
|
|
'body'=>'<table class="box-left">'.$summary.'</table>',
|
|
|
|
));
|
|
|
|
|
|
|
|
Style::add(array(
|
|
|
|
'type'=>'file',
|
|
|
|
'data'=>'css/list.css',
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
|
|
|
private function process(array $file) {
|
|
|
|
$data = file_get_contents($file['tmp_name']);
|
|
|
|
|
|
|
|
if (! $data)
|
|
|
|
return;
|
|
|
|
|
|
|
|
$start = $end = FALSE;
|
|
|
|
$return = array();
|
|
|
|
foreach (preg_split("/\n/",$data) as $line) {
|
|
|
|
// Items start after "Item ID"
|
|
|
|
if (! $start && preg_match('/^Item ID,/',$line)) {
|
|
|
|
$start = true;
|
|
|
|
continue;
|
|
|
|
// Items end after "Subtotal"
|
|
|
|
} elseif ($start && ! $end && preg_match('/^Subtotal:,/',$line)) {
|
|
|
|
$end = true;
|
|
|
|
continue;
|
|
|
|
// If we havent started or not ended, continue
|
|
|
|
} elseif (! $start || $end) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
list($id,$ref,$unknown,$linedata,$q,$cost,$total) = explode(',',$line);
|
|
|
|
|
|
|
|
// Extract the phone number from the $linedata
|
|
|
|
@list($service,$description) = explode(':',(preg_replace('/([0-9]+)\s+-\s+(.*)$/',"$1:$2",$linedata)));
|
|
|
|
|
|
|
|
// If the description says Monthly Charge, we know its the monthly fee.
|
|
|
|
if (preg_match('/^Monthly Charge/',$description))
|
|
|
|
$return[$service]['cost'] = preg_replace('/\$/','',$total);
|
|
|
|
// If the description says VISP credit, we know this is commission.
|
|
|
|
elseif (preg_match('/^VISP Credit/',$description))
|
|
|
|
$return[$service]['credit'] = preg_replace('/\$/','',$total);
|
2011-09-28 16:46:22 +10:00
|
|
|
// If the description says Excess, we know this is excess charges.
|
2010-11-30 09:41:08 +11:00
|
|
|
elseif (preg_match('/^Excess usage/',$description))
|
|
|
|
$return[$service]['excess'] = preg_replace('/\$/','',$total);
|
|
|
|
else
|
|
|
|
$return[$service]['info'] = $line;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $return;
|
|
|
|
}
|
2011-07-14 19:09:03 +10:00
|
|
|
|
2011-10-11 10:38:21 +11:00
|
|
|
/**
|
|
|
|
* List services that need to be invoiced.
|
|
|
|
*/
|
|
|
|
public function action_listinvoicesoon() {
|
|
|
|
Block::add(array(
|
|
|
|
'title'=>_('Services to Invoice'),
|
|
|
|
'body'=>Table::display(
|
2012-01-02 12:35:47 +11:00
|
|
|
ORM::factory('service')->list_invoicesoon(ORM::factory('invoice')->config('GEN_SOON_DAYS')),
|
2011-10-11 10:38:21 +11:00
|
|
|
25,
|
|
|
|
array(
|
|
|
|
'id'=>array('label'=>'ID','url'=>'user/service/view/'),
|
|
|
|
'service_name()'=>array('label'=>'Details'),
|
|
|
|
'recur_schedule'=>array('label'=>'Billing'),
|
|
|
|
'date_next_invoice'=>array('label'=>'Next Invoice'),
|
2012-03-30 15:13:01 +11:00
|
|
|
'price(TRUE,TRUE)'=>array('label'=>'Price','class'=>'right'),
|
2012-02-22 19:15:46 +11:00
|
|
|
'charges_new()'=>array('label'=>'Charges','class'=>'right'),
|
2012-08-01 22:43:33 +10:00
|
|
|
'status'=>array('label'=>'Active'),
|
2011-10-11 10:38:21 +11:00
|
|
|
'account->accnum()'=>array('label'=>'Cust ID'),
|
|
|
|
'account->name()'=>array('label'=>'Customer'),
|
|
|
|
),
|
|
|
|
array(
|
|
|
|
'page'=>TRUE,
|
|
|
|
'type'=>'select',
|
|
|
|
'form'=>'user/service/view',
|
|
|
|
)),
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
2011-12-17 10:31:35 +11:00
|
|
|
public function action_update() {
|
|
|
|
$id = $this->request->param('id');
|
2011-07-14 19:09:03 +10:00
|
|
|
$so = ORM::factory('service',$id);
|
|
|
|
|
|
|
|
if (! $so->loaded())
|
|
|
|
Request::current()->redirect('welcome/index');
|
|
|
|
|
|
|
|
if ($_POST) {
|
|
|
|
if (isset($_POST['plugin']) AND $_POST['plugin'])
|
|
|
|
if (! $so->plugin()->values($_POST['plugin'])->update()->saved())
|
|
|
|
throw new Kohana_Exception('Failed to save updates to plugin data for record :record',array(':record'=>$so->id()));
|
|
|
|
|
|
|
|
if (! $so->values($_POST)->update()->saved())
|
2012-03-30 15:13:01 +11:00
|
|
|
throw new Kohana_Exception('Failed to save updates to service data for record :record',array(':record'=>$so->id()));
|
2011-07-14 19:09:03 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
Block::add(array(
|
|
|
|
'title'=>sprintf('%s %s:%s',_('Update Service'),$so->id(),$so->name()),
|
2011-12-27 00:52:46 +11:00
|
|
|
'body'=>View::factory($this->viewpath())
|
2011-07-14 19:09:03 +10:00
|
|
|
->set('so',$so)
|
2011-07-22 11:04:20 +10:00
|
|
|
->set('mediapath',Route::get('default/media'))
|
2011-07-14 19:09:03 +10:00
|
|
|
->set('plugin_form',$so->admin_update()),
|
|
|
|
));
|
2011-07-22 11:04:20 +10:00
|
|
|
|
|
|
|
// @todo Investigate a better way of preparing for jscalendar
|
|
|
|
Script::add(array(
|
|
|
|
'type'=>'file',
|
|
|
|
'data'=>'js/dhtml.calendar.js',
|
|
|
|
));
|
|
|
|
Script::add(array(
|
|
|
|
'type'=>'file',
|
|
|
|
'data'=>'js/dhtml.calendar-setup.js',
|
|
|
|
));
|
|
|
|
Script::add(array(
|
|
|
|
'type'=>'file',
|
|
|
|
'data'=>'js/dhtml.calendar-en.js',
|
|
|
|
));
|
|
|
|
Script::add(array(
|
|
|
|
'type'=>'file',
|
|
|
|
'data'=>'js/dhtml.date_selector.js',
|
|
|
|
));
|
|
|
|
Style::add(array(
|
|
|
|
'type'=>'file',
|
|
|
|
'data'=>'css/dhtml.calendar.css',
|
|
|
|
));
|
2011-07-14 19:09:03 +10:00
|
|
|
}
|
2012-01-09 12:35:24 +11:00
|
|
|
|
|
|
|
public function action_view() {
|
|
|
|
list($id,$output) = Table::page(__METHOD__);
|
|
|
|
|
|
|
|
$so = ORM::factory('service',$id);
|
|
|
|
|
|
|
|
if (! $so->loaded() OR ! Auth::instance()->authorised($so->account_id,$so->affiliate_id)) {
|
|
|
|
$this->template->content = 'Unauthorised or doesnt exist?';
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2012-03-31 14:33:13 +11:00
|
|
|
$doutput = $loutput = '';
|
2012-01-09 12:35:24 +11:00
|
|
|
|
|
|
|
$loutput .= View::factory($this->viewpath())
|
|
|
|
->set('so',$so);
|
|
|
|
|
2012-03-31 14:33:13 +11:00
|
|
|
// Validate the transactions
|
|
|
|
$bt = NULL;
|
|
|
|
$save = (isset($_REQUEST['go']) && $_REQUEST['go']=1) ? 1 : 0;
|
|
|
|
$xsid=197;
|
|
|
|
foreach ($so->transactions()->where('item_type','=',0)->find_all() as $iio) {
|
|
|
|
if ($iio->service_id == $xsid AND $iio->invoice_id < 3070) continue;
|
|
|
|
if ($iio->quantity < 0 OR $iio->price_base < 0)
|
|
|
|
continue;
|
|
|
|
if (in_array($iio->id,array(960)))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if ($iio->invoice_id > 4000 OR $iio->product->prod_plugin_file=="ADSL")
|
|
|
|
$a = FALSE;
|
|
|
|
else
|
|
|
|
$a = TRUE;
|
|
|
|
|
|
|
|
if (is_null($bt))
|
|
|
|
$bt = $iio->date_start;
|
|
|
|
|
|
|
|
// $doutput .= sprintf('%s: BT is: %s [%s]<br/>',$iio->id,Config::date($bt),$bt);
|
|
|
|
|
|
|
|
$pdata = Period::details($iio->recurring_schedule,$a ? NULL : $iio->product->price_recurr_weekday,$bt,TRUE);
|
|
|
|
|
|
|
|
switch ($iio->recurring_schedule) {
|
|
|
|
case 1:
|
|
|
|
case 2:
|
|
|
|
case 4:
|
|
|
|
case 5:
|
|
|
|
if ($iio->date_start != $pdata['start_time']) {
|
|
|
|
$doutput .= sprintf('%s: Set start_time: %s [%s]<br/>',$iio->id,Config::date($pdata['start_time']),$pdata['start_time']);
|
|
|
|
$iio->date_start=$pdata['start_time'];
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($iio->date_stop != $pdata['end_time']) {
|
|
|
|
$doutput .= sprintf('%s: Set end_time: %s [%s] <br/>',$iio->id,Config::date($pdata['end_time']),$pdata['end_time']);
|
|
|
|
$iio->date_stop=$pdata['end_time'];
|
|
|
|
}
|
|
|
|
$bt = $pdata['end_time']+86400;
|
|
|
|
|
|
|
|
//$doutput .= sprintf('%s: BT now: %s (%s) [%s]<br/>',$iio->id,Config::date($bt),Config::date($pdata['end_time']),$bt);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
$doutput .= sprintf('%s: %s Not handled',$iio->id,$iio->recurring_schedule);
|
|
|
|
}
|
|
|
|
//$doutput .= '<br/>';
|
|
|
|
if ($save) {
|
|
|
|
$iio->save();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isset($_REQUEST['go']))
|
|
|
|
$this->request->redirect('admin/service/view/'.$so->id);
|
2012-01-09 12:35:24 +11:00
|
|
|
Block::add(array(
|
|
|
|
'title'=>sprintf('Transaction History for %s: %s',$so->id(),$so->name()),
|
|
|
|
'body'=>$loutput,
|
|
|
|
));
|
|
|
|
|
2012-03-31 14:33:13 +11:00
|
|
|
Block::add(array(
|
|
|
|
'title'=>sprintf('Transaction Debug for %s: %s',$so->id(),$so->name()),
|
|
|
|
'body'=>$doutput,
|
|
|
|
));
|
|
|
|
|
2012-01-09 12:35:24 +11:00
|
|
|
$output .= View::factory('service/user/view')
|
|
|
|
->set('so',$so);
|
|
|
|
|
|
|
|
Block::add(array(
|
|
|
|
'title'=>sprintf('%s: %s',$so->id(),$so->service_name()),
|
|
|
|
'body'=>$output,
|
|
|
|
));
|
2012-05-13 09:29:44 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
public function action_adslstat() {
|
|
|
|
$output = '';
|
|
|
|
$svs = ORM::factory('service')->list_bylistgroup('ADSL');
|
|
|
|
$stats = array();
|
|
|
|
// @todo This needs to be configurable.
|
|
|
|
$traffic = array(1000,2000,5000,10000,25000,50000,75000,100000);
|
|
|
|
$ts = 0;
|
|
|
|
|
|
|
|
foreach ($svs as $a=>$so) {
|
|
|
|
// Number of services
|
|
|
|
if (! isset($stats[$so->product->plugin()->speed]['c']))
|
|
|
|
$stats[$so->product->plugin()->speed]['c'] = 0;
|
|
|
|
|
|
|
|
$stats[$so->product->plugin()->speed]['c']++;
|
|
|
|
$ts++;
|
|
|
|
|
|
|
|
// Amount of traffic
|
|
|
|
$t = array_sum($so->plugin()->traffic_lastmonth(FALSE));
|
|
|
|
$a = 0;
|
|
|
|
foreach (array_reverse($traffic) as $i) {
|
|
|
|
if ($i < $t)
|
|
|
|
break;
|
|
|
|
$a = $i;
|
|
|
|
}
|
2012-01-09 12:35:24 +11:00
|
|
|
|
2012-05-13 09:29:44 +10:00
|
|
|
if (! isset($stats[$so->product->plugin()->speed]['d'][$a]))
|
|
|
|
$stats[$so->product->plugin()->speed]['d'][$a] = 0;
|
|
|
|
|
|
|
|
$stats[$so->product->plugin()->speed]['d'][$a]++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (count($stats)) {
|
|
|
|
$output .= View::factory($this->viewpath().'/head')
|
|
|
|
->set('name','SPEED');
|
|
|
|
$output .= View::factory($this->viewpath().'/head_data')
|
|
|
|
->set('name','#');
|
|
|
|
foreach ($traffic as $i)
|
|
|
|
$output .= View::factory($this->viewpath().'/head_data')
|
|
|
|
->set('name',$i);
|
|
|
|
$output .= View::factory($this->viewpath().'/head_data')
|
|
|
|
->set('name','Other');
|
|
|
|
$output .= View::factory($this->viewpath().'/head_end');
|
|
|
|
|
|
|
|
foreach ($stats as $speed => $details) {
|
|
|
|
$output .= View::factory($this->viewpath().'/body_head')
|
|
|
|
->set('count',$details['c'])
|
|
|
|
->set('percent',sprintf('%2.1f',$details['c']/$ts*100))
|
|
|
|
->set('speed',$speed);
|
|
|
|
|
|
|
|
foreach ($traffic as $i) {
|
|
|
|
$output .= View::factory($this->viewpath().'/body_data')
|
|
|
|
->set('count',$c=isset($details['d'][$i]) ? $details['d'][$i] : 0)
|
|
|
|
->set('percent',sprintf('%2.1f',$c/$details['c']*100));
|
|
|
|
}
|
|
|
|
|
|
|
|
$output .= View::factory($this->viewpath().'/body_data')
|
|
|
|
->set('count',$c=isset($details['d'][0]) ? $details['d'][0] : 0)
|
|
|
|
->set('percent',sprintf('%2.1f',$c/$details['c']*100));
|
|
|
|
$output .= View::factory($this->viewpath().'/body_end');
|
|
|
|
}
|
|
|
|
|
|
|
|
$output .= View::factory($this->viewpath().'/foot');
|
|
|
|
}
|
|
|
|
|
|
|
|
Block::add(array(
|
|
|
|
'title'=>_('ADSL Traffic Summary Stats - Last Month'),
|
|
|
|
'body'=>$output,
|
|
|
|
));
|
2012-01-09 12:35:24 +11:00
|
|
|
}
|
2010-11-30 09:41:08 +11:00
|
|
|
}
|
|
|
|
?>
|