commands.c 76.3 KB
Newer Older
aymeric's avatar
aymeric committed
1 2 3 4
/****************************************************************************
 *
 *  $Id: commands.c,v 1.39 2008/07/03 15:08:34 smorlat Exp $
 *
5
 *  Copyright (C) 2006-2009  Sandro Santilli <strk@keybit.net>
aymeric's avatar
aymeric committed
6 7
 *  Copyright (C) 2004  Simon MORLAT <simon.morlat@linphone.org>
 *
8
****************************************************************************
aymeric's avatar
aymeric committed
9
 *
10 11 12 13
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License
 * as published by the Free Software Foundation; either version 2
 * of the License, or (at your option) any later version.
aymeric's avatar
aymeric committed
14
 *
15 16 17 18
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
aymeric's avatar
aymeric committed
19
 *
20 21
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
22
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
aymeric's avatar
aymeric committed
23 24 25 26 27 28
 *
 ****************************************************************************/

#include <string.h>
#include <stdio.h>
#include <stdlib.h>
Jehan Monnier's avatar
Jehan Monnier committed
29
#ifndef _WIN32_WCE
aymeric's avatar
aymeric committed
30
#include <errno.h>
Jehan Monnier's avatar
Jehan Monnier committed
31
#endif /*_WIN32_WCE*/
aymeric's avatar
aymeric committed
32 33
#include <limits.h>
#include <ctype.h>
34
#include <linphone/core.h>
aymeric's avatar
aymeric committed
35
#include "linphonec.h"
36
#include <linphone/lpconfig.h>
aymeric's avatar
aymeric committed
37

38
#ifndef _WIN32
39
#include <sys/wait.h>
40
#include <unistd.h>
41 42
#endif

43 44 45
#define AUDIO 0
#define VIDEO 1

aymeric's avatar
aymeric committed
46 47
/***************************************************************************
 *
48
 *  Forward declarations
aymeric's avatar
aymeric committed
49 50 51 52 53 54 55 56 57
 *
 ***************************************************************************/

extern char *lpc_strip_blanks(char *input);

/* Command handlers */
static int lpc_cmd_help(LinphoneCore *, char *);
static int lpc_cmd_proxy(LinphoneCore *, char *);
static int lpc_cmd_call(LinphoneCore *, char *);
58
static int lpc_cmd_calls(LinphoneCore *, char *);
Simon Morlat's avatar
Simon Morlat committed
59
static int lpc_cmd_chat(LinphoneCore *, char *);
aymeric's avatar
aymeric committed
60
static int lpc_cmd_answer(LinphoneCore *, char *);
smorlat's avatar
smorlat committed
61
static int lpc_cmd_autoanswer(LinphoneCore *, char *);
aymeric's avatar
aymeric committed
62
static int lpc_cmd_terminate(LinphoneCore *, char *);
63
static int lpc_cmd_redirect(LinphoneCore *, char *);
aymeric's avatar
aymeric committed
64 65
static int lpc_cmd_call_logs(LinphoneCore *, char *);
static int lpc_cmd_ipv6(LinphoneCore *, char *);
66
static int lpc_cmd_transfer(LinphoneCore *, char *);
aymeric's avatar
aymeric committed
67 68 69 70 71 72
static int lpc_cmd_quit(LinphoneCore *, char *);
static int lpc_cmd_nat(LinphoneCore *, char *);
static int lpc_cmd_stun(LinphoneCore *, char *);
static int lpc_cmd_firewall(LinphoneCore *, char *);
static int lpc_cmd_friend(LinphoneCore *, char*);
static int lpc_cmd_soundcard(LinphoneCore *, char *);
73 74
static int lpc_cmd_webcam(LinphoneCore *, char *);
static int lpc_cmd_staticpic(LinphoneCore *, char *);
aymeric's avatar
aymeric committed
75 76
static int lpc_cmd_play(LinphoneCore *, char *);
static int lpc_cmd_record(LinphoneCore *, char *);
smorlat's avatar
smorlat committed
77
static int lpc_cmd_register(LinphoneCore *, char *);
smorlat's avatar
smorlat committed
78
static int lpc_cmd_unregister(LinphoneCore *, char *);
smorlat's avatar
smorlat committed
79
static int lpc_cmd_duration(LinphoneCore *lc, char *args);
smorlat's avatar
smorlat committed
80
static int lpc_cmd_status(LinphoneCore *lc, char *args);
81
static int lpc_cmd_ports(LinphoneCore *lc, char *args);
82
static int lpc_cmd_param(LinphoneCore *lc, char *args);
83
static int lpc_cmd_speak(LinphoneCore *lc, char *args);
84 85 86
static int lpc_cmd_acodec(LinphoneCore *lc, char *args);
static int lpc_cmd_vcodec(LinphoneCore *lc, char *args);
static int lpc_cmd_codec(int type, LinphoneCore *lc, char *args);
87
static int lpc_cmd_echocancellation(LinphoneCore *lc, char *args);
88
static int lpc_cmd_echolimiter(LinphoneCore *lc, char *args);
89 90
static int lpc_cmd_pause(LinphoneCore *lc, char *args);
static int lpc_cmd_resume(LinphoneCore *lc, char *args);
Simon Morlat's avatar
Simon Morlat committed
91 92
static int lpc_cmd_mute_mic(LinphoneCore *lc, char *args);
static int lpc_cmd_unmute_mic(LinphoneCore *lc, char *args);
93
static int lpc_cmd_playback_gain(LinphoneCore *lc, char *args);
Simon Morlat's avatar
Simon Morlat committed
94
static int lpc_cmd_rtp_no_xmit_on_audio_mute(LinphoneCore *lc, char *args);
Simon Morlat's avatar
Simon Morlat committed
95
#ifdef VIDEO_ENABLED
96
static int lpc_cmd_camera(LinphoneCore *lc, char *args);
97
static int lpc_cmd_video_window(LinphoneCore *lc, char *args);
98
static int lpc_cmd_preview_window(LinphoneCore *lc, char *args);
99
static int lpc_cmd_snapshot(LinphoneCore *lc, char *args);
100
static int lpc_cmd_preview_snapshot(LinphoneCore *lc, char *args);
101
static int lpc_cmd_vfureq(LinphoneCore *lc, char *arg);
Simon Morlat's avatar
Simon Morlat committed
102
#endif
103
static int lpc_cmd_states(LinphoneCore *lc, char *args);
104
static int lpc_cmd_identify(LinphoneCore *lc, char *args);
105
static int lpc_cmd_ringback(LinphoneCore *lc, char *args);
Simon Morlat's avatar
Simon Morlat committed
106
static int lpc_cmd_conference(LinphoneCore *lc, char *args);
Simon Morlat's avatar
Simon Morlat committed
107
static int lpc_cmd_zrtp_verified(LinphoneCore *lc, char *args);
108
static int lpc_cmd_zrtp_unverified(LinphoneCore *lc, char *args);
smorlat's avatar
smorlat committed
109

