mirror of
				https://github.com/asterisk/asterisk.git
				synced 2025-11-03 20:38:59 +00:00 
			
		
		
		
	git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@78649 65c4cc65-6c06-0410-ace0-fbb531ad65f3
		
			
				
	
	
		
			724 lines
		
	
	
		
			22 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			724 lines
		
	
	
		
			22 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 * Asterisk -- An open source telephony toolkit.
 | 
						|
 *
 | 
						|
 * Copyright (C) 2005 Anthony Minessale II (anthmct@yahoo.com)
 | 
						|
 * Copyright (C) 2005 - 2006, 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>
 | 
						|
 *
 | 
						|
 * \ingroup applications
 | 
						|
 */
 | 
						|
 | 
						|
#include "asterisk.h"
 | 
						|
 | 
						|
ASTERISK_FILE_VERSION(__FILE__, "$Revision$")
 | 
						|
 | 
						|
#include <stdlib.h>
 | 
						|
#include <stdio.h>
 | 
						|
#include <string.h>
 | 
						|
#include <unistd.h>
 | 
						|
#include <ctype.h>
 | 
						|
 | 
						|
#include "asterisk/file.h"
 | 
						|
#include "asterisk/logger.h"
 | 
						|
#include "asterisk/channel.h"
 | 
						|
#include "asterisk/audiohook.h"
 | 
						|
#include "asterisk/features.h"
 | 
						|
#include "asterisk/options.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"
 | 
						|
 | 
						|
#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"
 | 
						|
;
 | 
						|
 | 
						|
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 */
 | 
						|
} chanspy_opt_flags;
 | 
						|
 | 
						|
enum {
 | 
						|
	OPT_ARG_VOLUME = 0,
 | 
						|
	OPT_ARG_GROUP,
 | 
						|
	OPT_ARG_RECORD,
 | 
						|
	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('o', OPTION_READONLY),
 | 
						|
	AST_APP_OPTION('X', OPTION_EXIT),
 | 
						|
});
 | 
						|
 | 
						|
 | 
						|
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 = NULL;
 | 
						|
		
 | 
						|
	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;
 | 
						|
	}
 | 
						|
 | 
						|
	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;
 | 
						|
		
 | 
						|
	if (ast_write(chan, f)) {
 | 
						|
		ast_frfree(f);
 | 
						|
		return -1;
 | 
						|
	}
 | 
						|
 | 
						|
	if (csth->fd)
 | 
						|
		write(csth->fd, f->data, f->datalen);
 | 
						|
 | 
						|
	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, struct ast_channel *spychan, 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);
 | 
						|
 | 
						|
	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;
 | 
						|
}
 | 
						|
 | 
						|
static int channel_spy(struct ast_channel *chan, struct ast_channel *spyee, 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;
 | 
						|
 | 
						|
	if (ast_check_hangup(chan) || ast_check_hangup(spyee))
 | 
						|
		return 0;
 | 
						|
 | 
						|
	name = ast_strdupa(spyee->name);
 | 
						|
	ast_verb(2, "Spying on channel %s\n", name);
 | 
						|
 | 
						|
	memset(&csth, 0, sizeof(csth));
 | 
						|
 | 
						|
	ast_audiohook_init(&csth.spy_audiohook, AST_AUDIOHOOK_TYPE_SPY, "ChanSpy");
 | 
						|
	
 | 
						|
	if (start_spying(spyee, chan, &csth.spy_audiohook)) {
 | 
						|
		ast_audiohook_destroy(&csth.spy_audiohook);
 | 
						|
		return 0;
 | 
						|
	}
 | 
						|
 | 
						|
	if (ast_test_flag(flags, OPTION_WHISPER)) {
 | 
						|
		ast_audiohook_init(&csth.whisper_audiohook, AST_AUDIOHOOK_TYPE_WHISPER, "ChanSpy");
 | 
						|
		start_spying(spyee, chan, &csth.whisper_audiohook);
 | 
						|
	}
 | 
						|
 | 
						|
	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);
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	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);
 | 
						|
	
 | 
						|
	if (option_verbose >= 2)
 | 
						|
		ast_verbose(VERBOSE_PREFIX_2 "Done Spying on channel %s\n", name);
 | 
						|
	
 | 
						|
	return running;
 | 
						|
}
 | 
						|
 | 
						|
static struct ast_channel *next_channel(const struct ast_channel *last, const char *spec,
 | 
						|
					const char *exten, const char *context)
 | 
						|
{
 | 
						|
	struct ast_channel *this;
 | 
						|
 | 
						|
	redo:
 | 
						|
	if (spec)
 | 
						|
		this = ast_walk_channel_by_name_prefix_locked(last, spec, strlen(spec));
 | 
						|
	else if (exten)
 | 
						|
		this = ast_walk_channel_by_exten_locked(last, exten, context);
 | 
						|
	else
 | 
						|
		this = ast_channel_walk_locked(last);
 | 
						|
 | 
						|
	if (this) {
 | 
						|
		ast_channel_unlock(this);
 | 
						|
		if (!strncmp(this->name, "Zap/pseudo", 10))
 | 
						|
			goto redo;
 | 
						|
	}
 | 
						|
 | 
						|
	return this;
 | 
						|
}
 | 
						|
 | 
						|
