1227 lines
34 KiB
C
1227 lines
34 KiB
C
/* $Id: isdnconf.c,v 1.26 1999/05/13 11:39:09 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.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
|
|
*
|
|
*/
|
|
|
|
#include "isdnconf.h"
|
|
|
|
/*****************************************************************************/
|
|
#define ZAUNPFAHL 1 /* FIXME: Michi: Offset */
|
|
/*****************************************************************************/
|
|
|
|
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 (_number != NULL && (ptr = get_areacode(_number,NULL,C_NO_ERROR)) != NULL)
|
|
print_msg(PRT_NORMAL,"Location:\t%s\n",ptr);
|
|
|
|
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);
|
|
|
|
area = area_diff_string(NULL,_number);
|
|
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);
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
|
|
static char *printrate(RATE Rate)
|
|
{
|
|
static char message[BUFSIZ];
|
|
|
|
|
|
if (Rate.Basic > 0)
|
|
sprintf(message, "%s %s + %s/%3.1fs = %s %s + %s/Min (%s)",
|
|
currency, double2str(Rate.Basic, 5, 3, DEB),
|
|
double2str(Rate.Price, 5, 3, DEB),
|
|
Rate.Duration,
|
|
currency, double2str(Rate.Basic, 5, 3, DEB),
|
|
double2str(60 * Rate.Price / Rate.Duration, 5, 3, DEB),
|
|
explainRate(&Rate));
|
|
else
|
|
sprintf(message, "%s %s/%3.1fs = %s %s/Min (%s)",
|
|
currency, double2str(Rate.Price, 5, 3, DEB),
|
|
Rate.Duration,
|
|
currency, double2str(60 * Rate.Price / Rate.Duration, 5, 3, DEB),
|
|
explainRate(&Rate));
|
|
|
|
return(message);
|
|
} /* printrate */
|
|
|
|
|
|
#define MAXPROVIDER 1000
|
|
|
|
typedef struct {
|
|
int prefix;
|
|
double rate;
|
|
char *explain;
|
|
char *msg;
|
|
} SORT;
|
|
|
|
static SORT sort[MAXPROVIDER];
|
|
|
|
|
|
static int compare(SORT *s1, SORT *s2)
|
|
{
|
|
return(s1->rate > s2->rate);
|
|
} /* compare */
|
|
|
|
|
|
static void showRates(RATE Rate, char *message)
|
|
{
|
|
if (Rate.Basic > 0)
|
|
sprintf(message, "%s %s + %s/%3.1fs = %s %s + %s/Min (%s)",
|
|
currency, double2str(Rate.Basic, 5, 3, DEB),
|
|
double2str(Rate.Price, 5, 3, DEB),
|
|
Rate.Duration,
|
|
currency, double2str(Rate.Basic, 5, 3, DEB),
|
|
double2str(60 * Rate.Price / Rate.Duration, 5, 3, DEB),
|
|
explainRate(&Rate));
|
|
else
|
|
sprintf(message, "%s %s/%3.1fs = %s %s/Min (%s)",
|
|
currency, double2str(Rate.Price, 5, 3, DEB),
|
|
Rate.Duration,
|
|
currency, double2str(60 * Rate.Price / Rate.Duration, 5, 3, DEB),
|
|
explainRate(&Rate));
|
|
} /* showRates */
|
|
|
|
|
|
static void showLCR(int duration)
|
|
{
|
|
auto int tz, hour, provider, lastprovider = UNKNOWN, lasthour = UNKNOWN, *p;
|
|
auto int useds = 0, maxhour, leastprovider = UNKNOWN;
|
|
auto int ignoreprovider = UNKNOWN;
|
|
auto RATE Rate;
|
|
auto int probe[] = { REGIOCALL, GERMANCALL, C_NETZ, D1_NETZ, D2_NETZ, E_PLUS_NETZ, E2_NETZ, INTERNET, AUKUNFT_IN, AUSKUNFT_AUS, 0 };
|
|
auto int used[MAXPROVIDER];
|
|
auto int hours[MAXPROVIDER];
|
|
auto char lastmessage[BUFSIZ], message[BUFSIZ], *px;
|
|
auto struct tm *tm;
|
|
auto time_t werktag, wochenende;
|
|
|
|
|
|
print_msg(PRT_NORMAL, "Least-Cost-Routing-Table [Verbindungsdauer:%d Sekunden]:\n\n", duration);
|
|
|
|
time(&werktag);
|
|
tm = localtime(&werktag);
|
|
|
|
if ((tm->tm_wday == 6) || (tm->tm_wday == 0)) {
|
|
wochenende = werktag;
|
|
werktag += (2 * 24 * 60 * 60);
|
|
}
|
|
else
|
|
wochenende = werktag + ((6 - tm->tm_wday) * 24 * 60 * 60);
|
|
|
|
|
|
retry:
|
|
memset(used, 0, sizeof(used));
|
|
memset(hours, 0, sizeof(hours));
|
|
|
|
for (tz = 0; tz < 2; tz++) { /* Werktag .. Wochendende */
|
|
|
|
switch (tz) {
|
|
case 0 : print_msg(PRT_NORMAL, "Werktag:\n");
|
|
tm = localtime(&werktag);
|
|
break;
|
|
|
|
case 1 : print_msg(PRT_NORMAL, "Wochenende/Feiertag:\n");
|
|
tm = localtime(&wochenende);
|
|
break;
|
|
} /* switch */
|
|
|
|
p = probe;
|
|
|
|
while (*p) {
|
|
|
|
switch (*p) {
|
|
case REGIOCALL : print_msg(PRT_NORMAL, "\tRegioCall:\n"); break;
|
|
case GERMANCALL : print_msg(PRT_NORMAL, "\tGermanCall:\n"); break;
|
|
case C_NETZ : print_msg(PRT_NORMAL, "\tC Mobilfunk:\n"); break;
|
|
case D1_NETZ : print_msg(PRT_NORMAL, "\tD1 Mobilfunk:\n"); break;
|
|
case D2_NETZ : print_msg(PRT_NORMAL, "\tD2 Mobilfunk:\n"); break;
|
|
case E_PLUS_NETZ : print_msg(PRT_NORMAL, "\tEplus Mobilfunk:\n"); break;
|
|
case E2_NETZ : print_msg(PRT_NORMAL, "\tE2 Mobilfunk:\n"); break;
|
|
case INTERNET : print_msg(PRT_NORMAL, "\tInternet:\n"); break;
|
|
case AUKUNFT_IN : print_msg(PRT_NORMAL, "\tAuskunft Inland:\n"); break;
|
|
case AUSKUNFT_AUS : print_msg(PRT_NORMAL, "\tAuskunft Ausland:\n"); break;
|
|
} /* switch */
|
|
|
|
lastprovider = UNKNOWN;
|
|
lasthour = UNKNOWN;
|
|
|
|
hour = 7;
|
|
|
|
while (1) {
|
|
|
|
tm->tm_hour = hour;
|
|
tm->tm_sec = tm->tm_min = 0;
|
|
memset(&Rate, 0, sizeof(Rate));
|
|
Rate.zone = *p;
|
|
Rate.start = mktime(tm);
|
|
Rate.now = Rate.start + duration - ZAUNPFAHL;
|
|
|
|
provider = getLeastCost(&Rate, ignoreprovider);
|
|
showRates(Rate, message);
|
|
|
|
#if 0
|
|
print_msg(PRT_NORMAL, "DEBUG::tz=%d, zone=%d, Hour=%02d, P=%d, %s lasthour=%d, lastprovider=%d, now=%s", tz, *p, hour, provider, getProvidername(provider), lasthour, lastprovider, ctime(&Rate.start));
|
|
#endif
|
|
|
|
if (lastprovider == UNKNOWN) {
|
|
lastprovider = provider;
|
|
strcpy(lastmessage, message);
|
|
} /* if */
|
|
|
|
if (lasthour == UNKNOWN)
|
|
lasthour = hour;
|
|
|
|
if (provider != lastprovider) {
|
|
px = getProvidername(lastprovider);
|
|
|
|
print_msg(PRT_NORMAL, "\t\t%02d:00 .. %02d:59 010%02d:%s%*s(%s)\n",
|
|
lasthour, hour - 1, lastprovider, px, 14 - strlen(px), "", lastmessage);
|
|
|
|
used[lastprovider] = 1;
|
|
|
|
if (lasthour >= hour)
|
|
hours[lastprovider] += ((24 - lasthour) + hour);
|
|
else
|
|
hours[lastprovider] += hour - lasthour;
|
|
|
|
lastprovider = provider;
|
|
lasthour = hour;
|
|
strcpy(lastmessage, message);
|
|
} /* if */
|
|
|
|
hour++;
|
|
if (hour == 24)
|
|
hour = 0;
|
|
else if (hour == 7)
|
|
break;
|
|
} /* for */
|
|
|
|
px = getProvidername(lastprovider);
|
|
|
|
if ((lasthour == 7) && (hour == 7))
|
|
print_msg(PRT_NORMAL, "\t\timmer 010%02d:%s%*s(%s)\n",
|
|
lastprovider, px, 14 - strlen(px), "", lastmessage);
|
|
else
|
|
print_msg(PRT_NORMAL, "\t\t%02d:00 .. %02d:59 010%02d:%s%*s(%s)\n",
|
|
lasthour, hour - 1, lastprovider, px, 14 - strlen(px), "", lastmessage);
|
|
|
|
used[lastprovider] = 1;
|
|
|
|
if (lasthour >= hour)
|
|
hours[lastprovider] += ((24 - lasthour) + hour);
|
|
else
|
|
hours[lastprovider] += hour - lasthour;
|
|
|
|
p++;
|
|
} /* while */
|
|
} /* for */
|
|
|
|
print_msg(PRT_NORMAL, "\nProvider(s) used:\n");
|
|
|
|
maxhour = 9999999;
|
|
useds = 0;
|
|
|
|
for (provider = 0; provider < MAXPROVIDER; provider++)
|
|
if (used[provider]) {
|
|
print_msg(PRT_NORMAL, "010%02d:%s\t(%d hours)\n", provider, getProvidername(provider), hours[provider]);
|
|
useds++;
|
|
|
|
if (hours[provider] < maxhour) {
|
|
maxhour = hours[provider];
|
|
leastprovider = provider;
|
|
} /* if */
|
|
} /* if */
|
|
|
|
#if 1
|
|
if (useds == 6) {
|
|
if (ignoreprovider != leastprovider) {
|
|
|
|
print_msg(PRT_NORMAL, "OOOPS: More than 5 providers used. Retry with 010%02d:%s ignored\n",
|
|
leastprovider, getProvidername(leastprovider));
|
|
|
|
ignoreprovider = leastprovider;
|
|
goto retry;
|
|
} /* if */
|
|
} /* if */
|
|
#endif
|
|
} /* showLCR */
|
|
|
|
|
|
int main(int argc, char *argv[], char *envp[])
|
|
{
|
|
int c, n = 0;
|
|
int Cnt = 0;
|
|
section *conf_dat = NULL;
|
|
char *myname = basename(argv[0]);
|
|
FILE *fp;
|
|
RATE Rate;
|
|
char *msg;
|
|
char country[BUFSIZ];
|
|
|
|
|
|
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')
|
|
{
|
|
char *ptr;
|
|
int len, i, zone = UNKNOWN, zone2 = UNKNOWN, duration = TESTDURATION;
|
|
|
|
|
|
initHoliday(holifile, NULL);
|
|
initRate("/etc/isdn/rate.conf", "/usr/lib/isdn/rate-de.dat", "/usr/lib/isdn/countries-de.dat", NULL, NULL);
|
|
/* initRate(NULL, "/usr/lib/isdn/rate-de.dat", NULL); */
|
|
currency = strdup("DEM");
|
|
|
|
if (1 /* FIXME: (*areacode == '.') || (ptr = get_areacode(areacode,&len,quiet?C_NO_ERROR|C_NO_WARN:0)) != NULL */ )
|
|
{
|
|
if (!isdnmon)
|
|
{
|
|
register char *p;
|
|
|
|
|
|
if ((p = strchr(areacode, ','))) {
|
|
duration = atoi(p + 1);
|
|
*p = 0;
|
|
} /* if */
|
|
|
|
if (*areacode == '.') {
|
|
if (strlen(areacode) > 1) {
|
|
zone = atoi(areacode + 1);
|
|
ptr = "";
|
|
}
|
|
else {
|
|
showLCR(duration);
|
|
exit(0);
|
|
} /* else */
|
|
}
|
|
else {
|
|
if (isalpha(*areacode)) {
|
|
if (abroad(areacode, country))
|
|
strcpy(areacode, country);
|
|
} /* if */
|
|
|
|
zone = area_diff(NULL, areacode);
|
|
|
|
switch (zone) {
|
|
case AREA_ERROR :
|
|
case AREA_UNKNOWN : zone = UNKNOWN; break;
|
|
case AREA_LOCAL : zone = CITYCALL; break;
|
|
case AREA_R50 : zone = REGIOCALL; break;
|
|
case AREA_FAR : zone = GERMANCALL; break;
|
|
case AREA_ABROAD : zone = UNKNOWN; break;
|
|
} /* switch */
|
|
} /* else */
|
|
|
|
memset(&Rate, 0, sizeof(Rate));
|
|
|
|
time(&Rate.start);
|
|
Rate.now = Rate.start + duration - ZAUNPFAHL;
|
|
|
|
if (zone != UNKNOWN)
|
|
zone2 = zone;
|
|
#if 0
|
|
else
|
|
zone2 = getZone(DTAG, areacode);
|
|
#endif
|
|
|
|
if (zone2 == UNKNOWN) {
|
|
if (abroad(areacode, country))
|
|
print_msg(PRT_NORMAL, "Ein %d Sekunden langes Gespraech nach %s (%s) kostet am %s",
|
|
duration, country, areacode, ctime(&Rate.start));
|
|
else
|
|
print_msg(PRT_NORMAL, "Ein %d Sekunden langes Gespraech in Zone Welt (%s) kostet am %s",
|
|
duration, areacode, ctime(&Rate.start));
|
|
}
|
|
else {
|
|
auto int ll;
|
|
auto char *p;
|
|
|
|
if ((p = get_areacode(areacode, &ll, C_NO_WARN | C_NO_EXPAND | C_NO_ERROR)) != 0)
|
|
print_msg(PRT_NORMAL, "Ein %d Sekunden langes Gespraech nach %s (Zone %d) kostet am %s",
|
|
duration, p, zone2, ctime(&Rate.start));
|
|
else
|
|
print_msg(PRT_NORMAL, "Ein %d Sekunden langes Gespraech in Zone %d kostet am %s",
|
|
duration, zone2, ctime(&Rate.start));
|
|
} /* else */
|
|
|
|
for (Rate.prefix = 0; Rate.prefix < MAXPROVIDER; Rate.prefix++) {
|
|
if (zone != UNKNOWN)
|
|
Rate.zone = zone;
|
|
else
|
|
Rate.zone = getZone(Rate.prefix, areacode);
|
|
|
|
if (Rate.zone != UNKNOWN) {
|
|
|
|
time(&Rate.start);
|
|
Rate.now = Rate.start + duration - ZAUNPFAHL;
|
|
|
|
if (getRate(&Rate, &msg) != UNKNOWN) {
|
|
sort[n].prefix = Rate.prefix;
|
|
sort[n].rate = Rate.Charge;
|
|
sort[n].msg = strdup(msg);
|
|
sort[n].explain = strdup(printrate(Rate));
|
|
n++;
|
|
} /* if */
|
|
} /* if */
|
|
} /* for */
|
|
|
|
qsort((void *)sort, n, sizeof(SORT), compare);
|
|
|
|
for (i = 0; i < n; i++)
|
|
print_msg(PRT_NORMAL, "010%02d %s %8.3f (%s)\n", sort[i].prefix, currency, sort[i].rate, sort[i].explain);
|
|
#if 0
|
|
print_msg(PRT_NORMAL, "getLeastCost=%d\n", getLeastCost(&Rate, UNKNOWN));
|
|
#endif
|
|
exit(0);
|
|
}
|
|
|
|
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;
|
|
}
|
|
|