aymeric's avatar
aymeric committed
110 111 112 113 114 115
/* Command handler helpers */
static void linphonec_proxy_add(LinphoneCore *lc);
static void linphonec_proxy_display(LinphoneProxyConfig *lc);
static void linphonec_proxy_list(LinphoneCore *lc);
static void linphonec_proxy_remove(LinphoneCore *lc, int index);
static  int linphonec_proxy_use(LinphoneCore *lc, int index);
smorlat's avatar
smorlat committed
116
static void linphonec_proxy_show(LinphoneCore *lc,int index);
aymeric's avatar
aymeric committed
117 118 119 120
static void linphonec_friend_display(LinphoneFriend *fr);
static int linphonec_friend_list(LinphoneCore *lc, char *arg);
static void linphonec_display_command_help(LPC_COMMAND *cmd);
static int linphonec_friend_call(LinphoneCore *lc, unsigned int num);
121
#ifndef _WIN32
aymeric's avatar
aymeric committed
122
static int linphonec_friend_add(LinphoneCore *lc, const char *name, const char *addr);
123
#endif
aymeric's avatar
aymeric committed
124
static int linphonec_friend_delete(LinphoneCore *lc, int num);
125
static int linphonec_friend_delete(LinphoneCore *lc, int num);
126 127 128
static void linphonec_codec_list(int type, LinphoneCore *lc);
static void linphonec_codec_enable(int type, LinphoneCore *lc, int index);
static void linphonec_codec_disable(int type, LinphoneCore *lc, int index);
129
static void lpc_display_call_states(LinphoneCore *lc);
smorlat's avatar
smorlat committed
130

aymeric's avatar
aymeric committed
131 132 133
/* Command table management */
static LPC_COMMAND *lpc_find_command(const char *name);

134 135
void linphonec_out(const char *fmt,...);

136 137
VideoParams lpc_video_params={-1,-1,-1,-1,NULL,TRUE,FALSE};
VideoParams lpc_preview_params={-1,-1,-1,-1,NULL,TRUE,FALSE};
138

aymeric's avatar
aymeric committed
139 140 141 142 143 144 145 146 147
/***************************************************************************
 *
 *  Global variables
 *
 ***************************************************************************/

/*
 * Commands table.
 */
