save
[protos/libecoli.git] / lib / ecoli_node_option.c
1 /*
2  * Copyright (c) 2016, Olivier MATZ <zer0@droids-corp.org>
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions are met:
6  *
7  *     * Redistributions of source code must retain the above copyright
8  *       notice, this list of conditions and the following disclaimer.
9  *     * Redistributions in binary form must reproduce the above copyright
10  *       notice, this list of conditions and the following disclaimer in the
11  *       documentation and/or other materials provided with the distribution.
12  *     * Neither the name of the University of California, Berkeley nor the
13  *       names of its contributors may be used to endorse or promote products
14  *       derived from this software without specific prior written permission.
15  *
16  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND ANY
17  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
18  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
19  * DISCLAIMED. IN NO EVENT SHALL THE REGENTS AND CONTRIBUTORS BE LIABLE FOR ANY
20  * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
21  * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
22  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
23  * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
24  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
25  * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26  */
27
28 #include <stdio.h>
29 #include <stdlib.h>
30 #include <string.h>
31 #include <assert.h>
32 #include <stdarg.h>
33
34 #include <ecoli_malloc.h>
35 #include <ecoli_log.h>
36 #include <ecoli_strvec.h>
37 #include <ecoli_node.h>
38 #include <ecoli_node_option.h>
39 #include <ecoli_node_str.h>
40 #include <ecoli_test.h>
41
42 struct ec_node_option {
43         struct ec_node gen;
44         struct ec_node *child;
45 };
46
47 static struct ec_parsed *ec_node_option_parse(const struct ec_node *gen_node,
48         const struct ec_strvec *strvec)
49 {
50         struct ec_node_option *node = (struct ec_node_option *)gen_node;
51         struct ec_parsed *parsed = NULL, *child_parsed;
52         struct ec_strvec *match_strvec;
53
54         parsed = ec_parsed_new();
55         if (parsed == NULL)
56                 goto fail;
57
58         child_parsed = ec_node_parse_strvec(node->child, strvec);
59         if (child_parsed == NULL)
60                 goto fail;
61
62         if (ec_parsed_matches(child_parsed)) {
63                 ec_parsed_add_child(parsed, child_parsed);
64                 match_strvec = ec_strvec_dup(child_parsed->strvec);
65         } else {
66                 ec_parsed_free(child_parsed);
67                 match_strvec = ec_strvec_new();
68         }
69
70         if (match_strvec == NULL)
71                 goto fail;
72
73         ec_parsed_set_match(parsed, gen_node, match_strvec);
74
75         return parsed;
76
77  fail:
78         ec_parsed_free(parsed);
79         return NULL;
80 }
81
82 static struct ec_completed *ec_node_option_complete(const struct ec_node *gen_node,
83         const struct ec_strvec *strvec)
84 {
85         struct ec_node_option *node = (struct ec_node_option *)gen_node;
86
87         return ec_node_complete_strvec(node->child, strvec);
88 }
89
90 static void ec_node_option_free_priv(struct ec_node *gen_node)
91 {
92         struct ec_node_option *node = (struct ec_node_option *)gen_node;
93
94         ec_node_free(node->child);
95 }
96
97 static struct ec_node_type ec_node_option_type = {
98         .name = "option",
99         .parse = ec_node_option_parse,
100         .complete = ec_node_option_complete,
101         .size = sizeof(struct ec_node_option),
102         .free_priv = ec_node_option_free_priv,
103 };
104
105 EC_NODE_TYPE_REGISTER(ec_node_option_type);
106
107 struct ec_node *ec_node_option(const char *id, struct ec_node *child)
108 {
109         struct ec_node *gen_node = NULL;
110         struct ec_node_option *node = NULL;
111
112         if (child == NULL)
113                 return NULL;
114
115         gen_node = __ec_node_new(&ec_node_option_type, id);
116         if (gen_node == NULL) {
117                 ec_node_free(child);
118                 return NULL;
119         }
120         node = (struct ec_node_option *)gen_node;
121
122         node->child = child;
123
124         child->parent = gen_node;
125         TAILQ_INSERT_TAIL(&gen_node->children, child, next);
126
127         return &node->gen;
128 }
129
130 static int ec_node_option_testcase(void)
131 {
132         struct ec_node *node;
133         int ret = 0;
134
135         node = ec_node_option(NULL, ec_node_str(NULL, "foo"));
136         if (node == NULL) {
137                 ec_log(EC_LOG_ERR, "cannot create node\n");
138                 return -1;
139         }
140         ret |= EC_TEST_CHECK_PARSE(node, 1, "foo");
141         ret |= EC_TEST_CHECK_PARSE(node, 1, "foo", "bar");
142         ret |= EC_TEST_CHECK_PARSE(node, 0, "bar");
143         ret |= EC_TEST_CHECK_PARSE(node, 0);
144         ec_node_free(node);
145
146         /* test completion */
147         node = ec_node_option(NULL, ec_node_str(NULL, "foo"));
148         if (node == NULL) {
149                 ec_log(EC_LOG_ERR, "cannot create node\n");
150                 return -1;
151         }
152         ret |= EC_TEST_CHECK_COMPLETE(node,
153                 "", EC_NODE_ENDLIST,
154                 "foo", EC_NODE_ENDLIST,
155                 "foo");
156         ret |= EC_TEST_CHECK_COMPLETE(node,
157                 "f", EC_NODE_ENDLIST,
158                 "oo", EC_NODE_ENDLIST,
159                 "oo");
160         ret |= EC_TEST_CHECK_COMPLETE(node,
161                 "b", EC_NODE_ENDLIST,
162                 EC_NODE_ENDLIST,
163                 "");
164         ec_node_free(node);
165
166         return ret;
167 }
168
169 static struct ec_test ec_node_option_test = {
170         .name = "node_option",
171         .test = ec_node_option_testcase,
172 };
173
174 EC_TEST_REGISTER(ec_node_option_test);