pass state to completed api
[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_parsed.h>
39 #include <ecoli_completed.h>
40 #include <ecoli_node_option.h>
41 #include <ecoli_node_str.h>
42 #include <ecoli_test.h>
43
44 struct ec_node_option {
45         struct ec_node gen;
46         struct ec_node *child;
47 };
48
49 static int
50 ec_node_option_parse(const struct ec_node *gen_node,
51                 struct ec_parsed *state,
52                 const struct ec_strvec *strvec)
53 {
54         struct ec_node_option *node = (struct ec_node_option *)gen_node;
55         int ret;
56
57         ret = ec_node_parse_child(node->child, state, strvec);
58         if (ret == EC_PARSED_NOMATCH)
59                 return 0;
60         else if (ret < 0)
61                 return ret;
62
63         return ret;
64 }
65
66 static struct ec_completed *
67 ec_node_option_complete(const struct ec_node *gen_node,
68                         struct ec_parsed *state,
69                         const struct ec_strvec *strvec)
70 {
71         struct ec_node_option *node = (struct ec_node_option *)gen_node;
72
73         return ec_node_complete_child(node->child, state, strvec);
74 }
75
76 static void ec_node_option_free_priv(struct ec_node *gen_node)
77 {
78         struct ec_node_option *node = (struct ec_node_option *)gen_node;
79
80         ec_node_free(node->child);
81 }
82
83 static struct ec_node_type ec_node_option_type = {
84         .name = "option",
85         .parse = ec_node_option_parse,
86         .complete = ec_node_option_complete,
87         .size = sizeof(struct ec_node_option),
88         .free_priv = ec_node_option_free_priv,
89 };
90
91 EC_NODE_TYPE_REGISTER(ec_node_option_type);
92
93 struct ec_node *ec_node_option(const char *id, struct ec_node *child)
94 {
95         struct ec_node *gen_node = NULL;
96         struct ec_node_option *node = NULL;
97
98         if (child == NULL)
99                 return NULL;
100
101         gen_node = __ec_node(&ec_node_option_type, id);
102         if (gen_node == NULL) {
103                 ec_node_free(child);
104                 return NULL;
105         }
106         node = (struct ec_node_option *)gen_node;
107
108         node->child = child;
109
110         child->parent = gen_node;
111         TAILQ_INSERT_TAIL(&gen_node->children, child, next);
112
113         return &node->gen;
114 }
115
116 static int ec_node_option_testcase(void)
117 {
118         struct ec_node *node;
119         int ret = 0;
120
121         node = ec_node_option(NULL, ec_node_str(NULL, "foo"));
122         if (node == NULL) {
123                 ec_log(EC_LOG_ERR, "cannot create node\n");
124                 return -1;
125         }
126         ret |= EC_TEST_CHECK_PARSE(node, 1, "foo");
127         ret |= EC_TEST_CHECK_PARSE(node, 1, "foo", "bar");
128         ret |= EC_TEST_CHECK_PARSE(node, 0, "bar");
129         ret |= EC_TEST_CHECK_PARSE(node, 0);
130         ec_node_free(node);
131
132         /* test completion */
133         node = ec_node_option(NULL, ec_node_str(NULL, "foo"));
134         if (node == NULL) {
135                 ec_log(EC_LOG_ERR, "cannot create node\n");
136                 return -1;
137         }
138         ret |= EC_TEST_CHECK_COMPLETE(node,
139                 "", EC_NODE_ENDLIST,
140                 "foo", EC_NODE_ENDLIST,
141                 "foo");
142         ret |= EC_TEST_CHECK_COMPLETE(node,
143                 "f", EC_NODE_ENDLIST,
144                 "oo", EC_NODE_ENDLIST,
145                 "oo");
146         ret |= EC_TEST_CHECK_COMPLETE(node,
147                 "b", EC_NODE_ENDLIST,
148                 EC_NODE_ENDLIST,
149                 "");
150         ec_node_free(node);
151
152         return ret;
153 }
154
155 static struct ec_test ec_node_option_test = {
156         .name = "node_option",
157         .test = ec_node_option_testcase,
158 };
159
160 EC_TEST_REGISTER(ec_node_option_test);