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.
709 lines
27 KiB
709 lines
27 KiB
package NGCP::Panel::Controller::Contract;
|
|
use NGCP::Panel::Utils::Generic qw(:all);
|
|
use Sipwise::Base;
|
|
|
|
use parent 'Catalyst::Controller';
|
|
|
|
use NGCP::Panel::Form;
|
|
use NGCP::Panel::Utils::Message;
|
|
use NGCP::Panel::Utils::Navigation;
|
|
use NGCP::Panel::Utils::Contract;
|
|
use NGCP::Panel::Utils::ProfilePackages;
|
|
use NGCP::Panel::Utils::Subscriber;
|
|
use NGCP::Panel::Utils::DateTime;
|
|
use NGCP::Panel::Utils::BillingMappings qw();
|
|
|
|
sub auto :Does(ACL) :ACLDetachTo('/denied_page') :AllowedRole(admin) {
|
|
my ($self, $c) = @_;
|
|
$c->log->debug(__PACKAGE__ . '::auto');
|
|
NGCP::Panel::Utils::Navigation::check_redirect_chain(c => $c);
|
|
return 1;
|
|
}
|
|
|
|
sub contract_list :Chained('/') :PathPart('contract') :CaptureArgs(0) {
|
|
my ($self, $c) = @_;
|
|
|
|
my $now = NGCP::Panel::Utils::DateTime::current_local;
|
|
$c->stash->{contract_dt_columns} = NGCP::Panel::Utils::Datatables::set_columns($c, [
|
|
{ name => "id", int_search => 1, title => $c->loc("#") },
|
|
{ name => "external_id", strict_search => 1, title => $c->loc("External #") },
|
|
{ name => "contact.email", search => 0, title => $c->loc("Contact Email") },
|
|
{ name => "product.name", search => 0, title => $c->loc("Product") },
|
|
{ name => 'billing_profile_name', accessor => "billing_profile_name", search => 0, title => $c->loc('Billing Profile'),
|
|
literal_sql => NGCP::Panel::Utils::BillingMappings::get_actual_billing_mapping_stmt(c => $c, now => $now, projection => 'billing_profile.name' ) },
|
|
{ name => "status", search => 0, title => $c->loc("Status") },
|
|
{ title => $c->loc("Contact Email"), search => 1, no_column => 1 },
|
|
]);
|
|
|
|
my $rs = NGCP::Panel::Utils::Contract::get_contract_rs(
|
|
schema => $c->model('DB'),
|
|
now => $now
|
|
);
|
|
my $rs_all = NGCP::Panel::Utils::Contract::get_contract_rs(
|
|
schema => $c->model('DB'),
|
|
now => $now,
|
|
include_terminated => 1,
|
|
);
|
|
unless($c->user->is_superuser) {
|
|
$rs = $rs->search({
|
|
'contact.reseller_id' => $c->user->reseller_id,
|
|
}, {
|
|
join => 'contact',
|
|
});
|
|
}
|
|
|
|
my @product_ids = map { $_->id; } $c->model('DB')->resultset('products')->search_rs({ 'class' => ['pstnpeering','sippeering','reseller'] })->all;
|
|
$rs = $rs->search({
|
|
'product_id' => { -in => [ @product_ids ] },
|
|
});
|
|
$c->stash(contract_select_rs => $rs);
|
|
$c->stash(contract_select_all_rs => $rs_all);
|
|
$c->stash(now => $now);
|
|
$c->stash(ajax_uri => $c->uri_for_action("/contract/ajax"));
|
|
$c->stash(template => 'contract/list.tt');
|
|
}
|
|
|
|
sub root :Chained('contract_list') :PathPart('') :Args(0) {
|
|
my ($self, $c) = @_;
|
|
}
|
|
|
|
sub base :Chained('contract_list') :PathPart('') :CaptureArgs(1) {
|
|
my ($self, $c, $contract_id) = @_;
|
|
|
|
unless($contract_id && is_int($contract_id)) {
|
|
NGCP::Panel::Utils::Message::error(
|
|
c => $c,
|
|
log => 'Invalid contract id detected!',
|
|
desc => $c->loc('Invalid contract id detected!'),
|
|
);
|
|
NGCP::Panel::Utils::Navigation::back_or($c, $c->uri_for('/contract'));
|
|
}
|
|
|
|
my $contract_rs = $c->stash->{contract_select_rs}
|
|
->search({
|
|
'me.id' => $contract_id,
|
|
},undef);
|
|
my $contract_terminated_rs = $c->stash->{contract_select_all_rs}
|
|
->search({
|
|
'me.id' => $contract_id,
|
|
},undef);
|
|
|
|
my $contract_first = $contract_rs->first;
|
|
|
|
unless(defined($contract_first)) {
|
|
NGCP::Panel::Utils::Message::error(
|
|
c => $c,
|
|
log => 'Contract does not exist',
|
|
desc => $c->loc('Contract does not exist'),
|
|
);
|
|
NGCP::Panel::Utils::Navigation::back_or($c, $c->uri_for('/contract'));
|
|
}
|
|
|
|
my $now = $c->stash->{now};
|
|
my $billing_mapping = NGCP::Panel::Utils::BillingMappings::get_actual_billing_mapping(c => $c, contract => $contract_first, now => $now, );
|
|
|
|
my $billing_mappings_ordered = NGCP::Panel::Utils::BillingMappings::billing_mappings_ordered($contract_first->billing_mappings,$now,$billing_mapping);
|
|
my $future_billing_mappings = NGCP::Panel::Utils::BillingMappings::billing_mappings_ordered(NGCP::Panel::Utils::BillingMappings::future_billing_mappings($contract_first->billing_mappings,$now));
|
|
|
|
$c->stash(contract => $contract_first);
|
|
$c->stash(contract_rs => $contract_rs);
|
|
$c->stash(contract_terminated_rs => $contract_terminated_rs);
|
|
$c->stash(billing_mapping => $billing_mapping );
|
|
$c->stash(billing_mappings_ordered_result => $billing_mappings_ordered ); # all billings mappings are displayed in the details page
|
|
|
|
$c->stash(future_billing_mappings => $future_billing_mappings ); # only editable billing mappings are displayed in the edit dialog
|
|
return;
|
|
}
|
|
|
|
sub edit :Chained('base') :PathPart('edit') :Args(0) {
|
|
my ($self, $c) = @_;
|
|
|
|
my $posted = ($c->request->method eq 'POST');
|
|
|
|
my $contract = $c->stash->{contract};
|
|
my $billing_mapping = $c->stash->{billing_mapping};
|
|
my $now = $c->stash->{now};
|
|
my $billing_profile = $billing_mapping->billing_profile;
|
|
my $params = {};
|
|
unless($posted) {
|
|
$params->{billing_profile}{id} = $billing_mapping->billing_profile->id;
|
|
$params->{billing_profiles} = [ map { { $_->get_inflated_columns }; } $c->stash->{future_billing_mappings}->all ];
|
|
$params->{contact}{id} = $contract->contact_id;
|
|
$params->{external_id} = $contract->external_id;
|
|
$params->{status} = $contract->status;
|
|
$params->{max_subscribers} = $contract->max_subscribers;
|
|
}
|
|
$params = merge($params, $c->session->{created_objects});
|
|
my ($form, $is_peering_reseller);
|
|
if ( NGCP::Panel::Utils::Contract::is_peering_product( c => $c, product => $contract->product ) ) {
|
|
$form = NGCP::Panel::Form::get("NGCP::Panel::Form::Contract::Peering", $c);
|
|
$is_peering_reseller = 1;
|
|
} elsif ( NGCP::Panel::Utils::Contract::is_reseller_product( c => $c, product => $contract->product ) ) {
|
|
$form = NGCP::Panel::Form::get("NGCP::Panel::Form::Contract::Reseller", $c);
|
|
$is_peering_reseller = 1;
|
|
} else {
|
|
$form = NGCP::Panel::Form::get("NGCP::Panel::Form::Contract::Contract", $c);
|
|
$is_peering_reseller = 0;
|
|
}
|
|
$form->process(
|
|
posted => $posted,
|
|
params => $c->req->params,
|
|
item => $params,
|
|
);
|
|
NGCP::Panel::Utils::Navigation::check_form_buttons(
|
|
c => $c, form => $form,
|
|
fields => {
|
|
'contact.create' => ( $is_peering_reseller
|
|
? $c->uri_for('/contact/create/noreseller')
|
|
: $c->uri_for('/contact/create')),
|
|
'billing_profile.create' => $c->uri_for('/billing/create'),
|
|
'billing_profiles.profile.create' => $c->uri_for('/billing/create'),
|
|
'subscriber_email_template.create' => $c->uri_for('/emailtemplate/create'),
|
|
'passreset_email_template.create' => $c->uri_for('/emailtemplate/create'),
|
|
'invoice_email_template.create' => $c->uri_for('/emailtemplate/create'),
|
|
},
|
|
back_uri => $c->req->uri,
|
|
);
|
|
if($posted && $form->validated) {
|
|
try {
|
|
my $schema = $c->model('DB');
|
|
$schema->set_transaction_isolation('READ COMMITTED');
|
|
$schema->txn_do(sub {
|
|
foreach(qw/contact billing_profile/){
|
|
$form->values->{$_.'_id'} = $form->values->{$_}{id} || undef;
|
|
delete $form->values->{$_};
|
|
}
|
|
$form->values->{modify_timestamp} = $now; #problematic for ON UPDATE current_timestamp columns
|
|
|
|
my $mappings_to_create = [];
|
|
my $delete_mappings = 0;
|
|
my $set_package = ($form->values->{billing_profile_definition} // 'id') eq 'package';
|
|
NGCP::Panel::Utils::BillingMappings::prepare_billing_mappings(
|
|
c => $c,
|
|
resource => $form->values,
|
|
old_resource => { $contract->get_inflated_columns },
|
|
mappings_to_create => $mappings_to_create,
|
|
now => $now,
|
|
delete_mappings => \$delete_mappings,
|
|
err_code => sub {
|
|
my ($err,@fields) = @_;
|
|
die( [$err, "showdetails"] );
|
|
});
|
|
|
|
my $old_status = $contract->status;
|
|
my $old_package = $contract->profile_package;
|
|
|
|
$contract->update($form->values);
|
|
NGCP::Panel::Utils::BillingMappings::append_billing_mappings(c => $c,
|
|
contract => $contract,
|
|
mappings_to_create => $mappings_to_create,
|
|
now => $now,
|
|
delete_mappings => $delete_mappings,
|
|
);
|
|
|
|
my $balance = NGCP::Panel::Utils::ProfilePackages::catchup_contract_balances(c => $c,
|
|
contract => $contract,
|
|
old_package => $old_package,
|
|
now => $now);
|
|
$balance = NGCP::Panel::Utils::ProfilePackages::resize_actual_contract_balance(c => $c,
|
|
contract => $contract,
|
|
old_package => $old_package,
|
|
balance => $balance,
|
|
now => $now,
|
|
profiles_added => ($set_package ? scalar @$mappings_to_create : 0),
|
|
);
|
|
|
|
if ($is_peering_reseller &&
|
|
defined $contract->contact->reseller_id) {
|
|
die( ["Cannot use this contact for peering or reseller contracts.", "showdetails"] );
|
|
}
|
|
|
|
# if status changed, populate it down the chain
|
|
if($contract->status ne $old_status) {
|
|
NGCP::Panel::Utils::Contract::recursively_lock_contract(
|
|
c => $c,
|
|
contract => $contract,
|
|
);
|
|
}
|
|
|
|
delete $c->session->{created_objects}->{contact};
|
|
delete $c->session->{created_objects}->{billing_profile};
|
|
});
|
|
NGCP::Panel::Utils::Message::info(
|
|
c => $c,
|
|
data => { $contract->get_inflated_columns },
|
|
desc => $c->loc('Contract successfully changed!'),
|
|
);
|
|
} catch($e) {
|
|
NGCP::Panel::Utils::Message::error(
|
|
c => $c,
|
|
error => $e,
|
|
data => { $contract->get_inflated_columns },
|
|
desc => $c->loc('Failed to update contract'),
|
|
);
|
|
}
|
|
NGCP::Panel::Utils::Navigation::back_or($c, $c->uri_for('/contract'));
|
|
}
|
|
|
|
$c->stash(form => $form);
|
|
$c->stash(edit_flag => 1);
|
|
}
|
|
|
|
sub terminate :Chained('base') :PathPart('terminate') :Args(0) {
|
|
my ($self, $c) = @_;
|
|
my $contract = $c->stash->{contract};
|
|
|
|
if ($contract->id == 1) {
|
|
NGCP::Panel::Utils::Message::error(
|
|
c => $c,
|
|
desc => $c->loc('Cannot terminate contract with the id 1'),
|
|
);
|
|
NGCP::Panel::Utils::Navigation::back_or($c, $c->uri_for('/contract'));
|
|
}
|
|
|
|
try {
|
|
my $old_status = $contract->status;
|
|
my $schema = $c->model('DB');
|
|
$schema->txn_do(sub {
|
|
$contract->voip_contract_preferences->delete;
|
|
$contract->update({
|
|
status => 'terminated',
|
|
terminate_timestamp => NGCP::Panel::Utils::DateTime::current_local,
|
|
});
|
|
$contract = $c->stash->{contract_terminated_rs}->first;
|
|
# if status changed, populate it down the chain
|
|
if($contract->status ne $old_status) {
|
|
NGCP::Panel::Utils::Contract::recursively_lock_contract(
|
|
c => $c,
|
|
contract => $contract,
|
|
schema => $schema,
|
|
);
|
|
}
|
|
});
|
|
NGCP::Panel::Utils::Message::info(
|
|
c => $c,
|
|
data => { $contract->get_inflated_columns },
|
|
desc => $c->loc('Contract successfully terminated'),
|
|
);
|
|
} catch ($e) {
|
|
NGCP::Panel::Utils::Message::error(
|
|
c => $c,
|
|
error => $e,
|
|
data => { $contract->get_inflated_columns },
|
|
desc => $c->loc('Failed to terminate contract'),
|
|
);
|
|
};
|
|
NGCP::Panel::Utils::Navigation::back_or($c, $c->uri_for('/contract'));
|
|
}
|
|
|
|
sub ajax :Chained('contract_list') :PathPart('ajax') :Args(0) {
|
|
my ($self, $c) = @_;
|
|
|
|
my $res = $c->stash->{contract_select_rs};
|
|
NGCP::Panel::Utils::Datatables::process($c, $res, $c->stash->{contract_dt_columns}, undef, {
|
|
'count_limit' => 1000,
|
|
'extra_or' => [ {
|
|
'me.id' => { in => [ map { $_->id } $res->search({
|
|
'contact.email' => { like => [ NGCP::Panel::Utils::Datatables::get_search_string_pattern($c) ]->[0] },
|
|
},{
|
|
rows => 1001,
|
|
page => 1,
|
|
join => 'contact',
|
|
})->all ] },
|
|
}, ],
|
|
});
|
|
$c->detach( $c->view("JSON") );
|
|
}
|
|
|
|
sub billingmappings_ajax :Chained('base') :PathPart('billingmappings/ajax') :Args(0) {
|
|
my ($self, $c) = @_;
|
|
$c->stash(timeline_data => {
|
|
contract => { $c->stash->{contract}->get_columns },
|
|
events => NGCP::Panel::Utils::BillingMappings::get_billingmappings_timeline_data($c,$c->stash->{contract}),
|
|
});
|
|
$c->detach( $c->view("JSON") );
|
|
}
|
|
|
|
sub peering_list :Chained('contract_list') :PathPart('peering') :CaptureArgs(0) {
|
|
my ($self, $c) = @_;
|
|
|
|
my @product_ids = map { $_->id; } $c->model('DB')->resultset('products')->search_rs({ 'class' => ['sippeering'] })->all;
|
|
my $base_rs = $c->stash->{contract_select_rs};
|
|
$c->stash->{peering_rs} = $base_rs->search({
|
|
'product_id' => { -in => [ @product_ids ] },
|
|
});
|
|
|
|
$c->stash(ajax_uri => $c->uri_for_action("/contract/peering_ajax"));
|
|
}
|
|
|
|
sub peering_root :Chained('peering_list') :PathPart('') :Args(0) {
|
|
|
|
}
|
|
|
|
sub peering_ajax :Chained('peering_list') :PathPart('ajax') :Args(0) {
|
|
my ($self, $c) = @_;
|
|
|
|
my $rs = $c->stash->{peering_rs};
|
|
NGCP::Panel::Utils::Datatables::process($c, $rs, $c->stash->{contract_dt_columns});
|
|
$c->detach( $c->view("JSON") );
|
|
}
|
|
|
|
sub peering_create :Chained('peering_list') :PathPart('create') :Args(0) {
|
|
my ($self, $c) = @_;
|
|
|
|
my $posted = ($c->request->method eq 'POST');
|
|
my $params = {};
|
|
$params = merge($params, $c->session->{created_objects});
|
|
unless ($self->is_valid_noreseller_contact($c, $params->{contact}{id})) {
|
|
delete $params->{contact};
|
|
}
|
|
$c->stash->{type} = 'sippeering';
|
|
$c->stash(ajax_uri => $c->uri_for_action("/contract/ajax"));
|
|
my $form = NGCP::Panel::Form::get("NGCP::Panel::Form::Contract::Peering", $c);
|
|
$form->process(
|
|
posted => $posted,
|
|
params => $c->request->params,
|
|
item => $params
|
|
);
|
|
NGCP::Panel::Utils::Navigation::check_form_buttons(
|
|
c => $c,
|
|
form => $form,
|
|
fields => {'contact.create' => $c->uri_for('/contact/create/noreseller'),
|
|
'billing_profile.create' => $c->uri_for('/billing/create/noreseller'),
|
|
'billing_profiles.profile.create' => $c->uri_for('/billing/create/noreseller')},
|
|
back_uri => $c->req->uri,
|
|
);
|
|
if($posted && $form->validated) {
|
|
try {
|
|
my $schema = $c->model('DB');
|
|
$schema->set_transaction_isolation('READ COMMITTED');
|
|
$schema->txn_do(sub {
|
|
foreach(qw/contact billing_profile/){
|
|
$form->values->{$_.'_id'} = $form->values->{$_}{id} || undef;
|
|
delete $form->values->{$_};
|
|
}
|
|
$form->values->{external_id} = $form->field('external_id')->value;
|
|
$form->values->{create_timestamp} = $form->values->{modify_timestamp} = NGCP::Panel::Utils::DateTime::current_local;
|
|
$form->values->{product_id} = $schema->resultset('products')->search_rs({ class => $c->stash->{type} })->first->id;
|
|
|
|
my $mappings_to_create = [];
|
|
NGCP::Panel::Utils::BillingMappings::prepare_billing_mappings(
|
|
c => $c,
|
|
resource => $form->values,
|
|
mappings_to_create => $mappings_to_create,
|
|
err_code => sub {
|
|
my ($err,@fields) = @_;
|
|
die( [$err, "showdetails"] );
|
|
});
|
|
|
|
my $contract = $schema->resultset('contracts')->create($form->values);
|
|
NGCP::Panel::Utils::BillingMappings::append_billing_mappings(c => $c,
|
|
contract => $contract,
|
|
mappings_to_create => $mappings_to_create,
|
|
);
|
|
|
|
NGCP::Panel::Utils::ProfilePackages::create_initial_contract_balances(c => $c,
|
|
contract => $contract,
|
|
);
|
|
|
|
if (defined $contract->contact->reseller_id) {
|
|
my $contact_id = $contract->contact->id;
|
|
die( ["Cannot use this contact (#$contact_id) for peering contracts.", "showdetails"] );
|
|
}
|
|
|
|
$c->session->{created_objects}->{contract} = { id => $contract->id };
|
|
delete $c->session->{created_objects}->{contact};
|
|
delete $c->session->{created_objects}->{billing_profile};
|
|
NGCP::Panel::Utils::Message::info(
|
|
c => $c,
|
|
cname => 'peering_create',
|
|
desc => $c->loc('Contract #[_1] successfully created', $contract->id),
|
|
);
|
|
});
|
|
} catch($e) {
|
|
NGCP::Panel::Utils::Message::error(
|
|
c => $c,
|
|
error => $e,
|
|
desc => $c->loc('Failed to create contract'),
|
|
);
|
|
}
|
|
NGCP::Panel::Utils::Navigation::back_or($c, $c->uri_for('/contract'));
|
|
}
|
|
|
|
$c->stash(create_flag => 1);
|
|
$c->stash(form => $form);
|
|
}
|
|
|
|
sub reseller_list :Chained('contract_list') :PathPart('reseller') :CaptureArgs(0) {
|
|
my ($self, $c) = @_;
|
|
|
|
my @product_ids = map { $_->id; } $c->model('DB')->resultset('products')->search_rs({ 'class' => ['reseller'] })->all;
|
|
my $base_rs = $c->stash->{contract_select_rs};
|
|
$c->stash->{reseller_rs} = $base_rs->search({
|
|
'product_id' => { -in => [ @product_ids ] },
|
|
});
|
|
|
|
$c->stash(ajax_uri => $c->uri_for_action("/contract/reseller_ajax"));
|
|
}
|
|
|
|
sub reseller_root :Chained('reseller_list') :PathPart('') :Args(0) {
|
|
|
|
}
|
|
|
|
sub reseller_ajax :Chained('reseller_list') :PathPart('ajax') :Args(0) {
|
|
my ($self, $c) = @_;
|
|
|
|
my $rs = $c->stash->{reseller_rs};
|
|
NGCP::Panel::Utils::Datatables::process($c, $rs, $c->stash->{contract_dt_columns});
|
|
$c->detach( $c->view("JSON") );
|
|
}
|
|
|
|
sub reseller_ajax_contract_filter :Chained('reseller_list') :PathPart('ajax/contract') :Args(1) {
|
|
my ($self, $c, $contract_id) = @_;
|
|
|
|
unless($contract_id && is_int($contract_id)) {
|
|
$contract_id //= '';
|
|
NGCP::Panel::Utils::Message::error(
|
|
c => $c,
|
|
data => { id => $contract_id },
|
|
desc => $c->loc('Invalid contract id detected'),
|
|
);
|
|
$c->response->redirect($c->uri_for());
|
|
return;
|
|
}
|
|
my $now = $c->stash->{now};
|
|
my $rs = NGCP::Panel::Utils::Contract::get_contract_rs(
|
|
schema => $c->model('DB'),
|
|
now => $now)->search_rs({
|
|
'me.id' => $contract_id,
|
|
});
|
|
my $contract_columns = NGCP::Panel::Utils::Datatables::set_columns($c, [
|
|
{ name => "id", search => 1, title => $c->loc("#") },
|
|
{ name => "external_id", search => 1, title => $c->loc("External #") },
|
|
{ name => "contact.email", search => 1, title => $c->loc("Contact Email") },
|
|
{ name => 'billing_profile_name', accessor => "billing_profile_name", search => 0, title => $c->loc('Billing Profile'),
|
|
literal_sql => NGCP::Panel::Utils::BillingMappings::get_actual_billing_mapping_stmt(c => $c, now => $now, projection => 'billing_profile.name' ) },
|
|
{ name => "status", search => 1, title => $c->loc("Status") },
|
|
{ name => "max_subscribers", search => 1, title => $c->loc("Max. Subscribers") },
|
|
]);
|
|
NGCP::Panel::Utils::Datatables::process($c, $rs, $contract_columns);
|
|
$c->detach( $c->view("JSON") );
|
|
}
|
|
|
|
sub reseller_create :Chained('reseller_list') :PathPart('create') :Args(0) {
|
|
my ($self, $c) = @_;
|
|
|
|
my $posted = ($c->request->method eq 'POST');
|
|
my $params = {};
|
|
$params = merge($params, $c->session->{created_objects});
|
|
unless ($self->is_valid_noreseller_contact($c, $params->{contact}{id})) {
|
|
delete $params->{contact};
|
|
}
|
|
$c->stash->{type} = 'reseller';
|
|
$c->stash(ajax_uri => $c->uri_for_action("/contract/ajax"));
|
|
my $form = NGCP::Panel::Form::get("NGCP::Panel::Form::Contract::Reseller", $c);
|
|
$form->process(
|
|
posted => $posted,
|
|
params => $c->request->params,
|
|
item => $params
|
|
);
|
|
NGCP::Panel::Utils::Navigation::check_form_buttons(
|
|
c => $c,
|
|
form => $form,
|
|
fields => {'contact.create' => $c->uri_for('/contact/create/noreseller'),
|
|
'billing_profile.create' => $c->uri_for('/billing/create/noreseller'),
|
|
'billing_profiles.profile.create' => $c->uri_for('/billing/create/noreseller'),
|
|
},
|
|
back_uri => $c->req->uri,
|
|
);
|
|
if($posted && $form->validated) {
|
|
try {
|
|
my $schema = $c->model('DB');
|
|
$schema->set_transaction_isolation('READ COMMITTED');
|
|
$schema->txn_do(sub {
|
|
foreach(qw/contact billing_profile/){
|
|
$form->values->{$_.'_id'} = $form->values->{$_}{id} || undef;
|
|
delete $form->values->{$_};
|
|
}
|
|
$form->values->{external_id} = $form->field('external_id')->value;
|
|
$form->values->{create_timestamp} = $form->values->{modify_timestamp} = NGCP::Panel::Utils::DateTime::current_local;
|
|
$form->values->{product_id} = $schema->resultset('products')->search_rs({ class => $c->stash->{type} })->first->id;
|
|
|
|
my $mappings_to_create = [];
|
|
NGCP::Panel::Utils::BillingMappings::prepare_billing_mappings(
|
|
c => $c,
|
|
resource => $form->values,
|
|
mappings_to_create => $mappings_to_create,
|
|
err_code => sub {
|
|
my ($err,@fields) = @_;
|
|
die( [$err, "showdetails"] );
|
|
});
|
|
|
|
my $contract = $schema->resultset('contracts')->create($form->values);
|
|
NGCP::Panel::Utils::BillingMappings::append_billing_mappings(c => $c,
|
|
contract => $contract,
|
|
mappings_to_create => $mappings_to_create,
|
|
);
|
|
|
|
NGCP::Panel::Utils::ProfilePackages::create_initial_contract_balances(c => $c,
|
|
contract => $contract,
|
|
);
|
|
|
|
if (defined $contract->contact->reseller_id) {
|
|
my $contact_id = $contract->contact->id;
|
|
die( ["Cannot use this contact (#$contact_id) for reseller contracts.", "showdetails"] );
|
|
}
|
|
|
|
$c->session->{created_objects}->{contract} = { id => $contract->id };
|
|
delete $c->session->{created_objects}->{contact};
|
|
delete $c->session->{created_objects}->{billing_profile};
|
|
NGCP::Panel::Utils::Message::info(
|
|
c => $c,
|
|
cname => 'reseller_create',
|
|
desc => $c->loc('Contract #[_1] successfully created', $contract->id),
|
|
);
|
|
});
|
|
} catch($e) {
|
|
NGCP::Panel::Utils::Message::error(
|
|
c => $c,
|
|
error => $e,
|
|
desc => $c->loc('Failed to create contract'),
|
|
);
|
|
}
|
|
NGCP::Panel::Utils::Navigation::back_or($c, $c->uri_for('/contract'));
|
|
}
|
|
|
|
$c->stash(create_flag => 1);
|
|
$c->stash(form => $form);
|
|
}
|
|
|
|
sub is_valid_noreseller_contact {
|
|
my ($self, $c, $contact_id) = @_;
|
|
my $contact = $c->model('DB')->resultset('contacts')->search_rs({
|
|
'id' => $contact_id,
|
|
'reseller_id' => undef,
|
|
'status' => { '!=' => 'terminated' },
|
|
})->first;
|
|
if( $contact ) {
|
|
return 1;
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
sub all_contracts_list :Chained('contract_list') :PathPart('all_contracts') :CaptureArgs(0) {
|
|
my ($self, $c) = @_;
|
|
|
|
my $now = NGCP::Panel::Utils::DateTime::current_local;
|
|
$c->stash->{contract_dt_columnsX} = NGCP::Panel::Utils::Datatables::set_columns($c, [
|
|
{ name => "id", search => 1, title => $c->loc("#") },
|
|
{ name => "external_id", search => 1, title => $c->loc("External #") },
|
|
{ name => "contact.email", search => 1, title => $c->loc("Contact Email") },
|
|
#{ name => 'billing_profile_name', accessor => "billing_profile_name", search => 0, title => $c->loc('Billing Profile'),
|
|
# literal_sql => NGCP::Panel::Utils::BillingMappings::get_actual_billing_mapping_stmt(c => $c, now => $now, projection => 'billing_profile.name' ) },
|
|
{ name => "status", search => 1, title => $c->loc("Status") },
|
|
{ name => "product.name", search => 0, title => $c->loc("Product") },
|
|
]);
|
|
|
|
my $rs_all_contracts = NGCP::Panel::Utils::Contract::get_contract_rs(
|
|
schema => $c->model('DB'),
|
|
now => $now,
|
|
include_terminated => 0,
|
|
);
|
|
|
|
$c->stash(rs_all_contracts => $rs_all_contracts);
|
|
|
|
#my @product_ids = map { $_->id; } $c->model('DB')->resultset('products')->search_rs({ 'class' => ['sippeering'] })->all;
|
|
#my $base_rs = $c->stash->{contract_select_rs};
|
|
#$c->stash->{peering_rs} = $base_rs->search({
|
|
# 'product_id' => { -in => [ @product_ids ] },
|
|
#});
|
|
|
|
$c->stash(ajax_uri => $c->uri_for_action("/contract/all_contracts_ajax"));
|
|
}
|
|
|
|
sub all_contracts_root :Chained('all_contracts_list') :PathPart('') :Args(0) {
|
|
|
|
}
|
|
|
|
sub all_contracts_ajax :Chained('all_contracts_list') :PathPart('ajax') :Args(0) {
|
|
my ($self, $c) = @_;
|
|
|
|
my $rs = $c->stash->{rs_all_contracts};
|
|
NGCP::Panel::Utils::Datatables::process($c, $rs, $c->stash->{contract_dt_columnsX});
|
|
$c->detach( $c->view("JSON") );
|
|
}
|
|
|
|
1;
|
|
|
|
=head1 NAME
|
|
|
|
NGCP::Panel::Controller::Contract - Catalyst Controller
|
|
|
|
=head1 DESCRIPTION
|
|
|
|
View and edit Contracts. Optionally filter them by only peering contracts.
|
|
|
|
=head1 METHODS
|
|
|
|
=head2 contract_list
|
|
|
|
Basis for contracts.
|
|
|
|
=head2 root
|
|
|
|
Display contracts through F<contract/list.tt> template.
|
|
|
|
=head2 create
|
|
|
|
Show modal dialog to create a new contract.
|
|
|
|
=head2 base
|
|
|
|
Capture id of existing contract. Used for L</edit> and L</delete>. Stash "contract" and "contract_result".
|
|
|
|
=head2 edit
|
|
|
|
Show modal dialog to edit a contract.
|
|
|
|
=head2 delete
|
|
|
|
Delete a contract.
|
|
|
|
=head2 ajax
|
|
|
|
Get contracts from the database and output them as JSON.
|
|
The output format is meant for parsing with datatables.
|
|
|
|
The selected rows should be billing.billing_mappings JOIN billing.contracts with only one billing_mapping per contract (the one that fits best with the time).
|
|
|
|
=head2 peering_list
|
|
|
|
Basis for peering_contracts.
|
|
|
|
=head2 peering_root
|
|
|
|
Display contracts through F<contract/list.tt> template. Use L</peering_ajax> as data source.
|
|
|
|
=head2 peering_ajax
|
|
|
|
Similar to L</ajax>. Only select contracts, where billing.product is of class "sippeering".
|
|
|
|
=head2 peering_create
|
|
|
|
Similar to L</create> but sets product_id of billing_mapping to match the
|
|
product of class "sippeering".
|
|
|
|
=head1 AUTHOR
|
|
|
|
Andreas Granig,,,
|
|
|
|
=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:
|