| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  |  * Asterisk -- An open source telephony toolkit. | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * Copyright (C) 2005, Anthony Minessale II | 
					
						
							| 
									
										
										
										
											2006-01-03 22:16:23 +00:00
										 |  |  |  * Copyright (C) 2005 - 2006, Digium, Inc. | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * Mark Spencer <markster@digium.com> | 
					
						
							|  |  |  |  * Kevin P. Fleming <kpfleming@digium.com> | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  |  * Based on app_muxmon.c provided by | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  |  * Anthony Minessale II <anthmct@yahoo.com> | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  |  * 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. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  |  * This program is free software, distributed under the terms of | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  |  * the GNU General Public License Version 2. See the LICENSE file | 
					
						
							|  |  |  |  * at the top of the source tree. | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-24 20:12:06 +00:00
										 |  |  | /*! \file
 | 
					
						
							| 
									
										
										
										
											2005-12-30 21:18:06 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2005-10-31 15:33:27 +00:00
										 |  |  |  * \brief MixMonitor() - Record a call and mix the audio during the recording | 
					
						
							| 
									
										
										
										
											2005-11-06 15:09:47 +00:00
										 |  |  |  * \ingroup applications | 
					
						
							| 
									
										
										
										
											2005-12-30 21:18:06 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * \author Mark Spencer <markster@digium.com> | 
					
						
							|  |  |  |  * \author Kevin P. Fleming <kpfleming@digium.com> | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \note Based on app_muxmon.c provided by | 
					
						
							|  |  |  |  * Anthony Minessale II <anthmct@yahoo.com> | 
					
						
							| 
									
										
										
										
											2005-10-24 20:12:06 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-06-07 18:54:56 +00:00
										 |  |  | #include "asterisk.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ASTERISK_FILE_VERSION(__FILE__, "$Revision$") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-11-20 23:16:15 +00:00
										 |  |  | #include "asterisk/paths.h"	/* use ast_config_AST_MONITOR_DIR */
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | #include "asterisk/file.h"
 | 
					
						
							| 
									
										
										
										
											2007-08-08 19:30:52 +00:00
										 |  |  | #include "asterisk/audiohook.h"
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | #include "asterisk/pbx.h"
 | 
					
						
							|  |  |  | #include "asterisk/module.h"
 | 
					
						
							|  |  |  | #include "asterisk/cli.h"
 | 
					
						
							|  |  |  | #include "asterisk/app.h"
 | 
					
						
							| 
									
										
										
										
											2007-11-22 03:50:04 +00:00
										 |  |  | #include "asterisk/channel.h"
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | #define get_volfactor(x) x ? ((x > 0) ? (1 << x) : ((1 << abs(x)) * -1)) : 0
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static const char *app = "MixMonitor"; | 
					
						
							|  |  |  | static const char *synopsis = "Record a call and mix the audio during the recording"; | 
					
						
							|  |  |  | static const char *desc = "" | 
					
						
							| 
									
										
										
										
											2007-11-06 19:04:45 +00:00
										 |  |  | "  MixMonitor(<file>.<ext>[,<options>[,<command>]]):\n" | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | "Records the audio on the current channel to the specified file.\n" | 
					
						
							|  |  |  | "If the filename is an absolute path, uses that path, otherwise\n" | 
					
						
							|  |  |  | "creates the file in the configured monitoring directory from\n" | 
					
						
							|  |  |  | "asterisk.conf.\n\n" | 
					
						
							|  |  |  | "Valid options:\n" | 
					
						
							|  |  |  | " a      - Append to the file instead of overwriting it.\n" | 
					
						
							|  |  |  | " b      - Only save audio to the file while the channel is bridged.\n" | 
					
						
							| 
									
										
										
										
											2007-01-25 01:54:39 +00:00
										 |  |  | "          Note: Does not include conferences or sounds played to each bridged\n" | 
					
						
							|  |  |  | "                party.\n" | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | " v(<x>) - Adjust the heard volume by a factor of <x> (range -4 to 4)\n"	 | 
					
						
							|  |  |  | " V(<x>) - Adjust the spoken volume by a factor of <x> (range -4 to 4)\n"	 | 
					
						
							|  |  |  | " W(<x>) - Adjust the both heard and spoken volumes by a factor of <x>\n" | 
					
						
							|  |  |  | "         (range -4 to 4)\n\n"	 | 
					
						
							|  |  |  | "<command> will be executed when the recording is over\n" | 
					
						
							| 
									
										
										
										
											2007-06-27 23:26:46 +00:00
										 |  |  | "Any strings matching ^{X} will be unescaped to ${X}.\n" | 
					
						
							|  |  |  | "All variables will be evaluated at the time MixMonitor is called.\n" | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | "The variable MIXMONITOR_FILENAME will contain the filename used to record.\n" | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | ""; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-16 17:37:03 +00:00
										 |  |  | static const char *stop_app = "StopMixMonitor"; | 
					
						
							|  |  |  | static const char *stop_synopsis = "Stop recording a call through MixMonitor"; | 
					
						
							|  |  |  | static const char *stop_desc = "" | 
					
						
							| 
									
										
										
										
											2007-11-06 19:04:45 +00:00
										 |  |  | "  StopMixMonitor():\n" | 
					
						
							| 
									
										
										
										
											2006-02-16 17:37:03 +00:00
										 |  |  | "Stops the audio recording that was started with a call to MixMonitor()\n" | 
					
						
							|  |  |  | "on the current channel.\n" | 
					
						
							|  |  |  | ""; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-04-14 14:08:19 +00:00
										 |  |  | struct module_symbols *me; | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | static const char *mixmonitor_spy_type = "MixMonitor"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct mixmonitor { | 
					
						
							| 
									
										
										
										
											2007-08-08 19:30:52 +00:00
										 |  |  | 	struct ast_audiohook audiohook; | 
					
						
							| 
									
										
										
										
											2007-01-22 19:22:07 +00:00
										 |  |  | 	char *filename; | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 	char *post_process; | 
					
						
							| 
									
										
										
										
											2006-06-12 21:37:29 +00:00
										 |  |  | 	char *name; | 
					
						
							| 
									
										
										
											
												After some study, thought, comparing, etc. I've backed out the previous universal mod to make ast_flags a 64 bit thing. Instead, I added a 64-bit version of ast_flags (ast_flags64), and 64-bit versions of the test-flag, set-flag, etc. macros, and an app_parse_options64 routine, and I use these in app_dial alone, to eliminate the 30-option limit it had grown to meet. There is room now for 32 more options and flags. I was heavily tempted to implement some of the other ideas that were presented, but this solution does not intro any new versions of dial, doesn't have a different API, has a minimal/zero impact on code outside of dial, and doesn't seriously (I hope) affect the code structure of dial. It's the best I can think of right now. My goal was NOT to rewrite dial. I leave that to a future, coordinated effort.
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@75983 65c4cc65-6c06-0410-ace0-fbb531ad65f3
											
										 
											2007-07-19 23:24:27 +00:00
										 |  |  | 	unsigned int flags; | 
					
						
							| 
									
										
										
										
											2007-10-31 17:56:21 +00:00
										 |  |  | 	struct ast_channel *chan; | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | enum { | 
					
						
							| 
									
										
										
										
											2005-11-03 21:19:11 +00:00
										 |  |  | 	MUXFLAG_APPEND = (1 << 1), | 
					
						
							|  |  |  | 	MUXFLAG_BRIDGED = (1 << 2), | 
					
						
							|  |  |  | 	MUXFLAG_VOLUME = (1 << 3), | 
					
						
							|  |  |  | 	MUXFLAG_READVOLUME = (1 << 4), | 
					
						
							|  |  |  | 	MUXFLAG_WRITEVOLUME = (1 << 5), | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | } mixmonitor_flags; | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-11-03 21:19:11 +00:00
										 |  |  | enum { | 
					
						
							|  |  |  | 	OPT_ARG_READVOLUME = 0, | 
					
						
							|  |  |  | 	OPT_ARG_WRITEVOLUME, | 
					
						
							|  |  |  | 	OPT_ARG_VOLUME, | 
					
						
							|  |  |  | 	OPT_ARG_ARRAY_SIZE, | 
					
						
							|  |  |  | } mixmonitor_args; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | AST_APP_OPTIONS(mixmonitor_opts, { | 
					
						
							|  |  |  | 	AST_APP_OPTION('a', MUXFLAG_APPEND), | 
					
						
							|  |  |  | 	AST_APP_OPTION('b', MUXFLAG_BRIDGED), | 
					
						
							|  |  |  | 	AST_APP_OPTION_ARG('v', MUXFLAG_READVOLUME, OPT_ARG_READVOLUME), | 
					
						
							|  |  |  | 	AST_APP_OPTION_ARG('V', MUXFLAG_WRITEVOLUME, OPT_ARG_WRITEVOLUME), | 
					
						
							|  |  |  | 	AST_APP_OPTION_ARG('W', MUXFLAG_VOLUME, OPT_ARG_VOLUME), | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-08-08 19:30:52 +00:00
										 |  |  | static int startmon(struct ast_channel *chan, struct ast_audiohook *audiohook)  | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2007-08-08 19:30:52 +00:00
										 |  |  | 	struct ast_channel *peer = NULL; | 
					
						
							|  |  |  | 	int res = 0; | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 	if (!chan) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-08-08 19:30:52 +00:00
										 |  |  | 	ast_audiohook_attach(chan, audiohook); | 
					
						
							| 
									
										
										
										
											2006-06-12 21:37:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 	if (!res && ast_test_flag(chan, AST_FLAG_NBRIDGE) && (peer = ast_bridged_channel(chan))) | 
					
						
							|  |  |  | 		ast_softhangup(peer, AST_SOFTHANGUP_UNBRIDGE);	 | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | #define SAMPLES_PER_FRAME 160
 | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | static void *mixmonitor_thread(void *obj)  | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct mixmonitor *mixmonitor = obj; | 
					
						
							| 
									
										
										
										
											2007-01-22 19:22:07 +00:00
										 |  |  | 	struct ast_filestream *fs = NULL; | 
					
						
							|  |  |  | 	unsigned int oflags; | 
					
						
							|  |  |  | 	char *ext; | 
					
						
							|  |  |  | 	int errflag = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-12-14 14:48:38 +00:00
										 |  |  | 	ast_verb(2, "Begin MixMonitor Recording %s\n", mixmonitor->name); | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2007-08-08 19:30:52 +00:00
										 |  |  | 	ast_audiohook_lock(&mixmonitor->audiohook); | 
					
						
							| 
									
										
										
										
											2006-03-04 11:51:45 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-08-08 19:30:52 +00:00
										 |  |  | 	while (mixmonitor->audiohook.status == AST_AUDIOHOOK_STATUS_RUNNING) { | 
					
						
							|  |  |  | 		struct ast_frame *fr = NULL; | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-08-08 19:30:52 +00:00
										 |  |  | 		ast_audiohook_trigger_wait(&mixmonitor->audiohook); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (mixmonitor->audiohook.status != AST_AUDIOHOOK_STATUS_RUNNING) | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 			break; | 
					
						
							| 
									
										
										
										
											2007-08-08 19:30:52 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		if (!(fr = ast_audiohook_read_frame(&mixmonitor->audiohook, SAMPLES_PER_FRAME, AST_AUDIOHOOK_DIRECTION_BOTH, AST_FORMAT_SLINEAR))) | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-10-31 17:56:21 +00:00
										 |  |  | 		if (!ast_test_flag(mixmonitor, MUXFLAG_BRIDGED) || ast_bridged_channel(mixmonitor->chan)) { | 
					
						
							|  |  |  | 			/* Initialize the file if not already done so */ | 
					
						
							|  |  |  | 			if (!fs && !errflag) { | 
					
						
							|  |  |  | 				oflags = O_CREAT | O_WRONLY; | 
					
						
							|  |  |  | 				oflags |= ast_test_flag(mixmonitor, MUXFLAG_APPEND) ? O_APPEND : O_TRUNC; | 
					
						
							|  |  |  | 				 | 
					
						
							|  |  |  | 				if ((ext = strrchr(mixmonitor->filename, '.'))) | 
					
						
							|  |  |  | 					*(ext++) = '\0'; | 
					
						
							|  |  |  | 				else | 
					
						
							|  |  |  | 					ext = "raw"; | 
					
						
							|  |  |  | 				 | 
					
						
							|  |  |  | 				if (!(fs = ast_writefile(mixmonitor->filename, ext, NULL, oflags, 0, 0644))) { | 
					
						
							|  |  |  | 					ast_log(LOG_ERROR, "Cannot open %s.%s\n", mixmonitor->filename, ext); | 
					
						
							|  |  |  | 					errflag = 1; | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2007-10-31 17:56:21 +00:00
										 |  |  | 			 | 
					
						
							|  |  |  | 			/* Write out frame */ | 
					
						
							|  |  |  | 			if (fs) | 
					
						
							|  |  |  | 				ast_writestream(fs, fr); | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2007-08-08 19:30:52 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		/* All done! free it. */ | 
					
						
							|  |  |  | 		ast_frame_free(fr, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2006-06-12 21:37:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-08-08 19:30:52 +00:00
										 |  |  | 	ast_audiohook_detach(&mixmonitor->audiohook); | 
					
						
							|  |  |  | 	ast_audiohook_unlock(&mixmonitor->audiohook); | 
					
						
							|  |  |  | 	ast_audiohook_destroy(&mixmonitor->audiohook); | 
					
						
							| 
									
										
										
										
											2006-01-03 17:25:39 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-12-14 14:48:38 +00:00
										 |  |  | 	ast_verb(2, "End MixMonitor Recording %s\n", mixmonitor->name); | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-11-26 17:23:28 +00:00
										 |  |  | 	if (fs) | 
					
						
							|  |  |  | 		ast_closestream(fs); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-06-13 13:40:58 +00:00
										 |  |  | 	if (mixmonitor->post_process) { | 
					
						
							| 
									
										
										
										
											2007-12-14 14:48:38 +00:00
										 |  |  | 		ast_verb(2, "Executing [%s]\n", mixmonitor->post_process); | 
					
						
							| 
									
										
										
										
											2006-06-12 21:37:29 +00:00
										 |  |  | 		ast_safe_system(mixmonitor->post_process); | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-06-06 21:20:11 +00:00
										 |  |  | 	ast_free(mixmonitor); | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
											
												After some study, thought, comparing, etc. I've backed out the previous universal mod to make ast_flags a 64 bit thing. Instead, I added a 64-bit version of ast_flags (ast_flags64), and 64-bit versions of the test-flag, set-flag, etc. macros, and an app_parse_options64 routine, and I use these in app_dial alone, to eliminate the 30-option limit it had grown to meet. There is room now for 32 more options and flags. I was heavily tempted to implement some of the other ideas that were presented, but this solution does not intro any new versions of dial, doesn't have a different API, has a minimal/zero impact on code outside of dial, and doesn't seriously (I hope) affect the code structure of dial. It's the best I can think of right now. My goal was NOT to rewrite dial. I leave that to a future, coordinated effort.
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@75983 65c4cc65-6c06-0410-ace0-fbb531ad65f3
											
										 
											2007-07-19 23:24:27 +00:00
										 |  |  | static void launch_monitor_thread(struct ast_channel *chan, const char *filename, unsigned int flags, | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 				  int readvol, int writevol, const char *post_process)  | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	pthread_t thread; | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 	struct mixmonitor *mixmonitor; | 
					
						
							| 
									
										
										
										
											2007-11-13 01:19:53 +00:00
										 |  |  | 	char postprocess2[1024] = ""; | 
					
						
							| 
									
										
										
										
											2006-06-13 13:40:58 +00:00
										 |  |  | 	size_t len; | 
					
						
							| 
									
										
										
										
											2006-06-12 21:37:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-01-22 19:22:07 +00:00
										 |  |  | 	len = sizeof(*mixmonitor) + strlen(chan->name) + strlen(filename) + 2; | 
					
						
							| 
									
										
										
										
											2006-06-13 13:40:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-11-12 18:44:36 +00:00
										 |  |  | 	postprocess2[0] = 0; | 
					
						
							| 
									
										
										
										
											2006-06-13 13:40:58 +00:00
										 |  |  | 	/* If a post process system command is given attach it to the structure */ | 
					
						
							|  |  |  | 	if (!ast_strlen_zero(post_process)) { | 
					
						
							|  |  |  | 		char *p1, *p2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		p1 = ast_strdupa(post_process); | 
					
						
							|  |  |  | 		for (p2 = p1; *p2 ; p2++) { | 
					
						
							|  |  |  | 			if (*p2 == '^' && *(p2+1) == '{') { | 
					
						
							|  |  |  | 				*p2 = '$'; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		pbx_substitute_variables_helper(chan, p1, postprocess2, sizeof(postprocess2) - 1); | 
					
						
							|  |  |  | 		if (!ast_strlen_zero(postprocess2)) | 
					
						
							|  |  |  | 			len += strlen(postprocess2) + 1; | 
					
						
							| 
									
										
										
										
											2006-06-12 21:37:29 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-06-13 13:40:58 +00:00
										 |  |  | 	/* Pre-allocate mixmonitor structure and spy */ | 
					
						
							| 
									
										
										
										
											2007-06-06 21:20:11 +00:00
										 |  |  | 	if (!(mixmonitor = ast_calloc(1, len))) { | 
					
						
							| 
									
										
										
										
											2006-06-13 13:40:58 +00:00
										 |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-06-12 21:37:29 +00:00
										 |  |  | 	/* Copy over flags and channel name */ | 
					
						
							|  |  |  | 	mixmonitor->flags = flags; | 
					
						
							| 
									
										
										
										
											2007-10-31 17:56:21 +00:00
										 |  |  | 	mixmonitor->chan = chan; | 
					
						
							| 
									
										
										
										
											2006-06-13 13:40:58 +00:00
										 |  |  | 	mixmonitor->name = (char *) mixmonitor + sizeof(*mixmonitor); | 
					
						
							|  |  |  | 	strcpy(mixmonitor->name, chan->name); | 
					
						
							|  |  |  | 	if (!ast_strlen_zero(postprocess2)) { | 
					
						
							| 
									
										
										
										
											2007-01-29 23:45:39 +00:00
										 |  |  | 		mixmonitor->post_process = mixmonitor->name + strlen(mixmonitor->name) + strlen(filename) + 2; | 
					
						
							| 
									
										
										
										
											2006-06-13 13:40:58 +00:00
										 |  |  | 		strcpy(mixmonitor->post_process, postprocess2); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2006-06-12 21:37:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-01-22 19:22:07 +00:00
										 |  |  | 	mixmonitor->filename = (char *) mixmonitor + sizeof(*mixmonitor) + strlen(chan->name) + 1; | 
					
						
							|  |  |  | 	strcpy(mixmonitor->filename, filename); | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-06-12 21:37:29 +00:00
										 |  |  | 	/* Setup the actual spy before creating our thread */ | 
					
						
							| 
									
										
										
										
											2007-08-08 19:30:52 +00:00
										 |  |  | 	if (ast_audiohook_init(&mixmonitor->audiohook, AST_AUDIOHOOK_TYPE_SPY, mixmonitor_spy_type)) { | 
					
						
							| 
									
										
										
										
											2007-12-12 20:05:13 +00:00
										 |  |  | 		ast_free(mixmonitor); | 
					
						
							| 
									
										
										
										
											2007-08-08 19:30:52 +00:00
										 |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2006-06-12 21:37:29 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-08-08 19:30:52 +00:00
										 |  |  | 	ast_set_flag(&mixmonitor->audiohook, AST_AUDIOHOOK_TRIGGER_WRITE); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-10-31 18:03:17 +00:00
										 |  |  | 	if (readvol) | 
					
						
							|  |  |  | 		mixmonitor->audiohook.options.read_volume = readvol; | 
					
						
							|  |  |  | 	if (writevol) | 
					
						
							|  |  |  | 		mixmonitor->audiohook.options.write_volume = writevol; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-08-08 19:30:52 +00:00
										 |  |  | 	if (startmon(chan, &mixmonitor->audiohook)) { | 
					
						
							| 
									
										
										
										
											2006-06-12 21:37:29 +00:00
										 |  |  | 		ast_log(LOG_WARNING, "Unable to add '%s' spy to channel '%s'\n", | 
					
						
							| 
									
										
										
										
											2007-08-08 19:30:52 +00:00
										 |  |  | 			mixmonitor_spy_type, chan->name); | 
					
						
							|  |  |  | 		ast_audiohook_destroy(&mixmonitor->audiohook); | 
					
						
							| 
									
										
										
										
											2007-06-06 21:20:11 +00:00
										 |  |  | 		ast_free(mixmonitor); | 
					
						
							| 
									
										
										
										
											2006-06-12 21:37:29 +00:00
										 |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-05-24 18:30:19 +00:00
										 |  |  | 	ast_pthread_create_detached_background(&thread, NULL, mixmonitor_thread, mixmonitor); | 
					
						
							| 
									
										
										
										
											2006-06-12 21:37:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | static int mixmonitor_exec(struct ast_channel *chan, void *data) | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 	int x, readvol = 0, writevol = 0; | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 	struct ast_flags flags = {0}; | 
					
						
							| 
									
										
										
										
											2007-06-24 16:20:18 +00:00
										 |  |  | 	char *parse, *tmp, *slash; | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 	AST_DECLARE_APP_ARGS(args, | 
					
						
							|  |  |  | 		AST_APP_ARG(filename); | 
					
						
							|  |  |  | 		AST_APP_ARG(options); | 
					
						
							|  |  |  | 		AST_APP_ARG(post_process); | 
					
						
							|  |  |  | 	); | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2005-10-26 19:48:14 +00:00
										 |  |  | 	if (ast_strlen_zero(data)) { | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 		ast_log(LOG_WARNING, "MixMonitor requires an argument (filename)\n"); | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-05-10 13:22:15 +00:00
										 |  |  | 	parse = ast_strdupa(data); | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 	AST_STANDARD_APP_ARGS(args, parse); | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 	if (ast_strlen_zero(args.filename)) { | 
					
						
							| 
									
										
										
										
											2005-11-04 22:11:51 +00:00
										 |  |  | 		ast_log(LOG_WARNING, "MixMonitor requires an argument (filename)\n"); | 
					
						
							| 
									
										
										
										
											2005-10-19 18:19:02 +00:00
										 |  |  | 		return -1; | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-11-13 16:03:10 +00:00
										 |  |  | 	if (args.options) { | 
					
						
							| 
									
										
										
										
											2005-11-03 21:19:11 +00:00
										 |  |  | 		char *opts[OPT_ARG_ARRAY_SIZE] = { NULL, }; | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-11-03 21:19:11 +00:00
										 |  |  | 		ast_app_parse_options(mixmonitor_opts, &flags, opts, args.options); | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		if (ast_test_flag(&flags, MUXFLAG_READVOLUME)) { | 
					
						
							| 
									
										
										
										
											2005-11-03 21:19:11 +00:00
										 |  |  | 			if (ast_strlen_zero(opts[OPT_ARG_READVOLUME])) { | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 				ast_log(LOG_WARNING, "No volume level was provided for the heard volume ('v') option.\n"); | 
					
						
							| 
									
										
										
										
											2005-11-03 21:19:11 +00:00
										 |  |  | 			} else if ((sscanf(opts[OPT_ARG_READVOLUME], "%d", &x) != 1) || (x < -4) || (x > 4)) { | 
					
						
							|  |  |  | 				ast_log(LOG_NOTICE, "Heard volume must be a number between -4 and 4, not '%s'\n", opts[OPT_ARG_READVOLUME]); | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 			} else { | 
					
						
							|  |  |  | 				readvol = get_volfactor(x); | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 		if (ast_test_flag(&flags, MUXFLAG_WRITEVOLUME)) { | 
					
						
							| 
									
										
										
										
											2005-11-03 21:19:11 +00:00
										 |  |  | 			if (ast_strlen_zero(opts[OPT_ARG_WRITEVOLUME])) { | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 				ast_log(LOG_WARNING, "No volume level was provided for the spoken volume ('V') option.\n"); | 
					
						
							| 
									
										
										
										
											2005-11-03 21:19:11 +00:00
										 |  |  | 			} else if ((sscanf(opts[OPT_ARG_WRITEVOLUME], "%d", &x) != 1) || (x < -4) || (x > 4)) { | 
					
						
							|  |  |  | 				ast_log(LOG_NOTICE, "Spoken volume must be a number between -4 and 4, not '%s'\n", opts[OPT_ARG_WRITEVOLUME]); | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 			} else { | 
					
						
							|  |  |  | 				writevol = get_volfactor(x); | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 		 | 
					
						
							|  |  |  | 		if (ast_test_flag(&flags, MUXFLAG_VOLUME)) { | 
					
						
							| 
									
										
										
										
											2005-11-03 21:19:11 +00:00
										 |  |  | 			if (ast_strlen_zero(opts[OPT_ARG_VOLUME])) { | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 				ast_log(LOG_WARNING, "No volume level was provided for the combined volume ('W') option.\n"); | 
					
						
							| 
									
										
										
										
											2005-11-03 21:19:11 +00:00
										 |  |  | 			} else if ((sscanf(opts[OPT_ARG_VOLUME], "%d", &x) != 1) || (x < -4) || (x > 4)) { | 
					
						
							|  |  |  | 				ast_log(LOG_NOTICE, "Combined volume must be a number between -4 and 4, not '%s'\n", opts[OPT_ARG_VOLUME]); | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 			} else { | 
					
						
							|  |  |  | 				readvol = writevol = get_volfactor(x); | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* if not provided an absolute path, use the system-configured monitoring directory */ | 
					
						
							|  |  |  | 	if (args.filename[0] != '/') { | 
					
						
							|  |  |  | 		char *build; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		build = alloca(strlen(ast_config_AST_MONITOR_DIR) + strlen(args.filename) + 3); | 
					
						
							|  |  |  | 		sprintf(build, "%s/%s", ast_config_AST_MONITOR_DIR, args.filename); | 
					
						
							|  |  |  | 		args.filename = build; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-06-24 16:20:18 +00:00
										 |  |  | 	tmp = ast_strdupa(args.filename); | 
					
						
							|  |  |  | 	if ((slash = strrchr(tmp, '/'))) | 
					
						
							|  |  |  | 		*slash = '\0'; | 
					
						
							|  |  |  | 	ast_mkdir(tmp, 0777); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 	pbx_builtin_setvar_helper(chan, "MIXMONITOR_FILENAME", args.filename); | 
					
						
							| 
									
										
										
										
											2007-11-13 16:03:10 +00:00
										 |  |  | 	launch_monitor_thread(chan, args.filename, flags.flags, readvol, writevol, args.post_process); | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-16 17:37:03 +00:00
										 |  |  | static int stop_mixmonitor_exec(struct ast_channel *chan, void *data) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2007-08-08 19:30:52 +00:00
										 |  |  | 	ast_audiohook_detach_source(chan, mixmonitor_spy_type); | 
					
						
							| 
									
										
										
										
											2006-02-16 17:37:03 +00:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-10-11 19:03:06 +00:00
										 |  |  | static char *handle_cli_mixmonitor(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a) | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct ast_channel *chan; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-10-11 19:03:06 +00:00
										 |  |  | 	switch (cmd) { | 
					
						
							|  |  |  | 	case CLI_INIT: | 
					
						
							|  |  |  | 		e->command = "mixmonitor [start|stop]"; | 
					
						
							|  |  |  | 		e->usage = | 
					
						
							|  |  |  | 			"Usage: mixmonitor <start|stop> <chan_name> [args]\n" | 
					
						
							|  |  |  | 			"       The optional arguments are passed to the MixMonitor\n" | 
					
						
							|  |  |  | 			"       application when the 'start' command is used.\n"; | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	case CLI_GENERATE: | 
					
						
							|  |  |  | 		return ast_complete_channels(a->line, a->word, a->pos, a->n, 2); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (a->argc < 3) | 
					
						
							|  |  |  | 		return CLI_SHOWUSAGE; | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-10-11 19:03:06 +00:00
										 |  |  | 	if (!(chan = ast_get_channel_by_name_prefix_locked(a->argv[2], strlen(a->argv[2])))) { | 
					
						
							|  |  |  | 		ast_cli(a->fd, "No channel matching '%s' found.\n", a->argv[2]); | 
					
						
							|  |  |  | 		/* Technically this is a failure, but we don't want 2 errors printing out */ | 
					
						
							|  |  |  | 		return CLI_SUCCESS; | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-10-11 19:03:06 +00:00
										 |  |  | 	if (!strcasecmp(a->argv[1], "start")) { | 
					
						
							|  |  |  | 		mixmonitor_exec(chan, a->argv[3]); | 
					
						
							| 
									
										
										
										
											2007-08-08 19:30:52 +00:00
										 |  |  | 		ast_channel_unlock(chan); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		ast_channel_unlock(chan); | 
					
						
							|  |  |  | 		ast_audiohook_detach_source(chan, mixmonitor_spy_type); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-10-11 19:03:06 +00:00
										 |  |  | 	return CLI_SUCCESS; | 
					
						
							| 
									
										
										
										
											2006-02-16 17:37:03 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-09-18 19:54:18 +00:00
										 |  |  | static struct ast_cli_entry cli_mixmonitor[] = { | 
					
						
							| 
									
										
										
										
											2007-10-22 20:05:18 +00:00
										 |  |  | 	AST_CLI_DEFINE(handle_cli_mixmonitor, "Execute a MixMonitor command") | 
					
						
							| 
									
										
										
										
											2005-10-28 23:01:13 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | static int unload_module(void) | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2005-10-18 22:52:21 +00:00
										 |  |  | 	int res; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-09-18 19:54:18 +00:00
										 |  |  | 	ast_cli_unregister_multiple(cli_mixmonitor, sizeof(cli_mixmonitor) / sizeof(struct ast_cli_entry)); | 
					
						
							|  |  |  | 	res = ast_unregister_application(stop_app); | 
					
						
							| 
									
										
										
										
											2005-10-18 22:52:21 +00:00
										 |  |  | 	res |= ast_unregister_application(app); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	return res; | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | static int load_module(void) | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2005-10-18 22:52:21 +00:00
										 |  |  | 	int res; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-09-18 19:54:18 +00:00
										 |  |  | 	ast_cli_register_multiple(cli_mixmonitor, sizeof(cli_mixmonitor) / sizeof(struct ast_cli_entry)); | 
					
						
							|  |  |  | 	res = ast_register_application(app, mixmonitor_exec, synopsis, desc); | 
					
						
							| 
									
										
										
										
											2006-02-16 17:37:03 +00:00
										 |  |  | 	res |= ast_register_application(stop_app, stop_mixmonitor_exec, stop_synopsis, stop_desc); | 
					
						
							| 
									
										
										
										
											2005-10-18 22:52:21 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return res; | 
					
						
							| 
									
										
										
										
											2005-10-16 02:59:13 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | AST_MODULE_INFO_STANDARD(ASTERISK_GPL_KEY, "Mixed Audio Monitoring Application"); |