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.
		
		
		
		
		
			
		
			
				
					
					
						
							720 lines
						
					
					
						
							28 KiB
						
					
					
				
			
		
		
	
	
							720 lines
						
					
					
						
							28 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 ) ) {
 | |
|         if (!$c->license('reseller')) {
 | |
|              $c->detach('/denied_page');
 | |
|             return;
 | |
|         }
 | |
|         $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 ( NGCP::Panel::Utils::Contract::is_reseller_product( c => $c, product => $contract->product ) ) {
 | |
|         if (!$c->license('reseller')) {
 | |
|                 $c->detach('/denied_page');
 | |
|             return;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     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) :Does(License) :RequiresLicense('batch_provisioning') :LicenseDetachTo('/denied_page') {
 | |
|     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:
 |