You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1140 lines
38 KiB
1140 lines
38 KiB
package NGCP::Panel::Controller::Billing;
|
|
use NGCP::Panel::Utils::Generic qw(:all);
|
|
use Sipwise::Base;
|
|
use DateTime::Format::ISO8601;
|
|
|
|
use parent 'Catalyst::Controller';
|
|
|
|
use NGCP::Panel::Form;
|
|
|
|
use NGCP::Panel::Utils::Contract;
|
|
use NGCP::Panel::Utils::Message;
|
|
use NGCP::Panel::Utils::Navigation;
|
|
use NGCP::Panel::Utils::Datatables;
|
|
use NGCP::Panel::Utils::DateTime;
|
|
use NGCP::Panel::Utils::Billing;
|
|
|
|
sub auto :Private {
|
|
my ($self, $c) = @_;
|
|
$c->log->debug(__PACKAGE__ . '::auto');
|
|
NGCP::Panel::Utils::Navigation::check_redirect_chain(c => $c);
|
|
return 1;
|
|
}
|
|
|
|
sub profile_list :Chained('/') :PathPart('billing') :CaptureArgs(0) :Does(ACL) :ACLDetachTo('/denied_page') :AllowedRole(admin) :AllowedRole(reseller) :AllowedRole(ccareadmin) :AllowedRole(ccare) {
|
|
my ( $self, $c ) = @_;
|
|
|
|
my $dispatch_role = $c->user->roles =~ /admin$/ ? 'admin' : 'reseller';
|
|
my $dispatch_to = '_profile_resultset_' . $dispatch_role;
|
|
my $profiles_rs = $self->$dispatch_to($c);
|
|
$c->stash(profiles_rs => $profiles_rs);
|
|
$c->stash->{profile_dt_columns} = NGCP::Panel::Utils::Datatables::set_columns($c, [
|
|
{ name => "id", "search" => 1, "title" => $c->loc("#") },
|
|
{ name => "name", "search" => 1, "title" => $c->loc("Name") },
|
|
{ name => "reseller.name", "search" => 1, "title" => $c->loc("Reseller") },
|
|
NGCP::Panel::Utils::Billing::get_datatable_cols($c),
|
|
]);
|
|
|
|
$c->stash(template => 'billing/list.tt');
|
|
}
|
|
|
|
sub profile_list_restricted :Chained('profile_list') :PathPart('') :CaptureArgs(0) :Does(License) :RequiresLicense('billing') :LicenseDetachTo('/denied_page') :Does(ACL) :ACLDetachTo('/denied_page') :AllowedRole(admin) :AllowedRole(reseller) {
|
|
my ($self, $c) = @_;
|
|
}
|
|
|
|
sub _profile_resultset_admin {
|
|
my ($self, $c) = @_;
|
|
my $rs = $c->model('DB')->resultset('billing_profiles')->search({
|
|
'me.status' => { '!=' => 'terminated' },
|
|
},
|
|
{ '+select' => [ { '' => \[ NGCP::Panel::Utils::Billing::get_contract_count_stmt(1000) ] , -as => 'contract_cnt' },
|
|
{ '' => \[ NGCP::Panel::Utils::Billing::get_contract_exists_stmt() ] , -as => 'contract_exists' },
|
|
{ '' => \[ NGCP::Panel::Utils::Billing::get_package_count_stmt() ] , -as => 'package_cnt' }, ],
|
|
});
|
|
return $rs;
|
|
}
|
|
|
|
sub _profile_resultset_reseller {
|
|
my ($self, $c) = @_;
|
|
my $rs = $c->model('DB')->resultset('admins')
|
|
->find($c->user->id)->reseller->billing_profiles
|
|
->search_rs({
|
|
'me.status' => { '!=' => 'terminated' },
|
|
},
|
|
{ '+select' => [ { '' => \[ NGCP::Panel::Utils::Billing::get_contract_count_stmt(1000) ] , -as => 'contract_cnt' },
|
|
{ '' => \[ NGCP::Panel::Utils::Billing::get_contract_exists_stmt() ] , -as => 'contract_exists' },
|
|
{ '' => \[ NGCP::Panel::Utils::Billing::get_package_count_stmt() ] , -as => 'package_cnt' }, ],
|
|
});
|
|
return $rs;
|
|
}
|
|
|
|
sub root :Chained('profile_list') :PathPart('') :Args(0) {
|
|
my ($self, $c) = @_;
|
|
}
|
|
|
|
sub ajax :Chained('profile_list') :PathPart('ajax') :Args(0) {
|
|
my ($self, $c) = @_;
|
|
|
|
my $resultset = $c->stash->{profiles_rs};
|
|
NGCP::Panel::Utils::Datatables::process($c, $resultset, $c->stash->{profile_dt_columns});
|
|
|
|
$c->detach( $c->view("JSON") );
|
|
}
|
|
|
|
sub ajax_no_prepaid :Chained('profile_list') :PathPart('ajax_no_prepaid') :Args(0) {
|
|
my ($self, $c) = @_;
|
|
|
|
my $resultset = $c->stash->{profiles_rs}->search_rs({ prepaid => 0 });
|
|
NGCP::Panel::Utils::Datatables::process($c, $resultset, $c->stash->{profile_dt_columns});
|
|
|
|
$c->detach( $c->view("JSON") );
|
|
}
|
|
|
|
sub ajax_filter_reseller :Chained('profile_list') :PathPart('ajax/filter_reseller') :Args(1) {
|
|
my ($self, $c, $reseller_id) = @_;
|
|
|
|
my $resultset = $c->stash->{profiles_rs}->search({
|
|
'me.reseller_id' => $reseller_id,
|
|
});
|
|
NGCP::Panel::Utils::Datatables::process($c, $resultset, $c->stash->{profile_dt_columns});
|
|
$c->detach( $c->view("JSON") );
|
|
}
|
|
|
|
sub base :Chained('profile_list_restricted') :PathPart('') :CaptureArgs(1) {
|
|
my ($self, $c, $profile_id) = @_;
|
|
|
|
unless($profile_id && is_int($profile_id)) {
|
|
NGCP::Panel::Utils::Message::error(
|
|
c => $c,
|
|
data => { id => $profile_id },
|
|
desc => $c->loc('Invalid profile id detected!'),
|
|
);
|
|
$c->response->redirect($c->uri_for());
|
|
return;
|
|
}
|
|
|
|
my $res = $c->stash->{profiles_rs}->find($profile_id);
|
|
unless(defined($res)) {
|
|
NGCP::Panel::Utils::Message::error(
|
|
c => $c,
|
|
data => { id => $profile_id },
|
|
desc => $c->loc('Billing Profile does not exist!'),
|
|
);
|
|
$c->response->redirect($c->uri_for());
|
|
return;
|
|
}
|
|
$c->stash(profile => {$res->get_inflated_columns});
|
|
$c->stash(profile_result => $res);
|
|
}
|
|
|
|
sub edit :Chained('base') :PathPart('edit') {
|
|
my ($self, $c ) = @_;
|
|
$c->forward('process_edit', [0] );
|
|
}
|
|
sub process_edit :Private {
|
|
my ($self, $c, $duplicate) = @_;
|
|
|
|
my $posted = ($c->request->method eq 'POST');
|
|
my $form;
|
|
my $params = $c->stash->{profile};
|
|
$params->{reseller}{id} = delete $params->{reseller_id};
|
|
$params = merge($params, $c->session->{created_objects});
|
|
if( $duplicate ) {
|
|
NGCP::Panel::Utils::Billing::get_billing_profile_uniq_params( params => $params );
|
|
if($c->user->is_superuser) {
|
|
$form = NGCP::Panel::Form::get("NGCP::Panel::Form::BillingProfile::Admin", $c);
|
|
} else {
|
|
$form = NGCP::Panel::Form::get("NGCP::Panel::Form::BillingProfile::Reseller", $c);
|
|
}
|
|
} else {
|
|
$form = NGCP::Panel::Form::get("NGCP::Panel::Form::BillingProfile::Reseller", $c);
|
|
}
|
|
$form->process(
|
|
posted => $posted,
|
|
params => $c->request->params,
|
|
item => $params,
|
|
);
|
|
NGCP::Panel::Utils::Navigation::check_form_buttons(
|
|
c => $c,
|
|
form => $form,
|
|
fields => {
|
|
'reseller.create' => $c->uri_for('/reseller/create'),
|
|
},
|
|
back_uri => $c->req->uri,
|
|
);
|
|
if($posted && $form->validated) {
|
|
try {
|
|
if (exists $form->values->{reseller}) {
|
|
if($c->user->is_superuser) {
|
|
$form->values->{reseller_id} = $form->values->{reseller}{id};
|
|
} else {
|
|
$form->values->{reseller_id} = $c->user->reseller_id;
|
|
}
|
|
delete $form->values->{reseller};
|
|
}
|
|
$form->values->{modify_timestamp} = NGCP::Panel::Utils::DateTime::current_local;
|
|
my $old_prepaid = $c->stash->{profile_result}->prepaid;
|
|
|
|
my $schema = $c->model('DB');
|
|
$schema->txn_do(sub {
|
|
|
|
$c->stash->{profile_result}->update($form->values);
|
|
|
|
NGCP::Panel::Utils::Billing::switch_prepaid(c => $c,
|
|
profile_id => $c->stash->{profile_result}->id,
|
|
old_prepaid => $old_prepaid,
|
|
new_prepaid => $c->stash->{profile_result}->prepaid,
|
|
);
|
|
|
|
});
|
|
|
|
delete $c->session->{created_objects}->{reseller};
|
|
NGCP::Panel::Utils::Message::info(
|
|
c => $c,
|
|
desc => $c->loc('Billing profile successfully updated'),
|
|
);
|
|
} catch($e) {
|
|
NGCP::Panel::Utils::Message::error(
|
|
c => $c,
|
|
error => $e,
|
|
desc => $c->loc('Failed to update billing profile'),
|
|
);
|
|
}
|
|
NGCP::Panel::Utils::Navigation::back_or($c, $c->uri_for('/billing'));
|
|
}
|
|
$c->stash( 'duplicate_flag' => 1 ) if $duplicate;
|
|
$c->stash( 'edit_flag' => 1 );
|
|
$c->stash( 'form' => $form );
|
|
}
|
|
|
|
sub create :Chained('profile_list_restricted') :PathPart('create') :Args(0) {
|
|
my ($self, $c, $no_reseller) = @_;
|
|
$c->forward('process_create', [$no_reseller, 0 ]);
|
|
}
|
|
|
|
sub duplicate :Chained('base') :PathPart('duplicate') {
|
|
my ($self, $c, $no_reseller) = @_;
|
|
my $posted = ($c->request->method eq 'POST');
|
|
if(!$posted){
|
|
$c->forward('process_edit', [1] );
|
|
}else{
|
|
$c->forward('process_create', [ $no_reseller, 1 ] );
|
|
}
|
|
}
|
|
sub process_create :Private {
|
|
my ($self, $c, $no_reseller, $duplicate ) = @_;
|
|
|
|
my $schema = $c->model('DB');
|
|
my $posted = ($c->request->method eq 'POST');
|
|
my $form;
|
|
my $params = {};
|
|
$params->{reseller}{id} = delete $params->{reseller_id};
|
|
$params = merge($params, $c->session->{created_objects});
|
|
$c->stash->{no_prepaid_billing_profile} = $no_reseller;
|
|
if($c->user->is_superuser && !$no_reseller) {
|
|
$form = NGCP::Panel::Form::get("NGCP::Panel::Form::BillingProfile::Admin", $c);
|
|
} else {
|
|
$form = NGCP::Panel::Form::get("NGCP::Panel::Form::BillingProfile::Reseller", $c);
|
|
}
|
|
$form->process(
|
|
posted => $posted,
|
|
params => $c->request->params,
|
|
item => $params,
|
|
);
|
|
NGCP::Panel::Utils::Navigation::check_form_buttons(
|
|
c => $c,
|
|
form => $form,
|
|
fields => {
|
|
'reseller.create' => $c->uri_for('/reseller/create'),
|
|
},
|
|
back_uri => $c->req->uri,
|
|
);
|
|
if($posted && $form->validated) {
|
|
try {
|
|
if($c->user->is_superuser && $no_reseller) {
|
|
$form->values->{reseller_id} = $c->user->reseller_id;
|
|
} elsif($c->user->is_superuser) {
|
|
$form->values->{reseller_id} = $form->values->{reseller}{id};
|
|
} else {
|
|
$form->values->{reseller_id} = $c->user->reseller_id;
|
|
}
|
|
$form->values->{create_timestamp} = $form->values->{modify_timestamp} = NGCP::Panel::Utils::DateTime::current_local;
|
|
delete $form->values->{reseller};
|
|
delete $form->values->{id} if $duplicate;
|
|
my $profile = $c->model('DB')->resultset('billing_profiles')->create($form->values);
|
|
|
|
if( $duplicate ) {
|
|
|
|
NGCP::Panel::Utils::Billing::clone_billing_profile_tackles(
|
|
c => $c,
|
|
profile_old => $c->stash->{'profile_result'},
|
|
profile_new => $profile,
|
|
#profile_new => $c->stash->{profiles_rs}->find(
|
|
# $c->session->{created_objects}->{billing_profile}->{id},
|
|
#),
|
|
schema => $schema,
|
|
);
|
|
}
|
|
|
|
$c->session->{created_objects}->{billing_profile} = { id => $profile->id };
|
|
delete $c->session->{created_objects}->{reseller};
|
|
NGCP::Panel::Utils::Message::info(
|
|
c => $c,
|
|
desc => $c->loc('Billing profile successfully created'),
|
|
);
|
|
} catch($e) {
|
|
NGCP::Panel::Utils::Message::error(
|
|
c => $c,
|
|
error => $e,
|
|
desc => $c->loc('Failed to create billing profile'),
|
|
);
|
|
}
|
|
NGCP::Panel::Utils::Navigation::back_or($c, $c->uri_for('/billing'));
|
|
}
|
|
|
|
$c->stash(create_flag => 1);
|
|
$c->stash(form => $form);
|
|
}
|
|
|
|
sub create_without_reseller :Chained('profile_list_restricted') :PathPart('create/noreseller') :Args(0) {
|
|
my ($self, $c) = @_;
|
|
|
|
$self->create($c, 1);
|
|
}
|
|
|
|
sub terminate :Chained('base') :PathPart('terminate') :Args(0) {
|
|
my ($self, $c) = @_;
|
|
my $profile = $c->stash->{profile_result};
|
|
|
|
if ($profile->id == 1) {
|
|
NGCP::Panel::Utils::Message::error(
|
|
c => $c,
|
|
desc => $c->loc('Cannot terminate default billing profile with the id 1'),
|
|
);
|
|
NGCP::Panel::Utils::Navigation::back_or($c, $c->uri_for('/billing'));
|
|
}
|
|
|
|
try {
|
|
#todo: putting the profile fetch into a transaction wouldn't help since the count columns a prone to phantom reads...
|
|
if($profile->get_column('contract_exists')) {
|
|
die(['Cannnot terminate billing profile that is still used in profile mappings', "showdetails"]);
|
|
}
|
|
unless($profile->get_column('package_cnt') == 0) {
|
|
die(['Cannnot terminate billing profile that is still used in profile packages', "showdetails"]);
|
|
}
|
|
$profile->update({
|
|
status => 'terminated',
|
|
terminate_timestamp => NGCP::Panel::Utils::DateTime::current_local,
|
|
});
|
|
NGCP::Panel::Utils::Message::info(
|
|
c => $c,
|
|
data => $c->stash->{profile},
|
|
desc => $c->loc('Billing profile successfully terminated'),
|
|
);
|
|
} catch ($e) {
|
|
NGCP::Panel::Utils::Message::error(
|
|
c => $c,
|
|
error => $e,
|
|
data => $c->stash->{profile},
|
|
desc => $c->loc('Failed to terminate billing profile'),
|
|
);
|
|
};
|
|
NGCP::Panel::Utils::Navigation::back_or($c, $c->uri_for('/billing'));
|
|
}
|
|
|
|
sub fees_list :Chained('base') :PathPart('fees') :CaptureArgs(0) {
|
|
my ($self, $c) = @_;
|
|
|
|
$c->stash->{fee_dt_columns} = NGCP::Panel::Utils::Datatables::set_columns($c, [
|
|
{ name => 'id', search => 1, title => $c->loc('#') },
|
|
{ name => 'source', search => 1, title => $c->loc('Source Pattern') },
|
|
{ name => 'destination', search => 1, title => $c->loc('Destination Pattern') },
|
|
{ name => 'match_mode', search => 0, title => $c->loc('Match Mode'),
|
|
custom_renderer => 'function ( data, type, full, opt ) {'.
|
|
'if(full.match_mode == "regex_longest_pattern"){return "' . $c->loc('Regular expression - longest pattern') . '";}'.
|
|
'else if(full.match_mode == "regex_longest_match"){return "' . $c->loc('Regular expression - longest match') . '";}'.
|
|
'else if(full.match_mode == "prefix"){return "' . $c->loc('Prefix string') . '";}'.
|
|
'else if(full.match_mode == "exact_destination"){return "' . $c->loc('Exact string (destination)') . '";}'.
|
|
'}',
|
|
},
|
|
{ name => 'direction', search => 1, title => $c->loc('Direction') },
|
|
{ name => 'billing_zone.detail', search => 1, title => $c->loc('Billing Zone') },
|
|
]);
|
|
$c->stash(template => 'billing/fees.tt');
|
|
}
|
|
|
|
sub fees :Chained('fees_list') :PathPart('') :Args(0) {
|
|
my ($self, $c) = @_;
|
|
|
|
}
|
|
|
|
sub fees_base :Chained('fees_list') :PathPart('') :CaptureArgs(1) {
|
|
my ($self, $c, $fee_id) = @_;
|
|
|
|
unless($fee_id && is_int($fee_id)) {
|
|
$fee_id //= '';
|
|
NGCP::Panel::Utils::Message::error(
|
|
c => $c,
|
|
data => $c->stash->{profile},
|
|
log => $fee_id,
|
|
desc => $c->loc('Invalid billing fee id detected!'),
|
|
);
|
|
$c->response->redirect($c->uri_for($c->stash->{profile}->{id}, 'fees'));
|
|
return;
|
|
}
|
|
|
|
my $res = $c->stash->{'profile_result'}->billing_fees
|
|
->search(undef, {join => 'billing_zone',})
|
|
->find($fee_id);
|
|
unless(defined($res)) {
|
|
NGCP::Panel::Utils::Message::error(
|
|
c => $c,
|
|
data => $c->stash->{profile},
|
|
log => $fee_id,
|
|
desc => $c->loc('Billing Fee does not exist!'),
|
|
);
|
|
$c->response->redirect($c->uri_for($c->stash->{profile}->{id}, 'fees'));
|
|
return;
|
|
}
|
|
$c->stash(fee => {$res->get_columns}); #get_columns should not be used
|
|
$c->stash->{fee}->{'billing_zone_id'} = $res->billing_zone->id
|
|
if (defined $res->billing_zone);
|
|
$c->stash(fee_result => $res);
|
|
}
|
|
|
|
sub fees_ajax :Chained('fees_list') :PathPart('ajax') :Args(0) {
|
|
my ($self, $c) = @_;
|
|
|
|
my $resultset = $c->stash->{'profile_result'}->billing_fees;
|
|
NGCP::Panel::Utils::Datatables::process($c, $resultset, $c->stash->{fee_dt_columns});
|
|
$c->detach( $c->view("JSON") );
|
|
}
|
|
|
|
sub fees_create :Chained('fees_list') :PathPart('create') :Args(0) {
|
|
my ($self, $c) = @_;
|
|
|
|
my $posted = ($c->request->method eq 'POST');
|
|
my $params = {};
|
|
$params = merge($params, $c->session->{created_objects});
|
|
my $profile_id = $c->stash->{profile}->{id};
|
|
my $form = NGCP::Panel::Form::get("NGCP::Panel::Form::BillingFee", $c);
|
|
$form->process(
|
|
posted => $posted,
|
|
params => $c->request->params,
|
|
item => $params,
|
|
);
|
|
NGCP::Panel::Utils::Navigation::check_form_buttons(
|
|
c => $c, form => $form,
|
|
fields => {'billing_zone.create' => $c->uri_for("$profile_id/zones/create")},
|
|
back_uri => $c->req->uri,
|
|
);
|
|
if($form->validated) {
|
|
$form->values->{match_mode} ||= 'regex_longest_pattern';
|
|
if (not defined $form->values->{source}) {
|
|
if ($form->values->{match_mode} eq 'regex_longest_pattern') {
|
|
$form->values->{source} = '.';
|
|
} else {
|
|
$form->values->{source} = '';
|
|
}
|
|
}
|
|
my $schema = $c->model('DB');
|
|
$schema->txn_do(sub {
|
|
NGCP::Panel::Utils::Billing::insert_unique_billing_fees(
|
|
c => $c,
|
|
schema => $schema,
|
|
profile => $c->stash->{'profile_result'},
|
|
fees => [$form->values],
|
|
return_created => 1,
|
|
);
|
|
});
|
|
delete $c->session->{created_objects}->{billing_zone};
|
|
NGCP::Panel::Utils::Message::info(
|
|
c => $c,
|
|
desc => $c->loc('Billing Fee successfully created!'),
|
|
);
|
|
$c->response->redirect($c->uri_for($c->stash->{profile}->{id}, 'fees'));
|
|
return;
|
|
}
|
|
|
|
$c->stash(create_flag => 1);
|
|
$c->stash(form => $form);
|
|
}
|
|
|
|
sub fees_upload :Chained('fees_list') :PathPart('upload') :Args(0) {
|
|
my ($self, $c) = @_;
|
|
|
|
my $form = NGCP::Panel::Form::get("NGCP::Panel::Form::BillingFeeUpload", $c);
|
|
my $upload = $c->req->upload('upload_fees');
|
|
my $posted = $c->req->method eq 'POST';
|
|
my @params = ( upload_fees => $posted ? $upload : undef, );
|
|
$form->process(
|
|
posted => $posted,
|
|
params => { @params },
|
|
action => $c->uri_for_action('/billing/fees_upload', $c->req->captures),
|
|
);
|
|
if($form->validated) {
|
|
|
|
# TODO: check by formhandler?
|
|
unless($upload) {
|
|
NGCP::Panel::Utils::Message::error(
|
|
c => $c,
|
|
desc => $c->loc('No Billing Fee file specified!'),
|
|
);
|
|
$c->response->redirect($c->uri_for($c->stash->{profile}->{id}, 'fees'));
|
|
return;
|
|
}
|
|
if ($c->req->params->{purge_existing}) {
|
|
$c->stash->{'profile_result'}->billing_fees_raw->delete;
|
|
$c->stash->{'profile_result'}->billing_fees->delete;
|
|
}
|
|
my $data = $upload->slurp;
|
|
my($fees, $fails, $text_success);
|
|
try {
|
|
my $schema = $c->model('DB');
|
|
$schema->txn_do(sub {
|
|
( $fees, $fails, $text_success ) = NGCP::Panel::Utils::Billing::process_billing_fees(
|
|
c => $c,
|
|
data => \$data,
|
|
profile => $c->stash->{'profile_result'},
|
|
schema => $schema,
|
|
);
|
|
});
|
|
|
|
NGCP::Panel::Utils::Message::info(
|
|
c => $c,
|
|
desc => $$text_success,
|
|
);
|
|
} catch($e) {
|
|
NGCP::Panel::Utils::Message::error(
|
|
c => $c,
|
|
error => $e,
|
|
desc => $c->loc('Failed to upload Billing Fees'),
|
|
);
|
|
}
|
|
|
|
$c->response->redirect($c->uri_for($c->stash->{profile}->{id}, 'fees'));
|
|
return;
|
|
}
|
|
|
|
$c->stash(create_flag => 1);
|
|
$c->stash(form => $form);
|
|
}
|
|
|
|
sub fees_download :Chained('fees_list') :PathPart('download') :Args(0) {
|
|
my ($self, $c) = @_;
|
|
my $schema = $c->model('DB');
|
|
my $data = NGCP::Panel::Utils::Billing::combine_billing_fees(
|
|
c => $c,
|
|
profile => $c->stash->{'profile_result'},
|
|
schema => $schema,
|
|
);
|
|
$c->response->header ('Content-Disposition' => 'attachment; filename="billing_fees_'.$c->stash->{profile}->{id}.'.txt"');
|
|
$c->response->content_type('text/csv');
|
|
$c->response->body($$data);
|
|
return;
|
|
}
|
|
|
|
sub fees_edit :Chained('fees_base') :PathPart('edit') :Args(0) {
|
|
my ($self, $c) = @_;
|
|
|
|
my $profile_id = $c->stash->{profile}->{id};
|
|
my $posted = ($c->request->method eq 'POST');
|
|
my $params = $c->stash->{fee};
|
|
$params->{billing_zone}{id} = delete $params->{billing_zone_id};
|
|
$params = merge($params, $c->session->{created_objects});
|
|
my $form = NGCP::Panel::Form::get("NGCP::Panel::Form::BillingFee", $c);
|
|
$form->process(
|
|
posted => $posted,
|
|
params => $c->request->params,
|
|
item => $params,
|
|
);
|
|
NGCP::Panel::Utils::Navigation::check_form_buttons(
|
|
c => $c, form => $form,
|
|
fields => {'billing_zone.create' => $c->uri_for("$profile_id/zones/create")},
|
|
back_uri => $c->req->uri,
|
|
);
|
|
if($posted && $form->validated) {
|
|
$form->values->{match_mode} ||= 'regex_longest_pattern';
|
|
if (not defined $form->values->{source}) {
|
|
if ($form->values->{match_mode} eq 'regex_longest_pattern') {
|
|
$form->values->{source} = '.';
|
|
} else {
|
|
$form->values->{source} = '';
|
|
}
|
|
}
|
|
$form->values->{billing_zone_id} = $form->values->{billing_zone}{id};
|
|
delete $form->values->{billing_zone};
|
|
$c->stash->{'fee_result'}
|
|
->update($form->values);
|
|
delete $c->session->{created_objects}->{billing_zone};
|
|
NGCP::Panel::Utils::Message::info(
|
|
c => $c,
|
|
desc => $c->loc('Billing fee successfully changed!'),
|
|
);
|
|
$c->response->redirect($c->uri_for($c->stash->{profile}->{id}, 'fees'));
|
|
return;
|
|
}
|
|
|
|
$c->stash(edit_fee_flag => 1);
|
|
$c->stash(form => $form);
|
|
}
|
|
|
|
sub fees_delete :Chained('fees_base') :PathPart('delete') :Args(0) {
|
|
my ($self, $c) = @_;
|
|
|
|
unless ( defined($c->stash->{'fee_result'}) ) {
|
|
NGCP::Panel::Utils::Message::error(
|
|
c => $c,
|
|
data => $c->stash->{profile},
|
|
desc => $c->loc('Billing fee not found!'),
|
|
);
|
|
return;
|
|
}
|
|
$c->stash->{'fee_result'}->delete;
|
|
|
|
NGCP::Panel::Utils::Message::info(
|
|
c => $c,
|
|
data => $c->stash->{profile},
|
|
desc => $c->loc('Billing fee successfully deleted!'),
|
|
);
|
|
$c->response->redirect($c->uri_for($c->stash->{profile}->{id}, 'fees'));
|
|
}
|
|
|
|
sub zones_list :Chained('base') :PathPart('zones') :CaptureArgs(0) {
|
|
my ($self, $c) = @_;
|
|
|
|
$c->stash->{zone_dt_columns} = NGCP::Panel::Utils::Datatables::set_columns($c, [
|
|
{ name => 'id', search => 1, title => $c->loc('#') },
|
|
{ name => 'zone', search => 1, title => $c->loc('Zone') },
|
|
{ name => 'detail', search => 1, title => $c->loc('Zone Details') },
|
|
]);
|
|
|
|
$c->stash( zones_root_uri =>
|
|
$c->uri_for_action('/billing/zones', [$c->req->captures->[0]])
|
|
);
|
|
|
|
$c->stash(template => 'billing/zones.tt');
|
|
}
|
|
|
|
sub zones_ajax :Chained('zones_list') :PathPart('ajax') :Args(0) {
|
|
my ($self, $c) = @_;
|
|
|
|
my $resultset = $c->stash->{'profile_result'}->billing_zones;
|
|
NGCP::Panel::Utils::Datatables::process($c, $resultset, $c->stash->{zone_dt_columns});
|
|
$c->detach( $c->view("JSON") );
|
|
}
|
|
|
|
sub zones_create :Chained('zones_list') :PathPart('create') :Args(0) {
|
|
my ($self, $c) = @_;
|
|
|
|
my $form = NGCP::Panel::Form::get("NGCP::Panel::Form::BillingZone", $c);
|
|
my $posted = ($c->request->method eq 'POST');
|
|
$form->process(
|
|
posted => $posted,
|
|
params => $c->request->params,
|
|
);
|
|
NGCP::Panel::Utils::Navigation::check_form_buttons(
|
|
c => $c,
|
|
form => $form,
|
|
fields => {},
|
|
back_uri => $c->req->uri,
|
|
);
|
|
|
|
if($posted && $form->validated) {
|
|
try {
|
|
my $zone = $c->stash->{'profile_result'}->billing_zones->create($form->values);
|
|
$c->session->{created_objects}->{billing_zone} = { id => $zone->id };
|
|
NGCP::Panel::Utils::Message::info(
|
|
c => $c,
|
|
desc => $c->loc('Billing Zone successfully created'),
|
|
);
|
|
} catch($e) {
|
|
NGCP::Panel::Utils::Message::error(
|
|
c => $c,
|
|
error => $e,
|
|
desc => $c->loc('Failed to create billing zone'),
|
|
);
|
|
}
|
|
NGCP::Panel::Utils::Navigation::back_or($c, $c->stash->{zones_root_uri});
|
|
}
|
|
|
|
$c->stash(form => $form);
|
|
$c->stash(create_flag => 1);
|
|
}
|
|
|
|
sub zones :Chained('zones_list') :PathPart('') :Args(0) {
|
|
}
|
|
|
|
sub zones_base :Chained('zones_list') :PathPart('') :CaptureArgs(1) {
|
|
my ($self, $c, $zone_id) = @_;
|
|
|
|
unless($zone_id && is_int($zone_id)) {
|
|
$zone_id //= '';
|
|
NGCP::Panel::Utils::Message::error(
|
|
c => $c,
|
|
data => $c->stash->{profile},
|
|
log => $zone_id,
|
|
desc => $c->loc('Invalid billing zone id detected'),
|
|
);
|
|
NGCP::Panel::Utils::Navigation::back_or($c, $c->stash->{zones_root_uri});
|
|
}
|
|
|
|
my $res = $c->stash->{'profile_result'}->billing_zones
|
|
->find($zone_id);
|
|
unless(defined($res)) {
|
|
NGCP::Panel::Utils::Message::error(
|
|
c => $c,
|
|
data => $c->stash->{profile},
|
|
log => $zone_id,
|
|
desc => $c->loc('Billing zone does not exist!'),
|
|
);
|
|
NGCP::Panel::Utils::Navigation::back_or($c, $c->stash->{zones_root_uri});
|
|
}
|
|
$c->stash(zone_result => $res);
|
|
}
|
|
|
|
sub zones_delete :Chained('zones_base') :PathPart('delete') :Args(0) {
|
|
my ($self, $c) = @_;
|
|
|
|
my $zone_info = { $c->stash->{zone_result}->get_inflated_columns };
|
|
try {
|
|
$c->stash->{zone_result}->billing_fees->delete_all;
|
|
$c->stash->{zone_result}->delete;
|
|
NGCP::Panel::Utils::Message::info(
|
|
c => $c,
|
|
data => $zone_info,
|
|
desc => $c->loc('Billing zone successfully deleted'),
|
|
);
|
|
} catch($e) {
|
|
NGCP::Panel::Utils::Message::error(
|
|
c => $c,
|
|
error => $e,
|
|
data => $zone_info,
|
|
desc => $c->loc('Failed to delete billing zone'),
|
|
);
|
|
}
|
|
NGCP::Panel::Utils::Navigation::back_or($c, $c->stash->{zones_root_uri});
|
|
}
|
|
|
|
sub peaktimes_list :Chained('base') :PathPart('peaktimes') :CaptureArgs(0) {
|
|
my ($self, $c) = @_;
|
|
$c->stash(peaktimes_root_uri =>
|
|
$c->uri_for_action('/billing/peaktimes', [$c->req->captures->[0]])
|
|
);
|
|
|
|
my $rs = $c->stash->{profile_result}->billing_peaktime_weekdays;
|
|
$rs = $rs->search(undef, {order_by => 'start'});
|
|
|
|
$c->stash->{special_dt_columns} = NGCP::Panel::Utils::Datatables::set_columns($c, [
|
|
{ name => 'id', search => 1, title => $c->loc('#') },
|
|
{ name => 'start', search => 1, title => $c->loc('Start Date') },
|
|
{ name => 'end', search => 1, title => $c->loc('End Date') },
|
|
]);
|
|
|
|
$c->stash(weekdays_result => $rs);
|
|
$c->stash(template => 'billing/peaktimes.tt');
|
|
}
|
|
|
|
sub peaktimes :Chained('peaktimes_list') :PathPart('') :Args(0) {
|
|
my ($self, $c) = @_;
|
|
$self->load_weekdays($c);
|
|
}
|
|
|
|
sub peaktime_weekdays_base :Chained('peaktimes_list') :PathPart('weekday') :CaptureArgs(1) {
|
|
my ($self, $c, $weekday_id) = @_;
|
|
unless (defined $weekday_id && $weekday_id >= 0 && $weekday_id <= 6) {
|
|
$weekday_id //= '';
|
|
NGCP::Panel::Utils::Message::error(
|
|
c => $c,
|
|
data => $c->stash->{profile},
|
|
log => $weekday_id,
|
|
desc => $c->loc('This weekday does not exist'),
|
|
);
|
|
$c->response->redirect($c->uri_for_action(
|
|
"/billing/peaktimes", [$c->req->captures->[0]],
|
|
));
|
|
}
|
|
$c->stash(weekday_id => $weekday_id);
|
|
}
|
|
|
|
sub peaktime_weekdays_edit :Chained('peaktime_weekdays_base') :PathPart('edit') :Args(0) {
|
|
my ($self, $c) = @_;
|
|
|
|
my $form = NGCP::Panel::Form::get("NGCP::Panel::Form::BillingPeaktimeWeekdays", $c);
|
|
$form->process(
|
|
posted => ($c->request->method eq 'POST'),
|
|
params => $c->request->params,
|
|
);
|
|
if($form->validated) {
|
|
$form->values->{weekday} = $c->stash->{weekday_id};
|
|
$form->values->{start} = '00:00:00' unless($form->values->{start});
|
|
$form->values->{end} = '23:59:59' unless($form->values->{end});
|
|
$c->stash->{'weekdays_result'}
|
|
->create($form->values);
|
|
NGCP::Panel::Utils::Message::info(
|
|
c => $c,
|
|
data => { %{$c->request->params},
|
|
billing_profile_id => $c->stash->{profile}{id}, },
|
|
desc => $c->loc('Timerange has been successfully created'),
|
|
);
|
|
}
|
|
|
|
my $delete_param = $c->request->params->{delete};
|
|
if($delete_param) {
|
|
my $rs = $c->stash->{weekdays_result}
|
|
->find($delete_param);
|
|
unless ($rs) {
|
|
NGCP::Panel::Utils::Message::error(
|
|
c => $c,
|
|
desc => $c->loc('The timerange you wanted to delete does not exist'),
|
|
);
|
|
$c->response->redirect($c->uri_for_action(
|
|
"/billing/peaktimes", [$c->req->captures->[0]],
|
|
));
|
|
return;
|
|
}
|
|
$rs->delete();
|
|
NGCP::Panel::Utils::Message::info(
|
|
c => $c,
|
|
data => { $rs->get_inflated_columns },
|
|
desc => $c->loc('Timerange has been successfully deleted'),
|
|
);
|
|
}
|
|
|
|
$form = NGCP::Panel::Form::get("NGCP::Panel::Form::BillingPeaktimeWeekdays", $c)
|
|
unless $form->has_errors;
|
|
|
|
$self->load_weekdays($c);
|
|
$c->stash(weekday => $c->stash->{weekdays}->[$c->stash->{weekday_id}]);
|
|
$c->stash(form => $form);
|
|
$c->stash(edit_flag => 1);
|
|
}
|
|
|
|
sub load_weekdays {
|
|
my ($self, $c) = @_;
|
|
|
|
my @WEEKDAYS = @{NGCP::Panel::Utils::DateTime::get_weekday_names($c)};
|
|
|
|
my @weekdays;
|
|
for(0 .. 6) {
|
|
$weekdays[$_] = {
|
|
name => $WEEKDAYS[$_],
|
|
ranges => [],
|
|
edit_link => $c->uri_for_action("/billing/peaktime_weekdays_edit",
|
|
[$c->req->captures->[0], $_]),
|
|
};
|
|
}
|
|
|
|
foreach my $range ($c->stash->{weekdays_result}->all) {
|
|
push @{ $weekdays[$range->weekday]->{ranges} }, {
|
|
start => $range->start,
|
|
end => $range->end,
|
|
id => $range->id,
|
|
}
|
|
}
|
|
|
|
$c->stash(weekdays => \@weekdays);
|
|
}
|
|
|
|
sub peaktime_specials_ajax :Chained('peaktimes_list') :PathPart('ajax') :Args(0) {
|
|
my ($self, $c) = @_;
|
|
|
|
my $resultset = $c->stash->{'profile_result'}->billing_peaktime_specials;
|
|
NGCP::Panel::Utils::Datatables::process($c, $resultset, $c->stash->{special_dt_columns});
|
|
$c->detach( $c->view("JSON") );
|
|
}
|
|
|
|
sub peaktime_specials_base :Chained('peaktimes_list') :PathPart('date') :CaptureArgs(1) {
|
|
my ($self, $c, $special_id) = @_;
|
|
|
|
unless($special_id && is_int($special_id)) {
|
|
$special_id //= '';
|
|
NGCP::Panel::Utils::Message::error(
|
|
c => $c,
|
|
desc => $c->loc('Invalid peaktime date id detected!'),
|
|
);
|
|
$c->response->redirect($c->stash->{peaktimes_root_uri});
|
|
return;
|
|
}
|
|
|
|
my $res = $c->stash->{'profile_result'}->billing_peaktime_specials
|
|
->find($special_id);
|
|
unless(defined($res)) {
|
|
NGCP::Panel::Utils::Message::error(
|
|
c => $c,
|
|
data => $c->stash->{profile},
|
|
log => $special_id,
|
|
desc => $c->loc('Peaktime date does not exist!'),
|
|
);
|
|
$c->response->redirect($c->stash->{peaktimes_root_uri});
|
|
return;
|
|
}
|
|
$self->load_weekdays($c);
|
|
$c->stash(special_result => $res);
|
|
}
|
|
|
|
sub peaktime_specials_edit :Chained('peaktime_specials_base') :PathPart('edit') :Args(0) {
|
|
my ($self, $c) = @_;
|
|
|
|
my $data_res = $c->stash->{special_result};
|
|
my $posted = ($c->request->method eq 'POST');
|
|
my $form = NGCP::Panel::Form::get("NGCP::Panel::Form::BillingPeaktimeSpecial", $c);
|
|
my $params = { $data_res->get_inflated_columns };
|
|
$form->process(
|
|
posted => $posted,
|
|
params => $c->request->params,
|
|
item => $params,
|
|
);
|
|
NGCP::Panel::Utils::Navigation::check_form_buttons(
|
|
c => $c,
|
|
form => $form,
|
|
fields => {},
|
|
back_uri => $c->req->uri,
|
|
);
|
|
if($posted && $form->validated) {
|
|
try {
|
|
$c->stash->{special_result}->update($form->values);
|
|
NGCP::Panel::Utils::Message::info(
|
|
c => $c,
|
|
data => { %{$c->request->params},
|
|
billing_profile_id => $c->stash->{profile}{id}, },
|
|
desc => $c->loc('Special offpeak entry successfully updated'),
|
|
);
|
|
} catch($e) {
|
|
NGCP::Panel::Utils::Message::error(
|
|
c => $c,
|
|
error => $e,
|
|
data => { %{$c->request->params},
|
|
billing_profile_id => $c->stash->{profile}{id}, },
|
|
desc => $c->loc('Failed to update special offpeak entry'),
|
|
);
|
|
}
|
|
NGCP::Panel::Utils::Navigation::back_or($c, $c->stash->{peaktimes_root_uri});
|
|
}
|
|
|
|
$c->stash(peaktimes_special_editflag => 1);
|
|
$c->stash(peaktimes_special_form => $form);
|
|
}
|
|
|
|
sub peaktime_specials_delete :Chained('peaktime_specials_base') :PathPart('delete') :Args(0) {
|
|
my ($self, $c) = @_;
|
|
my $special_result_info = { $c->stash->{special_result}->get_inflated_columns };
|
|
try {
|
|
$c->stash->{special_result}->delete;
|
|
NGCP::Panel::Utils::Message::info(
|
|
c => $c,
|
|
data => $special_result_info,
|
|
desc => $c->loc('Special offpeak entry successfully deleted'),
|
|
);
|
|
} catch($e) {
|
|
NGCP::Panel::Utils::Message::error(
|
|
c => $c,
|
|
error => $e,
|
|
data => $special_result_info,
|
|
desc => $c->loc('Failed to delete special offpeak entry'),
|
|
);
|
|
}
|
|
NGCP::Panel::Utils::Navigation::back_or($c, $c->stash->{peaktimes_root_uri});
|
|
}
|
|
|
|
sub peaktime_specials_create :Chained('peaktimes_list') :PathPart('date/create') :Args(0) {
|
|
my ($self, $c) = @_;
|
|
$self->load_weekdays($c);
|
|
|
|
my $posted = ($c->request->method eq 'POST');
|
|
my $form = NGCP::Panel::Form::get("NGCP::Panel::Form::BillingPeaktimeSpecial", $c);
|
|
my $params = {};
|
|
$form->process(
|
|
posted => $posted,
|
|
params => $c->request->params,
|
|
item => $params,
|
|
);
|
|
NGCP::Panel::Utils::Navigation::check_form_buttons(
|
|
c => $c,
|
|
form => $form,
|
|
fields => {},
|
|
back_uri => $c->req->uri,
|
|
);
|
|
if($form->validated) {
|
|
try {
|
|
$c->stash->{'profile_result'}->billing_peaktime_specials
|
|
->create($form->values);
|
|
NGCP::Panel::Utils::Message::info(
|
|
c => $c,
|
|
data => { %{$c->request->params},
|
|
billing_profile_id => $c->stash->{profile}{id}, },
|
|
desc => $c->loc('Special offpeak entry successfully created'),
|
|
);
|
|
} catch($e) {
|
|
NGCP::Panel::Utils::Message::error(
|
|
c => $c,
|
|
error => $e,
|
|
data => { %{$c->request->params},
|
|
billing_profile_id => $c->stash->{profile}{id}, },
|
|
desc => $c->loc('Failed to create special offpeak entry'),
|
|
);
|
|
}
|
|
NGCP::Panel::Utils::Navigation::back_or($c, $c->stash->{peaktimes_root_uri});
|
|
}
|
|
|
|
$c->stash(peaktimes_special_form => $form);
|
|
$c->stash(peaktimes_special_createflag => 1);
|
|
}
|
|
|
|
|
|
1;
|
|
|
|
__END__
|
|
|
|
=head1 NAME
|
|
|
|
NGCP::Panel::Controller::Billing - Catalyst Controller
|
|
|
|
=head1 DESCRIPTION
|
|
|
|
Catalyst Controller.
|
|
|
|
=head1 METHODS
|
|
|
|
=head2 profile_list
|
|
|
|
basis for the billing controller
|
|
|
|
=head2 root
|
|
|
|
just shows a list of billing profiles using datatables
|
|
|
|
=head2 ajax
|
|
|
|
Get billing_profiles and output them as JSON.
|
|
|
|
=head2 base
|
|
|
|
Fetch a billing_profile by its id.
|
|
|
|
=head2 edit
|
|
|
|
Show a modal to edit one billing_profile.
|
|
|
|
=head2 create
|
|
|
|
Show a modal to add a new billing_profile.
|
|
|
|
=head2 fees_list
|
|
|
|
basis for the billing_fees logic. for a certain billing_profile identified
|
|
by base.
|
|
|
|
=head2 fees
|
|
|
|
Shows a list of billing_fees for one billing_profile using datatables.
|
|
|
|
=head2 fees_base
|
|
|
|
Fetch a billing_fee (identified by id).
|
|
|
|
=head2 fees_ajax
|
|
|
|
Get billing_fees and output them as JSON.
|
|
|
|
=head2 fees_create
|
|
|
|
Show a modal to add a new billing_fee.
|
|
|
|
=head2 fees_upload
|
|
|
|
Show a modal to upload a CSV file of billing_fees and add them to the
|
|
Database.
|
|
|
|
=head2 fees_edit
|
|
|
|
Show a modal to edit a billing_fee.
|
|
|
|
=head2 fees_delete
|
|
|
|
Delete a billing_fee.
|
|
|
|
=head2 zones_list
|
|
|
|
basis for billing zones. part of a certain billing profile.
|
|
|
|
=head2 zones_ajax
|
|
|
|
sends a JSON representation of billing_zones under the current billing profile.
|
|
|
|
=head2 zones_create
|
|
|
|
Show a modal to create a new billing_zone in the current billing profile.
|
|
|
|
=head2 zones
|
|
|
|
Show a datatables list of billing_zones in the current billing profile.
|
|
|
|
=head2 zones_base
|
|
|
|
Fetch a billing_zone (identified by id).
|
|
|
|
=head2 zones_delete
|
|
|
|
Delete a billing_zone (defined by zones_base).
|
|
|
|
=head2 peaktimes_list
|
|
|
|
basis for billing_peaktime_* time definitions. part of a certain billing_profile.
|
|
|
|
=head2 peaktimes
|
|
|
|
show a list with peaktime weekdays and peaktime dates.
|
|
|
|
=head2 peaktime_weekdays_base
|
|
|
|
Define a certain weekday by id (for further processing in chain).
|
|
|
|
=head2 peaktime_weekdays_edit
|
|
|
|
Show a modal to edit one weekday.
|
|
|
|
=head2 load_weekdays
|
|
|
|
creates a weekdays structure from the stash variable weekdays_result
|
|
puts the result under weekdays on stash (will be used by template)
|
|
|
|
=head2 peaktime_specials_ajax
|
|
|
|
Returns an ajax representation of billing_peaktime_specials under the current
|
|
billing_profile. The rows are modified so that the final form will be
|
|
(id, date, startend).
|
|
|
|
This depends on inflation being activated in the schema.
|
|
|
|
=head2 peaktime_specials_base
|
|
|
|
Get one billing_peaktime_special from the database for further processing.
|
|
|
|
=head2 peaktime_specials_edit
|
|
|
|
Edit one billing_peaktime_special per modal and the form
|
|
NGCP::Panel::Form::BillingPeaktimeSpecial.
|
|
|
|
=head2 peaktime_specials_delete
|
|
|
|
Delete a billing_peaktime_special.
|
|
|
|
=head2 peaktime_specials_create
|
|
|
|
Create a new billing_peaktime_special under the current billing_profile.
|
|
Uses NGCP::Panel::Form::BillingPeaktimeSpecial.
|
|
|
|
=head1 AUTHOR
|
|
|
|
Gerhard Jungwirth C<< <gjungwirth@sipwise.com> >>
|
|
|
|
=head1 LICENSE
|
|
|
|
This library is free software. You can redistribute it and/or modify
|
|
it under the same terms as Perl itself.
|
|
|
|
=cut
|
|
|
|
# vim: set tabstop=4 expandtab:
|