148
static LPC_COMMAND commands[] = {
149
	{ "help", lpc_cmd_help, "Print commands help.",
150 151 152
		"'help <command>'\t: displays specific help for command.\n"
		"'help advanced'\t: shows advanced commands.\n"
	},
153 154 155 156 157 158 159 160 161
	{ "answer", lpc_cmd_answer, "Answer a call",
		"'answer' : Answer the current incoming call\n"
		"'answer <call id>' : Answer the call with given id\n"
	},
	{ "autoanswer", lpc_cmd_autoanswer, "Show/set auto-answer mode",
		"'autoanswer'       \t: show current autoanswer mode\n"
		"'autoanswer enable'\t: enable autoanswer mode\n"
		"'autoanswer disable'\t: disable autoanswer mode\n"
	},
162 163
	{ "call", lpc_cmd_call, "Call a SIP uri or number",
#ifdef VIDEO_ENABLED
164 165 166 167
		"'call <sip-url or number>  [options]' \t: initiate a call to the specified destination.\n"
		"Options can be:\n"
		"--audio-only : initiate the call without video.\n"
		"--early-media : sends audio and video stream immediately when remote proposes early media.\n"
168 169 170
#else
		"'call <sip-url or number>' \t: initiate a call to the specified destination.\n"
#endif
171
		},
172
	{ "calls", lpc_cmd_calls, "Show all the current calls with their id and status.",
173
		NULL
174
	},
175
	{ "call-logs", lpc_cmd_call_logs, "Calls history", NULL
Simon Morlat's avatar
Simon Morlat committed
176
	},
177 178 179
#ifdef VIDEO_ENABLED
	{ "camera", lpc_cmd_camera, "Send camera output for current call.",
		"'camera on'\t: allow sending of local camera video to remote end.\n"
180
		"'camera off'\t: disable sending of local camera's video to remote end.\n"
aymeric's avatar
aymeric committed
181
	},
182 183 184 185
#endif
	{ "chat", lpc_cmd_chat, "Chat with a SIP uri",
		"'chat <sip-url> \"message\"' "
		": send a chat message \"message\" to the specified destination."
aymeric's avatar
aymeric committed
186
	},
187 188 189
	{ "conference", lpc_cmd_conference, "Create and manage an audio conference.",
		"'conference add <call id> : join the call with id 'call id' into the audio conference."
		"'conference rm <call id> : remove the call with id 'call id' from the audio conference."
aymeric's avatar
aymeric committed
190
	},
191
	{ "duration", lpc_cmd_duration, "Print duration in seconds of the last call.", NULL
aymeric's avatar
aymeric committed
192
	},
strk's avatar
strk committed
193
	{ "firewall", lpc_cmd_firewall, "Set firewall policy",
aymeric's avatar
aymeric committed
194 195 196
		"'firewall'        : show current firewall policy.\n"
		"'firewall none'   : use direct connection.\n"
		"'firewall nat'    : use nat address given with the 'nat' command.\n"
strk's avatar
strk committed
197
		"'firewall stun'   : use stun server given with the 'stun' command.\n"
Yann Diorcet's avatar
Yann Diorcet committed
198 199
		"'firewall ice'    : use ice.\n"
		"'firewall upnp'   : use uPnP IGD.\n"
aymeric's avatar
aymeric committed
200 201 202 203 204 205 206 207 208
	},
	{ "friend", lpc_cmd_friend, "Manage friends",
		"'friend list [<pattern>]'    : list friends.\n"
		"'friend call <index>'        : call a friend.\n"
		"'friend add <name> <addr>'   : add friend, <name> must be quoted to include\n"
	    "                               spaces, <addr> has \"sip:\" added if it isn't\n"
	    "                               there.  Don't use '<' '>' around <addr>.\n"
		"'friend delete <index>'      : remove friend, 'all' removes all\n"
	},
209 210 211 212 213 214
	{ "ipv6", lpc_cmd_ipv6, "Use IPV6",
		"'ipv6 status' : show ipv6 usage status.\n"
		"'ipv6 enable' : enable the use of the ipv6 network.\n"
		"'ipv6 disable' : do not use ipv6 network."
	},
	{ "mute", lpc_cmd_mute_mic,
215 216
		"Mute microphone and suspend voice transmission.",
		NULL
217 218 219 220 221 222 223 224
	},
	{ "nat", lpc_cmd_nat, "Set nat address",
		"'nat'        : show nat settings.\n"
		"'nat <addr>' : set nat address.\n"
	},
	{ "pause", lpc_cmd_pause, "pause a call",
		"'pause' : pause the current call\n"
	},
225 226 227 228
	{ "play", lpc_cmd_play, "play a wav file",
		"This command has two roles:\n"
		"Plays a file instead of capturing from soundcard - only available in file mode (see 'help soundcard')\n"
		"Specifies a wav file to be played to play music to far end when putting it on hold (pause)\n"
aymeric's avatar
aymeric committed
229 230
		"'play <wav file>'    : play a wav file."
	},
231
	{ "playbackgain", lpc_cmd_playback_gain,
232 233
		"Adjust playback gain.",
		NULL
234 235 236 237 238 239 240 241 242 243
	},
	{ "proxy", lpc_cmd_proxy, "Manage proxies",
		"'proxy list' : list all proxy setups.\n"
		"'proxy add' : add a new proxy setup.\n"
		"'proxy remove <index>' : remove proxy setup with number index.\n"
		"'proxy use <index>' : use proxy with number index as default proxy.\n"
		"'proxy unuse' : don't use a default proxy.\n"
		"'proxy show <index>' : show configuration and status of the proxy numbered by index.\n"
		"'proxy show default' : show configuration and status of the default proxy.\n"
	},
aymeric's avatar
aymeric committed
244 245 246 247
	{ "record", lpc_cmd_record, "record to a wav file",
		"This feature is available only in file mode (see 'help soundcard')\n"
		"'record <wav file>'    : record into wav file."
	},
248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273
	{ "resume", lpc_cmd_resume, "resume a call",
		"'resume' : resume the unique call\n"
		"'resume <call id>' : hold off the call with given id\n"
	},
	{ "soundcard", lpc_cmd_soundcard, "Manage soundcards",
		"'soundcard list' : list all sound devices.\n"
		"'soundcard show' : show current sound devices configuration.\n"
		"'soundcard use <index>' : select a sound device.\n"
		"'soundcard use files' : use .wav files instead of soundcard\n"
	},
	{ "stun", lpc_cmd_stun, "Set stun server address",
		"'stun'        : show stun settings.\n"
		"'stun <addr>' : set stun server address.\n"
	},
	{ "terminate", lpc_cmd_terminate, "Terminate a call",
		"'terminate' : Terminate the current call\n"
		"'terminate <call id>' : Terminate the call with supplied id\n"
		"'terminate <all>' : Terminate all the current calls\n"
	},
	{ "transfer", lpc_cmd_transfer,
		"Transfer a call to a specified destination.",
		"'transfer <sip-uri>' : transfers the current active call to the destination sip-uri\n"
		"'transfer <call id> <sip-uri>': transfers the call with 'id' to the destination sip-uri\n"
		"'transfer <call id1> --to-call <call id2>': transfers the call with 'id1' to the destination of call 'id2' (attended transfer)\n"
	},
	{ "unmute", lpc_cmd_unmute_mic,
274 275
		"Unmute microphone and resume voice transmission.",
		NULL
276 277 278 279 280 281 282 283 284
	},
	{ "webcam", lpc_cmd_webcam, "Manage webcams",
		"'webcam list' : list all known devices.\n"
		"'webcam use <index>' : select a video device.\n"
	},
	{ "quit", lpc_cmd_quit, "Exit linphonec", NULL
	},
	{ (char *)NULL, (lpc_cmd_handler)NULL, (char *)NULL, (char *)NULL
	}
285 286 287 288 289
};


