123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966 |
- /*
- * Asterisk -- An open source telephony toolkit.
- *
- * Copyright (C) 2005 Anthony Minessale II (anthmct@yahoo.com)
- * Copyright (C) 2005 - 2008, Digium, Inc.
- *
- * A license has been granted to Digium (via disclaimer) for the use of
- * this code.
- *
- * See http://www.asterisk.org for more information about
- * the Asterisk project. Please do not directly contact
- * any of the maintainers of this project for assistance;
- * the project provides a web site, mailing lists and IRC
- * channels for your use.
- *
- * This program is free software, distributed under the terms of
- * the GNU General Public License Version 2. See the LICENSE file
- * at the top of the source tree.
- */
- /*! \file
- *
- * \brief ChanSpy: Listen in on any channel.
- *
- * \author Anthony Minessale II <anthmct@yahoo.com>
- * \author Joshua Colp <jcolp@digium.com>
- * \author Russell Bryant <russell@digium.com>
- *
- * \ingroup applications
- */
- #include "asterisk.h"
- ASTERISK_FILE_VERSION(__FILE__, "$Revision$")
- #include <ctype.h>
- #include <errno.h>
- #include "asterisk/paths.h" /* use ast_config_AST_MONITOR_DIR */
- #include "asterisk/file.h"
- #include "asterisk/channel.h"
- #include "asterisk/audiohook.h"
- #include "asterisk/features.h"
- #include "asterisk/app.h"
- #include "asterisk/utils.h"
- #include "asterisk/say.h"
- #include "asterisk/pbx.h"
- #include "asterisk/translate.h"
- #include "asterisk/module.h"
- #include "asterisk/lock.h"
- #include "asterisk/options.h"
- #define AST_NAME_STRLEN 256
- static const char *tdesc = "Listen to a channel, and optionally whisper into it";
- static const char *app_chan = "ChanSpy";
- static const char *desc_chan =
- " ChanSpy([chanprefix][,options]): This application is used to listen to the\n"
- "audio from an Asterisk channel. This includes the audio coming in and\n"
- "out of the channel being spied on. If the 'chanprefix' parameter is specified,\n"
- "only channels beginning with this string will be spied upon.\n"
- " While spying, the following actions may be performed:\n"
- " - Dialing # cycles the volume level.\n"
- " - Dialing * will stop spying and look for another channel to spy on.\n"
- " - Dialing a series of digits followed by # builds a channel name to append\n"
- " to 'chanprefix'. For example, executing ChanSpy(Agent) and then dialing\n"
- " the digits '1234#' while spying will begin spying on the channel\n"
- " 'Agent/1234'.\n"
- " Note: The X option supersedes the three features above in that if a valid\n"
- " single digit extension exists in the correct context ChanSpy will\n"
- " exit to it. This also disables choosing a channel based on 'chanprefix'\n"
- " and a digit sequence.\n"
- " Options:\n"
- " b - Only spy on channels involved in a bridged call.\n"
- " g(grp) - Match only channels where their SPYGROUP variable is set to\n"
- " contain 'grp' in an optional : delimited list.\n"
- " q - Don't play a beep when beginning to spy on a channel, or speak the\n"
- " selected channel name.\n"
- " r[(basename)] - Record the session to the monitor spool directory. An\n"
- " optional base for the filename may be specified. The\n"
- " default is 'chanspy'.\n"
- " v([value]) - Adjust the initial volume in the range from -4 to 4. A\n"
- " negative value refers to a quieter setting.\n"
- " w - Enable 'whisper' mode, so the spying channel can talk to\n"
- " the spied-on channel.\n"
- " W - Enable 'private whisper' mode, so the spying channel can\n"
- " talk to the spied-on channel but cannot listen to that\n"
- " channel.\n"
- " o - Only listen to audio coming from this channel.\n"
- " X - Allow the user to exit ChanSpy to a valid single digit\n"
- " numeric extension in the current context or the context\n"
- " specified by the SPY_EXIT_CONTEXT channel variable. The\n"
- " name of the last channel that was spied on will be stored\n"
- " in the SPY_CHANNEL variable.\n"
- " e(ext) - Enable 'enforced' mode, so the spying channel can\n"
- " only monitor extensions whose name is in the 'ext' : \n"
- " delimited list.\n"
- ;
- static const char *app_ext = "ExtenSpy";
- static const char *desc_ext =
- " ExtenSpy(exten[@context][,options]): This application is used to listen to the\n"
- "audio from an Asterisk channel. This includes the audio coming in and\n"
- "out of the channel being spied on. Only channels created by outgoing calls for the\n"
- "specified extension will be selected for spying. If the optional context is not\n"
- "supplied, the current channel's context will be used.\n"
- " While spying, the following actions may be performed:\n"
- " - Dialing # cycles the volume level.\n"
- " - Dialing * will stop spying and look for another channel to spy on.\n"
- " Note: The X option superseeds the two features above in that if a valid\n"
- " single digit extension exists in the correct context it ChanSpy will\n"
- " exit to it.\n"
- " Options:\n"
- " b - Only spy on channels involved in a bridged call.\n"
- " g(grp) - Match only channels where their ${SPYGROUP} variable is set to\n"
- " contain 'grp' in an optional : delimited list.\n"
- " q - Don't play a beep when beginning to spy on a channel, or speak the\n"
- " selected channel name.\n"
- " r[(basename)] - Record the session to the monitor spool directory. An\n"
- " optional base for the filename may be specified. The\n"
- " default is 'chanspy'.\n"
- " v([value]) - Adjust the initial volume in the range from -4 to 4. A\n"
- " negative value refers to a quieter setting.\n"
- " w - Enable 'whisper' mode, so the spying channel can talk to\n"
- " the spied-on channel.\n"
- " W - Enable 'private whisper' mode, so the spying channel can\n"
- " talk to the spied-on channel but cannot listen to that\n"
- " channel.\n"
- " o - Only listen to audio coming from this channel.\n"
- " X - Allow the user to exit ChanSpy to a valid single digit\n"
- " numeric extension in the current context or the context\n"
- " specified by the SPY_EXIT_CONTEXT channel variable. The\n"
- " name of the last channel that was spied on will be stored\n"
- " in the SPY_CHANNEL variable.\n"
- ;
- enum {
- OPTION_QUIET = (1 << 0), /* Quiet, no announcement */
- OPTION_BRIDGED = (1 << 1), /* Only look at bridged calls */
- OPTION_VOLUME = (1 << 2), /* Specify initial volume */
- OPTION_GROUP = (1 << 3), /* Only look at channels in group */
- OPTION_RECORD = (1 << 4),
- OPTION_WHISPER = (1 << 5),
- OPTION_PRIVATE = (1 << 6), /* Private Whisper mode */
- OPTION_READONLY = (1 << 7), /* Don't mix the two channels */
- OPTION_EXIT = (1 << 8), /* Exit to a valid single digit extension */
- OPTION_ENFORCED = (1 << 9), /* Enforced mode */
- } chanspy_opt_flags;
- enum {
- OPT_ARG_VOLUME = 0,
- OPT_ARG_GROUP,
- OPT_ARG_RECORD,
- OPT_ARG_ENFORCED,
- OPT_ARG_ARRAY_SIZE,
- } chanspy_opt_args;
- AST_APP_OPTIONS(spy_opts, {
- AST_APP_OPTION('q', OPTION_QUIET),
- AST_APP_OPTION('b', OPTION_BRIDGED),
- AST_APP_OPTION('w', OPTION_WHISPER),
- AST_APP_OPTION('W', OPTION_PRIVATE),
- AST_APP_OPTION_ARG('v', OPTION_VOLUME, OPT_ARG_VOLUME),
- AST_APP_OPTION_ARG('g', OPTION_GROUP, OPT_ARG_GROUP),
- AST_APP_OPTION_ARG('r', OPTION_RECORD, OPT_ARG_RECORD),
- AST_APP_OPTION_ARG('e', OPTION_ENFORCED, OPT_ARG_ENFORCED),
- AST_APP_OPTION('o', OPTION_READONLY),
- AST_APP_OPTION('X', OPTION_EXIT),
- });
- static int next_unique_id_to_use = 0;
- struct chanspy_translation_helper {
- /* spy data */
- struct ast_audiohook spy_audiohook;
- struct ast_audiohook whisper_audiohook;
- int fd;
- int volfactor;
- };
- static void *spy_alloc(struct ast_channel *chan, void *data)
- {
- /* just store the data pointer in the channel structure */
- return data;
- }
- static void spy_release(struct ast_channel *chan, void *data)
- {
- /* nothing to do */
- }
- static int spy_generate(struct ast_channel *chan, void *data, int len, int samples)
- {
- struct chanspy_translation_helper *csth = data;
- struct ast_frame *f, *cur;
- ast_audiohook_lock(&csth->spy_audiohook);
- if (csth->spy_audiohook.status != AST_AUDIOHOOK_STATUS_RUNNING) {
- /* Channel is already gone more than likely */
- ast_audiohook_unlock(&csth->spy_audiohook);
- return -1;
- }
- if (ast_test_flag(&csth->spy_audiohook, OPTION_READONLY)) {
- /* Option 'o' was set, so don't mix channel audio */
- f = ast_audiohook_read_frame(&csth->spy_audiohook, samples, AST_AUDIOHOOK_DIRECTION_READ, AST_FORMAT_SLINEAR);
- } else {
- f = ast_audiohook_read_frame(&csth->spy_audiohook, samples, AST_AUDIOHOOK_DIRECTION_BOTH, AST_FORMAT_SLINEAR);
- }
- ast_audiohook_unlock(&csth->spy_audiohook);
- if (!f)
- return 0;
- for (cur = f; cur; cur = AST_LIST_NEXT(cur, frame_list)) {
- if (ast_write(chan, cur)) {
- ast_frfree(f);
- return -1;
- }
- if (csth->fd) {
- if (write(csth->fd, cur->data, cur->datalen) < 0) {
- ast_log(LOG_WARNING, "write() failed: %s\n", strerror(errno));
- }
- }
- }
- ast_frfree(f);
- return 0;
- }
- static struct ast_generator spygen = {
- .alloc = spy_alloc,
- .release = spy_release,
- .generate = spy_generate,
- };
- static int start_spying(struct ast_channel *chan, const char *spychan_name, struct ast_audiohook *audiohook)
- {
- int res = 0;
- struct ast_channel *peer = NULL;
- ast_log(LOG_NOTICE, "Attaching %s to %s\n", spychan_name, chan->name);
- ast_set_flag(audiohook, AST_AUDIOHOOK_TRIGGER_SYNC | AST_AUDIOHOOK_SMALL_QUEUE);
- res = ast_audiohook_attach(chan, audiohook);
- if (!res && ast_test_flag(chan, AST_FLAG_NBRIDGE) && (peer = ast_bridged_channel(chan))) {
- ast_softhangup(peer, AST_SOFTHANGUP_UNBRIDGE);
- }
- return res;
- }
- struct chanspy_ds {
- struct ast_channel *chan;
- char unique_id[20];
- ast_mutex_t lock;
- };
- static int channel_spy(struct ast_channel *chan, struct chanspy_ds *spyee_chanspy_ds,
- int *volfactor, int fd, const struct ast_flags *flags, char *exitcontext)
- {
- struct chanspy_translation_helper csth;
- int running = 0, res, x = 0;
- char inp[24] = {0};
- char *name;
- struct ast_frame *f;
- struct ast_silence_generator *silgen = NULL;
- struct ast_channel *spyee = NULL;
- const char *spyer_name;
- ast_channel_lock(chan);
- spyer_name = ast_strdupa(chan->name);
- ast_channel_unlock(chan);
- ast_mutex_lock(&spyee_chanspy_ds->lock);
- while ((spyee = spyee_chanspy_ds->chan) && ast_channel_trylock(spyee)) {
- /* avoid a deadlock here, just in case spyee is masqueraded and
- * chanspy_ds_chan_fixup() is called with the channel locked */
- DEADLOCK_AVOIDANCE(&spyee_chanspy_ds->lock);
- }
- ast_mutex_unlock(&spyee_chanspy_ds->lock);
- if (!spyee)
- return 0;
- /* We now hold the channel lock on spyee */
- if (ast_check_hangup(chan) || ast_check_hangup(spyee)) {
- ast_channel_unlock(spyee);
- return 0;
- }
- name = ast_strdupa(spyee->name);
- ast_verb(2, "Spying on channel %s\n", name);
- memset(&csth, 0, sizeof(csth));
- ast_copy_flags(&csth.spy_audiohook, flags, AST_FLAGS_ALL);
- ast_audiohook_init(&csth.spy_audiohook, AST_AUDIOHOOK_TYPE_SPY, "ChanSpy");
- if (start_spying(spyee, spyer_name, &csth.spy_audiohook)) {
- ast_audiohook_destroy(&csth.spy_audiohook);
- ast_channel_unlock(spyee);
- return 0;
- }
- if (ast_test_flag(flags, OPTION_WHISPER)) {
- ast_audiohook_init(&csth.whisper_audiohook, AST_AUDIOHOOK_TYPE_WHISPER, "ChanSpy");
- start_spying(spyee, spyer_name, &csth.whisper_audiohook);
- }
- ast_channel_unlock(spyee);
- spyee = NULL;
- csth.volfactor = *volfactor;
- if (csth.volfactor) {
- csth.spy_audiohook.options.read_volume = csth.volfactor;
- csth.spy_audiohook.options.write_volume = csth.volfactor;
- }
- csth.fd = fd;
- if (ast_test_flag(flags, OPTION_PRIVATE))
- silgen = ast_channel_start_silence_generator(chan);
- else
- ast_activate_generator(chan, &spygen, &csth);
- /* We can no longer rely on 'spyee' being an actual channel;
- it can be hung up and freed out from under us. However, the
- channel destructor will put NULL into our csth.spy.chan
- field when that happens, so that is our signal that the spyee
- channel has gone away.
- */
- /* Note: it is very important that the ast_waitfor() be the first
- condition in this expression, so that if we wait for some period
- of time before receiving a frame from our spying channel, we check
- for hangup on the spied-on channel _after_ knowing that a frame
- has arrived, since the spied-on channel could have gone away while
- we were waiting
- */
- while ((res = ast_waitfor(chan, -1) > -1) && csth.spy_audiohook.status == AST_AUDIOHOOK_STATUS_RUNNING) {
- if (!(f = ast_read(chan)) || ast_check_hangup(chan)) {
- running = -1;
- break;
- }
- if (ast_test_flag(flags, OPTION_WHISPER) && f->frametype == AST_FRAME_VOICE) {
- ast_audiohook_lock(&csth.whisper_audiohook);
- ast_audiohook_write_frame(&csth.whisper_audiohook, AST_AUDIOHOOK_DIRECTION_WRITE, f);
- ast_audiohook_unlock(&csth.whisper_audiohook);
- ast_frfree(f);
- continue;
- }
-
- res = (f->frametype == AST_FRAME_DTMF) ? f->subclass : 0;
- ast_frfree(f);
- if (!res)
- continue;
- if (x == sizeof(inp))
- x = 0;
- if (res < 0) {
- running = -1;
- break;
- }
- if (ast_test_flag(flags, OPTION_EXIT)) {
- char tmp[2];
- tmp[0] = res;
- tmp[1] = '\0';
- if (!ast_goto_if_exists(chan, exitcontext, tmp, 1)) {
- ast_debug(1, "Got DTMF %c, goto context %s\n", tmp[0], exitcontext);
- pbx_builtin_setvar_helper(chan, "SPY_CHANNEL", name);
- running = -2;
- break;
- } else {
- ast_debug(2, "Exit by single digit did not work in chanspy. Extension %s does not exist in context %s\n", tmp, exitcontext);
- }
- } else if (res >= '0' && res <= '9') {
- inp[x++] = res;
- }
- if (res == '*') {
- running = 0;
- break;
- } else if (res == '#') {
- if (!ast_strlen_zero(inp)) {
- running = atoi(inp);
- break;
- }
- (*volfactor)++;
- if (*volfactor > 4)
- *volfactor = -4;
- ast_verb(3, "Setting spy volume on %s to %d\n", chan->name, *volfactor);
- csth.volfactor = *volfactor;
- csth.spy_audiohook.options.read_volume = csth.volfactor;
- csth.spy_audiohook.options.write_volume = csth.volfactor;
- }
- }
- if (ast_test_flag(flags, OPTION_PRIVATE))
- ast_channel_stop_silence_generator(chan, silgen);
- else
- ast_deactivate_generator(chan);
- if (ast_test_flag(flags, OPTION_WHISPER)) {
- ast_audiohook_lock(&csth.whisper_audiohook);
- ast_audiohook_detach(&csth.whisper_audiohook);
- ast_audiohook_unlock(&csth.whisper_audiohook);
- ast_audiohook_destroy(&csth.whisper_audiohook);
- }
- ast_audiohook_lock(&csth.spy_audiohook);
- ast_audiohook_detach(&csth.spy_audiohook);
- ast_audiohook_unlock(&csth.spy_audiohook);
- ast_audiohook_destroy(&csth.spy_audiohook);
-
- ast_verb(2, "Done Spying on channel %s\n", name);
- return running;
- }
- /*!
- * \note This relies on the embedded lock to be recursive, as it may be called
- * due to a call to chanspy_ds_free with the lock held there.
- */
- static void chanspy_ds_destroy(void *data)
- {
- struct chanspy_ds *chanspy_ds = data;
- /* Setting chan to be NULL is an atomic operation, but we don't want this
- * value to change while this lock is held. The lock is held elsewhere
- * while it performs non-atomic operations with this channel pointer */
- ast_mutex_lock(&chanspy_ds->lock);
- chanspy_ds->chan = NULL;
- ast_mutex_unlock(&chanspy_ds->lock);
- }
- static void chanspy_ds_chan_fixup(void *data, struct ast_channel *old_chan, struct ast_channel *new_chan)
- {
- struct chanspy_ds *chanspy_ds = data;
-
- ast_mutex_lock(&chanspy_ds->lock);
- chanspy_ds->chan = new_chan;
- ast_mutex_unlock(&chanspy_ds->lock);
- }
- static const struct ast_datastore_info chanspy_ds_info = {
- .type = "chanspy",
- .destroy = chanspy_ds_destroy,
- .chan_fixup = chanspy_ds_chan_fixup,
- };
- static struct chanspy_ds *chanspy_ds_free(struct chanspy_ds *chanspy_ds)
- {
- struct ast_channel *chan;
- if (!chanspy_ds) {
- return NULL;
- }
- ast_mutex_lock(&chanspy_ds->lock);
- while ((chan = chanspy_ds->chan)) {
- struct ast_datastore *datastore;
- if (ast_channel_trylock(chan)) {
- DEADLOCK_AVOIDANCE(&chanspy_ds->lock);
- continue;
- }
- if ((datastore = ast_channel_datastore_find(chan, &chanspy_ds_info, chanspy_ds->unique_id))) {
- ast_channel_datastore_remove(chan, datastore);
- /* chanspy_ds->chan is NULL after this call */
- chanspy_ds_destroy(datastore->data);
- datastore->data = NULL;
- ast_channel_datastore_free(datastore);
- }
- ast_channel_unlock(chan);
- break;
- }
- ast_mutex_unlock(&chanspy_ds->lock);
- return NULL;
- }
- /*! \note Returns the channel in the chanspy_ds locked as well as the chanspy_ds locked */
- static struct chanspy_ds *setup_chanspy_ds(struct ast_channel *chan, struct chanspy_ds *chanspy_ds)
- {
- struct ast_datastore *datastore = NULL;
- ast_mutex_lock(&chanspy_ds->lock);
- if (!(datastore = ast_channel_datastore_alloc(&chanspy_ds_info, chanspy_ds->unique_id))) {
- ast_mutex_unlock(&chanspy_ds->lock);
- chanspy_ds = chanspy_ds_free(chanspy_ds);
- ast_channel_unlock(chan);
- return NULL;
- }
-
- chanspy_ds->chan = chan;
- datastore->data = chanspy_ds;
- ast_channel_datastore_add(chan, datastore);
- return chanspy_ds;
- }
- static struct chanspy_ds *next_channel(struct ast_channel *chan,
- const struct ast_channel *last, const char *spec,
- const char *exten, const char *context, struct chanspy_ds *chanspy_ds)
- {
- struct ast_channel *next;
- const size_t pseudo_len = strlen("DAHDI/pseudo");
- redo:
- if (!ast_strlen_zero(spec))
- next = ast_walk_channel_by_name_prefix_locked(last, spec, strlen(spec));
- else if (!ast_strlen_zero(exten))
- next = ast_walk_channel_by_exten_locked(last, exten, context);
- else
- next = ast_channel_walk_locked(last);
- if (!next)
- return NULL;
- if (!strncmp(next->name, "DAHDI/pseudo", pseudo_len)) {
- last = next;
- ast_channel_unlock(next);
- goto redo;
- } else if (next == chan) {
- last = next;
- ast_channel_unlock(next);
- goto redo;
- }
- return setup_chanspy_ds(next, chanspy_ds);
- }
- static int common_exec(struct ast_channel *chan, const struct ast_flags *flags,
- int volfactor, const int fd, const char *mygroup, const char *myenforced,
- const char *spec, const char *exten, const char *context)
- {
- char nameprefix[AST_NAME_STRLEN];
- char peer_name[AST_NAME_STRLEN + 5];
- char exitcontext[AST_MAX_CONTEXT] = "";
- signed char zero_volume = 0;
- int waitms;
- int res;
- char *ptr;
- int num;
- int num_spyed_upon = 1;
- struct chanspy_ds chanspy_ds = { 0, };
- if (ast_test_flag(flags, OPTION_EXIT)) {
- const char *c;
- if ((c = pbx_builtin_getvar_helper(chan, "SPY_EXIT_CONTEXT")))
- ast_copy_string(exitcontext, c, sizeof(exitcontext));
- else if (!ast_strlen_zero(chan->macrocontext))
- ast_copy_string(exitcontext, chan->macrocontext, sizeof(exitcontext));
- else
- ast_copy_string(exitcontext, chan->context, sizeof(exitcontext));
- }
- ast_mutex_init(&chanspy_ds.lock);
- snprintf(chanspy_ds.unique_id, sizeof(chanspy_ds.unique_id), "%d", ast_atomic_fetchadd_int(&next_unique_id_to_use, +1));
- if (chan->_state != AST_STATE_UP)
- ast_answer(chan);
- ast_set_flag(chan, AST_FLAG_SPYING); /* so nobody can spy on us while we are spying */
- waitms = 100;
- for (;;) {
- struct chanspy_ds *peer_chanspy_ds = NULL, *next_chanspy_ds = NULL;
- struct ast_channel *prev = NULL, *peer = NULL;
- if (!ast_test_flag(flags, OPTION_QUIET) && num_spyed_upon) {
- res = ast_streamfile(chan, "beep", chan->language);
- if (!res)
- res = ast_waitstream(chan, "");
- else if (res < 0) {
- ast_clear_flag(chan, AST_FLAG_SPYING);
- break;
- }
- if (!ast_strlen_zero(exitcontext)) {
- char tmp[2];
- tmp[0] = res;
- tmp[1] = '\0';
- if (!ast_goto_if_exists(chan, exitcontext, tmp, 1))
- goto exit;
- else
- ast_debug(2, "Exit by single digit did not work in chanspy. Extension %s does not exist in context %s\n", tmp, exitcontext);
- }
- }
- res = ast_waitfordigit(chan, waitms);
- if (res < 0) {
- ast_clear_flag(chan, AST_FLAG_SPYING);
- break;
- }
- if (!ast_strlen_zero(exitcontext)) {
- char tmp[2];
- tmp[0] = res;
- tmp[1] = '\0';
- if (!ast_goto_if_exists(chan, exitcontext, tmp, 1))
- goto exit;
- else
- ast_debug(2, "Exit by single digit did not work in chanspy. Extension %s does not exist in context %s\n", tmp, exitcontext);
- }
- /* reset for the next loop around, unless overridden later */
- waitms = 100;
- num_spyed_upon = 0;
- for (peer_chanspy_ds = next_channel(chan, prev, spec, exten, context, &chanspy_ds);
- peer_chanspy_ds;
- chanspy_ds_free(peer_chanspy_ds), prev = peer,
- peer_chanspy_ds = next_chanspy_ds ? next_chanspy_ds :
- next_channel(chan, prev, spec, exten, context, &chanspy_ds), next_chanspy_ds = NULL) {
- int igrp = !mygroup;
- int ienf = !myenforced;
- char *s;
- peer = peer_chanspy_ds->chan;
- ast_mutex_unlock(&peer_chanspy_ds->lock);
- if (peer == prev) {
- ast_channel_unlock(peer);
- chanspy_ds_free(peer_chanspy_ds);
- break;
- }
- if (ast_check_hangup(chan)) {
- ast_channel_unlock(peer);
- chanspy_ds_free(peer_chanspy_ds);
- break;
- }
- if (ast_test_flag(flags, OPTION_BRIDGED) && !ast_bridged_channel(peer)) {
- ast_channel_unlock(peer);
- continue;
- }
- if (ast_check_hangup(peer) || ast_test_flag(peer, AST_FLAG_SPYING)) {
- ast_channel_unlock(peer);
- continue;
- }
- if (mygroup) {
- int num_groups = 0;
- char dup_group[512];
- char *groups[25];
- const char *group;
- int x;
- if ((group = pbx_builtin_getvar_helper(peer, "SPYGROUP"))) {
- ast_copy_string(dup_group, group, sizeof(dup_group));
- num_groups = ast_app_separate_args(dup_group, ':', groups,
- ARRAY_LEN(groups));
- }
- for (x = 0; x < num_groups; x++) {
- if (!strcmp(mygroup, groups[x])) {
- igrp = 1;
- break;
- }
- }
- }
- if (!igrp) {
- ast_channel_unlock(peer);
- continue;
- }
- if (myenforced) {
- char ext[AST_CHANNEL_NAME + 3];
- char buffer[512];
- char *end;
- snprintf(buffer, sizeof(buffer) - 1, ":%s:", myenforced);
- ast_copy_string(ext + 1, peer->name, sizeof(ext) - 1);
- if ((end = strchr(ext, '-'))) {
- *end++ = ':';
- *end = '\0';
- }
- ext[0] = ':';
- if (strcasestr(buffer, ext)) {
- ienf = 1;
- }
- }
- if (!ienf) {
- continue;
- }
- strcpy(peer_name, "spy-");
- strncat(peer_name, peer->name, AST_NAME_STRLEN - 4 - 1);
- ptr = strchr(peer_name, '/');
- *ptr++ = '\0';
- for (s = peer_name; s < ptr; s++)
- *s = tolower(*s);
- /* We have to unlock the peer channel here to avoid a deadlock.
- * So, when we need to dereference it again, we have to lock the
- * datastore and get the pointer from there to see if the channel
- * is still valid. */
- ast_channel_unlock(peer);
- if (!ast_test_flag(flags, OPTION_QUIET)) {
- if (ast_fileexists(peer_name, NULL, NULL) != -1) {
- res = ast_streamfile(chan, peer_name, chan->language);
- if (!res)
- res = ast_waitstream(chan, "");
- if (res) {
- chanspy_ds_free(peer_chanspy_ds);
- break;
- }
- } else
- res = ast_say_character_str(chan, peer_name, "", chan->language);
- if ((num = atoi(ptr)))
- ast_say_digits(chan, atoi(ptr), "", chan->language);
- }
- res = channel_spy(chan, peer_chanspy_ds, &volfactor, fd, flags, exitcontext);
- num_spyed_upon++;
- if (res == -1) {
- chanspy_ds_free(peer_chanspy_ds);
- goto exit;
- } else if (res == -2) {
- res = 0;
- chanspy_ds_free(peer_chanspy_ds);
- goto exit;
- } else if (res > 1 && spec) {
- struct ast_channel *next;
- snprintf(nameprefix, AST_NAME_STRLEN, "%s/%d", spec, res);
- if ((next = ast_get_channel_by_name_prefix_locked(nameprefix, strlen(nameprefix)))) {
- peer_chanspy_ds = chanspy_ds_free(peer_chanspy_ds);
- next_chanspy_ds = setup_chanspy_ds(next, &chanspy_ds);
- } else {
- /* stay on this channel, if it is still valid */
- ast_mutex_lock(&peer_chanspy_ds->lock);
- if (peer_chanspy_ds->chan) {
- ast_channel_lock(peer_chanspy_ds->chan);
- next_chanspy_ds = peer_chanspy_ds;
- peer_chanspy_ds = NULL;
- } else {
- /* the channel is gone */
- ast_mutex_unlock(&peer_chanspy_ds->lock);
- next_chanspy_ds = NULL;
- }
- }
- peer = NULL;
- }
- }
- if (res == -1 || ast_check_hangup(chan))
- break;
- }
- exit:
- ast_clear_flag(chan, AST_FLAG_SPYING);
- ast_channel_setoption(chan, AST_OPTION_TXGAIN, &zero_volume, sizeof(zero_volume), 0);
- ast_mutex_lock(&chanspy_ds.lock);
- ast_mutex_unlock(&chanspy_ds.lock);
- ast_mutex_destroy(&chanspy_ds.lock);
- return res;
- }
- static int chanspy_exec(struct ast_channel *chan, void *data)
- {
- char *myenforced = NULL;
- char *mygroup = NULL;
- char *recbase = NULL;
- int fd = 0;
- struct ast_flags flags;
- int oldwf = 0;
- int volfactor = 0;
- int res;
- AST_DECLARE_APP_ARGS(args,
- AST_APP_ARG(spec);
- AST_APP_ARG(options);
- );
- char *opts[OPT_ARG_ARRAY_SIZE];
- data = ast_strdupa(data);
- AST_STANDARD_APP_ARGS(args, data);
- if (args.spec && !strcmp(args.spec, "all"))
- args.spec = NULL;
- if (args.options) {
- ast_app_parse_options(spy_opts, &flags, opts, args.options);
- if (ast_test_flag(&flags, OPTION_GROUP))
- mygroup = opts[OPT_ARG_GROUP];
- if (ast_test_flag(&flags, OPTION_RECORD) &&
- !(recbase = opts[OPT_ARG_RECORD]))
- recbase = "chanspy";
- if (ast_test_flag(&flags, OPTION_VOLUME) && opts[OPT_ARG_VOLUME]) {
- int vol;
- if ((sscanf(opts[OPT_ARG_VOLUME], "%30d", &vol) != 1) || (vol > 4) || (vol < -4))
- ast_log(LOG_NOTICE, "Volume factor must be a number between -4 and 4\n");
- else
- volfactor = vol;
- }
- if (ast_test_flag(&flags, OPTION_PRIVATE))
- ast_set_flag(&flags, OPTION_WHISPER);
- if (ast_test_flag(&flags, OPTION_ENFORCED))
- myenforced = opts[OPT_ARG_ENFORCED];
- } else
- ast_clear_flag(&flags, AST_FLAGS_ALL);
- oldwf = chan->writeformat;
- if (ast_set_write_format(chan, AST_FORMAT_SLINEAR) < 0) {
- ast_log(LOG_ERROR, "Could Not Set Write Format.\n");
- return -1;
- }
- if (recbase) {
- char filename[PATH_MAX];
- snprintf(filename, sizeof(filename), "%s/%s.%d.raw", ast_config_AST_MONITOR_DIR, recbase, (int) time(NULL));
- if ((fd = open(filename, O_CREAT | O_WRONLY | O_TRUNC, AST_FILE_MODE)) <= 0) {
- ast_log(LOG_WARNING, "Cannot open '%s' for recording\n", filename);
- fd = 0;
- }
- }
- res = common_exec(chan, &flags, volfactor, fd, mygroup, myenforced, args.spec, NULL, NULL);
- if (fd)
- close(fd);
- if (oldwf && ast_set_write_format(chan, oldwf) < 0)
- ast_log(LOG_ERROR, "Could Not Set Write Format.\n");
- return res;
- }
- static int extenspy_exec(struct ast_channel *chan, void *data)
- {
- char *ptr, *exten = NULL;
- char *mygroup = NULL;
- char *recbase = NULL;
- int fd = 0;
- struct ast_flags flags;
- int oldwf = 0;
- int volfactor = 0;
- int res;
- AST_DECLARE_APP_ARGS(args,
- AST_APP_ARG(context);
- AST_APP_ARG(options);
- );
- data = ast_strdupa(data);
- AST_STANDARD_APP_ARGS(args, data);
- if (!ast_strlen_zero(args.context) && (ptr = strchr(args.context, '@'))) {
- exten = args.context;
- *ptr++ = '\0';
- args.context = ptr;
- }
- if (ast_strlen_zero(args.context))
- args.context = ast_strdupa(chan->context);
- if (args.options) {
- char *opts[OPT_ARG_ARRAY_SIZE];
- ast_app_parse_options(spy_opts, &flags, opts, args.options);
- if (ast_test_flag(&flags, OPTION_GROUP))
- mygroup = opts[OPT_ARG_GROUP];
- if (ast_test_flag(&flags, OPTION_RECORD) &&
- !(recbase = opts[OPT_ARG_RECORD]))
- recbase = "chanspy";
- if (ast_test_flag(&flags, OPTION_VOLUME) && opts[OPT_ARG_VOLUME]) {
- int vol;
- if ((sscanf(opts[OPT_ARG_VOLUME], "%30d", &vol) != 1) || (vol > 4) || (vol < -4))
- ast_log(LOG_NOTICE, "Volume factor must be a number between -4 and 4\n");
- else
- volfactor = vol;
- }
- if (ast_test_flag(&flags, OPTION_PRIVATE))
- ast_set_flag(&flags, OPTION_WHISPER);
- } else
- ast_clear_flag(&flags, AST_FLAGS_ALL);
- oldwf = chan->writeformat;
- if (ast_set_write_format(chan, AST_FORMAT_SLINEAR) < 0) {
- ast_log(LOG_ERROR, "Could Not Set Write Format.\n");
- return -1;
- }
- if (recbase) {
- char filename[PATH_MAX];
- snprintf(filename, sizeof(filename), "%s/%s.%d.raw", ast_config_AST_MONITOR_DIR, recbase, (int) time(NULL));
- if ((fd = open(filename, O_CREAT | O_WRONLY | O_TRUNC, AST_FILE_MODE)) <= 0) {
- ast_log(LOG_WARNING, "Cannot open '%s' for recording\n", filename);
- fd = 0;
- }
- }
- res = common_exec(chan, &flags, volfactor, fd, mygroup, NULL, NULL, exten, args.context);
- if (fd)
- close(fd);
- if (oldwf && ast_set_write_format(chan, oldwf) < 0)
- ast_log(LOG_ERROR, "Could Not Set Write Format.\n");
- return res;
- }
- static int unload_module(void)
- {
- int res = 0;
- res |= ast_unregister_application(app_chan);
- res |= ast_unregister_application(app_ext);
- return res;
- }
- static int load_module(void)
- {
- int res = 0;
- res |= ast_register_application(app_chan, chanspy_exec, tdesc, desc_chan);
- res |= ast_register_application(app_ext, extenspy_exec, tdesc, desc_ext);
- return res;
- }
- AST_MODULE_INFO_STANDARD(ASTERISK_GPL_KEY, "Listen to the audio of an active channel");
|