Bug 6508 - if priority was a number, then findlabel fails
[asterisk/asterisk.git] / apps / app_channelredirect.c
1 /*
2  * Asterisk -- An open source telephony toolkit.
3  *
4  * Copyright (C) 2006, Sergey Basmanov
5  *
6  * See http://www.asterisk.org for more information about
7  * the Asterisk project. Please do not directly contact
8  * any of the maintainers of this project for assistance;
9  * the project provides a web site, mailing lists and IRC
10  * channels for your use.
11  *
12  * This program is free software, distributed under the terms of
13  * the GNU General Public License Version 2. See the LICENSE file
14  * at the top of the source tree.
15  */
16
17 /*! \file
18  *
19  * \brief ChannelRedirect application
20  *
21  * \author Sergey Basmanov <sergey_basmanov@mail.ru>
22  *
23  * \ingroup applications
24  */
25
26 #include <stdio.h>
27 #include <stdlib.h>
28 #include <unistd.h>
29 #include <string.h>
30
31 #include "asterisk.h"
32
33 ASTERISK_FILE_VERSION(__FILE__, "$Revision$")
34
35 #include "asterisk/file.h"
36 #include "asterisk/logger.h"
37 #include "asterisk/channel.h"
38 #include "asterisk/pbx.h"
39 #include "asterisk/module.h"
40 #include "asterisk/lock.h"
41 #include "asterisk/app.h"
42 #include "asterisk/features.h"
43
44 static char *tdesc = "Channel Redirect";
45 static char *app = "ChannelRedirect";
46 static char *synopsis = "Redirects given channel to a dialplan target.";
47 static char *descrip = 
48 "ChannelRedirect(channel|[[context|]extension|]priority):\n"
49 "  Sends the specified channel to the specified extension priority\n";
50
51 LOCAL_USER_DECL;
52
53 static int asyncgoto_exec(struct ast_channel *chan, void *data)
54 {
55         int res = -1;
56         struct localuser *u;
57         char *info, *context, *exten, *priority;
58         int prio = 1;
59         struct ast_channel *chan2 = NULL;
60
61         AST_DECLARE_APP_ARGS(args,
62                 AST_APP_ARG(channel);
63                 AST_APP_ARG(label);
64         );
65
66         if (ast_strlen_zero(data)) {
67                 ast_log(LOG_WARNING, "%s requires an argument (channel|[[context|]exten|]priority)\n", app);
68                 return -1;
69         }
70
71         LOCAL_USER_ADD(u);
72
73         info = ast_strdupa(data);
74         AST_STANDARD_APP_ARGS(args, info);
75
76         if (ast_strlen_zero(args.channel) || ast_strlen_zero(args.label)) {
77                 ast_log(LOG_WARNING, "%s requires an argument (channel|[[context|]exten|]priority)\n", app);
78                 goto quit;
79         }
80
81         chan2 = ast_get_channel_by_name_locked(args.channel);
82         if (!chan2) {
83                 ast_log(LOG_WARNING, "No such channel: %s\n", args.channel);
84                 goto quit;
85         }
86
87         /* Parsed right to left, so standard parsing won't work */
88         context = strsep(&args.label, "|");
89         exten = strsep(&args.label, "|");
90         if (exten) {
91                 priority = strsep(&args.label, "|");
92                 if (!priority) {
93                         priority = exten;
94                         exten = context;
95                         context = NULL;
96                 }
97         } else {
98                 priority = context;
99                 context = NULL;
100         }
101
102         /* ast_findlabel_extension does not convert numeric priorities; it only does a lookup */
103         if (!(prio = atoi(priority)) && !(prio = ast_findlabel_extension(chan2, S_OR(context, chan2->context),
104                                                                         S_OR(exten, chan2->exten), priority, chan2->cid.cid_num))) {
105                 ast_log(LOG_WARNING, "'%s' is not a known priority or label\n", priority);
106                 goto chanquit;
107         }
108
109         ast_log(LOG_DEBUG, "Attempting async goto (%s) to %s|%s|%d\n", args.channel, S_OR(context, chan2->context), S_OR(exten, chan2->exten), prio);
110
111         if (ast_async_goto_if_exists(chan2, S_OR(context, chan2->context), S_OR(exten, chan2->exten), prio))
112                 ast_log(LOG_WARNING, "%s failed for %s\n", app, args.channel);
113         else
114                 res = 0;
115
116  chanquit:
117         ast_mutex_unlock(&chan2->lock);
118  quit:
119         LOCAL_USER_REMOVE(u);
120
121         return res;
122 }
123
124 int unload_module(void)
125 {
126         int res;
127
128         res = ast_unregister_application(app);
129
130         STANDARD_HANGUP_LOCALUSERS;
131
132         return res;     
133 }
134
135 int load_module(void)
136 {
137         return ast_register_application(app, asyncgoto_exec, synopsis, descrip);
138 }
139
140 char *description(void)
141 {
142         return tdesc;
143 }
144
145 int usecount(void)
146 {
147         int res;
148
149         STANDARD_USECOUNT(res);
150
151         return res;
152 }
153
154 char *key()
155 {
156         return ASTERISK_GPL_KEY;
157 }