static LPC_COMMAND advanced_commands[] = {
	 { "codec", lpc_cmd_acodec, "Audio codec configuration",
290 291 292
            "'codec list' : list audio codecs\n"
            "'codec enable <index>' : enable available audio codec\n"
            "'codec disable <index>' : disable audio codec" },
Simon Morlat's avatar
Simon Morlat committed
293 294 295 296
	{ "vcodec", lpc_cmd_vcodec, "Video codec configuration",
		"'vcodec list' : list video codecs\n"
		"'vcodec enable <index>' : enable available video codec\n"
		"'vcodec disable <index>' : disable video codec" },
297 298 299 300
	{ "ec", lpc_cmd_echocancellation, "Echo cancellation",
	    "'ec on [<delay>] [<tail>] [<framesize>]' : turn EC on with given delay, tail length and framesize\n"
	    "'ec off' : turn echo cancellation (EC) off\n"
	    "'ec show' : show EC status" },
301 302 303 304
	{ "el", lpc_cmd_echolimiter, "Echo limiter",
	    "'el on turns on echo limiter (automatic half duplex, for cases where echo canceller cannot work)\n"
	    "'el off' : turn echo limiter off\n"
	    "'el show' : show echo limiter status" },
Simon Morlat's avatar
Simon Morlat committed
305
	{ "nortp-on-audio-mute", lpc_cmd_rtp_no_xmit_on_audio_mute,
306 307
		  "Set the rtp_no_xmit_on_audio_mute configuration parameter",
		  "   If set to 1 then rtp transmission will be muted when\n"
308 309
		  "   audio is muted , otherwise rtp is always sent."
		},
Simon Morlat's avatar
Simon Morlat committed
310
#ifdef VIDEO_ENABLED
311 312 313 314
	{ "vwindow", lpc_cmd_video_window, "Control video display window",
		"'vwindow show': shows video window\n"
		"'vwindow hide': hides video window\n"
		"'vwindow pos <x> <y>': Moves video window to x,y pixel coordinates\n"
315 316 317 318 319 320 321 322 323 324 325
		"'vwindow size <width> <height>': Resizes video window\n"
		"'vwindow id <window id>': embeds video display into supplied window id."
	},
	{ "pwindow", lpc_cmd_preview_window, "Control local camera video display (preview window)",
		"'pwindow show': shows the local camera video display\n"
		"'pwindow hide': hides the local camera video display\n"
		"'pwindow pos <x> <y>': Moves preview window to x,y pixel coordinates\n"
		"'pwindow size <width> <height>': Resizes preview window\n"
		"'pwindow id <window id>': embeds preview display into supplied window id.\n"
		"'pwindow integrated': integrate preview display within the video window of current call.\n"
		"'pwindow standalone': use standalone window for preview display."
326
	},
327 328 329
	{ "snapshot", lpc_cmd_snapshot, "Take a snapshot of currently received video stream",
		"'snapshot <file path>': take a snapshot and records it in jpeg format into the supplied path\n"
	},
330
	{ "preview-snapshot", lpc_cmd_preview_snapshot, "Take a snapshot of currently captured video stream",
Simon Morlat's avatar
Simon Morlat committed
331
		"'preview-snapshot <file path>': take a snapshot and records it in jpeg format into the supplied path\n"
332
	},
333 334
	{ "vfureq", lpc_cmd_vfureq, "Request the other side to send VFU for the current call",
		NULL
Simon Morlat's avatar
Simon Morlat committed
335
	},
Simon Morlat's avatar
Simon Morlat committed
336
#endif
337 338 339 340 341
	{ "states", lpc_cmd_states, "Show internal states of liblinphone, registrations and calls, according to linphonecore.h definitions",
		"'states global': shows global state of liblinphone \n"
		"'states calls': shows state of calls\n"
		"'states proxies': shows state of proxy configurations"
	},
342 343
	{ "register", lpc_cmd_register, "Register in one line to a proxy" , "register <sip identity> <sip proxy> <password>"
},
344
	{ "unregister", lpc_cmd_unregister, "Unregister from default proxy", NULL	},
345
	{ "status", lpc_cmd_status, "Print various status information",
346 347 348
			"'status register'  \t: print status concerning registration\n"
			"'status autoanswer'\t: tell whether autoanswer mode is enabled\n"
			"'status hook'      \t: print hook status\n" },
349
	{ "ports", lpc_cmd_ports, "Network ports configuration",
350 351
			"'ports'  \t: prints current used ports.\n"
			"'ports sip <port number>'\t: Sets the sip port.\n" },
352 353 354 355
	{ "param", lpc_cmd_param, "parameter set or read as normally given in .linphonerc",
			"'param <section> <parameter> [<value>]'  \t: reads [sets] given parameter.\n"
			"NOTES: - changes may become effective after (re)establishing a sip connection.\n"
			"       - upon exit, .linphonerc will reflect the updated state.\n" },
356 357 358 359 360 361 362 363 364
	{ "speak", lpc_cmd_speak, "Speak a sentence using espeak TTS engine",
			"This feature is available only in file mode. (see 'help soundcard')\n"
			"'speak <voice name> <sentence>'	: speak a text using the specified espeak voice.\n"
			"Example for english voice: 'speak default Hello my friend !'"
	},
	{ "staticpic", lpc_cmd_staticpic, "Manage static pictures when nowebcam",
		"'staticpic set' : Set path to picture that should be used.\n"
		"'staticpic fps' : Get/set frames per seconds for picture emission.\n"
	},
365 366 367 368
	{ "identify", lpc_cmd_identify, "Returns the user-agent string of far end",
		"'identify' \t: returns remote user-agent string for current call.\n"
		"'identify <id>' \t: returns remote user-agent string for call with supplied id.\n"
	},
369 370 371 372
	{ "ringback", lpc_cmd_ringback, "Specifies a ringback tone to be played to remote end during incoming calls",
		"'ringback <path of mono .wav file>'\t: Specifies a ringback tone to be played to remote end during incoming calls\n"
		"'ringback disable'\t: Disable playing of ringback tone to callers\n"
	},
373
	{ "redirect", lpc_cmd_redirect, "Redirect an incoming call",
374 375
		"'redirect <id> <redirect-uri>'\t: Redirect the specified call to the <redirect-uri>\n"
		"'redirect all <redirect-uri>'\t: Redirect all pending incoming calls to the <redirect-uri>\n"
376
	},
Simon Morlat's avatar
Simon Morlat committed
377 378 379
	{ "zrtp-set-verified", lpc_cmd_zrtp_verified,"Set ZRTP SAS verified.",
		"'Set ZRTP SAS verified'\n"
	},
380 381 382
	{ "zrtp-set-unverified", lpc_cmd_zrtp_unverified,"Set ZRTP SAS not verified.",
		"'Set ZRTP SAS not verified'\n"
	},
383
	{	NULL,NULL,NULL,NULL}
aymeric's avatar
aymeric committed
384 385
};

