change add to full str
[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 int
67 ec_node_option_complete(const struct ec_node *gen_node,
68                         struct ec_completed *completed,
69                         struct ec_parsed *parsed,
70                         const struct ec_strvec *strvec)
71 {
72         struct ec_node_option *node = (struct ec_node_option *)gen_node;
73
74         return ec_node_complete_child(node->child, completed, parsed, strvec);
75 }
76
77 static void ec_node_option_free_priv(struct ec_node *gen_node)
78 {
79         struct ec_node_option *node = (struct ec_node_option *)gen_node;
80
81         ec_node_free(node->child);
82 }
83
84 static struct ec_node_type ec_node_option_type = {
85         .name = "option",
86         .parse = ec_node_option_parse,
87         .complete = ec_node_option_complete,
88         .size = sizeof(struct ec_node_option),
89         .free_priv = ec_node_option_free_priv,
90 };
91
92 EC_NODE_TYPE_REGISTER(ec_node_option_type);
93
94 struct ec_node *ec_node_option(const char *id, struct ec_node *child)
95 {
96         struct ec_node *gen_node = NULL;
97         struct ec_node_option *node = NULL;
98
99         if (child == NULL)
100                 return NULL;
101
102         gen_node = __ec_node(&ec_node_option_type, id);
103         if (gen_node == NULL) {
104                 ec_node_free(child);
105                 return NULL;
106         }
107         node = (struct ec_node_option *)gen_node;
108
109         node->child = child;
110
111         child->parent = gen_node;
112         TAILQ_INSERT_TAIL(&gen_node->children, child, next);
113
114         return &node->gen;
115 }
116
117 /* LCOV_EXCL_START */
118 static int ec_node_option_testcase(void)
119 {
120         struct ec_node *node;
121         int ret = 0;
122
123         node = ec_node_option(NULL, ec_node_str(NULL, "foo"));
124         if (node == NULL) {
125                 ec_log(EC_LOG_ERR, "cannot create node\n");
126                 return -1;
127         }
128         ret |= EC_TEST_CHECK_PARSE(node, 1, "foo");
129         ret |= EC_TEST_CHECK_PARSE(node, 1, "foo", "bar");
130         ret |= EC_TEST_CHECK_PARSE(node, 0, "bar");
131         ret |= EC_TEST_CHECK_PARSE(node, 0);
132         ec_node_free(node);
133
134         /* test completion */
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_COMPLETE(node,
141                 "", EC_NODE_ENDLIST,
142                 "foo", EC_NODE_ENDLIST);
143         ret |= EC_TEST_CHECK_COMPLETE(node,
144                 "f", EC_NODE_ENDLIST,
145                 "foo", EC_NODE_ENDLIST);
146         ret |= EC_TEST_CHECK_COMPLETE(node,
147                 "b", EC_NODE_ENDLIST,
148                 EC_NODE_ENDLIST);
149         ec_node_free(node);
150
151         return ret;
152 }
153 /* LCOV_EXCL_STOP */
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);