#! /usr/bin/perl -w # # Written by Oron Peled # Copyright (C) 2007, Xorcom # This program is free software; you can redistribute and/or # modify it under the same terms as Perl itself. # # $Id$ # use strict; use File::Basename; BEGIN { my $dir = dirname($0); unshift(@INC, "$dir", "$dir/zconf"); } use Zaptel; use Zaptel::Xpp; use Zaptel::Config::Defaults; my %default_context = ( FXO => 'from-pstn', FXS => 'from-internal', IN => 'astbank-input', OUT => 'astbank-output', BRI_TE => 'from-pstn', BRI_NT => 'from-internal', PRI_TE => 'from-pstn', PRI_NT => 'from-internal', ); my %default_group = ( FXO => 0, FXS => 5, IN => 5, # FIXME: no need for group? OUT => 5, # FIXME: no need for group? BRI_TE => 0, BRI_NT => 5, PRI_TE => 0, PRI_NT => 5, ); my $fxs_default_start = 'ls'; my %default_zaptel_signalling = ( FXO => 'fxsks', FXS => "fxo$fxs_default_start", IN => "fxo$fxs_default_start", OUT => "fxo$fxs_default_start", ); my %default_zapata_signalling = ( FXO => 'fxs_ks', FXS => "fxo_$fxs_default_start", IN => "fxo_$fxs_default_start", OUT => "fxo_$fxs_default_start", ); my $base_exten = 4000; my $fxs_immediate = 'no'; my $lc_country = 'us'; my $loadzone = $lc_country; my $defaultzone = $lc_country; my $bri_sig_style = 'bri_ptmp'; my %zaptel_default_vars = ( base_exten => \$base_exten, fxs_immediate => \$fxs_immediate, fxs_default_start => \$fxs_default_start, lc_country => [ \$loadzone, \$defaultzone, ], context_lines => \$default_context{FXO}, context_phones => \$default_context{FXS}, context_input => \$default_context{IN}, context_output => \$default_context{OUT}, group_phones => [ \$default_group{FXS}, \$default_group{IN}, \$default_group{OUT}, ], group_lines => \$default_group{FXO}, ZAPBRI_SIGNALLING => \$bri_sig_style, ); sub map_zaptel_defaults { my %defaults = @_; foreach my $name (keys %defaults) { my $val = $defaults{$name}; my $ref = $zaptel_default_vars{$name}; my $type = ref $ref; my @vars = (); # Some broken shells (msh) export even variables # That where not defined. Work around that. next unless defined $val && $val ne ''; if($type eq 'SCALAR') { @vars = ($ref); } elsif($type eq 'ARRAY') { @vars = @$ref; } else { die "$0: Don't know how to map '$name' (type=$type)\n"; } foreach my $v (@vars) { $$v = $val; } } } my $zapconf_file; my $zapata_file; my @spans = Zaptel::spans(); sub bchan_range($) { my $span = shift || die; my $first_chan = ($span->chans())[0]; my $first_num = $first_chan->num(); my $range_start = $first_num; my @range; my $prev = undef; die unless $span->is_digital(); foreach my $c (@{$span->bchan_list()}) { my $curr = $c + $first_num; if(!defined($prev)) { $prev = $curr; } elsif($curr != $prev + 1) { push(@range, sprintf("%d-%d", $range_start, $prev)); $range_start = $curr; } $prev = $curr; } if($prev >= $first_num) { push(@range, sprintf("%d-%d", $range_start, $prev)); } return join(',', @range); } sub gen_zaptel_signalling($) { my $chan = shift || die; my $type = $chan->type; my $num = $chan->num; die "channel $num type $type is not an analog channel\n" if $chan->is_digital(); if($type eq 'EMPTY') { printf "# channel %d, %s, no module.\n", $num, $chan->fqn; return; } my $sig = $default_zaptel_signalling{$type} || die "unknown default zaptel signalling for chan $chan type $type"; if ($type eq 'IN') { printf "# astbanktype: input\n"; } elsif ($type eq 'OUT') { printf "# astbanktype: output\n"; } printf "$sig=$num\n"; } my $bri_te_last_timing = 1; sub gen_zaptel_digital($) { my $span = shift || die; my $num = $span->num() || die; die "Span #$num is analog" unless $span->is_digital(); my $termtype = $span->termtype() || die "$0: Span #$num -- unkown termtype [NT/TE]\n"; my $timing; my $lbo = 1; my $framing = $span->framing() || die "$0: No framing information for span #$num\n"; my $coding = $span->coding() || die "$0: No coding information for span #$num\n"; my $span_yellow = $span->yellow(); $span_yellow = (defined $span_yellow) ? ",$span_yellow" : ''; $timing = ($termtype eq 'NT') ? 0 : $bri_te_last_timing++; printf "span=%d,%d,%d,%s,%s\n", $num, $timing, $lbo, $framing, "$coding$span_yellow"; printf "# termtype: %s\n", lc($termtype); printf "bchan=%s\n", bchan_range($span); my $dchan = $span->dchan(); printf "dchan=%d\n", $dchan->num(); } sub gen_zaptelconf($) { my $file = shift || die; rename "$file", "$file.bak" or $! == 2 # ENOENT (No dependency on Errno.pm) or die "Failed to backup old config: $!\n"; open(F, ">$file") || die "$0: Failed to open $file: $!\n"; my $old = select F; printf "# Autogenerated by %s on %s -- do not hand edit\n", $0, scalar(localtime); print <<"HEAD"; # Zaptel Configuration File # # This file is parsed by the Zaptel Configurator, ztcfg # HEAD foreach my $span (@spans) { printf "# Span %d: %s %s\n", $span->num, $span->name, $span->description; if($span->is_digital()) { gen_zaptel_digital($span); } else { foreach my $chan ($span->chans()) { if(1 || !defined $chan->type) { my $type = $chan->probe_type; my $num = $chan->num; die "Failed probing type for channel $num" unless defined $type; $chan->type($type); } gen_zaptel_signalling($chan); } } print "\n"; } print <<"TAIL"; # Global data loadzone = $loadzone defaultzone = $defaultzone TAIL close F; select $old; } sub gen_zapata_digital($) { my $span = shift || die; my $num = $span->num() || die; die "Span #$num is analog" unless $span->is_digital(); my $type = $span->type() || die "$0: Span #$num -- unkown type\n"; my $termtype = $span->termtype() || die "$0: Span #$num -- unkown termtype [NT/TE]\n"; my $group = $default_group{"$type"}; my $context = $default_context{"$type"}; die "$0: missing default group (termtype=$termtype)\n" unless defined($group); die "$0: missing default context\n" unless $context; my $sig = $span->signalling || die "missing signalling info for span #$num type $type"; grep($bri_sig_style eq $_, 'bri', 'bri_ptmp', 'pri') or die "unknown signalling style for BRI"; if($bri_sig_style eq 'bri_ptmp') { $sig .= '_ptmp'; } $group .= "," . (10 + $num); # Invent unique group per span printf "group=$group\n"; printf "context=$context\n"; printf "switchtype = %s\n", $span->switchtype; printf "signalling = %s\n", $sig; printf "channel => %s\n", bchan_range($span); printf "group=\ncontext=default\n"; } sub gen_zapata_channel($) { my $chan = shift || die; my $type = $chan->type; my $num = $chan->num; die "channel $num type $type is not an analog channel\n" if $chan->is_digital(); my $exten = $base_exten + $num; my $sig = $default_zapata_signalling{$type}; my $context = $default_context{$type}; my $group = $default_group{$type}; my $callerid; my $immediate; return if $type eq 'EMPTY'; die "missing default_zapata_signalling for chan #$num type $type" unless $sig; $callerid = ($type eq 'FXO') ? 'asreceived' : sprintf "\"Channel %d\" <%04d>", $num, $exten; if($type eq 'IN') { $immediate = 'yes'; } # FIXME: $immediage should not be set for 'OUT' channels, but meanwhile # it's better to be compatible with genzaptelconf $immediate = 'yes' if $fxs_immediate eq 'yes' and $sig =~ /^fxo_/; my $signalling = $chan->signalling; $signalling = " " . $signalling if $signalling; my $info = $chan->info; $info = " " . $info if $info; printf ";;; line=\"%d %s%s%s\"\n", $num, $chan->fqn, $signalling, $info; printf "signalling=$sig\n"; printf "callerid=$callerid\n"; printf "mailbox=%04d\n", $exten unless $type eq 'FXO'; if(defined $group) { printf "group=$group\n"; } printf "context=$context\n"; printf "immediate=$immediate\n" if defined $immediate; printf "channel => %d\n", $num; # Reset following values to default printf "callerid=\n"; printf "mailbox=\n" unless $type eq 'FXO'; if(defined $group) { printf "group=\n"; } printf "context=default\n"; printf "immediate=no\n" if defined $immediate; print "\n"; } sub gen_zapataconf($) { my $file = shift || die; rename "$file", "$file.bak" or $! == 2 # ENOENT (No dependency on Errno.pm) or die "Failed to backup old config: $!\n"; open(F, ">$file") || die "$0: Failed to open $file: $!\n"; my $old = select F; printf "; Autogenerated by %s on %s -- do not hand edit\n", $0, scalar(localtime); print <<"HEAD"; ; Zaptel Channels Configurations (zapata.conf) ; ; This is not intended to be a complete zapata.conf. Rather, it is intended ; to be #include-d by /etc/zapata.conf that will include the global settings ; HEAD foreach my $span (@spans) { printf "; Span %d: %s %s\n", $span->num, $span->name, $span->description; if($span->is_digital()) { gen_zapata_digital($span); } else { foreach my $chan ($span->chans()) { gen_zapata_channel($chan); } } print "\n"; } close F; select $old; } sub set_defaults { my $zaptel_boot_debian = $ENV{ZAPTEL_BOOT_DEBIAN} || "/etc/default/zaptel"; my $zaptel_boot_fedora = $ENV{ZAPTEL_BOOT_FEDORA} || "/etc/sysconfig/zaptel"; # Source default files my %source_defaults; foreach my $defaults ($zaptel_boot_debian, $zaptel_boot_fedora) { %source_defaults = Zaptel::Config::Defaults::do_source( $defaults, keys(%zaptel_default_vars)) if -r $defaults; } map_zaptel_defaults(%source_defaults); $zapconf_file = $ENV{ZAPCONF_FILE} || "/etc/zaptel.conf"; $zapata_file = $ENV{ZAPATA_FILE} || "/etc/asterisk/zapata-channels.conf"; } set_defaults; gen_zaptelconf $zapconf_file; gen_zapataconf $zapata_file; __END__ =head1 NAME zapconf - Generate configuration for zaptel channels. =head1 SYNOPSIS zapconf =head1 DESCRIPTION This script generate configuration files for Zaptel hardware. Currently it generate two files: =over 4 =item /etc/zaptel.conf Configuration for ztcfg(1). It's location may be overriden by the environment variable ZAPCONF_FILE. =item /etc/asterisk/zapata-channels.conf Configuration for asterisk(1). It should be included in the main /etc/asterisk/zapata.conf. It's location may be overriden by the environment variable ZAPATA_FILE. =back