1 /* Copyright (c) 2013, Vsevolod Stakhov 2 * All rights reserved. 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 * * Redistributions of source code must retain the above copyright 7 * notice, this list of conditions and the following disclaimer. 8 * * Redistributions in binary form must reproduce the above copyright 9 * notice, this list of conditions and the following disclaimer in the 10 * documentation and/or other materials provided with the distribution. 11 * 12 * THIS SOFTWARE IS PROVIDED ''AS IS'' AND ANY 13 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED 14 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE 15 * DISCLAIMED. IN NO EVENT SHALL AUTHOR BE LIABLE FOR ANY 16 * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES 17 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; 18 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND 19 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 20 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS 21 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 22 */ 23 24 #ifndef UCL_H_ 25 #define UCL_H_ 26 27 #include <string.h> 28 #include <stddef.h> 29 #include <stdlib.h> 30 #include <stdint.h> 31 #include <stdbool.h> 32 #include <stdarg.h> 33 #include <stdio.h> 34 35 #ifdef _WIN32 36 # define UCL_EXTERN __declspec(dllexport) 37 #else 38 # define UCL_EXTERN 39 #endif 40 41 /** 42 * @mainpage 43 * This is a reference manual for UCL API. You may find the description of UCL format by following this 44 * [github repository](https://github.com/vstakhov/libucl). 45 * 46 * This manual has several main sections: 47 * - @ref structures 48 * - @ref utils 49 * - @ref parser 50 * - @ref emitter 51 */ 52 53 /** 54 * @file ucl.h 55 * @brief UCL parsing and emitting functions 56 * 57 * UCL is universal configuration language, which is a form of 58 * JSON with less strict rules that make it more comfortable for 59 * using as a configuration language 60 */ 61 #ifdef __cplusplus 62 extern "C" { 63 #endif 64 /* 65 * Memory allocation utilities 66 * UCL_ALLOC(size) - allocate memory for UCL 67 * UCL_FREE(size, ptr) - free memory of specified size at ptr 68 * Default: malloc and free 69 */ 70 #ifndef UCL_ALLOC 71 #define UCL_ALLOC(size) malloc(size) 72 #endif 73 #ifndef UCL_FREE 74 #define UCL_FREE(size, ptr) free(ptr) 75 #endif 76 77 #if __GNUC__ > 3 || (__GNUC__ == 3 && __GNUC_MINOR__ >= 4) 78 #define UCL_WARN_UNUSED_RESULT \ 79 __attribute__((warn_unused_result)) 80 #else 81 #define UCL_WARN_UNUSED_RESULT 82 #endif 83 84 /** 85 * @defgroup structures Structures and types 86 * UCL defines several enumeration types used for error reporting or specifying flags and attributes. 87 * 88 * @{ 89 */ 90 91 /** 92 * The common error codes returned by ucl parser 93 */ 94 typedef enum ucl_error { 95 UCL_EOK = 0, /**< No error */ 96 UCL_ESYNTAX, /**< Syntax error occurred during parsing */ 97 UCL_EIO, /**< IO error occurred during parsing */ 98 UCL_ESTATE, /**< Invalid state machine state */ 99 UCL_ENESTED, /**< Input has too many recursion levels */ 100 UCL_EMACRO, /**< Error processing a macro */ 101 UCL_EINTERNAL, /**< Internal unclassified error */ 102 UCL_ESSL /**< SSL error */ 103 } ucl_error_t; 104 105 /** 106 * #ucl_object_t may have one of specified types, some types are compatible with each other and some are not. 107 * For example, you can always convert #UCL_TIME to #UCL_FLOAT. Also you can convert #UCL_FLOAT to #UCL_INTEGER 108 * by loosing floating point. Every object may be converted to a string by #ucl_object_tostring_forced() function. 109 * 110 */ 111 typedef enum ucl_type { 112 UCL_OBJECT = 0, /**< UCL object - key/value pairs */ 113 UCL_ARRAY, /**< UCL array */ 114 UCL_INT, /**< Integer number */ 115 UCL_FLOAT, /**< Floating point number */ 116 UCL_STRING, /**< Null terminated string */ 117 UCL_BOOLEAN, /**< Boolean value */ 118 UCL_TIME, /**< Time value (floating point number of seconds) */ 119 UCL_USERDATA, /**< Opaque userdata pointer (may be used in macros) */ 120 UCL_NULL /**< Null value */ 121 } ucl_type_t; 122 123 /** 124 * You can use one of these types to serialise #ucl_object_t by using ucl_object_emit(). 125 */ 126 typedef enum ucl_emitter { 127 UCL_EMIT_JSON = 0, /**< Emit fine formatted JSON */ 128 UCL_EMIT_JSON_COMPACT, /**< Emit compacted JSON */ 129 UCL_EMIT_CONFIG, /**< Emit human readable config format */ 130 UCL_EMIT_YAML /**< Emit embedded YAML format */ 131 } ucl_emitter_t; 132 133 /** 134 * These flags defines parser behaviour. If you specify #UCL_PARSER_ZEROCOPY you must ensure 135 * that the input memory is not freed if an object is in use. Moreover, if you want to use 136 * zero-terminated keys and string values then you should not use zero-copy mode, as in this case 137 * UCL still has to perform copying implicitly. 138 */ 139 typedef enum ucl_parser_flags { 140 UCL_PARSER_KEY_LOWERCASE = 0x1, /**< Convert all keys to lower case */ 141 UCL_PARSER_ZEROCOPY = 0x2 /**< Parse input in zero-copy mode if possible */ 142 } ucl_parser_flags_t; 143 144 /** 145 * String conversion flags, that are used in #ucl_object_fromstring_common function. 146 */ 147 typedef enum ucl_string_flags { 148 UCL_STRING_ESCAPE = 0x1, /**< Perform JSON escape */ 149 UCL_STRING_TRIM = 0x2, /**< Trim leading and trailing whitespaces */ 150 UCL_STRING_PARSE_BOOLEAN = 0x4, /**< Parse passed string and detect boolean */ 151 UCL_STRING_PARSE_INT = 0x8, /**< Parse passed string and detect integer number */ 152 UCL_STRING_PARSE_DOUBLE = 0x10, /**< Parse passed string and detect integer or float number */ 153 UCL_STRING_PARSE_NUMBER = UCL_STRING_PARSE_INT|UCL_STRING_PARSE_DOUBLE , /**< 154 Parse passed string and detect number */ 155 UCL_STRING_PARSE = UCL_STRING_PARSE_BOOLEAN|UCL_STRING_PARSE_NUMBER, /**< 156 Parse passed string (and detect booleans and numbers) */ 157 UCL_STRING_PARSE_BYTES = 0x20 /**< Treat numbers as bytes */ 158 } ucl_string_flags_t; 159 160 /** 161 * Basic flags for an object 162 */ 163 typedef enum ucl_object_flags { 164 UCL_OBJECT_ALLOCATED_KEY = 1, /**< An object has key allocated internally */ 165 UCL_OBJECT_ALLOCATED_VALUE = 2, /**< An object has a string value allocated internally */ 166 UCL_OBJECT_NEED_KEY_ESCAPE = 4 /**< The key of an object need to be escaped on output */ 167 } ucl_object_flags_t; 168 169 /** 170 * UCL object structure. Please mention that the most of fields should not be touched by 171 * UCL users. In future, this structure may be converted to private one. 172 */ 173 typedef struct ucl_object_s { 174 /** 175 * Variant value type 176 */ 177 union { 178 int64_t iv; /**< Int value of an object */ 179 const char *sv; /**< String value of an object */ 180 double dv; /**< Double value of an object */ 181 struct ucl_object_s *av; /**< Array */ 182 void *ov; /**< Object */ 183 void* ud; /**< Opaque user data */ 184 } value; 185 const char *key; /**< Key of an object */ 186 struct ucl_object_s *next; /**< Array handle */ 187 struct ucl_object_s *prev; /**< Array handle */ 188 unsigned char* trash_stack[2]; /**< Pointer to allocated chunks */ 189 unsigned keylen; /**< Lenght of a key */ 190 unsigned len; /**< Size of an object */ 191 enum ucl_type type; /**< Real type */ 192 uint16_t ref; /**< Reference count */ 193 uint16_t flags; /**< Object flags */ 194 } ucl_object_t; 195 196 /** @} */ 197 198 /** 199 * @defgroup utils Utility functions 200 * A number of utility functions simplify handling of UCL objects 201 * 202 * @{ 203 */ 204 /** 205 * Copy and return a key of an object, returned key is zero-terminated 206 * @param obj CL object 207 * @return zero terminated key 208 */ 209 UCL_EXTERN char* ucl_copy_key_trash (ucl_object_t *obj); 210 211 /** 212 * Copy and return a string value of an object, returned key is zero-terminated 213 * @param obj CL object 214 * @return zero terminated string representation of object value 215 */ 216 UCL_EXTERN char* ucl_copy_value_trash (ucl_object_t *obj); 217 218 /** 219 * Creates a new object 220 * @return new object 221 */ 222 static inline ucl_object_t* ucl_object_new (void) UCL_WARN_UNUSED_RESULT; 223 static inline ucl_object_t * 224 ucl_object_new (void) 225 { 226 ucl_object_t *new; 227 new = malloc (sizeof (ucl_object_t)); 228 if (new != NULL) { 229 memset (new, 0, sizeof (ucl_object_t)); 230 new->ref = 1; 231 new->type = UCL_NULL; 232 } 233 return new; 234 } 235 236 /** 237 * Create new object with type specified 238 * @param type type of a new object 239 * @return new object 240 */ 241 static inline ucl_object_t* ucl_object_typed_new (unsigned int type) UCL_WARN_UNUSED_RESULT; 242 static inline ucl_object_t * 243 ucl_object_typed_new (unsigned int type) 244 { 245 ucl_object_t *new; 246 new = malloc (sizeof (ucl_object_t)); 247 if (new != NULL) { 248 memset (new, 0, sizeof (ucl_object_t)); 249 new->ref = 1; 250 new->type = (type <= UCL_NULL ? type : UCL_NULL); 251 } 252 return new; 253 } 254 255 /** 256 * Convert any string to an ucl object making the specified transformations 257 * @param str fixed size or NULL terminated string 258 * @param len length (if len is zero, than str is treated as NULL terminated) 259 * @param flags conversion flags 260 * @return new object 261 */ 262 UCL_EXTERN ucl_object_t * ucl_object_fromstring_common (const char *str, size_t len, 263 enum ucl_string_flags flags) UCL_WARN_UNUSED_RESULT; 264 265 /** 266 * Create a UCL object from the specified string 267 * @param str NULL terminated string, will be json escaped 268 * @return new object 269 */ 270 static inline ucl_object_t * 271 ucl_object_fromstring (const char *str) 272 { 273 return ucl_object_fromstring_common (str, 0, UCL_STRING_ESCAPE); 274 } 275 276 /** 277 * Create a UCL object from the specified string 278 * @param str fixed size string, will be json escaped 279 * @param len length of a string 280 * @return new object 281 */ 282 static inline ucl_object_t * 283 ucl_object_fromlstring (const char *str, size_t len) 284 { 285 return ucl_object_fromstring_common (str, len, UCL_STRING_ESCAPE); 286 } 287 288 /** 289 * Create an object from an integer number 290 * @param iv number 291 * @return new object 292 */ 293 static inline ucl_object_t * 294 ucl_object_fromint (int64_t iv) 295 { 296 ucl_object_t *obj; 297 298 obj = ucl_object_new (); 299 if (obj != NULL) { 300 obj->type = UCL_INT; 301 obj->value.iv = iv; 302 } 303 304 return obj; 305 } 306 307 /** 308 * Create an object from a float number 309 * @param dv number 310 * @return new object 311 */ 312 static inline ucl_object_t * 313 ucl_object_fromdouble (double dv) 314 { 315 ucl_object_t *obj; 316 317 obj = ucl_object_new (); 318 if (obj != NULL) { 319 obj->type = UCL_FLOAT; 320 obj->value.dv = dv; 321 } 322 323 return obj; 324 } 325 326 /** 327 * Create an object from a boolean 328 * @param bv bool value 329 * @return new object 330 */ 331 static inline ucl_object_t * 332 ucl_object_frombool (bool bv) 333 { 334 ucl_object_t *obj; 335 336 obj = ucl_object_new (); 337 if (obj != NULL) { 338 obj->type = UCL_BOOLEAN; 339 obj->value.iv = bv; 340 } 341 342 return obj; 343 } 344 345 /** 346 * Insert a object 'elt' to the hash 'top' and associate it with key 'key' 347 * @param top destination object (will be created automatically if top is NULL) 348 * @param elt element to insert (must NOT be NULL) 349 * @param key key to associate with this object (either const or preallocated) 350 * @param keylen length of the key (or 0 for NULL terminated keys) 351 * @param copy_key make an internal copy of key 352 * @return new value of top object 353 */ 354 UCL_EXTERN ucl_object_t* ucl_object_insert_key (ucl_object_t *top, ucl_object_t *elt, 355 const char *key, size_t keylen, bool copy_key) UCL_WARN_UNUSED_RESULT; 356 357 /** 358 * Replace a object 'elt' to the hash 'top' and associate it with key 'key', old object will be unrefed, 359 * if no object has been found this function works like ucl_object_insert_key() 360 * @param top destination object (will be created automatically if top is NULL) 361 * @param elt element to insert (must NOT be NULL) 362 * @param key key to associate with this object (either const or preallocated) 363 * @param keylen length of the key (or 0 for NULL terminated keys) 364 * @param copy_key make an internal copy of key 365 * @return new value of top object 366 */ 367 UCL_EXTERN ucl_object_t* ucl_object_replace_key (ucl_object_t *top, ucl_object_t *elt, 368 const char *key, size_t keylen, bool copy_key) UCL_WARN_UNUSED_RESULT; 369 370 /** 371 * Delete a object associated with key 'key', old object will be unrefered, 372 * @param top object 373 * @param key key associated to the object to remove 374 * @param keylen length of the key (or 0 for NULL terminated keys) 375 */ 376 UCL_EXTERN bool ucl_object_delete_keyl (ucl_object_t *top, const char *key, size_t keylen); 377 378 /** 379 * Delete a object associated with key 'key', old object will be unrefered, 380 * @param top object 381 * @param key key associated to the object to remove 382 */ 383 UCL_EXTERN bool ucl_object_delete_key (ucl_object_t *top, const char *key); 384 385 /** 386 * Insert a object 'elt' to the hash 'top' and associate it with key 'key', if the specified key exist, 387 * try to merge its content 388 * @param top destination object (will be created automatically if top is NULL) 389 * @param elt element to insert (must NOT be NULL) 390 * @param key key to associate with this object (either const or preallocated) 391 * @param keylen length of the key (or 0 for NULL terminated keys) 392 * @param copy_key make an internal copy of key 393 * @return new value of top object 394 */ 395 UCL_EXTERN ucl_object_t* ucl_object_insert_key_merged (ucl_object_t *top, ucl_object_t *elt, 396 const char *key, size_t keylen, bool copy_key) UCL_WARN_UNUSED_RESULT; 397 398 /** 399 * Append an element to the front of array object 400 * @param top destination object (will be created automatically if top is NULL) 401 * @param elt element to append (must NOT be NULL) 402 * @return new value of top object 403 */ 404 static inline ucl_object_t * ucl_array_append (ucl_object_t *top, 405 ucl_object_t *elt) UCL_WARN_UNUSED_RESULT; 406 static inline ucl_object_t * 407 ucl_array_append (ucl_object_t *top, ucl_object_t *elt) 408 { 409 ucl_object_t *head; 410 411 if (elt == NULL) { 412 return NULL; 413 } 414 415 if (top == NULL) { 416 top = ucl_object_typed_new (UCL_ARRAY); 417 top->value.av = elt; 418 elt->next = NULL; 419 elt->prev = elt; 420 top->len = 1; 421 } 422 else { 423 head = top->value.av; 424 if (head == NULL) { 425 top->value.av = elt; 426 elt->prev = elt; 427 } 428 else { 429 elt->prev = head->prev; 430 head->prev->next = elt; 431 head->prev = elt; 432 } 433 elt->next = NULL; 434 top->len ++; 435 } 436 437 return top; 438 } 439 440 /** 441 * Append an element to the start of array object 442 * @param top destination object (will be created automatically if top is NULL) 443 * @param elt element to append (must NOT be NULL) 444 * @return new value of top object 445 */ 446 static inline ucl_object_t * ucl_array_prepend (ucl_object_t *top, 447 ucl_object_t *elt) UCL_WARN_UNUSED_RESULT; 448 static inline ucl_object_t * 449 ucl_array_prepend (ucl_object_t *top, ucl_object_t *elt) 450 { 451 ucl_object_t *head; 452 453 if (elt == NULL) { 454 return NULL; 455 } 456 457 if (top == NULL) { 458 top = ucl_object_typed_new (UCL_ARRAY); 459 top->value.av = elt; 460 elt->next = NULL; 461 elt->prev = elt; 462 top->len = 1; 463 } 464 else { 465 head = top->value.av; 466 if (head == NULL) { 467 top->value.av = elt; 468 elt->prev = elt; 469 } 470 else { 471 elt->prev = head->prev; 472 head->prev = elt; 473 } 474 elt->next = head; 475 top->value.av = elt; 476 top->len ++; 477 } 478 479 return top; 480 } 481 482 /** 483 * Removes an element `elt` from the array `top`. Caller must unref the returned object when it is not 484 * needed. 485 * @param top array ucl object 486 * @param elt element to remove 487 * @return removed element or NULL if `top` is NULL or not an array 488 */ 489 static inline ucl_object_t * 490 ucl_array_delete (ucl_object_t *top, ucl_object_t *elt) 491 { 492 ucl_object_t *head; 493 494 if (top == NULL || top->type != UCL_ARRAY || top->value.av == NULL) { 495 return NULL; 496 } 497 head = top->value.av; 498 499 if (elt->prev == elt) { 500 top->value.av = NULL; 501 } 502 else if (elt == head) { 503 elt->next->prev = elt->prev; 504 top->value.av = elt->next; 505 } 506 else { 507 elt->prev->next = elt->next; 508 if (elt->next) { 509 elt->next->prev = elt->prev; 510 } 511 else { 512 head->prev = elt->prev; 513 } 514 } 515 elt->next = NULL; 516 elt->prev = elt; 517 top->len --; 518 519 return elt; 520 } 521 522 /** 523 * Returns the first element of the array `top` 524 * @param top array ucl object 525 * @return element or NULL if `top` is NULL or not an array 526 */ 527 static inline ucl_object_t * 528 ucl_array_head (ucl_object_t *top) 529 { 530 if (top == NULL || top->type != UCL_ARRAY || top->value.av == NULL) { 531 return NULL; 532 } 533 return top->value.av; 534 } 535 536 /** 537 * Returns the last element of the array `top` 538 * @param top array ucl object 539 * @return element or NULL if `top` is NULL or not an array 540 */ 541 static inline ucl_object_t * 542 ucl_array_tail (ucl_object_t *top) 543 { 544 if (top == NULL || top->type != UCL_ARRAY || top->value.av == NULL) { 545 return NULL; 546 } 547 return top->value.av->prev; 548 } 549 550 /** 551 * Removes the last element from the array `top`. Caller must unref the returned object when it is not 552 * needed. 553 * @param top array ucl object 554 * @return removed element or NULL if `top` is NULL or not an array 555 */ 556 static inline ucl_object_t * 557 ucl_array_pop_last (ucl_object_t *top) 558 { 559 return ucl_array_delete (top, ucl_array_tail (top)); 560 } 561 562 /** 563 * Removes the first element from the array `top`. Caller must unref the returned object when it is not 564 * needed. 565 * @param top array ucl object 566 * @return removed element or NULL if `top` is NULL or not an array 567 */ 568 static inline ucl_object_t * 569 ucl_array_pop_first (ucl_object_t *top) 570 { 571 return ucl_array_delete (top, ucl_array_head (top)); 572 } 573 574 /** 575 * Append a element to another element forming an implicit array 576 * @param head head to append (may be NULL) 577 * @param elt new element 578 * @return new head if applicable 579 */ 580 static inline ucl_object_t * ucl_elt_append (ucl_object_t *head, 581 ucl_object_t *elt) UCL_WARN_UNUSED_RESULT; 582 static inline ucl_object_t * 583 ucl_elt_append (ucl_object_t *head, ucl_object_t *elt) 584 { 585 586 if (head == NULL) { 587 elt->next = NULL; 588 elt->prev = elt; 589 head = elt; 590 } 591 else { 592 elt->prev = head->prev; 593 head->prev->next = elt; 594 head->prev = elt; 595 elt->next = NULL; 596 } 597 598 return head; 599 } 600 601 /** 602 * Converts an object to double value 603 * @param obj CL object 604 * @param target target double variable 605 * @return true if conversion was successful 606 */ 607 static inline bool 608 ucl_object_todouble_safe (ucl_object_t *obj, double *target) 609 { 610 if (obj == NULL) { 611 return false; 612 } 613 switch (obj->type) { 614 case UCL_INT: 615 *target = obj->value.iv; /* Probaly could cause overflow */ 616 break; 617 case UCL_FLOAT: 618 case UCL_TIME: 619 *target = obj->value.dv; 620 break; 621 default: 622 return false; 623 } 624 625 return true; 626 } 627 628 /** 629 * Unsafe version of \ref ucl_obj_todouble_safe 630 * @param obj CL object 631 * @return double value 632 */ 633 static inline double 634 ucl_object_todouble (ucl_object_t *obj) 635 { 636 double result = 0.; 637 638 ucl_object_todouble_safe (obj, &result); 639 return result; 640 } 641 642 /** 643 * Converts an object to integer value 644 * @param obj CL object 645 * @param target target integer variable 646 * @return true if conversion was successful 647 */ 648 static inline bool 649 ucl_object_toint_safe (ucl_object_t *obj, int64_t *target) 650 { 651 if (obj == NULL) { 652 return false; 653 } 654 switch (obj->type) { 655 case UCL_INT: 656 *target = obj->value.iv; 657 break; 658 case UCL_FLOAT: 659 case UCL_TIME: 660 *target = obj->value.dv; /* Loosing of decimal points */ 661 break; 662 default: 663 return false; 664 } 665 666 return true; 667 } 668 669 /** 670 * Unsafe version of \ref ucl_obj_toint_safe 671 * @param obj CL object 672 * @return int value 673 */ 674 static inline int64_t 675 ucl_object_toint (ucl_object_t *obj) 676 { 677 int64_t result = 0; 678 679 ucl_object_toint_safe (obj, &result); 680 return result; 681 } 682 683 /** 684 * Converts an object to boolean value 685 * @param obj CL object 686 * @param target target boolean variable 687 * @return true if conversion was successful 688 */ 689 static inline bool 690 ucl_object_toboolean_safe (ucl_object_t *obj, bool *target) 691 { 692 if (obj == NULL) { 693 return false; 694 } 695 switch (obj->type) { 696 case UCL_BOOLEAN: 697 *target = (obj->value.iv == true); 698 break; 699 default: 700 return false; 701 } 702 703 return true; 704 } 705 706 /** 707 * Unsafe version of \ref ucl_obj_toboolean_safe 708 * @param obj CL object 709 * @return boolean value 710 */ 711 static inline bool 712 ucl_object_toboolean (ucl_object_t *obj) 713 { 714 bool result = false; 715 716 ucl_object_toboolean_safe (obj, &result); 717 return result; 718 } 719 720 /** 721 * Converts an object to string value 722 * @param obj CL object 723 * @param target target string variable, no need to free value 724 * @return true if conversion was successful 725 */ 726 static inline bool 727 ucl_object_tostring_safe (ucl_object_t *obj, const char **target) 728 { 729 if (obj == NULL) { 730 return false; 731 } 732 733 switch (obj->type) { 734 case UCL_STRING: 735 *target = ucl_copy_value_trash (obj); 736 break; 737 default: 738 return false; 739 } 740 741 return true; 742 } 743 744 /** 745 * Unsafe version of \ref ucl_obj_tostring_safe 746 * @param obj CL object 747 * @return string value 748 */ 749 static inline const char * 750 ucl_object_tostring (ucl_object_t *obj) 751 { 752 const char *result = NULL; 753 754 ucl_object_tostring_safe (obj, &result); 755 return result; 756 } 757 758 /** 759 * Convert any object to a string in JSON notation if needed 760 * @param obj CL object 761 * @return string value 762 */ 763 static inline const char * 764 ucl_object_tostring_forced (ucl_object_t *obj) 765 { 766 return ucl_copy_value_trash (obj); 767 } 768 769 /** 770 * Return string as char * and len, string may be not zero terminated, more efficient that \ref ucl_obj_tostring as it 771 * allows zero-copy (if #UCL_PARSER_ZEROCOPY has been used during parsing) 772 * @param obj CL object 773 * @param target target string variable, no need to free value 774 * @param tlen target length 775 * @return true if conversion was successful 776 */ 777 static inline bool 778 ucl_object_tolstring_safe (ucl_object_t *obj, const char **target, size_t *tlen) 779 { 780 if (obj == NULL) { 781 return false; 782 } 783 switch (obj->type) { 784 case UCL_STRING: 785 *target = obj->value.sv; 786 *tlen = obj->len; 787 break; 788 default: 789 return false; 790 } 791 792 return true; 793 } 794 795 /** 796 * Unsafe version of \ref ucl_obj_tolstring_safe 797 * @param obj CL object 798 * @return string value 799 */ 800 static inline const char * 801 ucl_object_tolstring (ucl_object_t *obj, size_t *tlen) 802 { 803 const char *result = NULL; 804 805 ucl_object_tolstring_safe (obj, &result, tlen); 806 return result; 807 } 808 809 /** 810 * Return object identified by a key in the specified object 811 * @param obj object to get a key from (must be of type UCL_OBJECT) 812 * @param key key to search 813 * @return object matched the specified key or NULL if key is not found 814 */ 815 UCL_EXTERN ucl_object_t * ucl_object_find_key (ucl_object_t *obj, const char *key); 816 817 /** 818 * Return object identified by a fixed size key in the specified object 819 * @param obj object to get a key from (must be of type UCL_OBJECT) 820 * @param key key to search 821 * @param klen length of a key 822 * @return object matched the specified key or NULL if key is not found 823 */ 824 UCL_EXTERN ucl_object_t *ucl_object_find_keyl (ucl_object_t *obj, const char *key, size_t klen); 825 826 /** 827 * Returns a key of an object as a NULL terminated string 828 * @param obj CL object 829 * @return key or NULL if there is no key 830 */ 831 static inline const char * 832 ucl_object_key (ucl_object_t *obj) 833 { 834 return ucl_copy_key_trash (obj); 835 } 836 837 /** 838 * Returns a key of an object as a fixed size string (may be more efficient) 839 * @param obj CL object 840 * @param len target key length 841 * @return key pointer 842 */ 843 static inline const char * 844 ucl_object_keyl (ucl_object_t *obj, size_t *len) 845 { 846 *len = obj->keylen; 847 return obj->key; 848 } 849 850 /** 851 * Free ucl object 852 * @param obj ucl object to free 853 */ 854 UCL_EXTERN void ucl_object_free (ucl_object_t *obj); 855 856 /** 857 * Increase reference count for an object 858 * @param obj object to ref 859 */ 860 static inline ucl_object_t * 861 ucl_object_ref (ucl_object_t *obj) { 862 obj->ref ++; 863 return obj; 864 } 865 866 /** 867 * Decrease reference count for an object 868 * @param obj object to unref 869 */ 870 static inline void 871 ucl_object_unref (ucl_object_t *obj) { 872 if (obj != NULL && --obj->ref <= 0) { 873 ucl_object_free (obj); 874 } 875 } 876 /** 877 * Opaque iterator object 878 */ 879 typedef void* ucl_object_iter_t; 880 881 /** 882 * Get next key from an object 883 * @param obj object to iterate 884 * @param iter opaque iterator, must be set to NULL on the first call: 885 * ucl_object_iter_t it = NULL; 886 * while ((cur = ucl_iterate_object (obj, &it)) != NULL) ... 887 * @return the next object or NULL 888 */ 889 UCL_EXTERN ucl_object_t* ucl_iterate_object (ucl_object_t *obj, ucl_object_iter_t *iter, bool expand_values); 890 /** @} */ 891 892 893 /** 894 * @defgroup parser Parsing functions 895 * These functions are used to parse UCL objects 896 * 897 * @{ 898 */ 899 900 /** 901 * Macro handler for a parser 902 * @param data the content of macro 903 * @param len the length of content 904 * @param ud opaque user data 905 * @param err error pointer 906 * @return true if macro has been parsed 907 */ 908 typedef bool (*ucl_macro_handler) (const unsigned char *data, size_t len, void* ud); 909 910 /* Opaque parser */ 911 struct ucl_parser; 912 913 /** 914 * Creates new parser object 915 * @param pool pool to allocate memory from 916 * @return new parser object 917 */ 918 UCL_EXTERN struct ucl_parser* ucl_parser_new (int flags); 919 920 /** 921 * Register new handler for a macro 922 * @param parser parser object 923 * @param macro macro name (without leading dot) 924 * @param handler handler (it is called immediately after macro is parsed) 925 * @param ud opaque user data for a handler 926 */ 927 UCL_EXTERN void ucl_parser_register_macro (struct ucl_parser *parser, const char *macro, 928 ucl_macro_handler handler, void* ud); 929 930 /** 931 * Register new parser variable 932 * @param parser parser object 933 * @param var variable name 934 * @param value variable value 935 */ 936 UCL_EXTERN void ucl_parser_register_variable (struct ucl_parser *parser, const char *var, 937 const char *value); 938 939 /** 940 * Load new chunk to a parser 941 * @param parser parser structure 942 * @param data the pointer to the beginning of a chunk 943 * @param len the length of a chunk 944 * @param err if *err is NULL it is set to parser error 945 * @return true if chunk has been added and false in case of error 946 */ 947 UCL_EXTERN bool ucl_parser_add_chunk (struct ucl_parser *parser, const unsigned char *data, size_t len); 948 949 /** 950 * Load and add data from a file 951 * @param parser parser structure 952 * @param filename the name of file 953 * @param err if *err is NULL it is set to parser error 954 * @return true if chunk has been added and false in case of error 955 */ 956 UCL_EXTERN bool ucl_parser_add_file (struct ucl_parser *parser, const char *filename); 957 958 /** 959 * Get a top object for a parser 960 * @param parser parser structure 961 * @param err if *err is NULL it is set to parser error 962 * @return top parser object or NULL 963 */ 964 UCL_EXTERN ucl_object_t* ucl_parser_get_object (struct ucl_parser *parser); 965 966 /** 967 * Get the error string if failing 968 * @param parser parser object 969 */ 970 UCL_EXTERN const char *ucl_parser_get_error(struct ucl_parser *parser); 971 /** 972 * Free ucl parser object 973 * @param parser parser object 974 */ 975 UCL_EXTERN void ucl_parser_free (struct ucl_parser *parser); 976 977 /** 978 * Add new public key to parser for signatures check 979 * @param parser parser object 980 * @param key PEM representation of a key 981 * @param len length of the key 982 * @param err if *err is NULL it is set to parser error 983 * @return true if a key has been successfully added 984 */ 985 UCL_EXTERN bool ucl_pubkey_add (struct ucl_parser *parser, const unsigned char *key, size_t len); 986 987 /** 988 * Set FILENAME and CURDIR variables in parser 989 * @param parser parser object 990 * @param filename filename to set or NULL to set FILENAME to "undef" and CURDIR to getcwd() 991 * @param need_expand perform realpath() if this variable is true and filename is not NULL 992 * @return true if variables has been set 993 */ 994 UCL_EXTERN bool ucl_parser_set_filevars (struct ucl_parser *parser, const char *filename, 995 bool need_expand); 996 997 /** @} */ 998 999 /** 1000 * @defgroup emitter Emitting functions 1001 * These functions are used to serialise UCL objects to some string representation. 1002 * 1003 * @{ 1004 */ 1005 1006 /** 1007 * Structure using for emitter callbacks 1008 */ 1009 struct ucl_emitter_functions { 1010 /** Append a single character */ 1011 int (*ucl_emitter_append_character) (unsigned char c, size_t nchars, void *ud); 1012 /** Append a string of a specified length */ 1013 int (*ucl_emitter_append_len) (unsigned const char *str, size_t len, void *ud); 1014 /** Append a 64 bit integer */ 1015 int (*ucl_emitter_append_int) (int64_t elt, void *ud); 1016 /** Append floating point element */ 1017 int (*ucl_emitter_append_double) (double elt, void *ud); 1018 /** Opaque userdata pointer */ 1019 void *ud; 1020 }; 1021 1022 /** 1023 * Emit object to a string 1024 * @param obj object 1025 * @param emit_type if type is #UCL_EMIT_JSON then emit json, if type is 1026 * #UCL_EMIT_CONFIG then emit config like object 1027 * @return dump of an object (must be freed after using) or NULL in case of error 1028 */ 1029 UCL_EXTERN unsigned char *ucl_object_emit (ucl_object_t *obj, enum ucl_emitter emit_type); 1030 1031 /** 1032 * Emit object to a string 1033 * @param obj object 1034 * @param emit_type if type is #UCL_EMIT_JSON then emit json, if type is 1035 * #UCL_EMIT_CONFIG then emit config like object 1036 * @return dump of an object (must be freed after using) or NULL in case of error 1037 */ 1038 UCL_EXTERN bool ucl_object_emit_full (ucl_object_t *obj, enum ucl_emitter emit_type, 1039 struct ucl_emitter_functions *emitter); 1040 /** @} */ 1041 1042 #ifdef __cplusplus 1043 } 1044 #endif 1045 /* 1046 * XXX: Poorly named API functions, need to replace them with the appropriate 1047 * named function. All API functions *must* use naming ucl_object_*. Usage of 1048 * ucl_obj* should be avoided. 1049 */ 1050 #define ucl_obj_todouble_safe ucl_object_todouble_safe 1051 #define ucl_obj_todouble ucl_object_todouble 1052 #define ucl_obj_tostring ucl_object_tostring 1053 #define ucl_obj_tostring_safe ucl_object_tostring_safe 1054 #define ucl_obj_tolstring ucl_object_tolstring 1055 #define ucl_obj_tolstring_safe ucl_object_tolstring_safe 1056 #define ucl_obj_toint ucl_object_toint 1057 #define ucl_obj_toint_safe ucl_object_toint_safe 1058 #define ucl_obj_toboolean ucl_object_toboolean 1059 #define ucl_obj_toboolean_safe ucl_object_toboolean_safe 1060 #define ucl_obj_get_key ucl_object_find_key 1061 #define ucl_obj_get_keyl ucl_object_find_keyl 1062 #define ucl_obj_unref ucl_object_unref 1063 #define ucl_obj_ref ucl_object_ref 1064 #define ucl_obj_free ucl_object_free 1065 1066 #endif /* UCL_H_ */ 1067