Asterisk - The Open Source Telephony Project  18.5.0
func_dialgroup.c
Go to the documentation of this file.
1 /*
2  * Asterisk -- An open source telephony toolkit.
3  *
4  * Copyright (C) 2007, Tilghman Lesher
5  *
6  * Tilghman Lesher <[email protected]>
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 Dial group dialplan function
22  *
23  * \author Tilghman Lesher <[email protected]>
24  *
25  * \ingroup functions
26  */
27 
28 /*** MODULEINFO
29  <support_level>core</support_level>
30  ***/
31 
32 #include "asterisk.h"
33 
34 #include <sys/stat.h>
35 
36 #include "asterisk/module.h"
37 #include "asterisk/channel.h"
38 #include "asterisk/pbx.h"
39 #include "asterisk/utils.h"
40 #include "asterisk/app.h"
41 #include "asterisk/astobj2.h"
42 #include "asterisk/astdb.h"
43 
44 /*** DOCUMENTATION
45  <function name="DIALGROUP" language="en_US">
46  <synopsis>
47  Manages a group of users for dialing.
48  </synopsis>
49  <syntax>
50  <parameter name="group" required="true" />
51  <parameter name="op">
52  <para>The operation name, possible values are:</para>
53  <para><literal>add</literal> - add a channel name or interface (write-only)</para>
54  <para><literal>del</literal> - remove a channel name or interface (write-only)</para>
55  </parameter>
56  </syntax>
57  <description>
58  <para>Presents an interface meant to be used in concert with the Dial
59  application, by presenting a list of channels which should be dialled when
60  referenced.</para>
61  <para>When DIALGROUP is read from, the argument is interpreted as the particular
62  <replaceable>group</replaceable> for which a dial should be attempted. When DIALGROUP is written to
63  with no arguments, the entire list is replaced with the argument specified.</para>
64  <para>Functionality is similar to a queue, except that when no interfaces are
65  available, execution may continue in the dialplan. This is useful when
66  you want certain people to be the first to answer any calls, with immediate
67  fallback to a queue when the front line people are busy or unavailable, but
68  you still want front line people to log in and out of that group, just like
69  a queue.</para>
70  <para>Example:</para>
71  <para>exten => 1,1,Set(DIALGROUP(mygroup,add)=SIP/10)</para>
72  <para>exten => 1,n,Set(DIALGROUP(mygroup,add)=SIP/20)</para>
73  <para>exten => 1,n,Dial(${DIALGROUP(mygroup)})</para>
74  </description>
75  </function>
76  ***/
77 
79 
80 struct group_entry {
82 };
83 
84 struct group {
87 };
88 
89 static void group_destroy(void *vgroup)
90 {
91  struct group *group = vgroup;
92  ao2_ref(group->entries, -1);
93 }
94 
95 static int group_hash_fn(const void *obj, const int flags)
96 {
97  const struct group *g = obj;
98  return ast_str_hash(g->name);
99 }
100 
101 static int group_cmp_fn(void *obj1, void *name2, int flags)
102 {
103  struct group *g1 = obj1, *g2 = name2;
104  char *name = name2;
105  if (flags & OBJ_POINTER)
106  return strcmp(g1->name, g2->name) ? 0 : CMP_MATCH | CMP_STOP;
107  else
108  return strcmp(g1->name, name) ? 0 : CMP_MATCH | CMP_STOP;
109 }
110 
111 static int entry_hash_fn(const void *obj, const int flags)
112 {
113  const struct group_entry *e = obj;
114  return ast_str_hash(e->name);
115 }
116 
117 static int entry_cmp_fn(void *obj1, void *name2, int flags)
118 {
119  struct group_entry *e1 = obj1, *e2 = name2;
120  char *name = name2;
121  if (flags & OBJ_POINTER)
122  return strcmp(e1->name, e2->name) ? 0 : CMP_MATCH | CMP_STOP;
123  else
124  return strcmp(e1->name, name) ? 0 : CMP_MATCH | CMP_STOP;
125 }
126 
127 static int dialgroup_read(struct ast_channel *chan, const char *cmd, char *data, char *buf, size_t len)
128 {
129  struct ao2_iterator i;
130  struct group *grhead = ao2_find(group_container, data, 0);
131  struct group_entry *entry;
132  size_t bufused = 0;
133  int trunc_warning = 0;
134  int res = 0;
135 
136  if (!grhead) {
137  if (!ast_strlen_zero(cmd)) {
138  ast_log(LOG_WARNING, "No such dialgroup '%s'\n", data);
139  }
140  return -1;
141  }
142 
143  buf[0] = '\0';
144 
145  i = ao2_iterator_init(grhead->entries, 0);
146  while ((entry = ao2_iterator_next(&i))) {
147  int tmp = strlen(entry->name);
148  /* Ensure that we copy only complete names, not partials */
149  if (len - bufused > tmp + 2) {
150  if (bufused != 0)
151  buf[bufused++] = '&';
152  ast_copy_string(buf + bufused, entry->name, len - bufused);
153  bufused += tmp;
154  } else if (trunc_warning++ == 0) {
155  if (!ast_strlen_zero(cmd)) {
156  ast_log(LOG_WARNING, "Dialgroup '%s' is too large. Truncating list.\n", data);
157  } else {
158  res = 1;
159  ao2_ref(entry, -1);
160  break;
161  }
162  }
163  ao2_ref(entry, -1);
164  }
166  ao2_ref(grhead, -1);
167 
168  return res;
169 }
170 
171 static int dialgroup_refreshdb(struct ast_channel *chan, const char *cdialgroup)
172 {
173  int len = 500, res = 0;
174  char *buf = NULL;
175  char *new_buf;
176  char *dialgroup = ast_strdupa(cdialgroup);
177 
178  do {
179  len *= 2;
180  new_buf = ast_realloc(buf, len);
181  if (!new_buf) {
182  ast_free(buf);
183  return -1;
184  }
185  buf = new_buf;
186 
187  if ((res = dialgroup_read(chan, "", dialgroup, buf, len)) < 0) {
188  ast_free(buf);
189  return -1;
190  }
191  } while (res == 1);
192 
193  if (ast_strlen_zero(buf)) {
194  ast_db_del("dialgroup", cdialgroup);
195  } else {
196  ast_db_put("dialgroup", cdialgroup, buf);
197  }
198  ast_free(buf);
199  return 0;
200 }
201 
202 static int dialgroup_write(struct ast_channel *chan, const char *cmd, char *data, const char *cvalue)
203 {
204  struct group *grhead;
205  struct group_entry *entry;
206  int j, needrefresh = 1;
209  AST_APP_ARG(op);
210  );
211  AST_DECLARE_APP_ARGS(inter,
212  AST_APP_ARG(faces)[100];
213  );
214  char *value = ast_strdupa(cvalue);
215 
217  AST_NONSTANDARD_APP_ARGS(inter, value, '&');
218 
219  if (!(grhead = ao2_find(group_container, args.group, 0))) {
220  /* Create group */
221  grhead = ao2_alloc(sizeof(*grhead), group_destroy);
222  if (!grhead)
223  return -1;
226  if (!grhead->entries) {
227  ao2_ref(grhead, -1);
228  return -1;
229  }
230  ast_copy_string(grhead->name, args.group, sizeof(grhead->name));
231  ao2_link(group_container, grhead);
232  }
233 
234  if (ast_strlen_zero(args.op)) {
235  /* Wholesale replacement of the group */
236  args.op = "add";
237 
238  /* Remove all existing */
239  ao2_ref(grhead->entries, -1);
242  if (!grhead->entries) {
243  ao2_unlink(group_container, grhead);
244  ao2_ref(grhead, -1);
245  return -1;
246  }
247  }
248 
249  if (strcasecmp(args.op, "add") == 0) {
250  for (j = 0; j < inter.argc; j++) {
251  /* Eliminate duplicates */
252  if ((entry = ao2_find(grhead->entries, inter.faces[j], 0))) {
253  ao2_ref(entry, -1);
254  continue;
255  }
256  if ((entry = ao2_alloc(sizeof(*entry), NULL))) {
257  ast_copy_string(entry->name, inter.faces[j], sizeof(entry->name));
258  ao2_link(grhead->entries, entry);
259  ao2_ref(entry, -1);
260  } else {
261  ast_log(LOG_WARNING, "Unable to add '%s' to dialgroup '%s'\n", inter.faces[j], grhead->name);
262  }
263  }
264  } else if (strncasecmp(args.op, "del", 3) == 0) {
265  for (j = 0; j < inter.argc; j++) {
266  if ((entry = ao2_find(grhead->entries, inter.faces[j], OBJ_UNLINK))) {
267  ao2_ref(entry, -1);
268  } else {
269  ast_log(LOG_WARNING, "Interface '%s' not found in dialgroup '%s'\n", inter.faces[j], grhead->name);
270  }
271  }
272  } else {
273  ast_log(LOG_ERROR, "Unrecognized operation: %s\n", args.op);
274  needrefresh = 0;
275  }
276  ao2_ref(grhead, -1);
277 
278  if (needrefresh) {
279  dialgroup_refreshdb(chan, args.group);
280  }
281 
282  return 0;
283 }
284 
286  .name = "DIALGROUP",
287  .read = dialgroup_read,
288  .write = dialgroup_write,
289 };
290 
291 static int unload_module(void)
292 {
293  int res = ast_custom_function_unregister(&dialgroup_function);
294  ao2_ref(group_container, -1);
295  return res;
296 }
297 
298 static int load_module(void)
299 {
300  struct ast_db_entry *dbtree, *tmp;
301  char groupname[AST_MAX_EXTENSION], *ptr;
302 
303  group_container = ao2_container_alloc_hash(AO2_ALLOC_OPT_LOCK_MUTEX, 0, 37,
305  if (group_container) {
306  /* Refresh groups from astdb */
307  if ((dbtree = ast_db_gettree("dialgroup", NULL))) {
308  for (tmp = dbtree; tmp; tmp = tmp->next) {
309  ast_copy_string(groupname, tmp->key, sizeof(groupname));
310  if ((ptr = strrchr(groupname, '/'))) {
311  ptr++;
312  dialgroup_write(NULL, "", ptr, tmp->data);
313  }
314  }
315  ast_db_freetree(dbtree);
316  }
317  return ast_custom_function_register(&dialgroup_function);
318  } else {
320  }
321 }
322 
323 AST_MODULE_INFO_STANDARD(ASTERISK_GPL_KEY, "Dialgroup dialplan function");
const char * name
Definition: pbx.h:119
Main Channel structure associated with a channel.
#define AST_MODULE_INFO_STANDARD(keystr, desc)
Definition: module.h:567
static struct ast_custom_function dialgroup_function
Asterisk main include file. File version handling, generic pbx functions.
#define ast_realloc(p, len)
A wrapper for realloc()
Definition: astmm.h:228
char buf[BUFSIZE]
Definition: eagi_proxy.c:66
static int unload_module(void)
#define OBJ_POINTER
Definition: astobj2.h:1154
#define AST_STANDARD_APP_ARGS(args, parse)
Performs the &#39;standard&#39; argument separation process for an application.
#define LOG_WARNING
Definition: logger.h:274
void ast_db_freetree(struct ast_db_entry *entry)
Free structure created by ast_db_gettree()
Definition: main/db.c:598
static int tmp()
Definition: bt_open.c:389
void ao2_iterator_destroy(struct ao2_iterator *iter)
Destroy a container iterator.
const char * args
#define NULL
Definition: resample.c:96
int value
Definition: syslog.c:37
int ast_custom_function_unregister(struct ast_custom_function *acf)
Unregister a custom function.
Utility functions.
#define ast_strlen_zero(foo)
Definition: strings.h:52
static struct ao2_container * group_container
#define ast_log
Definition: astobj2.c:42
struct ast_db_entry * next
Definition: astdb.h:32
static int group_hash_fn(const void *obj, const int flags)
General Asterisk PBX channel definitions.
Data structure associated with a custom dialplan function.
Definition: pbx.h:118
char name[AST_CHANNEL_NAME]
#define AST_MAX_EXTENSION
Definition: channel.h:135
#define ao2_ref(o, delta)
Definition: astobj2.h:464
#define ast_strdupa(s)
duplicate a string in memory from the stack
Definition: astmm.h:300
static int dialgroup_read(struct ast_channel *chan, const char *cmd, char *data, char *buf, size_t len)
Core PBX routines and definitions.
#define LOG_ERROR
Definition: logger.h:285
#define ao2_container_alloc_hash(ao2_options, container_options, n_buckets, hash_fn, sort_fn, cmp_fn)
Definition: astobj2.h:1310
#define ao2_unlink(container, obj)
Definition: astobj2.h:1598
struct ast_db_entry * ast_db_gettree(const char *family, const char *keytree)
Get a list of values within the astdb tree.
Definition: main/db.c:531
static int load_module(void)
#define AST_NONSTANDARD_APP_ARGS(args, parse, sep)
Performs the &#39;nonstandard&#39; argument separation process for an application.
char name[AST_MAX_EXTENSION]
static int len(struct ast_channel *chan, const char *cmd, char *data, char *buf, size_t buflen)
#define ao2_iterator_next(iter)
Definition: astobj2.h:1933
#define ao2_alloc(data_size, destructor_fn)
Definition: astobj2.h:411
Definition: astdb.h:31
static int entry_hash_fn(const void *obj, const int flags)
char data[0]
Definition: astdb.h:34
#define ast_free(a)
Definition: astmm.h:182
#define AST_CHANNEL_NAME
Definition: channel.h:172
Module has failed to load, may be in an inconsistent state.
Definition: module.h:78
#define ao2_find(container, arg, flags)
Definition: astobj2.h:1756
static int entry_cmp_fn(void *obj1, void *name2, int flags)
static int dialgroup_refreshdb(struct ast_channel *chan, const char *cdialgroup)
static int group_cmp_fn(void *obj1, void *name2, int flags)
When we need to walk through a container, we use an ao2_iterator to keep track of the current positio...
Definition: astobj2.h:1841
int ast_db_del(const char *family, const char *key)
Delete entry in astdb.
Definition: main/db.c:429
void ast_copy_string(char *dst, const char *src, size_t size)
Size-limited null-terminating string copy.
Definition: strings.h:401
static void group_destroy(void *vgroup)
Definition: search.h:40
int ast_db_put(const char *family, const char *key, const char *value)
Store value addressed by family/key.
Definition: main/db.c:327
Generic container type.
static int dialgroup_write(struct ast_channel *chan, const char *cmd, char *data, const char *cvalue)
#define ASTERISK_GPL_KEY
The text the key() function should return.
Definition: module.h:46
Asterisk module definitions.
Persistant data storage (akin to *doze registry)
#define AST_DECLARE_APP_ARGS(name, arglist)
Declare a structure to hold an application&#39;s arguments.
Application convenience functions, designed to give consistent look and feel to Asterisk apps...
struct ao2_container * entries
struct ao2_iterator ao2_iterator_init(struct ao2_container *c, int flags) attribute_warn_unused_result
Create an iterator for a container.
#define ast_custom_function_register(acf)
Register a custom function.
Definition: pbx.h:1508
char * key
Definition: astdb.h:33
#define AST_APP_ARG(name)
Define an application argument.
static force_inline int attribute_pure ast_str_hash(const char *str)
Compute a hash value on a string.
Definition: strings.h:1206
#define ao2_link(container, obj)
Definition: astobj2.h:1549