Version 0.2.0 from FTP
[asterisk/asterisk.git] / apps / app_playback.c
1 /*
2  * Asterisk -- A telephony toolkit for Linux.
3  *
4  * Trivial application to playback a sound file
5  * 
6  * Copyright (C) 1999, Mark Spencer
7  *
8  * Mark Spencer <markster@linux-support.net>
9  *
10  * This program is free software, distributed under the terms of
11  * the GNU General Public License
12  */
13  
14 #include <asterisk/lock.h>
15 #include <asterisk/file.h>
16 #include <asterisk/logger.h>
17 #include <asterisk/channel.h>
18 #include <asterisk/pbx.h>
19 #include <asterisk/module.h>
20 #include <asterisk/translate.h>
21 #include <string.h>
22 #include <stdlib.h>
23 #include <pthread.h>
24
25 static char *tdesc = "Trivial Playback Application";
26
27 static char *app = "Playback";
28
29 static char *synopsis = "Play a file";
30
31 static char *descrip = 
32 "  Playback(filename[|option]):  Plays  back  a  given  filename (do not put\n"
33 "extension). Options may also be  included following a pipe symbol. The only\n"
34 "defined option at this time is 'skip',  which  causes  the  playback of the\n"
35 "message to  be  skipped  if  the  channel is not in the 'up' state (i.e. it\n"
36 "hasn't been  answered  yet. If 'skip' is specified, the application will\n"
37 "return immediately should the channel not be off hook.  Otherwise, unless\n"
38 "'noanswer' is specified, the channel channel will be answered before the sound\n"
39 "is played. Not all channels support playing messages while on hook. Returns -1\n"
40 "if the channel was hung up, or if the file does not exist. Returns 0 otherwise.\n";
41
42 STANDARD_LOCAL_USER;
43
44 LOCAL_USER_DECL;
45
46 static int playback_exec(struct ast_channel *chan, void *data)
47 {
48         int res = 0;
49         struct localuser *u;
50         char tmp[256];
51         char *options;
52         int option_skip=0;
53         int option_noanswer = 0;
54         if (!data || !strlen((char *)data)) {
55                 ast_log(LOG_WARNING, "Playback requires an argument (filename)\n");
56                 return -1;
57         }
58         strncpy(tmp, (char *)data, sizeof(tmp)-1);
59         strtok(tmp, "|");
60         options = strtok(NULL, "|");
61         if (options && !strcasecmp(options, "skip"))
62                 option_skip = 1;
63         if (options && !strcasecmp(options, "noanswer"))
64                 option_noanswer = 1;
65         LOCAL_USER_ADD(u);
66         if (chan->_state != AST_STATE_UP) {
67                 if (option_skip) {
68                         /* At the user's option, skip if the line is not up */
69                         LOCAL_USER_REMOVE(u);
70                         return 0;
71                 } else if (!option_noanswer)
72                         /* Otherwise answer unless we're supposed to send this while on-hook */
73                         res = ast_answer(chan);
74         }
75         if (!res) {
76                 ast_stopstream(chan);
77                 res = ast_streamfile(chan, tmp, chan->language);
78                 if (!res) 
79                         res = ast_waitstream(chan, "");
80                 else {
81                         ast_log(LOG_WARNING, "ast_streamfile failed on %s for %s\n", chan->name, (char *)data);
82                         res = 0;
83                 }
84                 ast_stopstream(chan);
85         }
86         LOCAL_USER_REMOVE(u);
87         return res;
88 }
89
90 int unload_module(void)
91 {
92         STANDARD_HANGUP_LOCALUSERS;
93         return ast_unregister_application(app);
94 }
95
96 int load_module(void)
97 {
98         return ast_register_application(app, playback_exec, synopsis, descrip);
99 }
100
101 char *description(void)
102 {
103         return tdesc;
104 }
105
106 int usecount(void)
107 {
108         int res;
109         STANDARD_USECOUNT(res);
110         return res;
111 }
112
113 char *key()
114 {
115         return ASTERISK_GPL_KEY;
116 }