386 387


aymeric's avatar
aymeric committed
388 389
/***************************************************************************
 *
390
 *  Public interface
aymeric's avatar
aymeric committed
391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421
 *
 ***************************************************************************/

/*
 * Main command dispatcher.
 * WARNING: modifies second argument!
 *
 * Always return 1 currently.
 */
int
linphonec_parse_command_line(LinphoneCore *lc, char *cl)
{
	char *ptr=cl;
	char *args=NULL;
	LPC_COMMAND *cmd;

	/* Isolate first word and args */
	while(*ptr && !isspace(*ptr)) ++ptr;
	if (*ptr)
	{
		*ptr='\0';
 		/* set args to first nonblank */
		args=ptr+1;
		while(*args && isspace(*args)) ++args;
	}

	/* Handle DTMF */
	if ( isdigit(*cl) || *cl == '#' || *cl == '*' )
	{
		while ( isdigit(*cl) || *cl == '#' || *cl == '*' )
		{
422 423
			if (linphone_core_get_current_call(lc))
				linphone_call_send_dtmf(linphone_core_get_current_call(lc), *cl);
424
			linphone_core_play_dtmf (lc,*cl,100);
smorlat's avatar
smorlat committed
425
			ms_sleep(1); // be nice
aymeric's avatar
aymeric committed
426 427 428 429 430 431 432 433 434 435 436
			++cl;
		}

		// discard spurious trailing chars
		return 1;
	}

	/* Handle other kind of commands */
	cmd=lpc_find_command(cl);
	if ( !cmd )
	{
437
		linphonec_out("'%s': Cannot understand this.\n", cl);
aymeric's avatar
aymeric committed
438 439 440 441 442
		return 1;
	}

	if ( ! cmd->func(lc, args) )
	{
443
		linphonec_out("Syntax error.\n");
aymeric's avatar
aymeric committed
444 445 446 447 448 449 450 451 452 453 454 455 456 457 458
		linphonec_display_command_help(cmd);
	}

	return 1;
}

/*
 * Generator function for command completion.
 * STATE let us know whether to start from scratch;
 * without any state (STATE==0), then we start at the
 * top of the list.
 */
char *
linphonec_command_generator(const char *text, int state)
{
459 460
	static size_t len;
	static int index, adv;
aymeric's avatar
aymeric committed
461 462 463 464 465
	char *name;

	if ( ! state )
	{
		index=0;
466
		adv=0;
467
		len=strlen(text);
aymeric's avatar
aymeric committed
468 469 470 471 472
	}
	/*
 	 * Return the next name which partially matches
	 * from the commands list
	 */
473 474
	if (adv==0){
		while ((name=commands[index].name))
aymeric's avatar
aymeric committed
475
		{
476 477 478 479 480 481
			++index; /* so next call get next command */

			if (strncmp(name, text, len) == 0)
			{
				return ortp_strdup(name);
			}
aymeric's avatar
aymeric committed
482
		}
483 484
		adv=1;
		index=0;
aymeric's avatar
aymeric committed
485
	}
486 487 488 489
	if (adv==1){
		while ((name=advanced_commands[index].name))
		{
			++index; /* so next call get next command */
aymeric's avatar
aymeric committed
490

491 492 493 494 495 496
			if (strncmp(name, text, len) == 0)
			{
				return ortp_strdup(name);
			}
		}
	}
aymeric's avatar
aymeric committed
497 498 499 500 501 502
	return NULL;
}


/***************************************************************************
 *
503
 *  Command handlers
aymeric's avatar
aymeric committed
504 505 506 507 508 509 510 511 512 513 514
 *
 ***************************************************************************/

static int
lpc_cmd_help(LinphoneCore *lc, char *arg)
{
	int i=0;
	LPC_COMMAND *cmd;

	if (!arg || !*arg)
	{
515 516
		linphonec_out("Commands are:\n");
		linphonec_out("---------------------------\n");
aymeric's avatar
aymeric committed
517 518 519

		while (commands[i].help)
		{
520
			linphonec_out("%10.10s\t%s\n", commands[i].name,
aymeric's avatar
aymeric committed
521 522 523
				commands[i].help);
			i++;
		}
524

525
		linphonec_out("---------------------------\n");
526 527
		linphonec_out("Type 'help <command>' for more details or\n");
		linphonec_out("     'help advanced' to list additional commands.\n");
aymeric's avatar
aymeric committed
528 529 530 531

		return 1;
	}

532 533 534 535 536 537
	if (strcmp(arg,"advanced")==0){
		linphonec_out("Advanced commands are:\n");
		linphonec_out("---------------------------\n");
		i=0;
		while (advanced_commands[i].help)
		{
jehan's avatar
jehan committed
538
			linphonec_out("%20.20s\t%s\n", advanced_commands[i].name,
539 540 541
				advanced_commands[i].help);
			i++;
		}
542

543 544 545 546 547
		linphonec_out("---------------------------\n");
		linphonec_out("Type 'help <command>' for more details.\n");

		return 1;
	}
548

aymeric's avatar
aymeric committed
549 550 551
	cmd=lpc_find_command(arg);
	if ( !cmd )
	{
552
		linphonec_out("No such command.\n");
aymeric's avatar
aymeric committed
553 554 555 556 557 558 559 560
		return 1;
	}

	linphonec_display_command_help(cmd);
	return 1;

}

smorlat's avatar
smorlat committed
561
static char callee_name[256]={0};
562
static char caller_name[256]={0};
smorlat's avatar
smorlat committed
563

