test_expr.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362
  1. /* _
  2. * ___ __ _ __ _ _ _(_)
  3. * / __|/ _` |/ _` | | | | |
  4. * \__ \ (_| | (_| | |_| | |
  5. * |___/\__,_|\__, |\__,_|_|
  6. * |___/
  7. *
  8. * Cross-platform library which helps to develop web servers or frameworks.
  9. *
  10. * Copyright (C) 2016-2021 Silvio Clecio <[email protected]>
  11. *
  12. * Sagui library is free software; you can redistribute it and/or
  13. * modify it under the terms of the GNU Lesser General Public
  14. * License as published by the Free Software Foundation; either
  15. * version 2.1 of the License, or (at your option) any later version.
  16. *
  17. * Sagui library is distributed in the hope that it will be useful,
  18. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  19. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  20. * Lesser General Public License for more details.
  21. *
  22. * You should have received a copy of the GNU Lesser General Public
  23. * License along with Sagui library; if not, write to the Free Software
  24. * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
  25. */
  26. #include <string.h>
  27. #include "sg_macros.h"
  28. #include "sg_assert.h"
  29. #include "sg_utils.h"
  30. #include "sg_expr.h"
  31. #include <sagui.h>
  32. #define EXPR_1 "test_mul(2, 3)"
  33. #define EXPR_2 "2 + 3"
  34. #define EXPR_3 "test_missing(2, 3)"
  35. #define EXPR_4 "foo=2, bar=3"
  36. #define EXPR_5 "abc)def"
  37. static double test__mul(void *cls, struct sg_expr_argument *args,
  38. const char *identifier) {
  39. if (!args)
  40. return NAN;
  41. ASSERT(strcmp(identifier, "test_mul") == 0);
  42. ASSERT(strcmp(cls, EXPR_1) == 0);
  43. return sg_expr_arg(args, 0) * sg_expr_arg(args, 1);
  44. }
  45. struct sg_expr_extension extensions[] = {
  46. {.func = test__mul, .identifier = "test_mul", .cls = EXPR_1},
  47. {.func = NULL, .identifier = NULL, .cls = NULL},
  48. };
  49. static void test__expr_clear(struct sg_expr *expr) {
  50. sg_expr_compile(expr, EXPR_1, strlen(EXPR_1), extensions);
  51. ASSERT(expr->handle);
  52. ASSERT(expr->funcs);
  53. ASSERT(expr->handle);
  54. sg__expr_clear(expr);
  55. ASSERT(!expr->handle);
  56. ASSERT(!expr->funcs);
  57. ASSERT(!expr->handle);
  58. }
  59. static void test__expr_func(void) {
  60. vec_expr_t args = vec_init();
  61. ASSERT(isnan(sg__expr_func(NULL, &args, extensions)));
  62. vec_push(&args, expr_const(2));
  63. vec_push(&args, expr_const(3));
  64. ASSERT(sg__expr_func(NULL, &args, extensions) == 6);
  65. vec_free(&args);
  66. }
  67. static void test_expr_new(void) {
  68. struct sg_expr *expr = sg_expr_new();
  69. ASSERT(expr);
  70. ASSERT(expr->vars);
  71. sg_expr_free(expr);
  72. }
  73. static void test_expr_free(void) {
  74. sg_expr_free(NULL);
  75. }
  76. static void test_expr_compile(struct sg_expr *expr) {
  77. ASSERT(sg_expr_compile(NULL, EXPR_1, strlen(EXPR_1), extensions) == EINVAL);
  78. ASSERT(sg_expr_compile(expr, NULL, strlen(EXPR_1), extensions) == EINVAL);
  79. ASSERT(sg_expr_compile(expr, EXPR_1, strlen(EXPR_1), NULL) == EINVAL);
  80. ASSERT(sg_expr_compile(expr, EXPR_3, strlen(EXPR_3), extensions) == EINVAL);
  81. ASSERT(sg_expr_compile(expr, EXPR_1, 0, extensions) == 0);
  82. sg_expr_clear(expr);
  83. ASSERT(sg_expr_compile(expr, EXPR_2, strlen(EXPR_2), NULL) == 0);
  84. ASSERT(sg_expr_compile(expr, EXPR_1, strlen(EXPR_1), extensions) == EALREADY);
  85. sg_expr_clear(expr);
  86. ASSERT(sg_expr_compile(expr, EXPR_1, strlen(EXPR_1), extensions) == 0);
  87. sg_expr_clear(expr);
  88. ASSERT(sg_expr_compile(expr, "", 0, extensions) == 0);
  89. }
  90. static void test_expr_clear(struct sg_expr *expr) {
  91. ASSERT(sg_expr_clear(NULL) == EINVAL);
  92. sg_expr_clear(expr);
  93. ASSERT(sg_expr_compile(expr, EXPR_1, strlen(EXPR_1), extensions) == 0);
  94. ASSERT(sg_expr_compile(expr, EXPR_1, strlen(EXPR_1), extensions) == EALREADY);
  95. sg_expr_clear(expr);
  96. ASSERT(sg_expr_compile(expr, EXPR_1, strlen(EXPR_1), extensions) == 0);
  97. sg_expr_clear(expr);
  98. sg_expr_clear(expr);
  99. }
  100. static void test_expr_eval(struct sg_expr *expr) {
  101. errno = 0;
  102. ASSERT(isnan(sg_expr_eval(NULL)));
  103. ASSERT(errno == EINVAL);
  104. sg_expr_clear(expr);
  105. errno = 0;
  106. ASSERT(isnan(sg_expr_eval(expr)));
  107. ASSERT(errno == EINVAL);
  108. ASSERT(sg_expr_compile(expr, EXPR_1, strlen(EXPR_1), extensions) == 0);
  109. errno = 0;
  110. ASSERT(!isnan(sg_expr_eval(expr)));
  111. ASSERT(errno == 0);
  112. }
  113. static void test_expr_var(struct sg_expr *expr) {
  114. const char *foo = "foo";
  115. const char *bar = "bar";
  116. size_t foo_len = strlen(foo);
  117. size_t bar_len = strlen(bar);
  118. errno = 0;
  119. ASSERT(isnan(sg_expr_var(NULL, foo, foo_len)));
  120. ASSERT(errno == EINVAL);
  121. errno = 0;
  122. ASSERT(isnan(sg_expr_var(expr, NULL, foo_len)));
  123. ASSERT(errno == EINVAL);
  124. errno = 0;
  125. ASSERT(isnan(sg_expr_var(expr, foo, 0)));
  126. ASSERT(errno == EINVAL);
  127. sg_expr_clear(expr);
  128. errno = 0;
  129. ASSERT(sg_expr_var(expr, foo, foo_len) == 0);
  130. ASSERT(errno == 0);
  131. ASSERT(sg_expr_compile(expr, EXPR_4, strlen(EXPR_4), extensions) == 0);
  132. sg_expr_eval(expr);
  133. errno = 0;
  134. ASSERT(sg_expr_var(expr, foo, foo_len) == 2);
  135. ASSERT(errno == 0);
  136. errno = 0;
  137. ASSERT(sg_expr_var(expr, bar, bar_len) == 3);
  138. ASSERT(errno == 0);
  139. sg_expr_clear(expr);
  140. ASSERT(sg_expr_var(expr, foo, foo_len) == 0);
  141. ASSERT(sg_expr_var(expr, bar, bar_len) == 0);
  142. }
  143. static void test_expr_set_var(struct sg_expr *expr) {
  144. const char *foo = "foo";
  145. const char *bar = "bar";
  146. size_t foo_len = strlen(foo);
  147. size_t bar_len = strlen(bar);
  148. ASSERT(sg_expr_set_var(NULL, foo, foo_len, 0) == EINVAL);
  149. ASSERT(sg_expr_set_var(expr, NULL, foo_len, 0) == EINVAL);
  150. ASSERT(sg_expr_set_var(expr, foo, 0, 0) == EINVAL);
  151. sg_expr_clear(expr);
  152. ASSERT(sg_expr_var(expr, foo, foo_len) == 0);
  153. ASSERT(sg_expr_set_var(expr, foo, foo_len, 2) == 0);
  154. ASSERT(sg_expr_var(expr, foo, foo_len) == 2);
  155. ASSERT(sg_expr_var(expr, bar, bar_len) == 0);
  156. ASSERT(sg_expr_set_var(expr, bar, bar_len, 3) == 0);
  157. ASSERT(sg_expr_var(expr, bar, bar_len) == 3);
  158. ASSERT(sg_expr_set_var(expr, foo, foo_len, 0) == 0);
  159. ASSERT(sg_expr_set_var(expr, bar, bar_len, 0) == 0);
  160. ASSERT(sg_expr_var(expr, foo, foo_len) == 0);
  161. ASSERT(sg_expr_var(expr, bar, bar_len) == 0);
  162. }
  163. static void test_expr_arg(void) {
  164. vec_expr_t args = vec_init();
  165. struct sg_expr_argument func_args = {.handle = &args};
  166. errno = 0;
  167. ASSERT(isnan(sg_expr_arg(NULL, 0)));
  168. ASSERT(errno == EINVAL);
  169. errno = 0;
  170. ASSERT(isnan(sg_expr_arg(&func_args, -1)));
  171. ASSERT(errno == EINVAL);
  172. vec_push(&args, expr_const(2));
  173. vec_push(&args, expr_const(3));
  174. errno = 0;
  175. ASSERT(sg_expr_arg(&func_args, 0) == 2);
  176. ASSERT(sg_expr_arg(&func_args, 1) == 3);
  177. ASSERT(isnan(sg_expr_arg(&func_args, 3)));
  178. ASSERT(errno == 0);
  179. vec_free(&args);
  180. }
  181. static void test_expr_near(struct sg_expr *expr) {
  182. errno = 0;
  183. ASSERT(sg_expr_near(NULL) == 0);
  184. ASSERT(errno == EINVAL);
  185. sg_expr_clear(expr);
  186. errno = 0;
  187. ASSERT(sg_expr_near(expr) == 0);
  188. ASSERT(errno == 0);
  189. ASSERT(sg_expr_compile(expr, EXPR_5, strlen(EXPR_5), extensions) == EINVAL);
  190. errno = 0;
  191. ASSERT(sg_expr_near(expr) == 4);
  192. ASSERT(errno == 0);
  193. }
  194. static void test_expr_err_type(struct sg_expr *expr) {
  195. errno = 0;
  196. ASSERT(sg_expr_err(NULL) == SG_EXPR_ERR_UNKNOWN);
  197. ASSERT(errno == EINVAL);
  198. expr->err = EXPR_ERR_UNKNOWN;
  199. errno = 0;
  200. ASSERT(sg_expr_err(expr) == SG_EXPR_ERR_UNKNOWN);
  201. ASSERT(errno == 0);
  202. expr->err = EXPR_ERR_UNEXPECTED_NUMBER;
  203. errno = 0;
  204. ASSERT(sg_expr_err(expr) == SG_EXPR_ERR_UNEXPECTED_NUMBER);
  205. ASSERT(errno == 0);
  206. expr->err = EXPR_ERR_UNEXPECTED_WORD;
  207. errno = 0;
  208. ASSERT(sg_expr_err(expr) == SG_EXPR_ERR_UNEXPECTED_WORD);
  209. ASSERT(errno == 0);
  210. expr->err = EXPR_ERR_UNEXPECTED_PARENS;
  211. errno = 0;
  212. ASSERT(sg_expr_err(expr) == SG_EXPR_ERR_UNEXPECTED_PARENS);
  213. ASSERT(errno == 0);
  214. expr->err = EXPR_ERR_MISS_EXPECTED_OPERAND;
  215. errno = 0;
  216. ASSERT(sg_expr_err(expr) == SG_EXPR_ERR_MISSING_OPERAND);
  217. ASSERT(errno == 0);
  218. expr->err = EXPR_ERR_UNKNOWN_OPERATOR;
  219. errno = 0;
  220. ASSERT(sg_expr_err(expr) == SG_EXPR_ERR_UNKNOWN_OPERATOR);
  221. ASSERT(errno == 0);
  222. expr->err = EXPR_ERR_INVALID_FUNC_NAME;
  223. errno = 0;
  224. ASSERT(sg_expr_err(expr) == SG_EXPR_ERR_INVALID_FUNC_NAME);
  225. ASSERT(errno == 0);
  226. expr->err = EXPR_ERR_BAD_PARENS;
  227. errno = 0;
  228. ASSERT(sg_expr_err(expr) == SG_EXPR_ERR_BAD_PARENS);
  229. ASSERT(errno == 0);
  230. expr->err = EXPR_ERR_TOO_FEW_FUNC_ARGS;
  231. errno = 0;
  232. ASSERT(sg_expr_err(expr) == SG_EXPR_ERR_TOO_FEW_FUNC_ARGS);
  233. ASSERT(errno == 0);
  234. expr->err = EXPR_ERR_FIRST_ARG_IS_NOT_VAR;
  235. errno = 0;
  236. ASSERT(sg_expr_err(expr) == SG_EXPR_ERR_FIRST_ARG_IS_NOT_VAR);
  237. ASSERT(errno == 0);
  238. expr->err = EXPR_ERR_BAD_VARIABLE_NAME;
  239. errno = 0;
  240. ASSERT(sg_expr_err(expr) == SG_EXPR_ERR_BAD_VARIABLE_NAME);
  241. ASSERT(errno == 0);
  242. expr->err = EXPR_ERR_BAD_ASSIGNMENT;
  243. errno = 0;
  244. ASSERT(sg_expr_err(expr) == SG_EXPR_ERR_BAD_ASSIGNMENT);
  245. ASSERT(errno == 0);
  246. }
  247. static void test_expr_strerror(struct sg_expr *expr) {
  248. errno = 0;
  249. ASSERT(!sg_expr_strerror(NULL));
  250. ASSERT(errno == EINVAL);
  251. expr->err = EXPR_ERR_UNKNOWN;
  252. errno = 0;
  253. ASSERT(strcmp(sg_expr_strerror(expr), _("Unknown.\n")) == 0);
  254. ASSERT(errno == 0);
  255. expr->err = EXPR_ERR_UNEXPECTED_NUMBER;
  256. errno = 0;
  257. ASSERT(strcmp(sg_expr_strerror(expr), _("Unexpected number.\n")) == 0);
  258. ASSERT(errno == 0);
  259. expr->err = EXPR_ERR_UNEXPECTED_WORD;
  260. errno = 0;
  261. ASSERT(strcmp(sg_expr_strerror(expr), _("Unexpected word.\n")) == 0);
  262. ASSERT(errno == 0);
  263. expr->err = EXPR_ERR_UNEXPECTED_PARENS;
  264. errno = 0;
  265. ASSERT(strcmp(sg_expr_strerror(expr), _("Unexpected parenthesis.\n")) == 0);
  266. ASSERT(errno == 0);
  267. expr->err = EXPR_ERR_MISS_EXPECTED_OPERAND;
  268. errno = 0;
  269. ASSERT(strcmp(sg_expr_strerror(expr), _("Missing expected operand.\n")) == 0);
  270. ASSERT(errno == 0);
  271. expr->err = EXPR_ERR_UNKNOWN_OPERATOR;
  272. errno = 0;
  273. ASSERT(strcmp(sg_expr_strerror(expr), _("Unknown operator.\n")) == 0);
  274. ASSERT(errno == 0);
  275. expr->err = EXPR_ERR_INVALID_FUNC_NAME;
  276. errno = 0;
  277. ASSERT(strcmp(sg_expr_strerror(expr), _("Invalid function name.\n")) == 0);
  278. ASSERT(errno == 0);
  279. expr->err = EXPR_ERR_BAD_PARENS;
  280. errno = 0;
  281. ASSERT(strcmp(sg_expr_strerror(expr), _("Bad parenthesis.\n")) == 0);
  282. ASSERT(errno == 0);
  283. expr->err = EXPR_ERR_TOO_FEW_FUNC_ARGS;
  284. errno = 0;
  285. ASSERT(strcmp(sg_expr_strerror(expr), _("Too few function arguments.\n")) ==
  286. 0);
  287. expr->err = EXPR_ERR_FIRST_ARG_IS_NOT_VAR;
  288. errno = 0;
  289. ASSERT(strcmp(sg_expr_strerror(expr),
  290. _("First argument is not a variable.\n")) == 0);
  291. ASSERT(errno == 0);
  292. expr->err = EXPR_ERR_BAD_VARIABLE_NAME;
  293. errno = 0;
  294. ASSERT(strcmp(sg_expr_strerror(expr), _("Bad variable name.\n")) == 0);
  295. ASSERT(errno == 0);
  296. expr->err = EXPR_ERR_BAD_ASSIGNMENT;
  297. errno = 0;
  298. ASSERT(strcmp(sg_expr_strerror(expr), _("Bad assignment.\n")) == 0);
  299. ASSERT(errno == 0);
  300. }
  301. static void test_expr_calc(void) {
  302. errno = 0;
  303. ASSERT(isnan(sg_expr_calc(NULL, strlen(EXPR_1))));
  304. ASSERT(errno == EINVAL);
  305. errno = 0;
  306. ASSERT(isnan(sg_expr_calc(EXPR_1, 0)));
  307. ASSERT(errno == EINVAL);
  308. errno = 0;
  309. ASSERT(isnan(sg_expr_calc(EXPR_3, 0)));
  310. ASSERT(errno == EINVAL);
  311. errno = 0;
  312. ASSERT(sg_expr_calc(EXPR_2, strlen(EXPR_2)) == 5);
  313. ASSERT(errno == 0);
  314. }
  315. int main(void) {
  316. struct sg_expr *expr = sg_expr_new();
  317. test__expr_clear(expr);
  318. test__expr_func();
  319. test_expr_new();
  320. test_expr_free();
  321. test_expr_compile(expr);
  322. test_expr_clear(expr);
  323. test_expr_eval(expr);
  324. test_expr_var(expr);
  325. test_expr_set_var(expr);
  326. test_expr_arg();
  327. test_expr_near(expr);
  328. test_expr_err_type(expr);
  329. test_expr_strerror(expr);
  330. test_expr_calc();
  331. sg_expr_free(expr);
  332. return EXIT_SUCCESS;
  333. }