Skip to content
Snippets Groups Projects
pbx.c 164 KiB
Newer Older
  • Learn to ignore specific revisions
  • Mark Spencer's avatar
    Mark Spencer committed
    {
    	return e ? e->app : NULL;
    }
    
    void *ast_get_extension_app_data(struct ast_exten *e)
    {
    	return e ? e->data : NULL;
    }
    
    
    const char *ast_get_switch_name(struct ast_sw *sw)
    
    Mark Spencer's avatar
    Mark Spencer committed
    {
    	return sw ? sw->name : NULL;
    }
    
    
    const char *ast_get_switch_data(struct ast_sw *sw)
    
    Mark Spencer's avatar
    Mark Spencer committed
    {
    	return sw ? sw->data : NULL;
    }
    
    
    const char *ast_get_switch_registrar(struct ast_sw *sw)
    
    Mark Spencer's avatar
    Mark Spencer committed
    {
    	return sw ? sw->registrar : NULL;
    }
    
    /*
     * Walking functions ...
     */
    struct ast_context *ast_walk_contexts(struct ast_context *con)
    {
    	if (!con)
    		return contexts;
    	else
    		return con->next;
    }
    
    struct ast_exten *ast_walk_context_extensions(struct ast_context *con,
    	struct ast_exten *exten)
    {
    	if (!exten)
    		return con ? con->root : NULL;
    	else
    		return exten->next;
    }
    
    struct ast_sw *ast_walk_context_switches(struct ast_context *con,
    	struct ast_sw *sw)
    {
    	if (!sw)
    		return con ? con->alts : NULL;
    	else
    		return sw->next;
    }
    
    struct ast_exten *ast_walk_extension_priorities(struct ast_exten *exten,
    	struct ast_exten *priority)
    {
    	if (!priority)
    		return exten;
    	else
    		return priority->peer;
    }
    
    struct ast_include *ast_walk_context_includes(struct ast_context *con,
    	struct ast_include *inc)
    {
    	if (!inc)
    		return con ? con->includes : NULL;
    	else
    		return inc->next;
    }
    
    struct ast_ignorepat *ast_walk_context_ignorepats(struct ast_context *con,
    	struct ast_ignorepat *ip)
    {
    	if (!ip)
    		return con ? con->ignorepats : NULL;
    	else
    		return ip->next;
    }
    
    
    int ast_context_verify_includes(struct ast_context *con)
    {
    	struct ast_include *inc;
    	int res = 0;
    
    	for (inc = ast_walk_context_includes(con, NULL); inc; inc = ast_walk_context_includes(con, inc))
    		if (!ast_context_find(inc->rname)) {
    			res = -1;
    			ast_log(LOG_WARNING, "Context '%s' tries includes non-existant context '%s'\n",
    					ast_get_context_name(con), inc->rname);
    		}
    	return res;
    }
    
    static int __ast_goto_if_exists(struct ast_channel *chan, char* context, char *exten, int priority, int async) 
    
    	int (*goto_func)(struct ast_channel *chan, const char *context, const char *exten, int priority) = NULL;
    
    
    		if (async) {
    			goto_func = ast_async_goto;
    		} else { 
    			goto_func = ast_explicit_goto;
    			priority--;
    			if(priority < 0)
    				priority = 0;
    		}
    
    		if (ast_exists_extension(chan, 
    								 context ? context : chan->context,
    								 exten ? exten : chan->exten,
    								 priority,
    								 chan->cid.cid_num)) {
    
    							 context ? context : chan->context,
    							 exten ? exten : chan->exten,
    							 priority);
    
    int ast_goto_if_exists(struct ast_channel *chan, char* context, char *exten, int priority) {
    	return __ast_goto_if_exists(chan, context, exten, priority, 0);
    }
    
    int ast_async_goto_if_exists(struct ast_channel *chan, char* context, char *exten, int priority) {
    	return __ast_goto_if_exists(chan, context, exten, priority, 1);
    }
    
    
    int ast_parseable_goto(struct ast_channel *chan, const char *goto_string) 
    {
    	char *s;
    	char *exten, *pri, *context;
    	char *stringp=NULL;
    	int ipri;
    	int mode = 0;
    
    	if (!goto_string || ast_strlen_zero(goto_string)) {
    		ast_log(LOG_WARNING, "Goto requires an argument (optional context|optional extension|priority)\n");
    		return -1;
    	}
    	s = ast_strdupa(goto_string);
    	stringp=s;
    	context = strsep(&stringp, "|");
    	exten = strsep(&stringp, "|");
    	if (!exten) {
    		/* Only a priority in this one */
    		pri = context;
    		exten = NULL;
    		context = NULL;
    	} else {
    		pri = strsep(&stringp, "|");
    		if (!pri) {
    			/* Only an extension and priority in this one */
    			pri = exten;
    			exten = context;
    			context = NULL;
    		}
    	}
    	if (*pri == '+') {
    		mode = 1;
    		pri++;
    	} else if (*pri == '-') {
    		mode = -1;
    		pri++;
    	}
    	if (sscanf(pri, "%i", &ipri) != 1) {
    		if ((ipri = ast_findlabel_extension(chan, context ? context : chan->context, (exten && strcasecmp(exten, "BYEXTENSION")) ? exten : chan->exten, 
    			pri, chan->cid.cid_num)) < 1) {
    			ast_log(LOG_WARNING, "Priority '%s' must be a number > 0, or valid label\n", pri);
    			return -1;
    		} else
    			mode = 0;
    	} 
    	/* At this point we have a priority and maybe an extension and a context */
    
    	if (exten && !strcasecmp(exten, "BYEXTENSION"))
    		exten = NULL;
    
    	if (mode) 
    		ipri = chan->priority + (ipri * mode);
    
    	/* This channel is currently in the PBX */
    
    	ast_explicit_goto(chan, context, exten, ipri - 1);