564

aymeric's avatar
aymeric committed
565 566 567 568 569 570 571 572
static int
lpc_cmd_call(LinphoneCore *lc, char *args)
{
	if ( ! args || ! *args )
	{
		return 0;
	}
	{
573
		LinphoneCall *call;
574
		LinphoneCallParams *cp=linphone_core_create_call_params (lc, NULL);
575
		char *opt1,*opt2;
576 577 578 579 580
		if ( linphone_core_in_call(lc) )
		{
			linphonec_out("Terminate or hold on the current call first.\n");
			return 1;
		}
581 582 583 584
		opt1=strstr(args,"--audio-only");
		opt2=strstr(args,"--early-media");
		if (opt1){
			opt1[0]='\0';
585
			while(--opt1 > args && opt1[0]==' ') opt1[0]='\0';
586 587
			linphone_call_params_enable_video (cp,FALSE);
		}
588 589
		if (opt2){
			opt2[0]='\0';
590
			while(--opt2 > args && opt2[0]==' ') opt2[0]='\0';
591 592
			linphone_call_params_enable_early_media_sending(cp,TRUE);
		}
593
		if ( NULL == (call=linphone_core_invite_with_params(lc, args,cp)) )
aymeric's avatar
aymeric committed
594
		{
595
			linphonec_out("Error from linphone_core_invite.\n");
aymeric's avatar
aymeric committed
596 597 598
		}
		else
		{
smorlat's avatar
smorlat committed
599
			snprintf(callee_name,sizeof(callee_name),"%s",args);
aymeric's avatar
aymeric committed
600
		}
Simon Morlat's avatar
Simon Morlat committed
601
		linphone_call_params_unref(cp);
aymeric's avatar
aymeric committed
602 603 604 605
	}
	return 1;
}

606
static int
607
lpc_cmd_calls(LinphoneCore *lc, char *args){
608
	const bctbx_list_t *calls = linphone_core_get_calls(lc);
609 610
	if(calls)
	{
611
		lpc_display_call_states(lc);
612 613 614 615 616 617 618 619
	}else
	{
		linphonec_out("No active call.\n");
	}
	return 1;
}


Simon Morlat's avatar
Simon Morlat committed
620 621 622 623 624 625
static int
lpc_cmd_chat(LinphoneCore *lc, char *args)
{
	char *arg1 = args;
	char *arg2 = NULL;
	char *ptr = args;
626
	LinphoneChatRoom *cr;
Simon Morlat's avatar
Simon Morlat committed
627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642

	if (!args) return 0;

	/* Isolate first and second arg */
	while(*ptr && !isspace(*ptr)) ++ptr;
	if ( *ptr )
	{
		*ptr='\0';
		arg2=ptr+1;
		while(*arg2 && isspace(*arg2)) ++arg2;
	}
	else
	{
		/* missing one parameter */
		return 0;
	}
Simon Morlat's avatar
Simon Morlat committed
643
	cr = linphone_core_get_chat_room_from_uri(lc,arg1);
Simon Morlat's avatar
Simon Morlat committed
644 645 646 647
	linphone_chat_room_send_message(cr,arg2);
	return 1;
}

648
const char *linphonec_get_callee(void){
smorlat's avatar
smorlat committed
649 650 651
	return callee_name;
}

652
const char *linphonec_get_caller(void){
653 654 655 656 657 658 659
	return caller_name;
}

void linphonec_set_caller(const char *caller){
	snprintf(caller_name,sizeof(caller_name)-1,"%s",caller);
}

aymeric's avatar
aymeric committed
660
static int
661
lpc_cmd_transfer(LinphoneCore *lc, char *args)
aymeric's avatar
aymeric committed
662
{
663 664
	if (args){
		LinphoneCall *call;
665
		LinphoneCall *call2;
666 667 668
		const char *refer_to=NULL;
		char arg1[256]={0};
		char arg2[266]={0};
669 670
		int id2=0;
		int n=sscanf(args,"%255s %265s %d",arg1,arg2,&id2);
671 672
		if (n==1 || isalpha(*arg1)){
			call=linphone_core_get_current_call(lc);
673
			if (call==NULL && bctbx_list_size(linphone_core_get_calls(lc))==1){
674 675 676 677 678 679 680
				call=(LinphoneCall*)linphone_core_get_calls(lc)->data;
			}
			refer_to=args;
			if (call==NULL){
				linphonec_out("No active call, please specify a call id among the ones listed by 'calls' command.\n");
				return 0;
			}
681
			linphone_call_transfer(call, refer_to);
682
		}else if (n==2){
683
			int id=atoi(arg1);
684 685 686
			refer_to=args+strlen(arg1)+1;
			call=linphonec_get_call(id);
			if (call==NULL) return 0;
687
			linphone_call_transfer(call, refer_to);
688
		}else if (n==3){
689
			int id=atoi(arg1);
690 691 692 693 694 695 696
			call=linphonec_get_call(id);
			call2=linphonec_get_call(id2);
			if (call==NULL || call2==NULL) return 0;
			if (strcmp(arg2,"--to-call")!=0){
				return 0;
			}
			linphonec_out("Performing attended transfer of call %i to call %i",id,id2);
697
			linphone_call_transfer_to_another(call,call2);
698
		}else return 0;
699 700 701
	}else{
		linphonec_out("Transfer command requires at least one argument\n");
		return 0;
aymeric's avatar
aymeric committed
702 703 704 705 706 707 708
	}
	return 1;
}

