package NGCP::Panel::Role::API::Customers; use NGCP::Panel::Utils::Generic qw(:all); use Sipwise::Base; use parent 'NGCP::Panel::Role::API'; use boolean qw(true); use Data::HAL qw(); use Data::HAL::Link qw(); use HTTP::Status qw(:constants); use NGCP::Panel::Utils::DateTime; use NGCP::Panel::Utils::Contract; use NGCP::Panel::Utils::ProfilePackages qw(); use NGCP::Panel::Utils::BillingMappings qw(); use NGCP::Panel::Utils::Preferences; use NGCP::Panel::Utils::Subscriber qw(); sub _item_rs { my ($self, $c, $now) = @_; # returns a contracts rs filtered based on role my $item_rs = NGCP::Panel::Utils::Contract::get_customer_rs( c => $c, include_terminated => 1, now => $now, ); return $item_rs; } sub get_form { my ($self, $c) = @_; return NGCP::Panel::Form::get("NGCP::Panel::Form::Contract::CustomerAPI", $c); } sub resource_from_item { my ($self, $c, $item, $form, $now) = @_; my %resource = $item->get_inflated_columns; $now //= NGCP::Panel::Utils::DateTime::current_local; my $billing_mapping = NGCP::Panel::Utils::BillingMappings::get_actual_billing_mapping(c => $c, now => $now, contract => $item); my $billing_profile_id = $billing_mapping->billing_profile->id; my $future_billing_profiles = NGCP::Panel::Utils::BillingMappings::resource_from_future_mappings($item); my $billing_profiles = NGCP::Panel::Utils::BillingMappings::resource_from_mappings($item); NGCP::Panel::Utils::ProfilePackages::catchup_contract_balances(c => $c, contract => $item, now => $now); $form //= $self->get_form($c); return unless $self->validate_form( c => $c, form => $form, resource => \%resource, run => 0, ); foreach my $field (qw/create_timestamp activate_timestamp modify_timestamp terminate_timestamp/) { $resource{$field} = defined $resource{$field} ? NGCP::Panel::Utils::DateTime::to_string(NGCP::Panel::Utils::DateTime::from_string($resource{$field})) : undef; } # return the virtual "type" instead of the actual product id $resource{type} = $item->product->class; #$billing_mapping->product->class; $resource{billing_profiles} = $future_billing_profiles; $resource{all_billing_profiles} = $billing_profiles; $resource{id} = int($item->id); $resource{billing_profile_id} = int($billing_profile_id); $resource{billing_profile_definition} = 'id'; return \%resource; } sub hal_from_customer { my ($self, $c, $customer, $form, $now) = @_; my $is_adm = 0; if ($c->user->roles eq "admin" || $c->user->roles eq "reseller" || $c->user->roles eq "ccareadmin" || $c->user->roles eq "ccare") { $is_adm = 1; } my $resource = $self->resource_from_item($c, $customer, $form, $now); my @profile_links = (); my @network_links = (); foreach my $mapping ($customer->billing_mappings->all) { push(@profile_links,Data::HAL::Link->new(relation => 'ngcp:billingprofiles', href => sprintf("/api/billingprofiles/%d", $mapping->billing_profile->id))); if ($mapping->network_id) { push(@profile_links,Data::HAL::Link->new(relation => 'ngcp:billingnetworks', href => sprintf("/api/billingnetworks/%d", $mapping->network_id))); } } my $hal = Data::HAL->new( links => [ Data::HAL::Link->new( relation => 'curies', href => 'http://purl.org/sipwise/ngcp-api/#rel-{rel}', name => 'ngcp', templated => true, ), Data::HAL::Link->new(relation => 'collection', href => sprintf('/api/%s/', $self->resource_name)), Data::HAL::Link->new(relation => 'profile', href => 'http://purl.org/sipwise/ngcp-api/'), Data::HAL::Link->new(relation => 'self', href => sprintf("%s%d", $self->dispatch_path, $customer->id)), ($is_adm ? ( Data::HAL::Link->new(relation => 'ngcp:customercontacts', href => sprintf("/api/customercontacts/%d", $customer->contact->id)), Data::HAL::Link->new(relation => 'ngcp:customerpreferences', href => sprintf("/api/customerpreferences/%d", $customer->id)), Data::HAL::Link->new(relation => 'ngcp:customerfraudpreferences', href => sprintf("/api/customerfraudpreferences/%d", $customer->id)), @profile_links, @network_links, $customer->profile_package_id ? Data::HAL::Link->new(relation => 'ngcp:profilepackages', href => sprintf("/api/profilepackages/%d", $customer->profile_package_id)) : (), Data::HAL::Link->new(relation => 'ngcp:customerbalances', href => sprintf("/api/customerbalances/%d", $customer->id)), Data::HAL::Link->new(relation => 'ngcp:balanceintervals', href => sprintf("/api/balanceintervals/%d", $customer->id)), $customer->invoice_template ? (Data::HAL::Link->new(relation => 'ngcp:invoicetemplates', href => sprintf("/api/invoicetemplates/%d", $customer->invoice_template_id))) : (), $customer->subscriber_email_template_id ? (Data::HAL::Link->new(relation => 'ngcp:subscriberemailtemplates', href => sprintf("/api/emailtemplates/%d", $customer->subscriber_email_template_id))) : (), $customer->passreset_email_template_id ? (Data::HAL::Link->new(relation => 'ngcp:passresetemailtemplates', href => sprintf("/api/emailtemplates/%d", $customer->passreset_email_template_id))) : (), $customer->invoice_email_template_id ? (Data::HAL::Link->new(relation => 'ngcp:invoiceemailtemplates', href => sprintf("/api/emailtemplates/%d", $customer->invoice_email_template_id))) : (), (($c->user->roles eq "ccareadmin" || $c->user->roles eq "ccare") ? () : Data::HAL::Link->new(relation => 'ngcp:calls', href => sprintf("/api/calls/?customer_id=%d", $customer->id))), $self->get_journal_relation_link($c, $customer->id), ) : ()), ], relation => 'ngcp:'.$self->resource_name, ); $self->expand_fields($c, $resource); $hal->resource($resource); return $hal; } sub customer_by_id { my ($self, $c, $id, $now) = @_; my $customers = $self->item_rs($c,$now); return $customers->find($id); } sub update_customer { my ($self, $c, $customer, $old_resource, $resource, $form, $now) = @_; if ($customer->status eq 'terminated') { $self->error($c, HTTP_UNPROCESSABLE_ENTITY, 'Customer is already terminated and cannot be changed.'); return; } my $billing_mapping = NGCP::Panel::Utils::BillingMappings::get_actual_billing_mapping(c => $c, now => $now, contract => $customer, ); my $billing_profile = $billing_mapping->billing_profile; my $old_package = $customer->profile_package; #$old_resource->{prepaid} = $billing_profile->prepaid; $old_resource->{billing_mapping} = $billing_mapping; $form //= $self->get_form($c); # TODO: for some reason, formhandler lets missing contact_id slip thru $resource->{contact_id} //= undef; $resource->{type} //= $customer->product->class; return unless $self->validate_form( c => $c, form => $form, resource => $resource, ); my $mappings_to_create = []; my $delete_mappings = 0; my $set_package = ($resource->{billing_profile_definition} // 'id') eq 'package'; return unless NGCP::Panel::Utils::BillingMappings::prepare_billing_mappings( c => $c, resource => $resource, old_resource => $old_resource, mappings_to_create => $mappings_to_create, now => $now, delete_mappings => \$delete_mappings, err_code => sub { my ($err) = @_; $self->error($c, HTTP_UNPROCESSABLE_ENTITY, $err); }); delete $resource->{type}; $resource->{modify_timestamp} = $now; #problematic for ON UPDATE current_timestamp columns my $custcontact; if($old_resource->{contact_id} != $resource->{contact_id}) { $custcontact = $c->model('DB')->resultset('contacts') ->search({ 'me.status' => { '!=' => 'terminated' }, reseller_id => { '-not' => undef }, })->find($resource->{contact_id}); unless($custcontact) { $self->error($c, HTTP_UNPROCESSABLE_ENTITY, "Invalid 'contact_id', doesn't exist"); return; } unless($billing_profile->reseller_id == $custcontact->reseller_id) { $self->error($c, HTTP_UNPROCESSABLE_ENTITY, "Invalid 'contact_id', reseller doesn't match billing profile reseller"); return; } } else { $custcontact = $customer->contact; } my $tmplfields = $self->get_template_fields_spec(); foreach my $field (keys %$tmplfields){ my $oldtmpl = $old_resource->{$field} // 0; if($resource->{$field} && $oldtmpl != $resource->{$field}) { my $tmpl = $c->model('DB')->resultset($tmplfields->{$field}->[0]) ->search({ reseller_id => $custcontact->reseller_id }) ->find($resource->{$field}); unless($tmpl) { $self->error($c, HTTP_UNPROCESSABLE_ENTITY, "Invalid '$field', doesn't exist for reseller assigned to customer contact"); return; } } } my $old_ext_id = $customer->external_id // ''; if($resource->{status} eq "terminated") { $resource->{terminate_timestamp} = NGCP::Panel::Utils::DateTime::current_local; } try { $customer->update($resource); NGCP::Panel::Utils::BillingMappings::append_billing_mappings(c => $c, contract => $customer, mappings_to_create => $mappings_to_create, now => $now, delete_mappings => $delete_mappings, ); $customer = $self->customer_by_id($c, $customer->id, $now); my $balance = NGCP::Panel::Utils::ProfilePackages::catchup_contract_balances(c => $c, contract => $customer, old_package => $old_package, now => $now); #make balance_intervals.t work $balance = NGCP::Panel::Utils::ProfilePackages::resize_actual_contract_balance(c => $c, contract => $customer, old_package => $old_package, balance => $balance, now => $now, profiles_added => ($set_package ? scalar @$mappings_to_create : 0), ); $billing_mapping = NGCP::Panel::Utils::BillingMappings::get_actual_billing_mapping(c => $c, now => $now, contract => $customer, ); $billing_profile = $billing_mapping->billing_profile; if(($customer->external_id // '') ne $old_ext_id) { foreach my $sub($customer->voip_subscribers->all) { my $prov_sub = $sub->provisioning_voip_subscriber; next unless($prov_sub); NGCP::Panel::Utils::Subscriber::update_preferences( c => $c, prov_subscriber => $prov_sub, preferences => { ext_contract_id => $customer->external_id } ); } } NGCP::Panel::Utils::Subscriber::switch_prepaid_contract(c => $c, prepaid => $billing_profile->prepaid, contract => $customer, ); if($old_resource->{status} ne $resource->{status}) { if($customer->id == 1) { $self->error($c, HTTP_FORBIDDEN, "Cannot set customer status to '".$resource->{status}."' for customer id '1'"); return; } NGCP::Panel::Utils::Contract::recursively_lock_contract( c => $c, contract => $customer, ); } return $customer; # TODO: what about changed product, do we allow it? } catch($e) { $self->error($c, HTTP_INTERNAL_SERVER_ERROR, "Internal Server Error", "Failed to update customer contract id '".$customer->id."'", $e); }; } sub get_template_fields_spec{ return { 'invoice_template_id' => [qw/invoice_templates invoice_template/], 'subscriber_email_template_id' => [qw/email_templates subscriber_email_template/], 'passreset_email_template_id' => [qw/email_templates passreset_email_template/], 'invoice_email_template_id' => [qw/email_templates invoice_email_template/], }; } 1; # vim: set tabstop=4 expandtab: