X-Git-Url: http://git.freeside.biz/gitweb/?a=blobdiff_plain;f=FS%2FFS%2Fpart_export%2Fnetsapiens.pm;h=c6110f5ac523623455b7ef71407fff7a76e42ea0;hb=38e34bbc53a4222c7507e95914e1364a5a74623f;hp=10b53ef51e457f63d8172659c78f7fb879fc9884;hpb=2b968bffd937b19d2314aa5e304fccc8ec1e1350;p=freeside.git diff --git a/FS/FS/part_export/netsapiens.pm b/FS/FS/part_export/netsapiens.pm index 10b53ef51..c6110f5ac 100644 --- a/FS/FS/part_export/netsapiens.pm +++ b/FS/FS/part_export/netsapiens.pm @@ -1,139 +1,343 @@ package FS::part_export::netsapiens; +use base qw( FS::part_export ); -use vars qw(@ISA %info); -use URI; +use vars qw( $me %info ); use MIME::Base64; use Tie::IxHash; -use FS::part_export; +use Date::Format qw( time2str ); +use Regexp::Common qw( URI ); +use REST::Client; +use Carp qw(carp); -@ISA = qw(FS::part_export); +$me = '[FS::part_export::netsapiens]'; + +#These export options set default values for the various commands +#to create/update objects. Add more options as needed. + +my %tristate = ( type => 'select', options => [ '', 'yes', 'no' ]); + +tie my %subscriber_fields, 'Tie::IxHash', + 'admin_vmail' => { label=>'VMail Prov.', %tristate }, + 'dial_plan' => { label=>'Dial Translation' }, + 'dial_policy' => { label=>'Dial Permission' }, + 'call_limit' => { label=>'Call Limit' }, + 'domain_dir' => { label=>'Dir Lst', %tristate }, +; + +tie my %registrar_fields, 'Tie::IxHash', + 'authenticate_register' => { label=>'Authenticate Registration', %tristate }, + 'authentication_realm' => { label=>'Authentication Realm' }, +; + +tie my %dialplan_fields, 'Tie::IxHash', + 'responder' => { label=>'Application' }, #this could be nicer + 'from_name' => { label=>'Source Name Translation' }, + 'from_user' => { label=>'Source User Translation' }, +; + +my %features = ( + 'for' => 'Forward', + 'fnr' => 'Forward Not Registered', + 'fna' => 'Forward No Answer', + 'fbu' => 'Forward Busy', + 'dnd' => 'Do-Not-Disturb', + 'sim' => 'Simultaneous Ring', +); + +my %feature_param = ( + 'dnd' => 'n/a', + 'sim' => '$phonenum', +); tie my %options, 'Tie::IxHash', - 'login' => { label=>'NetSapiens tac2 API username' }, - 'password' => { label=>'NetSapiens tac2 API password' }, - 'url' => { label=>'NetSapiens tac2 URL' }, - 'domain' => { label=>'NetSapiens Domain' }, + 'login' => { label=>'NetSapiens tac2 User API username' }, + 'password' => { label=>'NetSapiens tac2 User API password' }, + 'url' => { label=>'NetSapiens tac2 User URL' }, + 'device_login' => { label=>'NetSapiens tac2 Device API username' }, + 'device_password' => { label=>'NetSapiens tac2 Device API password' }, + 'device_url' => { label=>'NetSapiens tac2 Device URL' }, + 'domain' => { label=>'NetSapiens Domain' }, + 'domain_no_tld' => { label=>'Omit TLD from domains', type=>'checkbox' }, + 'debug' => { label=>'Enable debugging', type=>'checkbox' }, + %subscriber_fields, + 'features' => { label => 'Default features', + type => 'select', + multiple => 1, + options => [ keys %features ], + option_label => sub { $features{$_[0]}; }, + }, + %registrar_fields, + %dialplan_fields, + 'did_countrycode' => { label=>'Use country code in DID destination', + type =>'checkbox' }, ; %info = ( - 'svc' => 'svc_phone', - 'desc' => 'Provision phone numbers to NetSapiens', - 'options' => \%options, - 'notes' => <<'END' -Requires installation of -REST::Client -from CPAN. + 'svc' => [qw( svc_phone part_device )], + 'desc' => 'Provision phone numbers to NetSapiens', + 'options' => \%options, + 'no_machine' => 1, + 'notes' => <<'END' END ); +# http://devguide.netsapiens.com/ + sub rebless { shift; } + +sub check_options { + my ($self, $options) = @_; + + my $rex = qr/$RE{URI}{HTTP}{-scheme => qr|https?|}/; # match any "http:" or "https:" URL + + for my $key (qw/url device_url/) { + if ($$options{$key} && ($$options{$key} !~ $rex)) { + return "Invalid (URL): " . $$options{$key}; + } + } + return ''; +} + + + sub ns_command { - my( $self, $method, $command, @args ) = @_; + my $self = shift; + $self->_ns_command('', @_); +} + +sub ns_device_command { + my $self = shift; + $self->_ns_command('device_', @_); +} + +sub _ns_command { + my( $self, $prefix, $method, $command ) = splice(@_,0,4); - eval 'use REST::Client'; - die $@ if $@; + # kludge to curb excessive paranoia in LWP 6.0+ + local $ENV{'PERL_LWP_SSL_VERIFY_HOSTNAME'} = 0; - my $ns = new REST::Client 'host'=>$self->option('url'); + my $ns = new REST::Client 'host'=>$self->option($prefix.'url'); - my $content = $method eq 'PUT' ? $ns->buildQuery( { @args } ) : ''; - $content =~ s/^\?//; + my @args = ( $command ); - warn $content; + if ( $method eq 'PUT' ) { + my $content = $ns->buildQuery( { @_ } ); + $content =~ s/^\?//; + push @args, $content; + } elsif ( $method eq 'GET' ) { + $args[0] .= $ns->buildQuery( { @_ } ); + } - my $auth = - encode_base64( $self->option('login'). ':'. $self->option('password') ); + warn "$me $method ". $self->option($prefix.'url'). join(', ', @args). "\n" + if $self->option('debug'); - $ns->$method( $command, $content, { 'Authorization' => "Basic $auth" } ); + my $auth = encode_base64( $self->option($prefix.'login'). ':'. + $self->option($prefix.'password') ); + push @args, { 'Authorization' => "Basic $auth" }; + $ns->$method( @args ); $ns; } +sub ns_domain { + my($self, $svc_phone) = (shift, shift); + my $domain = $svc_phone->domain || $self->option('domain'); + + $domain =~ s/\.\w{2,4}$// + if $self->option('domain_no_tld'); + + $domain; +} + sub ns_subscriber { my($self, $svc_phone) = (shift, shift); - my $domain = $self->option('domain'); + my $domain = $self->ns_domain($svc_phone); my $phonenum = $svc_phone->phonenum; "/domains_config/$domain/subscriber_config/$phonenum"; } +sub ns_registrar { + my($self, $svc_phone) = (shift, shift); + + $self->ns_subscriber($svc_phone). + '/registrar_config/'. $self->ns_devicename($svc_phone); +} + +sub ns_feature { + my($self, $svc_phone, $feature) = (shift, shift, shift); + + $self->ns_subscriber($svc_phone). + "/feature_config/$feature,*,*,*,*"; + +} + +sub ns_devicename { + my( $self, $svc_phone ) = (shift, shift); + + my $domain = $self->ns_domain($svc_phone); + #my $countrycode = $svc_phone->countrycode; + my $phonenum = $svc_phone->phonenum; + + #"sip:$countrycode$phonenum\@$domain"; + "sip:$phonenum\@$domain"; +} + +sub ns_dialplan { + my($self, $svc_phone) = (shift, shift); + + my $countrycode = $svc_phone->countrycode || '1'; + my $phonenum = $svc_phone->phonenum; + # Only in the dialplan destination, nowhere else + if ( $self->option('did_countrycode') ) { + $phonenum = $countrycode . $phonenum; + } + + #"/dialplans/DID+Table/dialplan_config/sip:$countrycode$phonenum\@*" + "/domains_config/admin-only/dialplans/DID+Table/dialplan_config/sip:$phonenum\@*,*,*,*,*,*,*"; +} + +sub ns_device { + my($self, $svc_phone, $phone_device ) = (shift, shift, shift); + + #my $countrycode = $svc_phone->countrycode; + #my $phonenum = $svc_phone->phonenum; + + "/phones_config/". lc($phone_device->mac_addr); +} + sub ns_create_or_update { my($self, $svc_phone, $dial_policy) = (shift, shift, shift); - my $domain = $self->option('domain'); - my $phonenum = $svc_phone->phonenum; + my $domain = $self->ns_domain($svc_phone); + #my $countrycode = $svc_phone->countrycode; + my $phonenum = $svc_phone->phonenum; + + #deal w/unaudited netsapiens services? + my $cust_main = $svc_phone->cust_svc->cust_pkg->cust_main; my( $firstname, $lastname ); if ( $svc_phone->phone_name =~ /^\s*(\S+)\s+(\S.*\S)\s*$/ ) { $firstname = $1; $lastname = $2; } else { - #deal w/unaudited netsapiens services? - my $cust_main = $svc_phone->cust_svc->cust_pkg->cust_main; $firstname = $cust_main->get('first'); $lastname = $cust_main->get('last'); } + my ($email) = ($cust_main->invoicing_list_emailonly, ''); + my $custnum = $cust_main->custnum; + + ### + # Piece 1 (already done) - User creation + ### + + $phonenum =~ /^(\d{3})/; + my $area_code = $1; + my $ns = $self->ns_command( 'PUT', $self->ns_subscriber($svc_phone), - 'subscriber_login' => $phonenum.'@'.$domain, - 'firstname' => $firstname, #4? - 'lastname' => $lastname, #5? - 'subscriber_pin' => $svc_phone->pin, #6? - 'dial_plan' => 'Default', #config? #7? - 'dial_policy' => $dial_policy, #8? -#no_answer_timeout30 -# simultaneous_ringyes -# gmt_offset-8 -# aor_schemesip: -# do_not_disturbyes -# email_vmail -# data_limit0 -# screen -# last_update2008-10-01 12:19:01.0 -# domain_diryes -# callid_name[*] -# admin_vmailyes -# subscriber_name -# rcv_broadcast -# directory_order1 -# accept -# rating_required -# date_created2008-02-22 08:38:01 -# message_waiting -# rate -# directory_listingno -# time_zoneUS/Pacific -# forward_no_answeryes -# vmail_sort_lifo -# modeover-capacity -# subscriber_groupn/a -# vmail_say_time -# presenceinactive -# directory_match826 -# language -# forward_busyyes -# callid_nmbr[*] -# vmail -# subscriber_login1007@vbox.netsapiens.com -# rejectyes -# forwardyes -# vmail_say_cidno -# email_address -# greeting_index - ); + 'subscriber_login' => $phonenum.'@'.$domain, + 'firstname' => $firstname, + 'lastname' => $lastname, + 'subscriber_pin' => $svc_phone->pin, + 'callid_name' => "$firstname $lastname", + 'callid_nmbr' => $phonenum, + 'callid_emgr' => $phonenum, + 'email_address' => $email, + 'area_code' => $area_code, + 'srv_code' => $custnum, + 'date_created' => time2str('%Y-%m-%d %H:%M:%S', time), + $self->options_named(keys %subscriber_fields), + # allow this to be overridden for suspend + ( $dial_policy ? ('dial_policy' => $dial_policy) : () ), + ); if ( $ns->responseCode !~ /^2/ ) { return $ns->responseCode. ' '. join(', ', $self->ns_parse_response( $ns->responseContent ) ); } + ### + # Piece 1.5 - feature creation + ### + foreach $feature (split /\s+/, $self->option('features') ) { + + my $param= exists($feature_param{$feature}) ? $feature_param{$feature} : ''; + $param = $phonenum if $param eq '$phonenum'; + + my $nsf = $self->ns_command( 'PUT', $self->ns_feature($svc_phone, $feature), + 'control' => 'd', #User Control, disable + 'expires' => 'never', + #'ts' => '', #? + 'parameters' => $param, + 'hour_match' => '*', + 'time_frame' => '*', + 'activation' => 'now', + ); + + if ( $nsf->responseCode !~ /^2/ ) { + return $nsf->responseCode. ' '. + join(', ', $self->ns_parse_response( $ns->responseContent ) ); + } + + } + + ### + # Piece 2 - sip device creation + ### + + my $ns2 = $self->ns_command( 'PUT', $self->ns_registrar($svc_phone), + 'termination_match' => $self->ns_devicename($svc_phone), + 'authentication_key'=> $svc_phone->sip_password, + 'srv_code' => $custnum, + $self->options_named(keys %registrar_fields), + ); + + if ( $ns2->responseCode !~ /^2/ ) { + return $ns2->responseCode. ' '. + join(', ', $self->ns_parse_response( $ns2->responseContent ) ); + } + + ### + # Piece 3 - DID mapping to user + ### + + my $ns3 = $self->ns_command( 'PUT', $self->ns_dialplan($svc_phone), + 'to_user' => $phonenum, + 'to_host' => $domain, + 'plan_description' => "$custnum: $lastname, $firstname", #config? + $self->options_named(keys %dialplan_fields), + ); + + if ( $ns3->responseCode !~ /^2/ ) { + return $ns3->responseCode. ' '. + join(', ', $self->ns_parse_response( $ns3->responseContent ) ); + } + ''; } sub ns_delete { my($self, $svc_phone) = (shift, shift); + # do the create steps in reverse order, though I'm not sure it matters + + my $ns3 = $self->ns_command( 'DELETE', $self->ns_dialplan($svc_phone) ); + + if ( $ns3->responseCode !~ /^2/ ) { + return $ns3->responseCode. ' '. + join(', ', $self->ns_parse_response( $ns3->responseContent ) ); + } + + my $ns2 = $self->ns_command( 'DELETE', $self->ns_registrar($svc_phone) ); + + if ( $ns2->responseCode !~ /^2/ ) { + return $ns2->responseCode. ' '. + join(', ', $self->ns_parse_response( $ns2->responseContent ) ); + } + my $ns = $self->ns_command( 'DELETE', $self->ns_subscriber($svc_phone) ); if ( $ns->responseCode !~ /^2/ ) { @@ -148,20 +352,18 @@ sub ns_delete { sub ns_parse_response { my( $self, $content ) = ( shift, shift ); + #try to screen-scrape something useful tie my %hash, Tie::IxHash; - #while ( $content =~ s/^.*?

\s*(.+?)<\/b>\s*<(\w+)>(.+?)<\/\2><\/p>//i ) { while ( $content =~ s/^.*?

\s*(.+?)<\/b>\s*(.+?)\s*<\/p>//is ) { ( $hash{$1} = $2 ) =~ s/^\s*<(\w+)>(.+?)<\/\1>/$2/is; } - #warn $content; #probably useless - %hash; } sub _export_insert { my($self, $svc_phone) = (shift, shift); - $self->ns_create_or_update($svc_phone, 'Permit All'); + $self->ns_create_or_update($svc_phone); } sub _export_replace { @@ -179,7 +381,7 @@ sub _export_delete { sub _export_suspend { my( $self, $svc_phone ) = (shift, shift); - $self->ns_create_or_udpate($svc_phone, 'Deny'); + $self->ns_create_or_update($svc_phone, 'Deny'); } sub _export_unsuspend { @@ -188,6 +390,78 @@ sub _export_unsuspend { $self->_export_insert($svc_phone); } +sub export_device_insert { + my( $self, $svc_phone, $phone_device ) = (shift, shift, shift); + + if ( $FS::svc_Common::noexport_hack ) { + carp 'export_device_insert() suppressed by noexport_hack' + if $self->option('debug'); + return; + } + + my $domain = $self->ns_domain($svc_phone); + my $countrycode = $svc_phone->countrycode; + my $phonenum = $svc_phone->phonenum; + + my $ns = $self->ns_device_command( + 'PUT', $self->ns_device($svc_phone, $phone_device), + 'line1_enable' => 'yes', + 'device1' => $self->ns_devicename($svc_phone), + 'line1_ext' => $phonenum, +, + #'line2_enable' => 'yes', + #'device2' => + #'line2_ext' => + + #'notes' => + 'server' => 'SiPbx', + 'domain' => $domain, + + 'brand' => $phone_device->part_device->devicename, + + ); + + if ( $ns->responseCode !~ /^2/ ) { + return $ns->responseCode. ' '. + join(', ', $self->ns_parse_response( $ns->responseContent ) ); + } + + ''; + +} + +sub export_device_delete { + my( $self, $svc_phone, $phone_device ) = (shift, shift, shift); + + if ( $FS::svc_Common::noexport_hack ) { + carp 'export_device_delete() suppressed by noexport_hack' + if $self->option('debug'); + return; + } + + my $ns = $self->ns_device_command( + 'DELETE', $self->ns_device($svc_phone, $phone_device), + ); + + if ( $ns->responseCode !~ /^2/ ) { + return $ns->responseCode. ' '. + join(', ', $self->ns_parse_response( $ns->responseContent ) ); + } + + ''; + +} + + +sub export_device_replace { + my( $self, $svc_phone, $new_phone_device, $old_phone_device ) = + (shift, shift, shift, shift); + + #? + $self->export_device_insert( $svc_phone, $new_phone_device ); + +} + sub export_links { my($self, $svc_phone, $arrayref) = (shift, shift, shift); #push @$arrayref, qq!