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.
296 lines
12 KiB
296 lines
12 KiB
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:
|