static int
lpc_cmd_terminate(LinphoneCore *lc, char *args)
{
709
	if (linphone_core_get_calls(lc)==NULL){
710
		linphonec_out("No active calls\n");
711 712
		return 1;
	}
713
	if (!args)
aymeric's avatar
aymeric committed
714
	{
715 716
		if ( -1 == linphone_core_terminate_call(lc, NULL) ){
			linphonec_out("Could not stop the active call.\n");
717 718
		}
		return 1;
aymeric's avatar
aymeric committed
719
	}
720

721 722 723 724 725 726
	if(strcmp(args,"all")==0){
		linphonec_out("We are going to stop all the calls.\n");
		linphone_core_terminate_all_calls(lc);
		return 1;
	}else{
		/*the argument is a linphonec call id */
727
		int id=atoi(args);
728 729
		LinphoneCall *call=linphonec_get_call(id);
		if (call){
730
			if (linphone_call_terminate(call)==-1){
731
				linphonec_out("Could not stop the call with id %li\n",id);
732
			}
733 734
		}else return 0;
		return 1;
735 736
	}
	return 0;
737

aymeric's avatar
aymeric committed
738 739
}

740 741
static int
lpc_cmd_redirect(LinphoneCore *lc, char *args){
742
	const bctbx_list_t *elem;
743 744 745 746 747 748
	int didit=0;
	if (!args) return 0;
	if ((elem=linphone_core_get_calls(lc))==NULL){
		linphonec_out("No active calls.\n");
		return 1;
	}
749 750 751 752
	if (strncmp(args, "all ", 4) == 0) {
		while(elem!=NULL){
			LinphoneCall *call=(LinphoneCall*)elem->data;
			if (linphone_call_get_state(call)==LinphoneCallIncomingReceived){
753
				if (linphone_call_redirect(call,args+4) != 0) {
754 755 756 757 758 759 760 761 762 763 764 765 766 767
					linphonec_out("Could not redirect call.\n");
					elem=elem->next;
				} else {
					didit=1;
					/*as the redirection closes the call, we need to re-check the call list that is invalidated.*/
					elem=linphone_core_get_calls(lc);
				}
			}else elem=elem->next;
		}
		if (didit==0){
			linphonec_out("There is no pending incoming call to redirect.\n");
		}
	} else {
		char space;
768
		int id;
769
		int charRead;
770
		if ( sscanf(args, "%d%c%n", &id, &space, &charRead) == 2 && space == ' ') {
771 772 773 774
			LinphoneCall * call = linphonec_get_call(id);
			if ( call != NULL ) {
				if (linphone_call_get_state(call)!=LinphoneCallIncomingReceived) {
					linphonec_out("The state of the call is not incoming, can't be redirected.\n");
775
				} else if (linphone_call_redirect(call,args+charRead) != 0) {
776 777
					linphonec_out("Could not redirect call.\n");
				}
778
			}
779
		} else return 0;
780 781 782 783
	}
	return 1;
}

aymeric's avatar
aymeric committed
784
static int
785
lpc_cmd_answer(LinphoneCore *lc, char *args){
786
	if (!args)
aymeric's avatar
aymeric committed
787
	{
Ghislain MARY's avatar
Ghislain MARY committed
788
		int nb=(int)bctbx_list_size(linphone_core_get_calls(lc));
789 790 791
		if (nb==1){
			//if just one call is present answer the only one in passing NULL to the linphone_core_accept_call ...
			if ( -1 == linphone_core_accept_call(lc, NULL) )
792
			{
793
				linphonec_out("Fail to accept incoming call\n");
794
			}
795 796 797 798 799
		}else if (nb==0){
			linphonec_out("There are no calls to answer.\n");
		}else{
			linphonec_out("Multiple calls in progress, please specify call id.\n");
			return 0;
800
		}
801 802
		return 1;
	}else{
803 804
		int id;
		if (sscanf(args,"%d",&id)==1){
805
			LinphoneCall *call=linphonec_get_call (id);
806
			if (linphone_call_accept(call)==-1){
807 808 809 810
				linphonec_out("Fail to accept call %i\n",id);
			}
		}else return 0;
		return 1;
811 812
	}
	return 0;
aymeric's avatar
aymeric committed
813 814
}

smorlat's avatar
smorlat committed
815 816 817
static int
lpc_cmd_autoanswer(LinphoneCore *lc, char *args)
{
818 819 820 821 822 823 824 825 826 827
	if ( ! args )
	{
		if ( linphonec_get_autoanswer() ) {
			linphonec_out("Auto answer is enabled. Use 'autoanswer disable' to disable.\n");
		} else {
			linphonec_out("Auto answer is disabled. Use 'autoanswer enable' to enable.\n");
		}
		return 1;
	}

smorlat's avatar
smorlat committed
828 829
	if (strstr(args,"enable")){
		linphonec_set_autoanswer(TRUE);
smorlat's avatar
smorlat committed
830
		linphonec_out("Auto answer enabled.\n");
smorlat's avatar
smorlat committed
831 832
	}else if (strstr(args,"disable")){
		linphonec_set_autoanswer(FALSE);
smorlat's avatar
smorlat committed
833
		linphonec_out("Auto answer disabled.\n");
smorlat's avatar
smorlat committed
834 835 836 837
	}else return 0;
	return 1;
}

aymeric's avatar
aymeric committed
838 839 840
static int
lpc_cmd_quit(LinphoneCore *lc, char *args)
{
smorlat's avatar
smorlat committed
841
	linphonec_main_loop_exit();
aymeric's avatar
aymeric committed
842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859
	return 1;
}

static int
lpc_cmd_nat(LinphoneCore *lc, char *args)
{
	bool_t use;
	const char *nat;

	if ( args ) args=lpc_strip_blanks(args);

	if ( args && *args )
	{
		linphone_core_set_nat_address(lc, args);
		/* linphone_core_set_firewall_policy(lc,LINPHONE_POLICY_USE_NAT_ADDRESS); */
	}

	nat = linphone_core_get_nat_address(lc);
Simon Morlat's avatar
Simon Morlat committed
860
	use = linphone_core_get_firewall_policy(lc)==LinphonePolicyUseNatAddress;
861
	linphonec_out("Nat address: %s%s\n", nat ? nat : "unspecified" , use ? "" : " (disabled - use 'firewall nat' to enable)");
aymeric's avatar
aymeric committed
862 863 864 865 866 867 868 869 870 871 872 873 874 875 876 877 878 879 880

	return 1;
}

