00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018
00019
00020
00021
00022
00023
00024
00025
00026
00027
00028
00029
00030
00031
00032
00033
00034
00035
00036 #include "asterisk.h"
00037
00038 ASTERISK_FILE_VERSION(__FILE__, "$Revision: 373220 $")
00039
00040 #include "asterisk/file.h"
00041 #include "asterisk/channel.h"
00042 #include "asterisk/pbx.h"
00043 #include "asterisk/module.h"
00044 #include "asterisk/lock.h"
00045 #include "asterisk/app.h"
00046 #include "asterisk/features.h"
00047 #include "asterisk/manager.h"
00048 #include "asterisk/callerid.h"
00049 #include "asterisk/cel.h"
00050
00051 #define PICKUPMARK "PICKUPMARK"
00052
00053
00054
00055
00056
00057
00058
00059
00060
00061
00062
00063
00064
00065
00066
00067
00068
00069
00070
00071
00072
00073
00074
00075
00076
00077
00078
00079
00080
00081
00082
00083
00084
00085
00086
00087
00088
00089
00090
00091
00092
00093
00094
00095
00096
00097
00098
00099
00100
00101
00102
00103
00104
00105
00106
00107
00108
00109
00110
00111
00112
00113
00114
00115
00116
00117
00118
00119 static const char app[] = "Pickup";
00120 static const char app2[] = "PickupChan";
00121
00122 struct pickup_by_name_args {
00123 const char *name;
00124 size_t len;
00125 };
00126
00127 static int pickup_by_name_cb(void *obj, void *arg, void *data, int flags)
00128 {
00129 struct ast_channel *target = obj;
00130 struct pickup_by_name_args *args = data;
00131
00132 ast_channel_lock(target);
00133 if (!strncasecmp(ast_channel_name(target), args->name, args->len) && ast_can_pickup(target)) {
00134
00135 return CMP_MATCH | CMP_STOP;
00136 }
00137 ast_channel_unlock(target);
00138
00139 return 0;
00140 }
00141
00142
00143 static struct ast_channel *my_ast_get_channel_by_name_locked(const char *channame)
00144 {
00145 char *chkchan;
00146 struct pickup_by_name_args pickup_args;
00147
00148
00149
00150
00151 if (strchr(channame, '-')) {
00152
00153 pickup_args.len = strlen(channame);
00154 pickup_args.name = channame;
00155 } else {
00156
00157
00158
00159
00160 pickup_args.len = strlen(channame) + 1;
00161 chkchan = ast_alloca(pickup_args.len + 1);
00162 strcpy(chkchan, channame);
00163 strcat(chkchan, "-");
00164 pickup_args.name = chkchan;
00165 }
00166
00167 return ast_channel_callback(pickup_by_name_cb, NULL, &pickup_args, 0);
00168 }
00169
00170
00171 static int pickup_by_channel(struct ast_channel *chan, char *pickup)
00172 {
00173 int res = -1;
00174 struct ast_channel *target;
00175
00176 target = my_ast_get_channel_by_name_locked(pickup);
00177 if (target) {
00178
00179 if (chan != target) {
00180 res = ast_do_pickup(chan, target);
00181 }
00182 ast_channel_unlock(target);
00183 target = ast_channel_unref(target);
00184 }
00185
00186 return res;
00187 }
00188
00189
00190 static int pickup_by_exten(struct ast_channel *chan, const char *exten, const char *context)
00191 {
00192 struct ast_channel *target = NULL;
00193 struct ast_channel_iterator *iter;
00194 int res = -1;
00195
00196 if (!(iter = ast_channel_iterator_by_exten_new(exten, context))) {
00197 return -1;
00198 }
00199
00200 while ((target = ast_channel_iterator_next(iter))) {
00201 ast_channel_lock(target);
00202 if ((chan != target) && ast_can_pickup(target)) {
00203 ast_log(LOG_NOTICE, "%s pickup by %s\n", ast_channel_name(target), ast_channel_name(chan));
00204 break;
00205 }
00206 ast_channel_unlock(target);
00207 target = ast_channel_unref(target);
00208 }
00209
00210 ast_channel_iterator_destroy(iter);
00211
00212 if (target) {
00213 res = ast_do_pickup(chan, target);
00214 ast_channel_unlock(target);
00215 target = ast_channel_unref(target);
00216 }
00217
00218 return res;
00219 }
00220
00221 static int find_by_mark(void *obj, void *arg, void *data, int flags)
00222 {
00223 struct ast_channel *target = obj;
00224 const char *mark = data;
00225 const char *tmp;
00226
00227 ast_channel_lock(target);
00228 tmp = pbx_builtin_getvar_helper(target, PICKUPMARK);
00229 if (tmp && !strcasecmp(tmp, mark) && ast_can_pickup(target)) {
00230
00231 return CMP_MATCH | CMP_STOP;
00232 }
00233 ast_channel_unlock(target);
00234
00235 return 0;
00236 }
00237
00238
00239 static int pickup_by_mark(struct ast_channel *chan, const char *mark)
00240 {
00241 struct ast_channel *target;
00242 int res = -1;
00243
00244
00245 target = ast_channel_callback(find_by_mark, NULL, (char *) mark, 0);
00246 if (target) {
00247 res = ast_do_pickup(chan, target);
00248 ast_channel_unlock(target);
00249 target = ast_channel_unref(target);
00250 }
00251
00252 return res;
00253 }
00254
00255 static int pickup_by_group(struct ast_channel *chan)
00256 {
00257 struct ast_channel *target;
00258 int res = -1;
00259
00260
00261 target = ast_pickup_find_by_group(chan);
00262 if (target) {
00263 ast_log(LOG_NOTICE, "pickup %s attempt by %s\n", ast_channel_name(target), ast_channel_name(chan));
00264 res = ast_do_pickup(chan, target);
00265 ast_channel_unlock(target);
00266 target = ast_channel_unref(target);
00267 }
00268
00269 return res;
00270 }
00271
00272
00273 static int pickup_exec(struct ast_channel *chan, const char *data)
00274 {
00275 char *tmp;
00276 char *exten;
00277 char *context;
00278
00279 if (ast_strlen_zero(data)) {
00280 return pickup_by_group(chan) ? 0 : -1;
00281 }
00282
00283
00284 tmp = ast_strdupa(data);
00285 while (!ast_strlen_zero(tmp) && (exten = strsep(&tmp, "&"))) {
00286 if ((context = strchr(exten, '@')))
00287 *context++ = '\0';
00288 if (!ast_strlen_zero(context) && !strcasecmp(context, PICKUPMARK)) {
00289 if (!pickup_by_mark(chan, exten)) {
00290
00291 return -1;
00292 }
00293 } else {
00294 if (ast_strlen_zero(context)) {
00295 context = (char *) ast_channel_context(chan);
00296 }
00297 if (!pickup_by_exten(chan, exten, context)) {
00298
00299 return -1;
00300 }
00301 }
00302 ast_log(LOG_NOTICE, "No target channel found for %s@%s.\n", exten, context);
00303 }
00304
00305
00306 return 0;
00307 }
00308
00309
00310 static int find_by_part(void *obj, void *arg, void *data, int flags)
00311 {
00312 struct ast_channel *target = obj;
00313 const char *part = data;
00314 int len = strlen(part);
00315
00316 ast_channel_lock(target);
00317 if (len <= strlen(ast_channel_name(target)) && !strncmp(ast_channel_name(target), part, len)
00318 && ast_can_pickup(target)) {
00319
00320 return CMP_MATCH | CMP_STOP;
00321 }
00322 ast_channel_unlock(target);
00323
00324 return 0;
00325 }
00326
00327
00328 static int pickup_by_part(struct ast_channel *chan, const char *part)
00329 {
00330 struct ast_channel *target;
00331 int res = -1;
00332
00333
00334 target = ast_channel_callback(find_by_part, NULL, (char *) part, 0);
00335 if (target) {
00336 res = ast_do_pickup(chan, target);
00337 ast_channel_unlock(target);
00338 target = ast_channel_unref(target);
00339 }
00340
00341 return res;
00342 }
00343
00344
00345 static int pickupchan_exec(struct ast_channel *chan, const char *data)
00346 {
00347 int partial_pickup = 0;
00348 char *pickup = NULL;
00349 char *parse = ast_strdupa(data);
00350 AST_DECLARE_APP_ARGS(args,
00351 AST_APP_ARG(channel);
00352 AST_APP_ARG(options);
00353 );
00354 AST_STANDARD_APP_ARGS(args, parse);
00355
00356 if (ast_strlen_zero(args.channel)) {
00357 ast_log(LOG_WARNING, "PickupChan requires an argument (channel)!\n");
00358
00359 return 0;
00360 }
00361
00362 if (!ast_strlen_zero(args.options) && strchr(args.options, 'p')) {
00363 partial_pickup = 1;
00364 }
00365
00366
00367 while (!ast_strlen_zero(args.channel) && (pickup = strsep(&args.channel, "&"))) {
00368 if (!strncasecmp(ast_channel_name(chan), pickup, strlen(pickup))) {
00369 ast_log(LOG_NOTICE, "Cannot pickup your own channel %s.\n", pickup);
00370 } else {
00371 if (partial_pickup) {
00372 if (!pickup_by_part(chan, pickup)) {
00373
00374 return -1;
00375 }
00376 } else if (!pickup_by_channel(chan, pickup)) {
00377
00378 return -1;
00379 }
00380 ast_log(LOG_NOTICE, "No target channel found for %s.\n", pickup);
00381 }
00382 }
00383
00384
00385 return 0;
00386 }
00387
00388 static int unload_module(void)
00389 {
00390 int res;
00391
00392 res = ast_unregister_application(app);
00393 res |= ast_unregister_application(app2);
00394
00395 return res;
00396 }
00397
00398 static int load_module(void)
00399 {
00400 int res;
00401
00402 res = ast_register_application_xml(app, pickup_exec);
00403 res |= ast_register_application_xml(app2, pickupchan_exec);
00404
00405 return res;
00406 }
00407
00408 AST_MODULE_INFO_STANDARD(ASTERISK_GPL_KEY, "Directed Call Pickup Application");