static int common_exec(struct ast_channel *chan, const struct ast_flags *flags,
 | 
						|
		       int volfactor, const int fd, const char *mygroup, const char *spec,
 | 
						|
		       const char *exten, const char *context)
 | 
						|
{
 | 
						|
	struct ast_channel *peer, *prev, *next;
 | 
						|
	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;
 | 
						|
 | 
						|
	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));
 | 
						|
	}
 | 
						|
 | 
						|
	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 (;;) {
 | 
						|
		if (!ast_test_flag(flags, OPTION_QUIET)) {
 | 
						|
			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;
 | 
						|
		peer = prev = next = NULL;
 | 
						|
 | 
						|
		for (peer = next_channel(peer, spec, exten, context);
 | 
						|
		     peer;
 | 
						|
		     prev = peer, peer = next ? next : next_channel(peer, spec, exten, context), next = NULL) {
 | 
						|
			const char *group;
 | 
						|
			int igrp = !mygroup;
 | 
						|
			char *groups[25];
 | 
						|
			int num_groups = 0;
 | 
						|
			char *dup_group;
 | 
						|
			int x;
 | 
						|
			char *s;
 | 
						|
				
 | 
						|
			if (peer == prev)
 | 
						|
				break;
 | 
						|
 | 
						|
			if (peer == chan)
 | 
						|
				continue;
 | 
						|
 | 
						|
			if (ast_test_flag(flags, OPTION_BRIDGED) && !ast_bridged_channel(peer))
 | 
						|
				continue;
 | 
						|
 | 
						|
			if (ast_check_hangup(peer) || ast_test_flag(peer, AST_FLAG_SPYING))
 | 
						|
				continue;
 | 
						|
 | 
						|
			if (mygroup) {
 | 
						|
				if ((group = pbx_builtin_getvar_helper(peer, "SPYGROUP"))) {
 | 
						|
					dup_group = ast_strdupa(group);
 | 
						|
					num_groups = ast_app_separate_args(dup_group, ':', groups,
 | 
						|
									   sizeof(groups) / sizeof(groups[0]));
 | 
						|
				}
 | 
						|
				
 | 
						|
				for (x = 0; x < num_groups; x++) {
 | 
						|
					if (!strcmp(mygroup, groups[x])) {
 | 
						|
						igrp = 1;
 | 
						|
						break;
 | 
						|
					}
 | 
						|
				}
 | 
						|
			}
 | 
						|
			
 | 
						|
			if (!igrp)
 | 
						|
				continue;
 | 
						|
 | 
						|
			strcpy(peer_name, "spy-");
 | 
						|
			strncat(peer_name, peer->name, AST_NAME_STRLEN);
 | 
						|
			ptr = strchr(peer_name, '/');
 | 
						|
			*ptr++ = '\0';
 | 
						|
			
 | 
						|
			for (s = peer_name; s < ptr; s++)
 | 
						|
				*s = tolower(*s);
 | 
						|
			
 | 
						|
			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)
 | 
						|
						break;
 | 
						|
				} else
 | 
						|
					res = ast_say_character_str(chan, peer_name, "", chan->language);
 | 
						|
				if ((num = atoi(ptr))) 
 | 
						|
					ast_say_digits(chan, atoi(ptr), "", chan->language);
 | 
						|
			}
 | 
						|
			
 | 
						|
			waitms = 5000;
 | 
						|
			res = channel_spy(chan, peer, &volfactor, fd, flags, exitcontext);
 | 
						|
			
 | 
						|
			if (res == -1) {
 | 
						|
				goto exit;
 | 
						|
			} else if (res == -2) {
 | 
						|
				res = 0;
 | 
						|
				goto exit;
 | 
						|
			} else if (res > 1 && spec) {
 | 
						|
				snprintf(nameprefix, AST_NAME_STRLEN, "%s/%d", spec, res);
 | 
						|
				if ((next = ast_get_channel_by_name_prefix_locked(nameprefix, strlen(nameprefix)))) {
 | 
						|
					ast_channel_unlock(next);
 | 
						|
				} else {
 | 
						|
					/* stay on this channel */
 | 
						|
					next = peer;
 | 
						|
				}
 | 
						|
				peer = NULL;
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
exit:
 | 
						|
 | 
						|
	ast_clear_flag(chan, AST_FLAG_SPYING);
 | 
						|
 | 
						|
	ast_channel_setoption(chan, AST_OPTION_TXGAIN, &zero_volume, sizeof(zero_volume), 0);
 | 
						|
 | 
						|
	return res;
 | 
						|
}
 | 
						|
 | 
						|
static int chanspy_exec(struct ast_channel *chan, void *data)
 | 
						|
{
 | 
						|
	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], "%d", &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[512];
 | 
						|
 | 
						|
		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, 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], "%d", &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[512];
 | 
						|
 | 
						|
		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, 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");
 |