isdn4k-utils/isdnlog/isdnconf/isdnconf.c

968 lines
25 KiB
C

/* $Id: isdnconf.c,v 1.40 1999/10/25 18:33:15 akool Exp $
*
* ISDN accounting for isdn4linux. (Report-module)
*
* Copyright 1996, 1999 by Stefan Luethje (luethje@sl-gw.lake.de)
* and Andreas Kool (akool@isdn4linux.de)
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2, or (at your option)
* any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*
* $Log: isdnconf.c,v $
* Revision 1.40 1999/10/25 18:33:15 akool
* isdnlog-3.57
* WARNING: Experimental version!
* Please use isdnlog-3.56 for production systems!
*
* Revision 1.39 1999/10/22 19:57:59 akool
* isdnlog-3.56 (for Karsten)
*
* Revision 1.38 1999/09/26 10:55:20 akool
* isdnlog-3.55
* - Patch from Oliver Lauer <Oliver.Lauer@coburg.baynet.de>
* added hup3 to option file
* - changed country-de.dat to ISO 3166 Countrycode / Airportcode
*
* Revision 1.37 1999/08/20 19:28:05 akool
* isdnlog-3.45
* - removed about 1 Mb of (now unused) data files
* - replaced areacodes and "vorwahl.dat" support by zone databases
* - fixed "Sonderrufnummern"
* - rate-de.dat :: V:1.10-Germany [20-Aug-1999 21:23:27]
*
* Revision 1.36 1999/07/03 10:24:02 akool
* fixed Makefile
*
* Revision 1.35 1999/06/28 19:15:53 akool
* isdnlog Version 3.38
* - new utility "isdnrate" started
*
* Revision 1.34 1999/06/22 19:40:37 akool
* zone-1.1 fixes
*
* Revision 1.33 1999/06/16 19:12:21 akool
* isdnlog Version 3.34
* fixed some memory faults
*
* Revision 1.32 1999/06/15 20:03:46 akool
* isdnlog Version 3.33
* - big step in using the new zone files
* - *This*is*not*a*production*ready*isdnlog*!!
* - Maybe the last release before the I4L meeting in Nuernberg
*
* Revision 1.31 1999/06/13 14:07:28 akool
* isdnlog Version 3.32
*
* - new option "-U1" (or "ignoreCOLP=1") to ignore CLIP/COLP Frames
* - TEI management decoded
*
* Revision 1.30 1999/06/09 19:58:12 akool
* isdnlog Version 3.31
* - Release 0.91 of zone-Database (aka "Verzonungstabelle")
* - "rate-de.dat" V:1.02-Germany [09-Jun-1999 21:45:26]
*
* Revision 1.29 1999/06/03 18:50:10 akool
* isdnlog Version 3.30
* - rate-de.dat V:1.02-Germany [03-Jun-1999 19:49:22]
* - small fixes
*
* Revision 1.28 1999/06/01 19:33:27 akool
* rate-de.dat V:1.02-Germany [01-Jun-1999 20:52:32]
*
* Revision 1.27 1999/05/22 10:18:18 akool
* isdnlog Version 3.29
*
* - processing of "sonderrufnummern" much more faster
* - detection for sonderrufnummern of other provider's implemented
* (like 01929:FreeNet)
* - Patch from Oliver Lauer <Oliver.Lauer@coburg.baynet.de>
* - Patch from Markus Schoepflin <schoepflin@ginit.de>
* - easter computing corrected
* - rate-de.dat 1.02-Germany [22-May-1999 11:37:33] (from rate-CVS)
* - countries-de.dat 1.02-Germany [22-May-1999 11:37:47] (from rate-CVS)
* - new option "-B" added (see README)
* (using "isdnlog -B16 ..." isdnlog now works in the Netherlands!)
*
* Revision 1.26 1999/05/13 11:39:09 akool
* isdnlog Version 3.28
*
* - "-u" Option corrected
* - "ausland.dat" removed
* - "countries-de.dat" fully integrated
* you should add the entry
* "COUNTRYFILE = /usr/lib/isdn/countries-de.dat"
* into section "[ISDNLOG]" of your config file!
* - rate-de.dat V:1.02-Germany [13-May-1999 12:26:24]
* - countries-de.dat V:1.02-Germany [13-May-1999 12:26:26]
*
* Revision 1.25 1999/05/09 18:24:10 akool
* isdnlog Version 3.25
*
* - README: isdnconf: new features explained
* - rate-de.dat: many new rates from the I4L-Tarifdatenbank-Crew
* - added the ability to directly enter a country-name into "rate-xx.dat"
*
* Revision 1.24 1999/05/04 19:32:23 akool
* isdnlog Version 3.24
*
* - fully removed "sondernummern.c"
* - removed "gcc -Wall" warnings in ASN.1 Parser
* - many new entries for "rate-de.dat"
* - better "isdnconf" utility
*
* Revision 1.23 1999/04/30 19:07:46 akool
* isdnlog Version 3.23
*
* - changed LCR probing duration from 181 seconds to 153 seconds
* - "rate-de.dat" filled with May, 1. rates
*
* Revision 1.22 1999/04/19 19:24:02 akool
* isdnlog Version 3.18
*
* - countries-at.dat added
* - spelling corrections in "countries-de.dat" and "countries-us.dat"
* - LCR-function of isdnconf now accepts a duration (isdnconf -c .,duration)
* - "rate-at.dat" and "rate-de.dat" extended/fixed
* - holiday.c and rate.c fixed (many thanks to reinelt@eunet.at)
*
* Revision 1.21 1999/04/17 14:10:56 akool
* isdnlog Version 3.17
*
* - LCR functions of "isdnconf" fixed
* - HINT's fixed
* - rate-de.dat: replaced "1-5" with "W" and "6-7" with "E"
*
* Revision 1.20 1999/04/15 19:14:29 akool
* isdnlog Version 3.15
*
* - reenable the least-cost-router functions of "isdnconf"
* try "isdnconf -c <areacode>" or even "isdnconf -c ."
* - README: "rate-xx.dat" documented
* - small fixes in processor.c and rate.c
* - "rate-de.dat" optimized
* - splitted countries.dat into countries-de.dat and countries-us.dat
*
* Revision 1.19 1999/04/14 13:16:18 akool
* isdnlog Version 3.14
*
* - "make install" now install's "rate-xx.dat", "rate.conf" and "ausland.dat"
* - "holiday-xx.dat" Version 1.1
* - many rate fixes (Thanks again to Michael Reinelt <reinelt@eunet.at>)
*
* Revision 1.18 1999/04/10 16:35:14 akool
* isdnlog Version 3.13
*
* WARNING: This is pre-ALPHA-dont-ever-use-Code!
* "tarif.dat" (aka "rate-xx.dat"): the next generation!
*
* You have to do the following to test this version:
* cp /usr/src/isdn4k-utils/isdnlog/holiday-de.dat /etc/isdn
* cp /usr/src/isdn4k-utils/isdnlog/rate-de.dat /usr/lib/isdn
* cp /usr/src/isdn4k-utils/isdnlog/samples/rate.conf.de /etc/isdn/rate.conf
*
* After that, add the following entries to your "/etc/isdn/isdn.conf" or
* "/etc/isdn/callerid.conf" file:
*
* [ISDNLOG]
* SPECIALNUMBERS = /usr/lib/isdn/sonderrufnummern.dat
* HOLIDAYS = /usr/lib/isdn/holiday-de.dat
* RATEFILE = /usr/lib/isdn/rate-de.dat
* RATECONF = /etc/isdn/rate.conf
*
* Please replace any "de" with your country code ("at", "ch", "nl")
*
* Good luck (Andreas Kool and Michael Reinelt)
*
* Revision 1.17 1999/04/03 12:46:54 akool
* - isdnlog Version 3.12
* - "%B" tag in ILABEL/OLABEL corrected
* - isdnlog now register's incoming calls when there are no free B-channels
* (idea from sergio@webmedia.es)
* - better "samples/rate.conf.de" (suppress provider without true call-by-call)
* - "tarif.dat" V:1.17 [03-Apr-99]
* - Added EWE-Tel rates from Reiner Klaproth <rk1@msjohan.dd.sn.schule.de>
* - isdnconf can now be used to generate a Least-cost-router table
* (try "isdnconf -c .")
* - isdnlog now simulate a RELEASE COMPLETE if nothing happpens after a SETUP
* - CHARGEMAX Patches from Oliver Lauer <Oliver.Lauer@coburg.baynet.de>
*
* Revision 1.16 1999/03/24 19:37:38 akool
* - isdnlog Version 3.10
* - moved "sondernnummern.c" from isdnlog/ to tools/
* - "holiday.c" and "rate.c" integrated
* - NetCologne rates from Oliver Flimm <flimm@ph-cip.uni-koeln.de>
* - corrected UUnet and T-Online rates
*
* Revision 1.15 1999/03/20 14:32:56 akool
* - isdnlog Version 3.08
* - more tesion)) Tarife from Michael Graw <Michael.Graw@bartlmae.de>
* - use "bunzip -f" from Franz Elsner <Elsner@zrz.TU-Berlin.DE>
* - show another "cheapest" hint if provider is overloaded ("OVERLOAD")
* - "make install" now makes the required entry
* [GLOBAL]
* AREADIFF = /usr/lib/isdn/vorwahl.dat
* - README: Syntax description of the new "rate-at.dat"
* - better integration of "sondernummern.c" from mario.joussen@post.rwth-aachen.de
* - server.c: buffer overrun fix from Michael.Weber@Post.RWTH-Aachen.DE (Michael Weber)
*
* Revision 1.14 1999/03/15 21:27:48 akool
* - isdnlog Version 3.06
* - README: explain some terms about LCR, corrected "-c" Option of "isdnconf"
* - isdnconf: added a small LCR-feature - simply try "isdnconf -c 069"
* - isdnlog: dont change CHARGEINT, if rate is't known!
* - sonderrufnummern 1.02 [15-Mar-99] :: added WorldCom
* - tarif.dat 1.09 [15-Mar-99] :: added WorldCom
* - isdnlog now correctly handles the new "Ortstarif-Zugang" of UUnet
*
* Revision 1.13 1999/03/07 18:18:42 akool
* - new 01805 tarif of DTAG
* - new March 1999 tarife
* - added new provider "01051 Telecom"
* - fixed a buffer overrun from Michael Weber <Michael.Weber@Post.RWTH-Aachen.DE>
* - fixed a bug using "sondernnummern.c"
* - fixed chargeint change over the time
* - "make install" now install's "sonderrufnummern.dat", "tarif.dat",
* "vorwahl.dat" and "tarif.conf"! Many thanks to
* Mario Joussen <mario.joussen@post.rwth-aachen.de>
* - Euracom Frames would now be ignored
* - fixed warnings in "sondernnummern.c"
* - "10plus" messages no longer send to syslog
*
* Revision 1.12 1998/09/22 20:59:08 luethje
* isdnrep: -fixed wrong provider report
* -fixed wrong html output for provider report
* -fixed strange html output
* kisdnlog: -fixed "1001 message window" bug ;-)
*
* Revision 1.11 1998/05/11 19:43:43 luethje
* Some changes for "vorwahlen.dat"
*
* Revision 1.10 1998/05/10 22:11:52 luethje
* Added support for VORWAHLEN2.EXE
*
* Revision 1.9 1997/05/25 19:40:53 luethje
* isdnlog: close all files and open again after kill -HUP
* isdnrep: support vbox version 2.0
* isdnconf: changes by Roderich Schupp <roderich@syntec.m.EUnet.de>
* conffile: ignore spaces at the end of a line
*
* Revision 1.8 1997/05/05 21:21:42 luethje
* bugfix for option -M
*
* Revision 1.6 1997/04/15 00:20:01 luethje
* replace variables: some bugfixes, README comleted
*
* Revision 1.5 1997/04/10 23:32:15 luethje
* Added the feature, that environment variables are allowed in the config files.
*
* Revision 1.4 1997/04/06 21:06:08 luethje
* problem with empty/not existing file resolved.
*
* Revision 1.3 1997/04/03 22:36:23 luethje
* splitt the file isdn.conf into callerid.conf and ~/.isdn (for editing).
*
* Revision 1.2 1997/03/23 20:58:31 luethje
* some bugfixes
*
*/
#define _ISDN_CONF_C_
#include "isdnconf.h"
int print_in_modules(const char *fmt, ...);
int print_msg(int Level, const char *fmt, ...);
int add_data(section **conf_dat);
int delete_data(char *, char *, section **conf_dat);
int look_data(section **conf_dat);
int find_data(char *_alias, char *_number, section *conf_dat);
const char* make_word(const char *in);
char* tmp_dup(const char *in);
int add_line(section **Ptr, const char *Name);
/*****************************************************************************/
static int and = 0;
static int add = 0;
static int del = 0;
static int msn = 0;
static int quiet = 0;
static int short_out= 0;
static int long_out= 0;
static int oneentry= 0;
static int isdnmon = 0;
static int match_flags = F_NO_HOLE_WORD;
static char areacode[SHORT_STRING_SIZE] = "";
static char si[SHORT_STRING_SIZE];
static char number[BUFSIZ] = "";
static char alias[BUFSIZ] = "";
static char conffile[BUFSIZ];
static char callerfile[BUFSIZ];
/*****************************************************************************/
void info(int chan, int reason, int state, char *msg)
{
/* DUMMY - dont needed here! */
} /* info */
/*****************************************************************************/
int add_line(section **Ptr, const char *Name)
{
char buf[BUFSIZ] = "";
if (fgets(buf,BUFSIZ,stdin) != NULL)
{
if (strlen(buf) > 1)
{
buf[strlen(buf)-1] = '\0';
if (Set_Entry(*Ptr,NULL,tmp_dup(Name),buf,C_WARN) == NULL)
{
print_msg(PRT_ERR,"%s\n","Can not allocate memory!");
return -1;
}
}
}
else
{
free_section(*Ptr);
*Ptr = NULL;
return -1;
}
return 0;
}
/*****************************************************************************/
int add_data(section **conf_dat)
{
int Cnt = 0;
section *NewPtr = NULL;
section *SubPtr = NULL;
while(1)
{
NewPtr = NULL;
if (Set_Section(&NewPtr,tmp_dup(msn?CONF_SEC_MSN:CONF_SEC_NUM),C_NOT_UNIQUE) == NULL)
{
print_msg(PRT_ERR,"%s\n","Can not allocate memory!");
return -1;
}
if (!quiet)
print_msg(PRT_NORMAL,"%s:\t\t",make_word(CONF_ENT_ALIAS));
if (add_line(&NewPtr,CONF_ENT_ALIAS))
break;
if (!quiet)
print_msg(PRT_NORMAL,"%s:\t\t",make_word(CONF_ENT_NUM));
if (add_line(&NewPtr,CONF_ENT_NUM))
break;
if (!quiet)
print_msg(PRT_NORMAL,"%s:\t\t",CONF_ENT_SI);
if (add_line(&NewPtr,CONF_ENT_SI))
break;
if (!quiet)
print_msg(PRT_NORMAL,"%s:\t\t",make_word(CONF_ENT_ZONE));
if (add_line(&NewPtr,CONF_ENT_ZONE))
break;
if (!quiet)
print_msg(PRT_NORMAL,"%s:\t",make_word(CONF_ENT_INTFAC));
if (add_line(&NewPtr,CONF_ENT_INTFAC))
break;
while(1)
{
SubPtr = NULL;
if (Set_Section(&SubPtr,tmp_dup(CONF_SEC_FLAG),C_NOT_UNIQUE) == NULL)
{
print_msg(PRT_ERR,"%s\n","Can not allocate memory!");
return -1;
}
if (!quiet)
print_msg(PRT_NORMAL,"%s:\t\t",make_word(CONF_ENT_FLAGS));
if (add_line(&SubPtr,CONF_ENT_FLAGS))
break;
if (!quiet)
print_msg(PRT_NORMAL,"%s:\t",make_word(CONF_ENT_PROG));
if (add_line(&SubPtr,CONF_ENT_PROG))
break;
if (!quiet)
print_msg(PRT_NORMAL,"%s:\t\t",make_word(CONF_ENT_USER));
if (add_line(&SubPtr,CONF_ENT_USER))
break;
if (!quiet)
print_msg(PRT_NORMAL,"%s:\t\t",make_word(CONF_ENT_GROUP));
if (add_line(&SubPtr,CONF_ENT_GROUP))
break;
if (!quiet)
print_msg(PRT_NORMAL,"%s:\t",make_word(CONF_ENT_INTVAL));
if (add_line(&SubPtr,CONF_ENT_INTVAL))
break;
if (!quiet)
print_msg(PRT_NORMAL,"%s:\t\t",make_word(CONF_ENT_TIME));
if (add_line(&SubPtr,CONF_ENT_TIME))
break;
Set_SubSection(NewPtr,tmp_dup(CONF_ENT_START),SubPtr,C_APPEND | C_WARN);
}
if (!quiet)
print_msg(PRT_NORMAL,"\n\n");
while(*conf_dat != NULL)
conf_dat = &((*conf_dat)->next);
*conf_dat = NewPtr;
Cnt++;
}
if (!quiet)
print_msg(PRT_NORMAL,"\n");
return Cnt;
}
/*****************************************************************************/
int delete_data(char * _alias , char * _number, section **conf_dat)
{
short_out = 1;
if (!quiet)
print_msg(PRT_NORMAL,"Following entry deleted!\n");
if (!quiet)
find_data(_alias,_number,*conf_dat);
Del_Section(conf_dat,NULL);
return 0;
}
/*****************************************************************************/
int find_data(char *_alias, char *_number, section *conf_dat)
{
auto char *ptr;
auto entry *CEPtr;
auto section *SPtr;
const char *area;
if (quiet)
{
print_msg(PRT_NORMAL,"%s",_alias?_alias:(number[0]?expand_number(number):alias));
}
else
{
print_msg(PRT_NORMAL,"%s:\t\t%s\n",make_word(CONF_ENT_ALIAS),_alias?_alias:S_UNKNOWN);
print_msg(PRT_NORMAL,"%s:\t\t%s\n",make_word(CONF_ENT_NUM),_number?_number:S_UNKNOWN);
#if 0 /* DELETE_ME AK:18-Aug-99 */
if (_number != NULL && (ptr = get_areacode(_number,NULL,C_NO_ERROR)) != NULL)
print_msg(PRT_NORMAL,"Location:\t%s\n",ptr);
#endif
if (!short_out)
{
ptr = (CEPtr = Get_Entry(conf_dat->entries,CONF_ENT_SI))?(CEPtr->value?CEPtr->value:"0"):"0";
print_msg(PRT_NORMAL,"%s:\t\t%s\n",CONF_ENT_SI,ptr);
#if 0 /* DELETE_ME AK:18-Aug-99 */
area = area_diff_string(NULL,_number);
#else
area = "";
#endif
ptr = (char*)(const char*) (area[0] != '\0'?area:(CEPtr = Get_Entry(conf_dat->entries,CONF_ENT_ZONE))?(CEPtr->value?CEPtr->value:""):"");
print_msg(PRT_NORMAL,"%s:\t\t%s\n",make_word(CONF_ENT_ZONE),ptr);
ptr = (CEPtr = Get_Entry(conf_dat->entries,CONF_ENT_INTFAC))?(CEPtr->value?CEPtr->value:""):"";
print_msg(PRT_NORMAL,"%s:\t%s\n",make_word(CONF_ENT_INTFAC),ptr);
}
if (long_out)
{
SPtr = Get_SubSection(conf_dat,CONF_ENT_START);
while (SPtr != NULL)
{
ptr = (CEPtr = Get_Entry(SPtr->entries,CONF_ENT_FLAGS))?(CEPtr->value?CEPtr->value:""):"";
print_msg(PRT_NORMAL,"%s:\t\t%s\n",make_word(CONF_ENT_FLAGS),ptr);
ptr = (CEPtr = Get_Entry(SPtr->entries,CONF_ENT_PROG))?(CEPtr->value?CEPtr->value:""):"";
print_msg(PRT_NORMAL,"%s:\t%s\n",make_word(CONF_ENT_PROG),ptr);
ptr = (CEPtr = Get_Entry(SPtr->entries,CONF_ENT_USER))?(CEPtr->value?CEPtr->value:""):"";
print_msg(PRT_NORMAL,"%s:\t%s\n",make_word(CONF_ENT_USER),ptr);
ptr = (CEPtr = Get_Entry(SPtr->entries,CONF_ENT_GROUP))?(CEPtr->value?CEPtr->value:""):"";
print_msg(PRT_NORMAL,"%s:\t%s\n",make_word(CONF_ENT_GROUP),ptr);
ptr = (CEPtr = Get_Entry(SPtr->entries,CONF_ENT_INTVAL))?(CEPtr->value?CEPtr->value:""):"";
print_msg(PRT_NORMAL,"%s:\t%s\n",make_word(CONF_ENT_INTVAL),ptr);
ptr = (CEPtr = Get_Entry(SPtr->entries,CONF_ENT_TIME))?(CEPtr->value?CEPtr->value:""):"";
print_msg(PRT_NORMAL,"%s:\t\t%s\n",make_word(CONF_ENT_TIME),ptr);
SPtr = SPtr->next;
}
}
}
print_msg(PRT_NORMAL,"\n");
return 0;
}
/*****************************************************************************/
int look_data(section **conf_dat)
{
int Cnt = 0;
auto entry *CEPtr;
char *_number = NULL, *_alias = NULL;
char _si[SHORT_STRING_SIZE];
section *old_conf_dat = NULL;
while (*conf_dat != NULL)
{
old_conf_dat = *conf_dat;
if (!strcmp((*conf_dat)->name,CONF_SEC_NUM) || !strcmp((*conf_dat)->name,CONF_SEC_MSN))
{
int Ret = 0;
free(_number);
free(_alias);
_number = _alias = NULL;
_si[0] = '\0';
if ((CEPtr = Get_Entry((*conf_dat)->entries,CONF_ENT_NUM)) != NULL) {
if (del)
_number = strdup(Replace_Variable(CEPtr->value));
else
_number = strdup(CEPtr->value);
}
if ((CEPtr = Get_Entry((*conf_dat)->entries,CONF_ENT_ALIAS)) != NULL) {
if (del)
_alias = strdup(Replace_Variable(CEPtr->value));
else
_alias = strdup(CEPtr->value);
}
if ((CEPtr = Get_Entry((*conf_dat)->entries,CONF_ENT_SI)) != NULL &&
CEPtr->value != NULL) {
if (del)
sprintf(_si,"%ld",strtol(Replace_Variable(CEPtr->value), NIL, 0));
else
sprintf(_si,"%ld",strtol(CEPtr->value, NIL, 0));
}
if (and)
Ret = 1;
else
Ret = 0;
if (number[0] && _number != NULL)
{
if (!num_match(_number,number))
Ret = 1;
else
Ret = 0;
}
if (alias[0] && _alias != NULL)
{
int Ret2 = 0;
if (!match(alias,_alias,match_flags))
Ret2 = 1;
if (and)
Ret &= Ret2;
else
Ret |= Ret2;
}
if (si[0])
{
int Ret2 = 0;
if (*_si == '\0' || !strcmp(_si,si))
Ret2 = 1;
if (and)
Ret &= Ret2;
else
Ret |= Ret2;
}
if (Ret == 1)
{
if (del)
{
delete_data(_alias,_number,conf_dat);
Cnt++;
}
else
{
find_data(_alias,_number,*conf_dat);
Cnt++;
}
if (oneentry)
break;
}
}
if (*conf_dat != NULL && old_conf_dat == *conf_dat)
conf_dat = &((*conf_dat)->next);
}
if (Cnt == 0 && quiet && !del)
find_data(NULL,_number,*conf_dat);
free(_number);
free(_alias);
return Cnt;
}
/*****************************************************************************/
char* tmp_dup(const char *in)
{
static char out[SHORT_STRING_SIZE];
strcpy(out,in);
return out;
}
/*****************************************************************************/
const char* make_word(const char *in)
{
int i = 0;
static char out[SHORT_STRING_SIZE];
out[0] ='\0';
while(in[i] != '\0')
out[i] = tolower(in[i]),i++;
out[i] ='\0';
out[0] = toupper(out[0]);
return out;
}
/*****************************************************************************/
int print_msg(int Level, const char *fmt, ...)
{
auto va_list ap;
auto char String[LONG_STRING_SIZE];
va_start(ap, fmt);
vsnprintf(String, LONG_STRING_SIZE, fmt, ap);
va_end(ap);
if (Level & PRT_ERR)
fprintf(stderr, "%s", String);
else
fprintf(stdout, "%s", String);
return 0;
}
/*****************************************************************************/
int print_in_modules(const char *fmt, ...)
{
auto va_list ap;
auto char String[LONG_STRING_SIZE];
va_start(ap, fmt);
(void)vsnprintf(String, LONG_STRING_SIZE, fmt, ap);
va_end(ap);
return print_msg(PRT_ERR, "%s", String);
}
/*****************************************************************************/
int main(int argc, char *argv[], char *envp[])
{
int c, len = 0;
int Cnt = 0;
section *conf_dat = NULL;
char *myname = basename(argv[0]);
FILE *fp;
char *ptr = "";
static char usage[] = "%s: usage: %s [ -%s ]\n";
static char options[] = "ADdn:a:t:f:c:wslimqgV1M:";
set_print_fct_for_tools(print_in_modules);
alias[0] = '\0';
number[0] = '\0';
sprintf(conffile,"%s%c%s",confdir(),C_SLASH,CONFFILE);
strcpy(callerfile,USERCONFFILE);
while ((c = getopt(argc, argv, options)) != EOF)
switch (c) {
case 'A' : add++;
break;
case 'D' : del++;
match_flags &= ~F_NO_HOLE_WORD;
break;
case 'V' : print_version(myname);
exit(0);
case 'd' : and++;
break;
case 'm' : msn++;
break;
case 'i' : match_flags |= F_IGNORE_CASE;
break;
case 'n' : strcpy(number, optarg);
break;
case 'a' : strcpy(alias, optarg);
break;
case 't' : sprintf(si,"%ld",strtol(optarg, NIL, 0));
break;
case 's' : short_out++;
break;
case 'l' : long_out++;
break;
case 'w' : match_flags &= ~F_NO_HOLE_WORD;
break;
case 'M' : isdnmon++;
oneentry++;
and++;
quiet++;
strcpy(areacode, optarg);
strcpy(number, optarg);
break;
case '1' : oneentry++;
break;
case 'q' : quiet++;
break;
case 'f' : strcpy(callerfile, optarg);
break;
case 'c' : strcpy(areacode, optarg);
break;
case 'g' : sprintf(callerfile,"%s%c%s",confdir(),C_SLASH,CALLERIDFILE);
break;
case '?' : print_msg(PRT_ERR, usage, myname, myname, options);
return(1);
}
if (isdnmon && (add || del || short_out || long_out))
{
print_msg(PRT_ERR,"Error: Can not do isdnmon output and delete or add or short or long!\n",conffile);
exit(7);
}
if (add || del)
{
if ((conf_dat = read_file(NULL, conffile, C_NOT_UNIQUE)) == NULL)
{
print_msg(PRT_ERR,"Error while reading file `%s': no rights on file, syntax error\nor emtpy (delete it first)!\n",conffile);
exit(2);
}
if (Set_Codes(conf_dat) != 0)
{
print_msg(PRT_ERR,"Error: Variables `%s' and `%s' are not set!\n",CONF_ENT_AREA,CONF_ENT_COUNTRY);
exit(5);
}
free_section(conf_dat);
conf_dat = NULL;
if (access(expand_file(callerfile),W_OK))
{
if (errno != ENOENT)
{
print_msg(PRT_ERR,"Error: Can not open file `%s' (%s)!\n",expand_file(callerfile),strerror(errno));
exit(6);
}
else
{
if (add)
{
if ((fp = fopen(expand_file(callerfile),"w")) == NULL)
{
print_msg(PRT_ERR,"Error: Can not open file `%s' (%s)!\n",expand_file(callerfile),strerror(errno));
exit(6);
}
else
print_msg(PRT_ERR,"Create file `%s'!\n",expand_file(callerfile),strerror(errno));
fclose(fp);
}
else
{
print_msg(PRT_ERR,"File `%s' doesn't exist!\n",expand_file(callerfile));
exit(6);
}
}
}
else
{
if ((conf_dat = read_file(NULL, expand_file(callerfile), C_NOT_UNIQUE)) == NULL)
{
print_msg(PRT_ERR,"Error while reading file `%s': no rights on file, syntax error\nor emtpy (delete it first)!\n",expand_file(callerfile));
exit(2);
}
}
}
else
{
if (read_isdnconf(&conf_dat) == NULL)
exit(2);
}
if (number[0] != '\0')
{
strcpy(number, expand_number(number));
if (isdnmon)
print_msg(PRT_NORMAL,"%s\t",number);
}
if (areacode[0] != '\0')
{
if (1) {
print_msg(PRT_NORMAL,"%s\t%d\t",ptr,len);
}
else
{
if (!isdnmon)
exit(3);
print_msg(PRT_NORMAL,"\t0\t");
}
}
if (optind < argc && !add)
{
if (!alias[0] && optind + 1 == argc)
strcpy(alias, argv[optind]);
else
{
print_msg(PRT_ERR,"Error: Can not set two strings for alias!\n");
exit(3);
}
}
if (!add)
{
if (!alias[0] && !number[0] && !si[0])
{
print_msg(PRT_ERR, usage, argv[0], argv[0], options);
exit(4);
}
}
else
{
if (alias[0] || number[0] || si[0])
{
print_msg(PRT_ERR, usage, argv[0], argv[0], options);
exit(5);
}
}
if (add && del)
{
print_msg(PRT_ERR,"Error: Can not do add and delete together!\n");
exit(1);
}
if (short_out && long_out)
{
print_msg(PRT_ERR,"Error: Can not do long and short output together!\n");
exit(1);
}
if (!add && !del && Replace_Variables(conf_dat))
exit(8);
if (add)
Cnt = add_data(&conf_dat);
else
Cnt = look_data(&conf_dat);
if ((add || del) && Cnt > 0)
{
if (conf_dat == NULL)
{
if (unlink(expand_file(callerfile)))
print_msg(PRT_ERR,"Error: Can not delete file `%s' (%s)!\n",expand_file(callerfile),strerror(errno));
else
print_msg(PRT_ERR,"Warning: File `%s' is empty. It is deleted now!\n", expand_file(callerfile));
}
else
if (write_file(conf_dat,expand_file(callerfile),myname,VERSION) == NULL)
exit(5);
}
if (del && !Cnt)
print_msg(PRT_ERR, "Warning: No entry deleted!\n");
free_section(conf_dat);
return 0;
}