123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370 |
- /*
- * Asterisk -- A telephony toolkit for Linux.
- *
- * DISA -- Direct Inward System Access Application 6/20/2001
- *
- * Copyright (C) 2001, Linux Support Services, Inc.
- *
- * Jim Dixon <jim@lambdatel.com>
- *
- * Made only slightly more sane by Mark Spencer <markster@digium.com>
- *
- * This program is free software, distributed under the terms of
- * the GNU General Public License
- */
-
- #include <asterisk/lock.h>
- #include <asterisk/file.h>
- #include <asterisk/logger.h>
- #include <asterisk/channel.h>
- #include <asterisk/indications.h>
- #include <asterisk/pbx.h>
- #include <asterisk/module.h>
- #include <asterisk/translate.h>
- #include <asterisk/ulaw.h>
- #include <asterisk/utils.h>
- #include <string.h>
- #include <stdlib.h>
- #include <stdio.h>
- #include <math.h>
- #include <sys/time.h>
- static char *tdesc = "DISA (Direct Inward System Access) Application";
- static char *app = "DISA";
- static char *synopsis = "DISA (Direct Inward System Access)";
- static char *descrip =
- "DISA (Direct Inward System Access) -- Allows someone from outside\n"
- "the telephone switch (PBX) to obtain an \"internal\" system dialtone\n"
- "and to place calls from it as if they were placing a call from within\n"
- "the switch. A user calls a number that connects to the DISA application\n"
- "and is given dialtone. The user enters their passcode, followed by the\n"
- "pound sign (#). If the passcode is correct, the user is then given\n"
- "system dialtone on which a call may be placed. Obviously, this type\n"
- "of access has SERIOUS security implications, and GREAT care must be\n"
- "taken NOT to compromise your security.\n\n"
- "There is a possibility of accessing DISA without password. Simply\n"
- "exchange your password with no-password.\n\n"
- " Example: exten => s,1,DISA,no-password|local\n\n"
- "but be aware of using this for your security compromising.\n\n"
- "The arguments to this application (in extensions.conf) allow either\n"
- "specification of a single global password (that everyone uses), or\n"
- "individual passwords contained in a file. It also allow specification\n"
- "of the context on which the user will be dialing. If no context is\n"
- "specified, the DISA application defaults the context to \"disa\"\n"
- "presumably that a normal system will have a special context set up\n"
- "for DISA use with some or a lot of restrictions. The arguments are\n"
- "one of the following:\n\n"
- " numeric-passcode\n"
- " numeric-passcode|context\n"
- " full-pathname-of-file-that-contains-passcodes\n\n"
- "The file that contains the passcodes (if used) allows specification\n"
- "of either just a passcode (defaulting to the \"disa\" context, or\n"
- "passcode|context on each line of the file. The file may contain blank\n"
- "lines, or comments starting with \"#\" or \";\". In addition, the\n"
- "above arguments may have |new-callerid-string appended to them, to\n"
- "specify a new (different) callerid to be used for this call, for\n"
- "example: numeric-passcode|context|\"My Phone\" <(234) 123-4567> or \n"
- "full-pathname-of-passcode-file|\"My Phone\" <(234) 123-4567>. Note that\n"
- "in the case of specifying the numeric-passcode, the context must be\n"
- "specified if the callerid is specified also.\n\n"
- "If login is successful, the application parses the dialed number in\n"
- "the specified (or default) context, and returns 0 with the new extension\n"
- "context filled-in and the priority set to 1, so that the PBX may\n"
- "re-apply the routing tables to it and complete the call normally.";
- STANDARD_LOCAL_USER;
- LOCAL_USER_DECL;
- static int firstdigittimeout = 20000; /* 20 seconds first digit timeout */
- static int digittimeout = 10000; /* 10 seconds subsequent digit timeout */
- static int ms_diff(struct timeval *tv1, struct timeval *tv2)
- {
- int ms;
-
- ms = (tv1->tv_sec - tv2->tv_sec) * 1000;
- ms += (tv1->tv_usec - tv2->tv_usec) / 1000;
- return(ms);
- }
- static void play_dialtone(struct ast_channel *chan)
- {
- const struct tone_zone_sound *ts = NULL;
- ts = ast_get_indication_tone(chan->zone, "dial");
- if (ts)
- ast_playtones_start(chan, 0, ts->data, 0);
- else
- ast_tonepair_start(chan, 350, 440, 0, 0);
- }
- static int disa_exec(struct ast_channel *chan, void *data)
- {
- int i,j,k,x,did_ignore;
- struct localuser *u;
- char tmp[256],arg2[256]="",exten[AST_MAX_EXTENSION],acctcode[20]="";
- char *ourcontext,*ourcallerid;
- struct ast_frame *f;
- struct timeval lastout, now, lastdigittime;
- int res;
- time_t rstart;
- FILE *fp;
- char *stringp=NULL;
- if (ast_set_write_format(chan,AST_FORMAT_ULAW))
- {
- ast_log(LOG_WARNING, "Unable to set write format to Mu-law on %s\n",chan->name);
- return -1;
- }
- if (ast_set_read_format(chan,AST_FORMAT_ULAW))
- {
- ast_log(LOG_WARNING, "Unable to set read format to Mu-law on %s\n",chan->name);
- return -1;
- }
- lastout.tv_sec = lastout.tv_usec = 0;
- if (!data || !strlen((char *)data)) {
- ast_log(LOG_WARNING, "disa requires an argument (passcode/passcode file)\n");
- return -1;
- }
- strncpy(tmp, (char *)data, sizeof(tmp)-1);
- stringp=tmp;
- strsep(&stringp, "|");
- ourcontext = strsep(&stringp, "|");
- /* if context specified, save 2nd arg and parse third */
- if (ourcontext) {
- strncpy(arg2,ourcontext, sizeof(arg2) - 1);
- ourcallerid = strsep(&stringp,"|");
- }
- /* if context not specified, use "disa" */
- else {
- arg2[0] = 0;
- ourcallerid = NULL;
- ourcontext = "disa";
- }
- LOCAL_USER_ADD(u);
- if (chan->_state != AST_STATE_UP)
- {
- /* answer */
- ast_answer(chan);
- }
- i = k = x = 0; /* k is 0 for pswd entry, 1 for ext entry */
- did_ignore = 0;
- exten[0] = 0;
- acctcode[0] = 0;
- /* can we access DISA without password? */
- ast_log(LOG_DEBUG, "Context: %s\n",ourcontext);
- if (!strcasecmp(tmp, "no-password"))
- {;
- k |= 1; /* We have the password */
- ast_log(LOG_DEBUG, "DISA no-password login success\n");
- }
- gettimeofday(&lastdigittime,NULL);
- play_dialtone(chan);
- for(;;)
- {
- gettimeofday(&now,NULL);
- /* if outa time, give em reorder */
- if (ms_diff(&now,&lastdigittime) >
- ((k&2) ? digittimeout : firstdigittimeout))
- {
- ast_log(LOG_DEBUG,"DISA %s entry timeout on chan %s\n",
- ((k&1) ? "extension" : "password"),chan->name);
- break;
- }
- if ((res = ast_waitfor(chan, -1) < 0)) {
- ast_log(LOG_DEBUG, "Waitfor returned %d\n", res);
- continue;
- }
-
- f = ast_read(chan);
- if (f == NULL)
- {
- LOCAL_USER_REMOVE(u);
- return -1;
- }
- if ((f->frametype == AST_FRAME_CONTROL) &&
- (f->subclass == AST_CONTROL_HANGUP))
- {
- ast_frfree(f);
- LOCAL_USER_REMOVE(u);
- return -1;
- }
- if (f->frametype == AST_FRAME_VOICE) {
- ast_frfree(f);
- continue;
- }
- /* if not DTMF, just do it again */
- if (f->frametype != AST_FRAME_DTMF)
- {
- ast_frfree(f);
- continue;
- }
- j = f->subclass; /* save digit */
- ast_frfree(f);
- if (i == 0)
- {
- k|=2; /* We have the first digit */
- ast_playtones_stop(chan);
- }
- gettimeofday(&lastdigittime,NULL);
- /* got a DTMF tone */
- if (i < AST_MAX_EXTENSION) /* if still valid number of digits */
- {
- if (!(k&1)) /* if in password state */
- {
- if (j == '#') /* end of password */
- {
- /* see if this is an integer */
- if (sscanf(tmp,"%d",&j) < 1)
- { /* nope, it must be a filename */
- fp = fopen(tmp,"r");
- if (!fp)
- {
- ast_log(LOG_WARNING,"DISA password file %s not found on chan %s\n",tmp,chan->name);
- LOCAL_USER_REMOVE(u);
- return -1;
- }
- tmp[0] = 0;
- while(fgets(tmp,sizeof(tmp) - 1,fp))
- {
- char *stringp=NULL,*stringp2;
- if (!tmp[0]) continue;
- if (tmp[strlen(tmp) - 1] == '\n')
- tmp[strlen(tmp) - 1] = 0;
- if (!tmp[0]) continue;
- /* skip comments */
- if (tmp[0] == '#') continue;
- if (tmp[0] == ';') continue;
- stringp=tmp;
- strsep(&stringp, "|");
- stringp2=strsep(&stringp, "|");
- if (stringp2) {
- ourcontext=stringp2;
- stringp2=strsep(&stringp, "|");
- if (stringp2) ourcallerid=stringp2;
- }
- /* password must be in valid format (numeric) */
- if (sscanf(tmp,"%d",&j) < 1) continue;
- /* if we got it */
- if (!strcmp(exten,tmp)) break;
- }
- fclose(fp);
- }
- /* compare the two */
- if (strcmp(exten,tmp))
- {
- ast_log(LOG_WARNING,"DISA on chan %s got bad password %s\n",chan->name,exten);
- goto reorder;
- }
- /* password good, set to dial state */
- ast_log(LOG_DEBUG,"DISA on chan %s password is good\n",chan->name);
- play_dialtone(chan);
- k|=1; /* In number mode */
- i = 0; /* re-set buffer pointer */
- exten[sizeof(acctcode)] = 0;
- strncpy(acctcode,exten, sizeof(acctcode) - 1);
- exten[0] = 0;
- ast_log(LOG_DEBUG,"Successful DISA log-in on chan %s\n",chan->name);
- continue;
- }
- }
- exten[i++] = j; /* save digit */
- exten[i] = 0;
- if (!(k&1)) continue; /* if getting password, continue doing it */
- /* if this exists */
- if (ast_ignore_pattern(ourcontext, exten)) {
- play_dialtone(chan);
- did_ignore = 1;
- } else
- if (did_ignore) {
- ast_playtones_stop(chan);
- did_ignore = 0;
- }
- /* if can do some more, do it */
- if (!ast_matchmore_extension(chan,ourcontext,exten,1, chan->callerid)) {
- break;
- }
- }
- }
- if ((k==3) && (ast_exists_extension(chan,ourcontext,exten,1, chan->callerid)))
- {
- ast_playtones_stop(chan);
- /* We're authenticated and have a valid extension */
- if (ourcallerid && *ourcallerid)
- {
- if (chan->callerid) free(chan->callerid);
- chan->callerid = strdup(ourcallerid);
- }
- strncpy(chan->exten, exten, sizeof(chan->exten) - 1);
- strncpy(chan->context, ourcontext, sizeof(chan->context) - 1);
- if (!ast_strlen_zero(acctcode))
- strncpy(chan->accountcode, acctcode, sizeof(chan->accountcode) - 1);
- chan->priority = 0;
- ast_cdr_init(chan->cdr,chan);
- LOCAL_USER_REMOVE(u);
- return 0;
- }
- reorder:
- ast_indicate(chan,AST_CONTROL_CONGESTION);
- /* something is invalid, give em reorder for several seconds */
- time(&rstart);
- while(time(NULL) < rstart + 10)
- {
- if (ast_waitfor(chan, -1) < 0)
- break;
- f = ast_read(chan);
- if (!f)
- break;
- ast_frfree(f);
- }
- ast_playtones_stop(chan);
- LOCAL_USER_REMOVE(u);
- return -1;
- }
- int unload_module(void)
- {
- STANDARD_HANGUP_LOCALUSERS;
- return ast_unregister_application(app);
- }
- int load_module(void)
- {
- return ast_register_application(app, disa_exec, synopsis, descrip);
- }
- char *description(void)
- {
- return tdesc;
- }
- int usecount(void)
- {
- int res;
- STANDARD_USECOUNT(res);
- return res;
- }
- char *key(void)
- {
- return ASTERISK_GPL_KEY;
- }
|