Asterisk - The Open Source Telephony Project GIT-master-8924258
All Data Structures Namespaces Files Functions Variables Typedefs Enumerations Enumerator Properties Macros Modules Pages
func_devstate.c
Go to the documentation of this file.
1/*
2 * Asterisk -- An open source telephony toolkit.
3 *
4 * Copyright (C) 2007, Digium, Inc.
5 *
6 * Russell Bryant <russell@digium.com>
7 *
8 * See http://www.asterisk.org for more information about
9 * the Asterisk project. Please do not directly contact
10 * any of the maintainers of this project for assistance;
11 * the project provides a web site, mailing lists and IRC
12 * channels for your use.
13 *
14 * This program is free software, distributed under the terms of
15 * the GNU General Public License Version 2. See the LICENSE file
16 * at the top of the source tree.
17 */
18
19/*! \file
20 *
21 * \brief Manually controlled blinky lights
22 *
23 * \author Russell Bryant <russell@digium.com>
24 *
25 * \ingroup functions
26 *
27 * \todo Delete the entry from AstDB when set to nothing like Set(DEVICE_STATE(Custom:lamp1)=)
28 *
29 * \note Props go out to Ahrimanes in \#asterisk for requesting this at 4:30 AM
30 * when I couldn't sleep. :)
31 */
32
33/*** MODULEINFO
34 <support_level>core</support_level>
35 ***/
36
37#include "asterisk.h"
38
39#include "asterisk/module.h"
40#include "asterisk/channel.h"
41#include "asterisk/pbx.h"
42#include "asterisk/utils.h"
45#include "asterisk/cli.h"
46#include "asterisk/astdb.h"
47#include "asterisk/app.h"
48
49/*** DOCUMENTATION
50 <function name="DEVICE_STATE" language="en_US">
51 <since>
52 <version>1.6.0</version>
53 </since>
54 <synopsis>
55 Get or Set a device state.
56 </synopsis>
57 <syntax>
58 <parameter name="device" required="true" />
59 </syntax>
60 <description>
61 <para>The DEVICE_STATE function can be used to retrieve the device state from any
62 device state provider. For example:</para>
63 <para>NoOp(SIP/mypeer has state ${DEVICE_STATE(SIP/mypeer)})</para>
64 <para>NoOp(Conference number 1234 has state ${DEVICE_STATE(MeetMe:1234)})</para>
65 <para>The DEVICE_STATE function can also be used to set custom device state from
66 the dialplan. The <literal>Custom:</literal> prefix must be used. For example:</para>
67 <para>Set(DEVICE_STATE(Custom:lamp1)=BUSY)</para>
68 <para>Set(DEVICE_STATE(Custom:lamp2)=NOT_INUSE)</para>
69 <para>You can subscribe to the status of a custom device state using a hint in
70 the dialplan:</para>
71 <para>exten => 1234,hint,Custom:lamp1</para>
72 <para>The possible values for both uses of this function are:</para>
73 <para>UNKNOWN | NOT_INUSE | INUSE | BUSY | INVALID | UNAVAILABLE | RINGING |
74 RINGINUSE | ONHOLD</para>
75 </description>
76 </function>
77 <function name="HINT" language="en_US">
78 <since>
79 <version>1.6.0</version>
80 </since>
81 <synopsis>
82 Get the devices set for a dialplan hint.
83 </synopsis>
84 <syntax>
85 <parameter name="extension" required="true" argsep="@">
86 <argument name="extension" required="true" />
87 <argument name="context" />
88 </parameter>
89 <parameter name="options">
90 <optionlist>
91 <option name="n">
92 <para>Retrieve name on the hint instead of list of devices.</para>
93 </option>
94 </optionlist>
95 </parameter>
96 </syntax>
97 <description>
98 <para>The HINT function can be used to retrieve the list of devices that are
99 mapped to a dialplan hint.</para>
100 <example title="Hint for extension 1234">
101 same => n,NoOp(Hint for Extension 1234 is ${HINT(1234)})
102 </example>
103 </description>
104 </function>
105 ***/
106
107
108static const char astdb_family[] = "CustomDevstate";
109
110static int devstate_read(struct ast_channel *chan, const char *cmd, char *data, char *buf, size_t len)
111{
113
114 return 0;
115}
116
117static int devstate_write(struct ast_channel *chan, const char *cmd, char *data, const char *value)
118{
119 size_t len = strlen("Custom:");
120 enum ast_device_state state_val;
121
122 if (strncasecmp(data, "Custom:", len)) {
123 ast_log(LOG_WARNING, "The DEVICE_STATE function can only be used to set 'Custom:' device state!\n");
124 return -1;
125 }
126 data += len;
127 if (ast_strlen_zero(data)) {
128 ast_log(LOG_WARNING, "DEVICE_STATE function called with no custom device name!\n");
129 return -1;
130 }
131
132 state_val = ast_devstate_val(value);
133
134 if (state_val == AST_DEVICE_UNKNOWN) {
135 ast_log(LOG_ERROR, "DEVICE_STATE function given invalid state value '%s'\n", value);
136 return -1;
137 }
138
140
141 ast_devstate_changed(state_val, AST_DEVSTATE_CACHABLE, "Custom:%s", data);
142
143 return 0;
144}
145
146enum {
147 HINT_OPT_NAME = (1 << 0),
148};
149
153
154static int hint_read(struct ast_channel *chan, const char *cmd, char *data, char *buf, size_t len)
155{
156 char *exten, *context;
158 AST_APP_ARG(exten);
160 );
161 struct ast_flags opts = { 0, };
162 int res;
163
164 if (ast_strlen_zero(data)) {
165 ast_log(LOG_WARNING, "The HINT function requires an extension\n");
166 return -1;
167 }
168
170
171 if (ast_strlen_zero(args.exten)) {
172 ast_log(LOG_WARNING, "The HINT function requires an extension\n");
173 return -1;
174 }
175
176 context = exten = args.exten;
177 strsep(&context, "@");
179 context = "default";
180
181 if (!ast_strlen_zero(args.options))
183
184 if (ast_test_flag(&opts, HINT_OPT_NAME))
185 res = ast_get_hint(NULL, 0, buf, len, chan, context, exten);
186 else
187 res = ast_get_hint(buf, len, NULL, 0, chan, context, exten);
188
189 return !res; /* ast_get_hint returns non-zero on success */
190}
191
192static enum ast_device_state custom_devstate_callback(const char *data)
193{
194 char buf[256] = "";
195
196 /* Ignore check_return warning from Coverity fow ast_db_get below */
197 ast_db_get(astdb_family, data, buf, sizeof(buf));
198
199 return ast_devstate_val(buf);
200}
201
202static char *handle_cli_devstate_list(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
203{
204 struct ast_db_entry *db_entry, *db_tree;
205
206 switch (cmd) {
207 case CLI_INIT:
208 e->command = "devstate list";
209 e->usage =
210 "Usage: devstate list\n"
211 " List all custom device states that have been set by using\n"
212 " the DEVICE_STATE dialplan function.\n";
213 return NULL;
214 case CLI_GENERATE:
215 return NULL;
216 }
217
218 if (a->argc != e->args)
219 return CLI_SHOWUSAGE;
220
221 ast_cli(a->fd, "\n"
222 "---------------------------------------------------------------------\n"
223 "--- Custom Device States --------------------------------------------\n"
224 "---------------------------------------------------------------------\n"
225 "---\n");
226
227 db_entry = db_tree = ast_db_gettree(astdb_family, NULL);
228 for (; db_entry; db_entry = db_entry->next) {
229 const char *dev_name = strrchr(db_entry->key, '/') + 1;
230 if (dev_name <= (const char *) 1)
231 continue;
232 ast_cli(a->fd, "--- Name: 'Custom:%s' State: '%s'\n"
233 "---\n", dev_name, db_entry->data);
234 }
235 ast_db_freetree(db_tree);
236 db_tree = NULL;
237
238 ast_cli(a->fd,
239 "---------------------------------------------------------------------\n"
240 "---------------------------------------------------------------------\n"
241 "\n");
242
243 return CLI_SUCCESS;
244}
245
246static char *handle_cli_devstate_change(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
247{
248 size_t len;
249 const char *dev, *state;
250 enum ast_device_state state_val;
251
252 switch (cmd) {
253 case CLI_INIT:
254 e->command = "devstate change";
255 e->usage =
256 "Usage: devstate change <device> <state>\n"
257 " Change a custom device to a new state.\n"
258 " The possible values for the state are:\n"
259 "UNKNOWN | NOT_INUSE | INUSE | BUSY | INVALID | UNAVAILABLE | RINGING\n"
260 "RINGINUSE | ONHOLD\n"
261 "\n"
262 "Examples:\n"
263 " devstate change Custom:mystate1 INUSE\n"
264 " devstate change Custom:mystate1 NOT_INUSE\n"
265 " \n";
266 return NULL;
267 case CLI_GENERATE:
268 {
269 static const char * const cmds[] = { "UNKNOWN", "NOT_INUSE", "INUSE", "BUSY",
270 "UNAVAILABLE", "RINGING", "RINGINUSE", "ONHOLD", NULL };
271
272 if (a->pos == e->args + 1)
273 return ast_cli_complete(a->word, cmds, a->n);
274
275 return NULL;
276 }
277 }
278
279 if (a->argc != e->args + 2)
280 return CLI_SHOWUSAGE;
281
282 len = strlen("Custom:");
283 dev = a->argv[e->args];
284 state = a->argv[e->args + 1];
285
286 if (strncasecmp(dev, "Custom:", len)) {
287 ast_cli(a->fd, "The devstate command can only be used to set 'Custom:' device state!\n");
288 return CLI_FAILURE;
289 }
290
291 dev += len;
292 if (ast_strlen_zero(dev))
293 return CLI_SHOWUSAGE;
294
295 state_val = ast_devstate_val(state);
296
297 if (state_val == AST_DEVICE_UNKNOWN)
298 return CLI_SHOWUSAGE;
299
300 ast_cli(a->fd, "Changing %s to %s\n", dev, state);
301
303
304 ast_devstate_changed(state_val, AST_DEVSTATE_CACHABLE, "Custom:%s", dev);
305
306 return CLI_SUCCESS;
307}
308
310 AST_CLI_DEFINE(handle_cli_devstate_list, "List currently known custom device states"),
311 AST_CLI_DEFINE(handle_cli_devstate_change, "Change a custom device state"),
312};
313
315 .name = "DEVICE_STATE",
316 .read = devstate_read,
317 .write = devstate_write,
318};
319
321 .name = "HINT",
322 .read = hint_read,
323};
324
325static int unload_module(void)
326{
327 int res = 0;
328
331 res |= ast_devstate_prov_del("Custom");
333
334 return res;
335}
336
337static int load_module(void)
338{
339 int res = 0;
340 struct ast_db_entry *db_entry, *db_tree;
341
342 /* Populate the device state cache on the system with all of the currently
343 * known custom device states. */
344 db_entry = db_tree = ast_db_gettree(astdb_family, NULL);
345 for (; db_entry; db_entry = db_entry->next) {
346 const char *dev_name = strrchr(db_entry->key, '/') + 1;
347 if (dev_name <= (const char *) 1)
348 continue;
350 AST_DEVSTATE_CACHABLE, "Custom:%s", dev_name);
351 }
352 ast_db_freetree(db_tree);
353 db_tree = NULL;
354
359
360 return res;
361}
362
363AST_MODULE_INFO(ASTERISK_GPL_KEY, AST_MODFLAG_LOAD_ORDER, "Gets or sets a device state in the dialplan",
364 .support_level = AST_MODULE_SUPPORT_CORE,
365 .load = load_module,
366 .unload = unload_module,
Persistent data storage (akin to *doze registry)
int ast_db_put(const char *family, const char *key, const char *value)
Store value addressed by family/key.
Definition: db.c:335
int ast_db_get(const char *family, const char *key, char *value, int valuelen)
Get key value specified by family/key.
Definition: db.c:421
struct ast_db_entry * ast_db_gettree(const char *family, const char *keytree)
Get a list of values within the astdb tree.
Definition: db.c:635
void ast_db_freetree(struct ast_db_entry *entry)
Free structure created by ast_db_gettree()
Definition: db.c:695
char * strsep(char **str, const char *delims)
Asterisk main include file. File version handling, generic pbx functions.
#define ast_log
Definition: astobj2.c:42
enum cc_state state
Definition: ccss.c:399
General Asterisk PBX channel definitions.
Standard Command Line Interface.
#define CLI_SHOWUSAGE
Definition: cli.h:45
#define CLI_SUCCESS
Definition: cli.h:44
int ast_cli_unregister_multiple(struct ast_cli_entry *e, int len)
Unregister multiple commands.
Definition: clicompat.c:30
#define AST_CLI_DEFINE(fn, txt,...)
Definition: cli.h:197
void ast_cli(int fd, const char *fmt,...)
Definition: clicompat.c:6
char * ast_cli_complete(const char *word, const char *const choices[], int pos)
Definition: main/cli.c:1853
@ CLI_INIT
Definition: cli.h:152
@ CLI_GENERATE
Definition: cli.h:153
#define CLI_FAILURE
Definition: cli.h:46
#define ast_cli_register_multiple(e, len)
Register multiple commands.
Definition: cli.h:265
Device state management.
int ast_devstate_prov_del(const char *label)
Remove device state provider.
Definition: devicestate.c:421
@ AST_DEVSTATE_CACHABLE
Definition: devicestate.h:70
const char * ast_devstate_str(enum ast_device_state devstate) attribute_pure
Convert device state to text string that is easier to parse.
Definition: devicestate.c:258
int ast_devstate_changed(enum ast_device_state state, enum ast_devstate_cache cachable, const char *fmt,...)
Tells Asterisk the State for Device is changed.
Definition: devicestate.c:513
enum ast_device_state ast_devstate_val(const char *val)
Convert device state from text to integer value.
Definition: devicestate.c:263
int ast_devstate_prov_add(const char *label, ast_devstate_prov_cb_type callback)
Add device state provider.
Definition: devicestate.c:394
ast_device_state
Device States.
Definition: devicestate.h:52
@ AST_DEVICE_UNKNOWN
Definition: devicestate.h:53
char buf[BUFSIZE]
Definition: eagi_proxy.c:66
static int devstate_write(struct ast_channel *chan, const char *cmd, char *data, const char *value)
static char * handle_cli_devstate_change(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
static struct ast_cli_entry cli_funcdevstate[]
static int hint_read(struct ast_channel *chan, const char *cmd, char *data, char *buf, size_t len)
@ HINT_OPT_NAME
static enum ast_device_state custom_devstate_callback(const char *data)
static struct ast_custom_function devstate_function
static struct ast_custom_function hint_function
static char * handle_cli_devstate_list(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
static const char astdb_family[]
static int load_module(void)
static int unload_module(void)
static int devstate_read(struct ast_channel *chan, const char *cmd, char *data, char *buf, size_t len)
static const struct ast_app_option hint_options[128]
static int len(struct ast_channel *chan, const char *cmd, char *data, char *buf, size_t buflen)
Application convenience functions, designed to give consistent look and feel to Asterisk apps.
#define AST_APP_ARG(name)
Define an application argument.
#define END_OPTIONS
#define AST_APP_OPTIONS(holder, options...)
Declares an array of options for an application.
#define AST_DECLARE_APP_ARGS(name, arglist)
Declare a structure to hold an application's arguments.
#define BEGIN_OPTIONS
#define AST_STANDARD_APP_ARGS(args, parse)
Performs the 'standard' argument separation process for an application.
#define AST_APP_OPTION(option, flagno)
Declares an application option that does not accept an argument.
int ast_app_parse_options(const struct ast_app_option *options, struct ast_flags *flags, char **args, char *optstr)
Parses a string containing application options and sets flags/arguments.
Definition: main/app.c:3066
#define LOG_ERROR
#define LOG_WARNING
A set of macros to manage forward-linked lists.
Asterisk module definitions.
@ AST_MODFLAG_LOAD_ORDER
Definition: module.h:331
#define AST_MODULE_INFO(keystr, flags_to_set, desc, fields...)
Definition: module.h:557
@ AST_MODPRI_DEVSTATE_PROVIDER
Definition: module.h:343
@ AST_MODULE_SUPPORT_CORE
Definition: module.h:121
#define ASTERISK_GPL_KEY
The text the key() function should return.
Definition: module.h:46
Core PBX routines and definitions.
#define ast_custom_function_register(acf)
Register a custom function.
Definition: pbx.h:1559
int ast_custom_function_unregister(struct ast_custom_function *acf)
Unregister a custom function.
int ast_get_hint(char *hint, int hintsize, char *name, int namesize, struct ast_channel *c, const char *context, const char *exten)
If an extension hint exists, return non-zero.
Definition: pbx.c:4152
#define NULL
Definition: resample.c:96
static force_inline int attribute_pure ast_strlen_zero(const char *s)
Definition: strings.h:65
void ast_copy_string(char *dst, const char *src, size_t size)
Size-limited null-terminating string copy.
Definition: strings.h:425
Main Channel structure associated with a channel.
descriptor for a cli entry.
Definition: cli.h:171
int args
This gets set in ast_cli_register()
Definition: cli.h:185
char * command
Definition: cli.h:186
const char * usage
Definition: cli.h:177
Data structure associated with a custom dialplan function.
Definition: pbx.h:118
const char * name
Definition: pbx.h:119
Definition: astdb.h:31
struct ast_db_entry * next
Definition: astdb.h:32
char * key
Definition: astdb.h:33
char data[0]
Definition: astdb.h:34
Structure used to handle boolean flags.
Definition: utils.h:199
int value
Definition: syslog.c:37
const char * args
static struct test_options options
static struct test_val a
Utility functions.
#define ast_test_flag(p, flag)
Definition: utils.h:63
#define ARRAY_LEN(a)
Definition: utils.h:666