* reseller phonebook entries * customer phonebook entries * subscriber phonebook entries * csv download/upload Change-Id: Ie2c08e1baf8467769a04e6769512417953643811changes/63/19863/19
parent
cb1a3a4cd8
commit
4dbd1e20a4
@ -0,0 +1,227 @@
|
||||
package NGCP::Panel::Controller::Phonebook;
|
||||
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::Phonebook;
|
||||
|
||||
sub auto :Does(ACL) :ACLDetachTo('/denied_page') :AllowedRole(admin) :AllowedRole(reseller) {
|
||||
my ($self, $c) = @_;
|
||||
$c->log->debug(__PACKAGE__ . '::auto');
|
||||
NGCP::Panel::Utils::Navigation::check_redirect_chain(c => $c);
|
||||
return 1;
|
||||
}
|
||||
|
||||
sub list :Chained('/') :PathPart('phonebook') :CaptureArgs(0) {
|
||||
my ( $self, $c ) = @_;
|
||||
|
||||
$c->stash->{phonebook_rs} = $c->model('DB')->resultset('reseller_phonebook');
|
||||
unless($c->user->roles eq "admin") {
|
||||
$c->stash->{phonebook_rs} = $c->stash->{phonebook_rs}->search({
|
||||
reseller_id => $c->user->reseller_id
|
||||
});
|
||||
}
|
||||
$c->stash->{phonebook_dt_columns} = NGCP::Panel::Utils::Datatables::set_columns($c, [
|
||||
{ name => 'id', search => 1, title => $c->loc('#') },
|
||||
$c->user->roles eq "admin"
|
||||
? { name => 'reseller.name', search => 1, title => $c->loc('Reseller') }
|
||||
: (),
|
||||
{ name => 'name', search => 1, title => $c->loc('Name') },
|
||||
{ name => 'number', search => 1, title => $c->loc('Number') },
|
||||
]);
|
||||
|
||||
$c->stash(template => 'phonebook/list.tt');
|
||||
}
|
||||
|
||||
sub root :Chained('list') :PathPart('') :Args(0) {
|
||||
my ($self, $c) = @_;
|
||||
}
|
||||
|
||||
sub ajax :Chained('list') :PathPart('ajax') :Args(0) {
|
||||
my ($self, $c) = @_;
|
||||
my $rs = $c->stash->{phonebook_rs};
|
||||
NGCP::Panel::Utils::Datatables::process($c, $rs, $c->stash->{phonebook_dt_columns});
|
||||
$c->detach( $c->view("JSON") );
|
||||
}
|
||||
|
||||
sub base :Chained('list') :PathPart('') :CaptureArgs(1) {
|
||||
my ($self, $c, $pb_id) = @_;
|
||||
|
||||
unless($pb_id && is_int($pb_id)) {
|
||||
NGCP::Panel::Utils::Message::error(
|
||||
c => $c,
|
||||
log => 'Invalid phonebook enry id detected',
|
||||
desc => $c->loc('Invalid phonebook entry id detected'),
|
||||
);
|
||||
NGCP::Panel::Utils::Navigation::back_or($c, $c->uri_for('/phonebook'));
|
||||
}
|
||||
|
||||
my $res = $c->stash->{phonebook_rs}->find($pb_id);
|
||||
unless(defined($res)) {
|
||||
NGCP::Panel::Utils::Message::error(
|
||||
c => $c,
|
||||
log => 'Phonebook entry does not exist',
|
||||
desc => $c->loc('Phonebook entry does not exist'),
|
||||
);
|
||||
NGCP::Panel::Utils::Navigation::back_or($c, $c->uri_for('/phonebook'));
|
||||
}
|
||||
$c->stash(phonebook_result => $res);
|
||||
}
|
||||
|
||||
sub create :Chained('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 $form;
|
||||
if($c->user->roles eq "admin") {
|
||||
$form = NGCP::Panel::Form::get("NGCP::Panel::Form::Phonebook::Admin", $c);
|
||||
} else {
|
||||
$form = NGCP::Panel::Form::get("NGCP::Panel::Form::Phonebook::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->roles eq "admin") {
|
||||
$form->values->{reseller_id} = $form->values->{reseller}{id};
|
||||
delete $form->values->{reseller};
|
||||
} else {
|
||||
$form->values->{reseller_id} = $c->user->reseller_id;
|
||||
}
|
||||
$c->stash->{phonebook_rs}->create($form->values);
|
||||
delete $c->session->{created_objects}->{reseller};
|
||||
NGCP::Panel::Utils::Message::info(
|
||||
c => $c,
|
||||
desc => $c->loc('Phonebook entry successfully created'),
|
||||
);
|
||||
} catch($e) {
|
||||
NGCP::Panel::Utils::Message::error(
|
||||
c => $c,
|
||||
error => $e,
|
||||
desc => $c->loc('Failed to create phonebook entry'),
|
||||
);
|
||||
}
|
||||
NGCP::Panel::Utils::Navigation::back_or($c, $c->uri_for('/phonebook'));
|
||||
}
|
||||
|
||||
$c->stash(form => $form);
|
||||
$c->stash(create_flag => 1);
|
||||
}
|
||||
|
||||
sub edit :Chained('base') :PathPart('edit') {
|
||||
my ($self, $c) = @_;
|
||||
|
||||
my $posted = ($c->request->method eq 'POST');
|
||||
my $params = { $c->stash->{phonebook_result}->get_inflated_columns };
|
||||
$params->{reseller}{id} = delete $params->{reseller_id};
|
||||
$params = merge($params, $c->session->{created_objects});
|
||||
my $form;
|
||||
if($c->user->roles eq "admin") {
|
||||
$form = NGCP::Panel::Form::get("NGCP::Panel::Form::Phonebook::Admin", $c);
|
||||
} else {
|
||||
$form = NGCP::Panel::Form::get("NGCP::Panel::Form::Phonebook::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->roles eq "admin") {
|
||||
$form->values->{reseller_id} = $form->values->{reseller}{id};
|
||||
delete $form->values->{reseller};
|
||||
}
|
||||
$c->stash->{phonebook_result}->update($form->values);
|
||||
delete $c->session->{created_objects}->{reseller};
|
||||
NGCP::Panel::Utils::Message::info(
|
||||
c => $c,
|
||||
desc => $c->loc('Phonebook entry successfully updated'),
|
||||
);
|
||||
} catch($e) {
|
||||
NGCP::Panel::Utils::Message::error(
|
||||
c => $c,
|
||||
error => $e,
|
||||
desc => $c->loc('Failed to update phonebook entry'),
|
||||
);
|
||||
}
|
||||
NGCP::Panel::Utils::Navigation::back_or($c, $c->uri_for('/phonebook'));
|
||||
}
|
||||
|
||||
$c->stash(form => $form);
|
||||
$c->stash(edit_flag => 1);
|
||||
}
|
||||
|
||||
sub delete :Chained('base') :PathPart('delete') {
|
||||
my ($self, $c) = @_;
|
||||
|
||||
try {
|
||||
$c->stash->{phonebook_result}->delete;
|
||||
NGCP::Panel::Utils::Message::info(
|
||||
c => $c,
|
||||
data => { $c->stash->{phonebook_result}->get_inflated_columns },
|
||||
desc => $c->loc('Phonebook entry successfully deleted'),
|
||||
);
|
||||
} catch($e) {
|
||||
NGCP::Panel::Utils::Message::error(
|
||||
c => $c,
|
||||
error => $e,
|
||||
desc => $c->loc('Failed to delete phonebook entry'),
|
||||
);
|
||||
}
|
||||
NGCP::Panel::Utils::Navigation::back_or($c, $c->uri_for('/phonebook'));
|
||||
}
|
||||
|
||||
sub upload_csv :Chained('list') :PathPart('upload_csv') :Args(0) {
|
||||
my ($self, $c) = @_;
|
||||
|
||||
my $form = NGCP::Panel::Form::get("NGCP::Panel::Form::Phonebook::Upload", $c);
|
||||
NGCP::Panel::Utils::Phonebook::ui_upload_csv(
|
||||
$c, $c->stash->{phonebook_rs}, $form, 'reseller', $c->user->reseller_id,
|
||||
$c->uri_for('/phonebook/upload_csv'), $c->uri_for('/phonebook')
|
||||
);
|
||||
|
||||
$c->stash(create_flag => 1);
|
||||
$c->stash(form => $form);
|
||||
return;
|
||||
}
|
||||
|
||||
sub download_csv :Chained('list') :PathPart('download_csv') :Args(0) {
|
||||
my ($self, $c) = @_;
|
||||
|
||||
$c->response->header ('Content-Disposition' => 'attachment; filename="reseller_phonebook_entries.csv"');
|
||||
$c->response->content_type('text/csv');
|
||||
$c->response->status(200);
|
||||
NGCP::Panel::Utils::Phonebook::download_csv(
|
||||
$c, $c->stash->{phonebook_rs}, 'reseller', $c->user->reseller_id
|
||||
);
|
||||
return;
|
||||
}
|
||||
|
||||
1;
|
@ -0,0 +1,22 @@
|
||||
package NGCP::Panel::Form::Phonebook::Admin;
|
||||
|
||||
use HTML::FormHandler::Moose;
|
||||
extends 'NGCP::Panel::Form::Phonebook::Reseller';
|
||||
|
||||
has_field 'reseller' => (
|
||||
type => '+NGCP::Panel::Field::Reseller',
|
||||
validate_when_empty => 1,
|
||||
element_attr => {
|
||||
rel => ['tooltip'],
|
||||
title => ['The reseller id to assign this phonebook entry to.']
|
||||
},
|
||||
);
|
||||
|
||||
has_block 'fields' => (
|
||||
tag => 'div',
|
||||
class => [qw/modal-body/],
|
||||
render_list => [qw/id reseller name number/],
|
||||
);
|
||||
|
||||
1;
|
||||
# vim: set tabstop=4 expandtab:
|
@ -0,0 +1,61 @@
|
||||
package NGCP::Panel::Form::Phonebook::Customer;
|
||||
|
||||
use HTML::FormHandler::Moose;
|
||||
extends 'HTML::FormHandler';
|
||||
|
||||
use HTML::FormHandler::Widget::Block::Bootstrap;
|
||||
|
||||
has '+widget_wrapper' => ( default => 'Bootstrap' );
|
||||
has_field 'submitid' => ( type => 'Hidden' );
|
||||
sub build_render_list {[qw/submitid fields actions/]}
|
||||
sub build_form_element_class {[qw(form-horizontal)]}
|
||||
|
||||
has_field 'id' => (
|
||||
type => 'Hidden',
|
||||
);
|
||||
|
||||
has_field 'contract_id' => (
|
||||
type => 'Hidden',
|
||||
);
|
||||
|
||||
has_field 'name' => (
|
||||
type => 'Text',
|
||||
required => 1,
|
||||
label => 'Name',
|
||||
element_attr => {
|
||||
rel => ['tooltip'],
|
||||
title => ['The full entry name "e.g. John Smith".'],
|
||||
},
|
||||
);
|
||||
|
||||
has_field 'number' => (
|
||||
type => 'Text',
|
||||
required => 1,
|
||||
label => 'Number',
|
||||
element_attr => {
|
||||
rel => ['tooltip'],
|
||||
title => ['The phonebook number, can be either as a numeric or a SIP number.' ],
|
||||
},
|
||||
);
|
||||
|
||||
has_field 'save' => (
|
||||
type => 'Submit',
|
||||
value => 'Save',
|
||||
element_class => [qw/btn btn-primary/],
|
||||
label => '',
|
||||
);
|
||||
|
||||
has_block 'fields' => (
|
||||
tag => 'div',
|
||||
class => [qw/modal-body/],
|
||||
render_list => [qw/id contract_id name number/],
|
||||
);
|
||||
|
||||
has_block 'actions' => (
|
||||
tag => 'div',
|
||||
class => [qw/modal-footer/],
|
||||
render_list => [qw/save/],
|
||||
);
|
||||
|
||||
1;
|
||||
# vim: set tabstop=4 expandtab:
|
@ -0,0 +1,61 @@
|
||||
package NGCP::Panel::Form::Phonebook::Reseller;
|
||||
|
||||
use HTML::FormHandler::Moose;
|
||||
extends 'HTML::FormHandler';
|
||||
|
||||
use HTML::FormHandler::Widget::Block::Bootstrap;
|
||||
|
||||
has '+widget_wrapper' => ( default => 'Bootstrap' );
|
||||
has_field 'submitid' => ( type => 'Hidden' );
|
||||
sub build_render_list {[qw/submitid fields actions/]}
|
||||
sub build_form_element_class {[qw(form-horizontal)]}
|
||||
|
||||
has_field 'id' => (
|
||||
type => 'Hidden',
|
||||
);
|
||||
|
||||
has_field 'reseller_id' => (
|
||||
type => 'Hidden',
|
||||
);
|
||||
|
||||
has_field 'name' => (
|
||||
type => 'Text',
|
||||
required => 1,
|
||||
label => 'Name',
|
||||
element_attr => {
|
||||
rel => ['tooltip'],
|
||||
title => ['The full entry name "e.g. John Smith".'],
|
||||
},
|
||||
);
|
||||
|
||||
has_field 'number' => (
|
||||
type => 'Text',
|
||||
required => 1,
|
||||
label => 'Number',
|
||||
element_attr => {
|
||||
rel => ['tooltip'],
|
||||
title => ['The phonebook number, can be either as a numeric or a SIP number.' ],
|
||||
},
|
||||
);
|
||||
|
||||
has_field 'save' => (
|
||||
type => 'Submit',
|
||||
value => 'Save',
|
||||
element_class => [qw/btn btn-primary/],
|
||||
label => '',
|
||||
);
|
||||
|
||||
has_block 'fields' => (
|
||||
tag => 'div',
|
||||
class => [qw/modal-body/],
|
||||
render_list => [qw/id reseller_id name number/],
|
||||
);
|
||||
|
||||
has_block 'actions' => (
|
||||
tag => 'div',
|
||||
class => [qw/modal-footer/],
|
||||
render_list => [qw/save/],
|
||||
);
|
||||
|
||||
1;
|
||||
# vim: set tabstop=4 expandtab:
|
@ -0,0 +1,71 @@
|
||||
package NGCP::Panel::Form::Phonebook::Subscriber;
|
||||
|
||||
use HTML::FormHandler::Moose;
|
||||
extends 'HTML::FormHandler';
|
||||
|
||||
use HTML::FormHandler::Widget::Block::Bootstrap;
|
||||
|
||||
has '+widget_wrapper' => ( default => 'Bootstrap' );
|
||||
has_field 'submitid' => ( type => 'Hidden' );
|
||||
sub build_render_list {[qw/submitid fields actions/]}
|
||||
sub build_form_element_class {[qw(form-horizontal)]}
|
||||
|
||||
has_field 'id' => (
|
||||
type => 'Hidden',
|
||||
);
|
||||
|
||||
has_field 'subscriber_id' => (
|
||||
type => 'Hidden',
|
||||
);
|
||||
|
||||
has_field 'name' => (
|
||||
type => 'Text',
|
||||
required => 1,
|
||||
label => 'Name',
|
||||
element_attr => {
|
||||
rel => ['tooltip'],
|
||||
title => ['The full entry name "e.g. John Smith".'],
|
||||
},
|
||||
);
|
||||
|
||||
has_field 'number' => (
|
||||
type => 'Text',
|
||||
required => 1,
|
||||
label => 'Number',
|
||||
element_attr => {
|
||||
rel => ['tooltip'],
|
||||
title => ['The phonebook number, can be either as a numeric or a SIP number.' ],
|
||||
},
|
||||
);
|
||||
|
||||
has_field 'shared' => (
|
||||
type => 'Boolean',
|
||||
required => 0,
|
||||
label => 'Shared',
|
||||
element_attr => {
|
||||
rel => ['tooltip'],
|
||||
title => ['Share the entry to other subscribers of the same customer.' ],
|
||||
},
|
||||
);
|
||||
|
||||
has_field 'save' => (
|
||||
type => 'Submit',
|
||||
value => 'Save',
|
||||
element_class => [qw/btn btn-primary/],
|
||||
label => '',
|
||||
);
|
||||
|
||||
has_block 'fields' => (
|
||||
tag => 'div',
|
||||
class => [qw/modal-body/],
|
||||
render_list => [qw/id subscriber_id name number shared/],
|
||||
);
|
||||
|
||||
has_block 'actions' => (
|
||||
tag => 'div',
|
||||
class => [qw/modal-footer/],
|
||||
render_list => [qw/save/],
|
||||
);
|
||||
|
||||
1;
|
||||
# vim: set tabstop=4 expandtab:
|
@ -0,0 +1,44 @@
|
||||
package NGCP::Panel::Form::Phonebook::Upload;
|
||||
use Sipwise::Base;
|
||||
use HTML::FormHandler::Moose;
|
||||
extends 'HTML::FormHandler';
|
||||
|
||||
use HTML::FormHandler::Widget::Block::Bootstrap;
|
||||
|
||||
has '+widget_wrapper' => ( default => 'Bootstrap' );
|
||||
has '+enctype' => ( default => 'multipart/form-data');
|
||||
has_field 'submitid' => ( type => 'Hidden' );
|
||||
sub build_render_list {[qw/submitid fields actions/]}
|
||||
sub build_form_element_class { [qw/form-horizontal/] }
|
||||
|
||||
has_field 'upload_phonebook' => (
|
||||
type => 'Upload',
|
||||
max_size => '2097152000', # 2GB
|
||||
);
|
||||
|
||||
has_field 'purge_existing' => (
|
||||
type => 'Boolean',
|
||||
);
|
||||
|
||||
has_field 'save' => (
|
||||
type => 'Submit',
|
||||
value => 'Upload',
|
||||
element_class => [qw/btn btn-primary/],
|
||||
do_label => 0,
|
||||
);
|
||||
|
||||
has_block 'fields' => (
|
||||
tag => 'div',
|
||||
class => [qw/modal-body/],
|
||||
render_list => [qw/upload_phonebook purge_existing/],
|
||||
);
|
||||
|
||||
has_block 'actions' => (
|
||||
tag => 'div',
|
||||
class => [qw/modal-footer/],
|
||||
render_list => [qw/save/],
|
||||
);
|
||||
|
||||
1;
|
||||
|
||||
# vim: set tabstop=4 expandtab:
|
@ -0,0 +1,277 @@
|
||||
package NGCP::Panel::Utils::Phonebook;
|
||||
use strict;
|
||||
use warnings;
|
||||
|
||||
use Sipwise::Base;
|
||||
use English;
|
||||
use NGCP::Panel::Utils::Generic qw(:all);
|
||||
use NGCP::Panel::Utils::Message;
|
||||
|
||||
sub get_reseller_phonebook {
|
||||
my ($c, $reseller_id) = @_;
|
||||
my @pb;
|
||||
|
||||
my $r_pb_rs = $c->model('DB')->resultset('reseller_phonebook')->search({
|
||||
reseller_id => $reseller_id,
|
||||
});
|
||||
|
||||
for my $r ($r_pb_rs->all) {
|
||||
push @pb, { name => $r->name, number => $r->number };
|
||||
}
|
||||
|
||||
return [ sort { $a->{name} cmp $b->{name} } @pb ];
|
||||
}
|
||||
|
||||
sub get_contract_phonebook {
|
||||
my ($c, $contract_id) = @_;
|
||||
my @pb;
|
||||
my %c_numbers;
|
||||
|
||||
my $contract = $c->model('DB')->resultset('contracts')->search({
|
||||
id => $contract_id,
|
||||
})->first;
|
||||
|
||||
my $r_pb_rs = $c->model('DB')->resultset('reseller_phonebook')->search({
|
||||
reseller_id => $contract->contact->reseller->id,
|
||||
});
|
||||
|
||||
my $c_pb_rs = $c->model('DB')->resultset('contract_phonebook')->search({
|
||||
contract_id => $contract_id,
|
||||
});
|
||||
|
||||
for my $r ($c_pb_rs->all) {
|
||||
push @pb, { name => $r->name, number => $r->number };
|
||||
$c_numbers{$r->number} = $r->name;
|
||||
}
|
||||
|
||||
for my $r ($r_pb_rs->all) {
|
||||
unless (exists $c_numbers{$r->number}) {
|
||||
push @pb, { name => $r->name, number => $r->number };
|
||||
}
|
||||
}
|
||||
|
||||
return [ sort { $a->{name} cmp $b->{name} } @pb ];
|
||||
}
|
||||
|
||||
sub get_subscriber_phonebook {
|
||||
my ($c, $subscriber_id) = @_;
|
||||
my @pb;
|
||||
my %c_numbers;
|
||||
|
||||
my $sub = $c->model('DB')->resultset('voip_subscribers')->search({
|
||||
id => $subscriber_id,
|
||||
})->first;
|
||||
|
||||
my $r_pb_rs = $c->model('DB')->resultset('reseller_phonebook')->search({
|
||||
reseller_id => $sub->contract->contact->reseller->id,
|
||||
});
|
||||
|
||||
my $c_pb_rs = $c->model('DB')->resultset('contract_phonebook')->search({
|
||||
contract_id => $sub->contract_id,
|
||||
});
|
||||
|
||||
my $a_pb_rs = $c->model('DB')->resultset('subscriber_phonebook')->search({
|
||||
shared => 1,
|
||||
'contract.id' => $sub->contract_id,
|
||||
},{
|
||||
join => { 'subscriber' => 'contract' },
|
||||
});
|
||||
|
||||
my $s_pb_rs = $c->model('DB')->resultset('subscriber_phonebook')->search({
|
||||
subscriber_id => $subscriber_id,
|
||||
});
|
||||
|
||||
for my $r ($s_pb_rs->all) {
|
||||
push @pb, { name => $r->name, number => $r->number };
|
||||
$c_numbers{$r->number} = $r->name;
|
||||
}
|
||||
|
||||
for my $r ($c_pb_rs->all) {
|
||||
unless (exists $c_numbers{$r->number}) {
|
||||
push @pb, { name => $r->name, number => $r->number };
|
||||
$c_numbers{$r->number} = $r->name;
|
||||
}
|
||||
}
|
||||
|
||||
for my $r ($a_pb_rs->all) {
|
||||
unless (exists $c_numbers{$r->number}) {
|
||||
push @pb, { name => $r->name, number => $r->number };
|
||||
$c_numbers{$r->number} = $r->name;
|
||||
}
|
||||
}
|
||||
|
||||
for my $r ($r_pb_rs->all) {
|
||||
unless (exists $c_numbers{$r->number}) {
|
||||
push @pb, { name => $r->name, number => $r->number };
|
||||
}
|
||||
}
|
||||
|
||||
return [ sort { $a->{name} cmp $b->{name} } @pb ];
|
||||
}
|
||||
|
||||
sub ui_upload_csv {
|
||||
my ($c, $rs, $form, $owner, $owner_id, $action, $back) = @_;
|
||||
|
||||
my $upload = $c->req->upload('upload_phonebook');
|
||||
my $posted = $c->req->method eq 'POST';
|
||||
my @params = ( upload_phonebook => $posted ? $upload : undef, );
|
||||
$form->process(
|
||||
posted => $posted,
|
||||
params => { @params },
|
||||
action => $action,
|
||||
);
|
||||
if($form->validated) {
|
||||
unless($upload) {
|
||||
NGCP::Panel::Utils::Message::error(
|
||||
c => $c,
|
||||
desc => $c->loc('No phonebook entries file specified!'),
|
||||
);
|
||||
$c->response->redirect($back);
|
||||
return;
|
||||
}
|
||||
my $data = $upload->slurp;
|
||||
my($entries, $fails, $text_success);
|
||||
try {
|
||||
my ($entries, $fails, $text) =
|
||||
upload_csv($c, $rs, $owner, $owner_id,
|
||||
$c->req->params->{purge_existing}, \$data);
|
||||
NGCP::Panel::Utils::Message::info(
|
||||
c => $c,
|
||||
desc => $$text,
|
||||
);
|
||||
} catch($e) {
|
||||
NGCP::Panel::Utils::Message::error(
|
||||
c => $c,
|
||||
error => $e,
|
||||
desc => $c->loc('Failed to upload Phonebook entries'),
|
||||
);
|
||||
}
|
||||
|
||||
$c->response->redirect($back);
|
||||
}
|
||||
|
||||
return;
|
||||
}
|
||||
|
||||
sub upload_csv {
|
||||
my ($c, $rs, $owner, $owner_id, $purge, $data) = @_;
|
||||
|
||||
my $csv = Text::CSV_XS->new({ allow_whitespace => 1, binary => 1, keep_meta_info => 1 });
|
||||
|
||||
my @cols = qw/name number/;
|
||||
my @fields ;
|
||||
my @fails = ();
|
||||
my $linenum = 0;
|
||||
my @entries;
|
||||
|
||||
$c->model('DB')->txn_do(sub {
|
||||
|
||||
open(my $fh, '<:encoding(utf8)', $data);
|
||||
|
||||
while ( my $line = <$fh> ){
|
||||
++$linenum;
|
||||
next unless length $line;
|
||||
unless($csv->parse($line)) {
|
||||
push @fails, $linenum;
|
||||
next;
|
||||
}
|
||||
@fields = $csv->fields();
|
||||
|
||||
my $row = {};
|
||||
# name,number
|
||||
if (scalar @fields == 2) {
|
||||
@{$row}{@cols} = @fields;
|
||||
# name,number,reseller_id (in case of admin uploads)
|
||||
} elsif ($c->user->roles eq "admin" &&
|
||||
$owner eq 'reseller' && scalar @fields == 3) {
|
||||
@{$row}{@cols,'reseller_id'} = @fields;
|
||||
# name,number,shared
|
||||
} elsif ($owner eq 'subscriber' && scalar @fields == 3) {
|
||||
@{$row}{@cols,'shared'} = @fields;
|
||||
# hmmm
|
||||
} else {
|
||||
push @fails, $linenum;
|
||||
next;
|
||||
}
|
||||
$row->{$owner.'_id'} //= $owner_id;
|
||||
push @entries, $row;
|
||||
unless ($purge) {
|
||||
$rs->update_or_create($row,{key=>'rel_u_idx'});
|
||||
}
|
||||
}
|
||||
|
||||
if ($purge) {
|
||||
my ($start, $end);
|
||||
$start = time;
|
||||
$rs->delete;
|
||||
$end = time;
|
||||
$c->log->debug("Purging phonebook entries took " . ($end - $start) . "s");
|
||||
$start = time;
|
||||
$rs->populate(\@entries);
|
||||
$end = time;
|
||||
$c->log->debug("Populating phonebook entries took " . ($end - $start) . "s");
|
||||
}
|
||||
});
|
||||
|
||||
my $text = $c->loc('Phonebook entries successfully uploaded');
|
||||
if (@fails) {
|
||||
$text .= $c->loc(", but skipped the following line numbers: ") . (join ", ", @fails);
|
||||
}
|
||||
|
||||
return ( \@entries, \@fails, \$text );
|
||||
}
|
||||
|
||||
sub download_csv {
|
||||
my ($c, $rs, $owner, $owner_id) = @_;
|
||||
|
||||
my @cols = qw/name number/;
|
||||
|
||||
if ($owner eq 'admin') {
|
||||
push @cols, 'reseller_id';
|
||||
} elsif ($owner eq 'subscriber') {
|
||||
push @cols, 'shared';
|
||||
}
|
||||
|
||||
my ($start, $end);
|
||||
$start = time;
|
||||
foreach my $row ($rs->all) {
|
||||
my %entry = $row->get_inflated_columns;
|
||||
delete $entry{id};
|
||||
$c->res->write_fh->write(join (",", @entry{@cols}) );
|
||||
$c->res->write_fh->write("\n");
|
||||
}
|
||||
$c->res->write_fh->close;
|
||||
$end = time;
|
||||
$c->log->debug("Creating phonebook entries CSV for download took " . ($end - $start) . "s");
|
||||
return 1;
|
||||
}
|
||||
|
||||
1;
|
||||
|
||||
=head1 NAME
|
||||
|
||||
NGCP::Panel::Utils::Phonebook
|
||||
|
||||
=head1 DESCRIPTION
|
||||
|
||||
A helper to manipulate the phonebook data
|
||||
|
||||
=head1 METHODS
|
||||
|
||||
=head2 get_reseller_phonebook
|
||||
|
||||
=head2 get_contract_phonebook
|
||||
|
||||
=head2 get_subscriber_phonebook
|
||||
|
||||
=head1 AUTHOR
|
||||
|
||||
Sipwise Development Team
|
||||
|
||||
=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:
|
@ -0,0 +1,29 @@
|
||||
[% site_config.title = c.loc('Phonebook') -%]
|
||||
[%
|
||||
helper.name = c.loc('Phonebook');
|
||||
helper.identifier = 'phonebook';
|
||||
helper.messages = messages;
|
||||
helper.dt_columns = phonebook_dt_columns;
|
||||
helper.length_change = 1;
|
||||
|
||||
helper.close_target = close_target;
|
||||
helper.create_flag = create_flag;
|
||||
helper.edit_flag = edit_flag;
|
||||
helper.form_object = form;
|
||||
helper.ajax_uri = c.uri_for_action( "/phonebook/ajax" );
|
||||
|
||||
UNLESS c.user.read_only;
|
||||
helper.dt_buttons = [
|
||||
{ name = c.loc('Edit'), uri = "/phonebook/'+full.id+'/edit", class = 'btn-small btn-primary', icon = 'icon-edit' },
|
||||
{ name = c.loc('Delete'), uri = "/phonebook/'+full.id+'/delete", class = 'btn-small btn-secondary', icon = 'icon-trash' },
|
||||
];
|
||||
helper.top_buttons = [
|
||||
{ name = c.loc('Create Phonebook Entry'), uri = c.uri_for('/phonebook/create'), icon = 'icon-star' },
|
||||
{ name = c.loc('Download CSV'), uri = c.uri_for('/phonebook/download_csv'), icon = 'icon-star' },
|
||||
{ name = c.loc('Upload CSV'), uri = c.uri_for('/phonebook/upload_csv'), icon = 'icon-star' },
|
||||
];
|
||||
END;
|
||||
|
||||
PROCESS 'helpers/datatables.tt';
|
||||
-%]
|
||||
[% # vim: set tabstop=4 syntax=html expandtab: -%]
|
Loading…
Reference in new issue