static int
lpc_cmd_stun(LinphoneCore *lc, char *args)
{
	bool_t use;
	const char *stun;

	if ( args ) args=lpc_strip_blanks(args);

	if ( args && *args )
	{
		linphone_core_set_stun_server(lc, args);
		/* linphone_core_set_firewall_policy(lc,LINPHONE_POLICY_USE_STUN); */
	}

	stun = linphone_core_get_stun_server(lc);
Simon Morlat's avatar
Simon Morlat committed
881
	use = linphone_core_get_firewall_policy(lc)==LinphonePolicyUseStun;
882
	linphonec_out("Stun server: %s%s\n", stun ? stun : "unspecified" , use? "" : " (disabled - use 'firewall stun' to enable)");
aymeric's avatar
aymeric committed
883 884 885 886 887 888 889 890 891 892 893 894 895 896 897

	return 1;
}

static int
lpc_cmd_firewall(LinphoneCore *lc, char *args)
{
	const char* setting=NULL;

	if ( args ) args=lpc_strip_blanks(args);

	if ( args && *args )
	{
		if (strcmp(args,"none")==0)
		{
Simon Morlat's avatar
Simon Morlat committed
898
			linphone_core_set_firewall_policy(lc,LinphonePolicyNoFirewall);
aymeric's avatar
aymeric committed
899
		}
900
		else if (strcmp(args,"upnp")==0)
Yann Diorcet's avatar
Yann Diorcet committed
901 902 903
		{
			linphone_core_set_firewall_policy(lc,LinphonePolicyUseUpnp);
		}
904 905 906 907 908 909 910 911 912 913
		else if (strcmp(args,"ice")==0)
		{
			setting = linphone_core_get_stun_server(lc);
			if ( ! setting )
			{
				linphonec_out("No stun server address is defined, use 'stun <address>' first\n");
				return 1;
			}
			linphone_core_set_firewall_policy(lc,LinphonePolicyUseIce);
		}
aymeric's avatar
aymeric committed
914 915 916 917 918
		else if (strcmp(args,"stun")==0)
		{
			setting = linphone_core_get_stun_server(lc);
			if ( ! setting )
			{
919
				linphonec_out("No stun server address is defined, use 'stun <address>' first\n");
aymeric's avatar
aymeric committed
920 921
				return 1;
			}
Simon Morlat's avatar
Simon Morlat committed
922
			linphone_core_set_firewall_policy(lc,LinphonePolicyUseStun);
aymeric's avatar
aymeric committed
923 924 925 926 927 928
		}
		else if (strcmp(args,"nat")==0)
		{
			setting = linphone_core_get_nat_address(lc);
			if ( ! setting )
			{
929
				linphonec_out("No nat address is defined, use 'nat <address>' first");
aymeric's avatar
aymeric committed
930 931
				return 1;
			}
Simon Morlat's avatar
Simon Morlat committed
932
			linphone_core_set_firewall_policy(lc,LinphonePolicyUseNatAddress);
aymeric's avatar
aymeric committed
933 934 935 936 937
		}
	}

	switch(linphone_core_get_firewall_policy(lc))
	{
Simon Morlat's avatar
Simon Morlat committed
938
		case LinphonePolicyNoFirewall:
939
			linphonec_out("No firewall\n");
aymeric's avatar
aymeric committed
940
			break;
Simon Morlat's avatar
Simon Morlat committed
941
		case LinphonePolicyUseStun:
942
			linphonec_out("Using stun server %s to discover firewall address\n", setting ? setting : linphone_core_get_stun_server(lc));
aymeric's avatar
aymeric committed
943
			break;
Simon Morlat's avatar
Simon Morlat committed
944
		case LinphonePolicyUseNatAddress:
945
			linphonec_out("Using supplied nat address %s.\n", setting ? setting : linphone_core_get_nat_address(lc));
aymeric's avatar
aymeric committed
946
			break;
947 948 949
		case LinphonePolicyUseIce:
			linphonec_out("Using ice with stun server %s to discover firewall address\n", setting ? setting : linphone_core_get_stun_server(lc));
			break;
Yann Diorcet's avatar
Yann Diorcet committed
950 951 952
		case LinphonePolicyUseUpnp:
			linphonec_out("Using uPnP IGD protocol\n");
			break;
aymeric's avatar
aymeric committed
953 954 955 956
	}
	return 1;
}

957
#ifndef _WIN32
aymeric's avatar
aymeric committed
958 959 960 961 962 963 964 965 966 967 968 969 970 971 972 973 974 975 976 977
/* Helper function for processing freind names */
static int
lpc_friend_name(char **args, char **name)
{
	/* Use space as a terminator unless quoted */
	if (('"' == **args) || ('\'' == **args)){
		char *end;
		char delim = **args;
		(*args)++;
		end = (*args);
		while ((delim != *end) && ('\0' != *end)) end++;
		if ('\0' == *end) {
			fprintf(stderr, "Mismatched quotes\n");
			return 0;
		}
		*name = *args;
		*end = '\0';
		*args = ++end;
	} else {
		*name = strsep(args, " ");
978

aymeric's avatar
aymeric committed
979 980 981 982 983 984 985 986
		if (NULL == *args) { /* Means there was no separator */
			fprintf(stderr, "Either name or address is missing\n");
			return 0;
		}
		if (NULL == *name) return 0;
	}
	return 1;
}
smorlat's avatar
smorlat committed
987
#endif
aymeric's avatar
aymeric committed
988 989 990 991 992 993 994 995 996 997 998 999 1000 1001 1002 1003 1004 1005 1006

static int
lpc_cmd_friend(LinphoneCore *lc, char *args)
{
	int friend_num;

	if ( args ) args=lpc_strip_blanks(args);

	if ( ! args || ! *args ) return 0;

	if ( !strncmp