xref: /freebsd/contrib/libucl/include/ucl.h (revision cddbc3b40812213ff00041f79174cac0be360a2a)
1 /* Copyright (c) 2013-2015, 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 #ifdef __GNUC__
85 #define UCL_DEPRECATED(func) func __attribute__ ((deprecated))
86 #elif defined(_MSC_VER)
87 #define UCL_DEPRECATED(func) __declspec(deprecated) func
88 #else
89 #define UCL_DEPRECATED(func) func
90 #endif
91 
92 /**
93  * @defgroup structures Structures and types
94  * UCL defines several enumeration types used for error reporting or specifying flags and attributes.
95  *
96  * @{
97  */
98 
99 /**
100  * The common error codes returned by ucl parser
101  */
102 typedef enum ucl_error {
103 	UCL_EOK = 0, /**< No error */
104 	UCL_ESYNTAX, /**< Syntax error occurred during parsing */
105 	UCL_EIO, /**< IO error occurred during parsing */
106 	UCL_ESTATE, /**< Invalid state machine state */
107 	UCL_ENESTED, /**< Input has too many recursion levels */
108 	UCL_EMACRO, /**< Error processing a macro */
109 	UCL_EINTERNAL, /**< Internal unclassified error */
110 	UCL_ESSL, /**< SSL error */
111 	UCL_EMERGE /**< A merge error occured */
112 } ucl_error_t;
113 
114 /**
115  * #ucl_object_t may have one of specified types, some types are compatible with each other and some are not.
116  * For example, you can always convert #UCL_TIME to #UCL_FLOAT. Also you can convert #UCL_FLOAT to #UCL_INTEGER
117  * by loosing floating point. Every object may be converted to a string by #ucl_object_tostring_forced() function.
118  *
119  */
120 typedef enum ucl_type {
121 	UCL_OBJECT = 0, /**< UCL object - key/value pairs */
122 	UCL_ARRAY, /**< UCL array */
123 	UCL_INT, /**< Integer number */
124 	UCL_FLOAT, /**< Floating point number */
125 	UCL_STRING, /**< Null terminated string */
126 	UCL_BOOLEAN, /**< Boolean value */
127 	UCL_TIME, /**< Time value (floating point number of seconds) */
128 	UCL_USERDATA, /**< Opaque userdata pointer (may be used in macros) */
129 	UCL_NULL /**< Null value */
130 } ucl_type_t;
131 
132 /**
133  * You can use one of these types to serialise #ucl_object_t by using ucl_object_emit().
134  */
135 typedef enum ucl_emitter {
136 	UCL_EMIT_JSON = 0, /**< Emit fine formatted JSON */
137 	UCL_EMIT_JSON_COMPACT, /**< Emit compacted JSON */
138 	UCL_EMIT_CONFIG, /**< Emit human readable config format */
139 	UCL_EMIT_YAML, /**< Emit embedded YAML format */
140 	UCL_EMIT_MSGPACK, /**< Emit msgpack output */
141 	UCL_EMIT_MAX /**< Unsupported emitter type */
142 } ucl_emitter_t;
143 
144 /**
145  * These flags defines parser behaviour. If you specify #UCL_PARSER_ZEROCOPY you must ensure
146  * that the input memory is not freed if an object is in use. Moreover, if you want to use
147  * zero-terminated keys and string values then you should not use zero-copy mode, as in this case
148  * UCL still has to perform copying implicitly.
149  */
150 typedef enum ucl_parser_flags {
151 	UCL_PARSER_DEFAULT = 0,       /**< No special flags */
152 	UCL_PARSER_KEY_LOWERCASE = (1 << 0), /**< Convert all keys to lower case */
153 	UCL_PARSER_ZEROCOPY = (1 << 1), /**< Parse input in zero-copy mode if possible */
154 	UCL_PARSER_NO_TIME = (1 << 2), /**< Do not parse time and treat time values as strings */
155 	UCL_PARSER_NO_IMPLICIT_ARRAYS = (1 << 3), /** Create explicit arrays instead of implicit ones */
156 	UCL_PARSER_SAVE_COMMENTS = (1 << 4), /** Save comments in the parser context */
157 	UCL_PARSER_DISABLE_MACRO = (1 << 5), /** Treat macros as comments */
158 	UCL_PARSER_NO_FILEVARS = (1 << 6) /** Do not set file vars */
159 } ucl_parser_flags_t;
160 
161 /**
162  * String conversion flags, that are used in #ucl_object_fromstring_common function.
163  */
164 typedef enum ucl_string_flags {
165 	UCL_STRING_RAW = 0x0,     /**< Treat string as is */
166 	UCL_STRING_ESCAPE = (1 << 0),  /**< Perform JSON escape */
167 	UCL_STRING_TRIM = (1 << 1),    /**< Trim leading and trailing whitespaces */
168 	UCL_STRING_PARSE_BOOLEAN = (1 << 2),    /**< Parse passed string and detect boolean */
169 	UCL_STRING_PARSE_INT = (1 << 3),    /**< Parse passed string and detect integer number */
170 	UCL_STRING_PARSE_DOUBLE = (1 << 4),    /**< Parse passed string and detect integer or float number */
171 	UCL_STRING_PARSE_TIME = (1 << 5), /**< Parse time strings */
172 	UCL_STRING_PARSE_NUMBER =  UCL_STRING_PARSE_INT|UCL_STRING_PARSE_DOUBLE|UCL_STRING_PARSE_TIME,  /**<
173 									Parse passed string and detect number */
174 	UCL_STRING_PARSE =  UCL_STRING_PARSE_BOOLEAN|UCL_STRING_PARSE_NUMBER,   /**<
175 									Parse passed string (and detect booleans and numbers) */
176 	UCL_STRING_PARSE_BYTES = (1 << 6)  /**< Treat numbers as bytes */
177 } ucl_string_flags_t;
178 
179 /**
180  * Basic flags for an object
181  */
182 typedef enum ucl_object_flags {
183 	UCL_OBJECT_ALLOCATED_KEY = (1 << 0), /**< An object has key allocated internally */
184 	UCL_OBJECT_ALLOCATED_VALUE = (1 << 1), /**< An object has a string value allocated internally */
185 	UCL_OBJECT_NEED_KEY_ESCAPE = (1 << 2), /**< The key of an object need to be escaped on output */
186 	UCL_OBJECT_EPHEMERAL = (1 << 3), /**< Temporary object that does not need to be freed really */
187 	UCL_OBJECT_MULTILINE = (1 << 4), /**< String should be displayed as multiline string */
188 	UCL_OBJECT_MULTIVALUE = (1 << 5), /**< Object is a key with multiple values */
189 	UCL_OBJECT_INHERITED = (1 << 6), /**< Object has been inherited from another */
190 	UCL_OBJECT_BINARY = (1 << 7) /**< Object contains raw binary data */
191 } ucl_object_flags_t;
192 
193 /**
194  * Duplicate policy types
195  */
196 enum ucl_duplicate_strategy {
197 	UCL_DUPLICATE_APPEND = 0, /**< Default policy to merge based on priorities */
198 	UCL_DUPLICATE_MERGE,     /**< Merge new object with old one */
199 	UCL_DUPLICATE_REWRITE,   /**< Rewrite old keys */
200 	UCL_DUPLICATE_ERROR      /**< Stop parsing on duplicate found */
201 };
202 
203 /**
204  * Input format type
205  */
206 enum ucl_parse_type {
207 	UCL_PARSE_UCL = 0, /**< Default ucl format */
208 	UCL_PARSE_MSGPACK, /**< Message pack input format */
209 	UCL_PARSE_CSEXP, /**< Canonical S-expressions */
210 	UCL_PARSE_AUTO /**< Try to detect parse type */
211 };
212 
213 /**
214  * UCL object structure. Please mention that the most of fields should not be touched by
215  * UCL users. In future, this structure may be converted to private one.
216  */
217 typedef struct ucl_object_s {
218 	/**
219 	 * Variant value type
220 	 */
221 	union {
222 		int64_t iv;							/**< Int value of an object */
223 		const char *sv;						/**< String value of an object */
224 		double dv;							/**< Double value of an object */
225 		void *av;							/**< Array					*/
226 		void *ov;							/**< Object					*/
227 		void* ud;							/**< Opaque user data		*/
228 	} value;
229 	const char *key;						/**< Key of an object		*/
230 	struct ucl_object_s *next;				/**< Array handle			*/
231 	struct ucl_object_s *prev;				/**< Array handle			*/
232 	uint32_t keylen;						/**< Length of a key		*/
233 	uint32_t len;							/**< Size of an object		*/
234 	uint32_t ref;							/**< Reference count		*/
235 	uint16_t flags;							/**< Object flags			*/
236 	uint16_t type;							/**< Real type				*/
237 	unsigned char* trash_stack[2];			/**< Pointer to allocated chunks */
238 } ucl_object_t;
239 
240 /**
241  * Destructor type for userdata objects
242  * @param ud user specified data pointer
243  */
244 typedef void (*ucl_userdata_dtor)(void *ud);
245 typedef const char* (*ucl_userdata_emitter)(void *ud);
246 
247 /** @} */
248 
249 /**
250  * @defgroup utils Utility functions
251  * A number of utility functions simplify handling of UCL objects
252  *
253  * @{
254  */
255 /**
256  * Copy and return a key of an object, returned key is zero-terminated
257  * @param obj CL object
258  * @return zero terminated key
259  */
260 UCL_EXTERN char* ucl_copy_key_trash (const ucl_object_t *obj);
261 
262 /**
263  * Copy and return a string value of an object, returned key is zero-terminated
264  * @param obj CL object
265  * @return zero terminated string representation of object value
266  */
267 UCL_EXTERN char* ucl_copy_value_trash (const ucl_object_t *obj);
268 
269 /**
270  * Creates a new object
271  * @return new object
272  */
273 UCL_EXTERN ucl_object_t* ucl_object_new (void) UCL_WARN_UNUSED_RESULT;
274 
275 /**
276  * Create new object with type specified
277  * @param type type of a new object
278  * @return new object
279  */
280 UCL_EXTERN ucl_object_t* ucl_object_typed_new (ucl_type_t type) UCL_WARN_UNUSED_RESULT;
281 
282 /**
283  * Create new object with type and priority specified
284  * @param type type of a new object
285  * @param priority priority of an object
286  * @return new object
287  */
288 UCL_EXTERN ucl_object_t* ucl_object_new_full (ucl_type_t type, unsigned priority)
289 	UCL_WARN_UNUSED_RESULT;
290 
291 /**
292  * Create new object with userdata dtor
293  * @param dtor destructor function
294  * @param emitter emitter for userdata
295  * @param ptr opaque pointer
296  * @return new object
297  */
298 UCL_EXTERN ucl_object_t* ucl_object_new_userdata (ucl_userdata_dtor dtor,
299 		ucl_userdata_emitter emitter, void *ptr) UCL_WARN_UNUSED_RESULT;
300 
301 /**
302  * Perform deep copy of an object copying everything
303  * @param other object to copy
304  * @return new object with refcount equal to 1
305  */
306 UCL_EXTERN ucl_object_t * ucl_object_copy (const ucl_object_t *other)
307 	UCL_WARN_UNUSED_RESULT;
308 
309 /**
310  * Return the type of an object
311  * @return the object type
312  */
313 UCL_EXTERN ucl_type_t ucl_object_type (const ucl_object_t *obj);
314 
315 /**
316  * Converts ucl object type to its string representation
317  * @param type type of object
318  * @return constant string describing type
319  */
320 UCL_EXTERN const char * ucl_object_type_to_string (ucl_type_t type);
321 
322 /**
323  * Converts string that represents ucl type to real ucl type enum
324  * @param input C string with name of type
325  * @param res resulting target
326  * @return true if `input` is a name of type stored in `res`
327  */
328 UCL_EXTERN bool ucl_object_string_to_type (const char *input, ucl_type_t *res);
329 
330 /**
331  * Convert any string to an ucl object making the specified transformations
332  * @param str fixed size or NULL terminated string
333  * @param len length (if len is zero, than str is treated as NULL terminated)
334  * @param flags conversion flags
335  * @return new object
336  */
337 UCL_EXTERN ucl_object_t * ucl_object_fromstring_common (const char *str, size_t len,
338 		enum ucl_string_flags flags) UCL_WARN_UNUSED_RESULT;
339 
340 /**
341  * Create a UCL object from the specified string
342  * @param str NULL terminated string, will be json escaped
343  * @return new object
344  */
345 UCL_EXTERN ucl_object_t *ucl_object_fromstring (const char *str) UCL_WARN_UNUSED_RESULT;
346 
347 /**
348  * Create a UCL object from the specified string
349  * @param str fixed size string, will be json escaped
350  * @param len length of a string
351  * @return new object
352  */
353 UCL_EXTERN ucl_object_t *ucl_object_fromlstring (const char *str,
354 		size_t len) UCL_WARN_UNUSED_RESULT;
355 
356 /**
357  * Create an object from an integer number
358  * @param iv number
359  * @return new object
360  */
361 UCL_EXTERN ucl_object_t* ucl_object_fromint (int64_t iv) UCL_WARN_UNUSED_RESULT;
362 
363 /**
364  * Create an object from a float number
365  * @param dv number
366  * @return new object
367  */
368 UCL_EXTERN ucl_object_t* ucl_object_fromdouble (double dv) UCL_WARN_UNUSED_RESULT;
369 
370 /**
371  * Create an object from a boolean
372  * @param bv bool value
373  * @return new object
374  */
375 UCL_EXTERN ucl_object_t* ucl_object_frombool (bool bv) UCL_WARN_UNUSED_RESULT;
376 
377 /**
378  * Insert a object 'elt' to the hash 'top' and associate it with key 'key'
379  * @param top destination object (must be of type UCL_OBJECT)
380  * @param elt element to insert (must NOT be NULL)
381  * @param key key to associate with this object (either const or preallocated)
382  * @param keylen length of the key (or 0 for NULL terminated keys)
383  * @param copy_key make an internal copy of key
384  * @return true if key has been inserted
385  */
386 UCL_EXTERN bool ucl_object_insert_key (ucl_object_t *top, ucl_object_t *elt,
387 		const char *key, size_t keylen, bool copy_key);
388 
389 /**
390  * Replace a object 'elt' to the hash 'top' and associate it with key 'key', old object will be unrefed,
391  * if no object has been found this function works like ucl_object_insert_key()
392  * @param top destination object (must be of type UCL_OBJECT)
393  * @param elt element to insert (must NOT be NULL)
394  * @param key key to associate with this object (either const or preallocated)
395  * @param keylen length of the key (or 0 for NULL terminated keys)
396  * @param copy_key make an internal copy of key
397  * @return true if key has been inserted
398  */
399 UCL_EXTERN bool ucl_object_replace_key (ucl_object_t *top, ucl_object_t *elt,
400 		const char *key, size_t keylen, bool copy_key);
401 
402 /**
403  * Merge the keys from one object to another object. Overwrite on conflict
404  * @param top destination object (must be of type UCL_OBJECT)
405  * @param elt element to insert (must be of type UCL_OBJECT)
406  * @param copy copy rather than reference the elements
407  * @return true if all keys have been merged
408  */
409 UCL_EXTERN bool ucl_object_merge (ucl_object_t *top, ucl_object_t *elt, bool copy);
410 
411 /**
412  * Delete a object associated with key 'key', old object will be unrefered,
413  * @param top object
414  * @param key key associated to the object to remove
415  * @param keylen length of the key (or 0 for NULL terminated keys)
416  */
417 UCL_EXTERN bool ucl_object_delete_keyl (ucl_object_t *top,
418 		const char *key, size_t keylen);
419 
420 /**
421  * Delete a object associated with key 'key', old object will be unrefered,
422  * @param top object
423  * @param key key associated to the object to remove
424  */
425 UCL_EXTERN bool ucl_object_delete_key (ucl_object_t *top,
426 		const char *key);
427 
428 
429 /**
430  * Removes `key` from `top` object, returning the object that was removed. This
431  * object is not released, caller must unref the returned object when it is no
432  * longer needed.
433  * @param top object
434  * @param key key to remove
435  * @param keylen length of the key (or 0 for NULL terminated keys)
436  * @return removed object or NULL if object has not been found
437  */
438 UCL_EXTERN ucl_object_t* ucl_object_pop_keyl (ucl_object_t *top, const char *key,
439 		size_t keylen) UCL_WARN_UNUSED_RESULT;
440 
441 /**
442  * Removes `key` from `top` object returning the object that was removed. This
443  * object is not released, caller must unref the returned object when it is no
444  * longer needed.
445  * @param top object
446  * @param key key to remove
447  * @return removed object or NULL if object has not been found
448  */
449 UCL_EXTERN ucl_object_t* ucl_object_pop_key (ucl_object_t *top, const char *key)
450 	UCL_WARN_UNUSED_RESULT;
451 
452 /**
453  * Insert a object 'elt' to the hash 'top' and associate it with key 'key', if
454  * the specified key exist, try to merge its content
455  * @param top destination object (must be of type UCL_OBJECT)
456  * @param elt element to insert (must NOT be NULL)
457  * @param key key to associate with this object (either const or preallocated)
458  * @param keylen length of the key (or 0 for NULL terminated keys)
459  * @param copy_key make an internal copy of key
460  * @return true if key has been inserted
461  */
462 UCL_EXTERN bool ucl_object_insert_key_merged (ucl_object_t *top, ucl_object_t *elt,
463 		const char *key, size_t keylen, bool copy_key);
464 
465 /**
466  * Append an element to the end of array object
467  * @param top destination object (must NOT be NULL)
468  * @param elt element to append (must NOT be NULL)
469  * @return true if value has been inserted
470  */
471 UCL_EXTERN bool ucl_array_append (ucl_object_t *top,
472 		ucl_object_t *elt);
473 
474 /**
475  * Append an element to the start of array object
476  * @param top destination object (must NOT be NULL)
477  * @param elt element to append (must NOT be NULL)
478  * @return true if value has been inserted
479  */
480 UCL_EXTERN bool ucl_array_prepend (ucl_object_t *top,
481 		ucl_object_t *elt);
482 
483 /**
484  * Merge all elements of second array into the first array
485  * @param top destination array (must be of type UCL_ARRAY)
486  * @param elt array to copy elements from (must be of type UCL_ARRAY)
487  * @param copy copy elements instead of referencing them
488  * @return true if arrays were merged
489  */
490 UCL_EXTERN bool ucl_array_merge (ucl_object_t *top, ucl_object_t *elt,
491 		bool copy);
492 
493 /**
494  * Removes an element `elt` from the array `top`, returning the object that was
495  * removed. This object is not released, caller must unref the returned object
496  * when it is no longer needed.
497  * @param top array ucl object
498  * @param elt element to remove
499  * @return removed element or NULL if `top` is NULL or not an array
500  */
501 UCL_EXTERN ucl_object_t* ucl_array_delete (ucl_object_t *top,
502 		ucl_object_t *elt);
503 
504 /**
505  * Returns the first element of the array `top`
506  * @param top array ucl object
507  * @return element or NULL if `top` is NULL or not an array
508  */
509 UCL_EXTERN const ucl_object_t* ucl_array_head (const ucl_object_t *top);
510 
511 /**
512  * Returns the last element of the array `top`
513  * @param top array ucl object
514  * @return element or NULL if `top` is NULL or not an array
515  */
516 UCL_EXTERN const ucl_object_t* ucl_array_tail (const ucl_object_t *top);
517 
518 /**
519  * Removes the last element from the array `top`, returning the object that was
520  * removed. This object is not released, caller must unref the returned object
521  * when it is no longer needed.
522  * @param top array ucl object
523  * @return removed element or NULL if `top` is NULL or not an array
524  */
525 UCL_EXTERN ucl_object_t* ucl_array_pop_last (ucl_object_t *top);
526 
527 /**
528  * Removes the first element from the array `top`, returning the object that was
529  * removed. This object is not released, caller must unref the returned object
530  * when it is no longer needed.
531  * @param top array ucl object
532  * @return removed element or NULL if `top` is NULL or not an array
533  */
534 UCL_EXTERN ucl_object_t* ucl_array_pop_first (ucl_object_t *top);
535 
536 /**
537  * Return object identified by index of the array `top`
538  * @param top object to get a key from (must be of type UCL_ARRAY)
539  * @param index array index to return
540  * @return object at the specified index or NULL if index is not found
541  */
542 UCL_EXTERN const ucl_object_t* ucl_array_find_index (const ucl_object_t *top,
543 		unsigned int index);
544 
545 /**
546  * Return the index of `elt` in the array `top`
547  * @param top object to get a key from (must be of type UCL_ARRAY)
548  * @param elt element to find index of (must NOT be NULL)
549  * @return index of `elt` in the array `top or (unsigned int)-1 if `elt` is not found
550  */
551 UCL_EXTERN unsigned int ucl_array_index_of (ucl_object_t *top,
552 		ucl_object_t *elt);
553 
554 /**
555  * Replace an element in an array with a different element, returning the object
556  * that was replaced. This object is not released, caller must unref the
557  * returned object when it is no longer needed.
558  * @param top destination object (must be of type UCL_ARRAY)
559  * @param elt element to append (must NOT be NULL)
560  * @param index array index in destination to overwrite with elt
561  * @return object that was replaced or NULL if index is not found
562  */
563 ucl_object_t *
564 ucl_array_replace_index (ucl_object_t *top, ucl_object_t *elt,
565 	unsigned int index);
566 
567 /**
568  * Append a element to another element forming an implicit array
569  * @param head head to append (may be NULL)
570  * @param elt new element
571  * @return the new implicit array
572  */
573 UCL_EXTERN ucl_object_t * ucl_elt_append (ucl_object_t *head,
574 		ucl_object_t *elt);
575 
576 /**
577  * Converts an object to double value
578  * @param obj CL object
579  * @param target target double variable
580  * @return true if conversion was successful
581  */
582 UCL_EXTERN bool ucl_object_todouble_safe (const ucl_object_t *obj, double *target);
583 
584 /**
585  * Unsafe version of \ref ucl_obj_todouble_safe
586  * @param obj CL object
587  * @return double value
588  */
589 UCL_EXTERN double ucl_object_todouble (const ucl_object_t *obj);
590 
591 /**
592  * Converts an object to integer value
593  * @param obj CL object
594  * @param target target integer variable
595  * @return true if conversion was successful
596  */
597 UCL_EXTERN bool ucl_object_toint_safe (const ucl_object_t *obj, int64_t *target);
598 
599 /**
600  * Unsafe version of \ref ucl_obj_toint_safe
601  * @param obj CL object
602  * @return int value
603  */
604 UCL_EXTERN int64_t ucl_object_toint (const ucl_object_t *obj);
605 
606 /**
607  * Converts an object to boolean value
608  * @param obj CL object
609  * @param target target boolean variable
610  * @return true if conversion was successful
611  */
612 UCL_EXTERN bool ucl_object_toboolean_safe (const ucl_object_t *obj, bool *target);
613 
614 /**
615  * Unsafe version of \ref ucl_obj_toboolean_safe
616  * @param obj CL object
617  * @return boolean value
618  */
619 UCL_EXTERN bool ucl_object_toboolean (const ucl_object_t *obj);
620 
621 /**
622  * Converts an object to string value
623  * @param obj CL object
624  * @param target target string variable, no need to free value
625  * @return true if conversion was successful
626  */
627 UCL_EXTERN bool ucl_object_tostring_safe (const ucl_object_t *obj, const char **target);
628 
629 /**
630  * Unsafe version of \ref ucl_obj_tostring_safe
631  * @param obj CL object
632  * @return string value
633  */
634 UCL_EXTERN const char* ucl_object_tostring (const ucl_object_t *obj);
635 
636 /**
637  * Convert any object to a string in JSON notation if needed
638  * @param obj CL object
639  * @return string value
640  */
641 UCL_EXTERN const char* ucl_object_tostring_forced (const ucl_object_t *obj);
642 
643 /**
644  * Return string as char * and len, string may be not zero terminated, more efficient that \ref ucl_obj_tostring as it
645  * allows zero-copy (if #UCL_PARSER_ZEROCOPY has been used during parsing)
646  * @param obj CL object
647  * @param target target string variable, no need to free value
648  * @param tlen target length
649  * @return true if conversion was successful
650  */
651 UCL_EXTERN bool ucl_object_tolstring_safe (const ucl_object_t *obj,
652 		const char **target, size_t *tlen);
653 
654 /**
655  * Unsafe version of \ref ucl_obj_tolstring_safe
656  * @param obj CL object
657  * @return string value
658  */
659 UCL_EXTERN const char* ucl_object_tolstring (const ucl_object_t *obj, size_t *tlen);
660 
661 /**
662  * Return object identified by a key in the specified object
663  * @param obj object to get a key from (must be of type UCL_OBJECT)
664  * @param key key to search
665  * @return object matching the specified key or NULL if key was not found
666  */
667 UCL_EXTERN const ucl_object_t* ucl_object_lookup (const ucl_object_t *obj,
668 		const char *key);
669 #define ucl_object_find_key ucl_object_lookup
670 
671 /**
672  * Return object identified by a key in the specified object, if the first key is
673  * not found then look for the next one. This process is repeated unless
674  * the next argument in the list is not NULL. So, `ucl_object_find_any_key(obj, key, NULL)`
675  * is equal to `ucl_object_find_key(obj, key)`
676  * @param obj object to get a key from (must be of type UCL_OBJECT)
677  * @param key key to search
678  * @param ... list of alternative keys to search (NULL terminated)
679  * @return object matching the specified key or NULL if key was not found
680  */
681 UCL_EXTERN const ucl_object_t* ucl_object_lookup_any (const ucl_object_t *obj,
682 		const char *key, ...);
683 #define ucl_object_find_any_key ucl_object_lookup_any
684 
685 /**
686  * Return object identified by a fixed size key in the specified object
687  * @param obj object to get a key from (must be of type UCL_OBJECT)
688  * @param key key to search
689  * @param klen length of a key
690  * @return object matching the specified key or NULL if key was not found
691  */
692 UCL_EXTERN const ucl_object_t* ucl_object_lookup_len (const ucl_object_t *obj,
693 		const char *key, size_t klen);
694 #define ucl_object_find_keyl ucl_object_lookup_len
695 
696 /**
697  * Return object identified by dot notation string
698  * @param obj object to search in
699  * @param path dot.notation.path to the path to lookup. May use numeric .index on arrays
700  * @return object matched the specified path or NULL if path is not found
701  */
702 UCL_EXTERN const ucl_object_t *ucl_object_lookup_path (const ucl_object_t *obj,
703 		const char *path);
704 #define ucl_lookup_path ucl_object_lookup_path
705 
706 /**
707  * Return object identified by object notation string using arbitrary delimiter
708  * @param obj object to search in
709  * @param path dot.notation.path to the path to lookup. May use numeric .index on arrays
710  * @param sep the sepatorator to use in place of . (incase keys have . in them)
711  * @return object matched the specified path or NULL if path is not found
712  */
713 UCL_EXTERN const ucl_object_t *ucl_object_lookup_path_char (const ucl_object_t *obj,
714 		const char *path, char sep);
715 #define ucl_lookup_path_char ucl_object_lookup_path_char
716 
717 /**
718  * Returns a key of an object as a NULL terminated string
719  * @param obj CL object
720  * @return key or NULL if there is no key
721  */
722 UCL_EXTERN const char* ucl_object_key (const ucl_object_t *obj);
723 
724 /**
725  * Returns a key of an object as a fixed size string (may be more efficient)
726  * @param obj CL object
727  * @param len target key length
728  * @return key pointer
729  */
730 UCL_EXTERN const char* ucl_object_keyl (const ucl_object_t *obj, size_t *len);
731 
732 /**
733  * Increase reference count for an object
734  * @param obj object to ref
735  * @return the referenced object
736  */
737 UCL_EXTERN ucl_object_t* ucl_object_ref (const ucl_object_t *obj);
738 
739 /**
740  * Free ucl object
741  * @param obj ucl object to free
742  */
743 UCL_DEPRECATED(UCL_EXTERN void ucl_object_free (ucl_object_t *obj));
744 
745 /**
746  * Decrease reference count for an object
747  * @param obj object to unref
748  */
749 UCL_EXTERN void ucl_object_unref (ucl_object_t *obj);
750 
751 /**
752  * Compare objects `o1` and `o2`
753  * @param o1 the first object
754  * @param o2 the second object
755  * @return values >0, 0 and <0 if `o1` is more than, equal and less than `o2`.
756  * The order of comparison:
757  * 1) Type of objects
758  * 2) Size of objects
759  * 3) Content of objects
760  */
761 UCL_EXTERN int ucl_object_compare (const ucl_object_t *o1,
762 		const ucl_object_t *o2);
763 
764 /**
765  * Compare objects `o1` and `o2` useful for sorting
766  * @param o1 the first object
767  * @param o2 the second object
768  * @return values >0, 0 and <0 if `o1` is more than, equal and less than `o2`.
769  * The order of comparison:
770  * 1) Type of objects
771  * 2) Size of objects
772  * 3) Content of objects
773  */
774 UCL_EXTERN int ucl_object_compare_qsort (const ucl_object_t **o1,
775 		const ucl_object_t **o2);
776 
777 /**
778  * Sort UCL array using `cmp` compare function
779  * @param ar
780  * @param cmp
781  */
782 UCL_EXTERN void ucl_object_array_sort (ucl_object_t *ar,
783 		int (*cmp)(const ucl_object_t **o1, const ucl_object_t **o2));
784 
785 /**
786  * Get the priority for specific UCL object
787  * @param obj any ucl object
788  * @return priority of an object
789  */
790 UCL_EXTERN unsigned int ucl_object_get_priority (const ucl_object_t *obj);
791 
792 /**
793  * Set explicit priority of an object.
794  * @param obj any ucl object
795  * @param priority new priroity value (only 4 least significant bits are considred)
796  */
797 UCL_EXTERN void ucl_object_set_priority (ucl_object_t *obj,
798 		unsigned int priority);
799 
800 /**
801  * Opaque iterator object
802  */
803 typedef void* ucl_object_iter_t;
804 
805 /**
806  * Get next key from an object
807  * @param obj object to iterate
808  * @param iter opaque iterator, must be set to NULL on the first call:
809  * ucl_object_iter_t it = NULL;
810  * while ((cur = ucl_iterate_object (obj, &it)) != NULL) ...
811  * @return the next object or NULL
812  */
813 UCL_EXTERN const ucl_object_t* ucl_object_iterate (const ucl_object_t *obj,
814 		ucl_object_iter_t *iter, bool expand_values);
815 #define ucl_iterate_object ucl_object_iterate
816 
817 /**
818  * Create new safe iterator for the specified object
819  * @param obj object to iterate
820  * @return new iterator object that should be used with safe iterators API only
821  */
822 UCL_EXTERN ucl_object_iter_t ucl_object_iterate_new (const ucl_object_t *obj)
823 	UCL_WARN_UNUSED_RESULT;
824 /**
825  * Reset initialized iterator to a new object
826  * @param obj new object to iterate
827  * @return modified iterator object
828  */
829 UCL_EXTERN ucl_object_iter_t ucl_object_iterate_reset (ucl_object_iter_t it,
830 		const ucl_object_t *obj);
831 
832 /**
833  * Get the next object from the `obj`. This fucntion iterates over arrays, objects
834  * and implicit arrays
835  * @param iter safe iterator
836  * @param expand_values expand explicit arrays and objects
837  * @return the next object in sequence
838  */
839 UCL_EXTERN const ucl_object_t* ucl_object_iterate_safe (ucl_object_iter_t iter,
840 		bool expand_values);
841 /**
842  * Iteration type enumerator
843  */
844 enum ucl_iterate_type {
845 	UCL_ITERATE_EXPLICIT = 1 << 0, /**< Iterate just explicit arrays and objects */
846 	UCL_ITERATE_IMPLICIT = 1 << 1,  /**< Iterate just implicit arrays */
847 	UCL_ITERATE_BOTH = (1 << 0) | (1 << 1),   /**< Iterate both explicit and implicit arrays*/
848 };
849 
850 /**
851  * Get the next object from the `obj`. This fucntion iterates over arrays, objects
852  * and implicit arrays if needed
853  * @param iter safe iterator
854  * @param
855  * @return the next object in sequence
856  */
857 UCL_EXTERN const ucl_object_t* ucl_object_iterate_full (ucl_object_iter_t iter,
858 		enum ucl_iterate_type type);
859 
860 /**
861  * Free memory associated with the safe iterator
862  * @param it safe iterator object
863  */
864 UCL_EXTERN void ucl_object_iterate_free (ucl_object_iter_t it);
865 
866 /** @} */
867 
868 
869 /**
870  * @defgroup parser Parsing functions
871  * These functions are used to parse UCL objects
872  *
873  * @{
874  */
875 
876 /**
877  * Macro handler for a parser
878  * @param data the content of macro
879  * @param len the length of content
880  * @param arguments arguments object
881  * @param ud opaque user data
882  * @param err error pointer
883  * @return true if macro has been parsed
884  */
885 typedef bool (*ucl_macro_handler) (const unsigned char *data, size_t len,
886 		const ucl_object_t *arguments,
887 		void* ud);
888 
889 /**
890  * Context dependent macro handler for a parser
891  * @param data the content of macro
892  * @param len the length of content
893  * @param arguments arguments object
894  * @param context previously parsed context
895  * @param ud opaque user data
896  * @param err error pointer
897  * @return true if macro has been parsed
898  */
899 typedef bool (*ucl_context_macro_handler) (const unsigned char *data, size_t len,
900 		const ucl_object_t *arguments,
901 		const ucl_object_t *context,
902 		void* ud);
903 
904 /* Opaque parser */
905 struct ucl_parser;
906 
907 /**
908  * Creates new parser object
909  * @param pool pool to allocate memory from
910  * @return new parser object
911  */
912 UCL_EXTERN struct ucl_parser* ucl_parser_new (int flags);
913 
914 /**
915  * Sets the default priority for the parser applied to chunks that does not
916  * specify priority explicitly
917  * @param parser parser object
918  * @param prio default priority (0 .. 16)
919  * @return true if parser's default priority was set
920  */
921 UCL_EXTERN bool ucl_parser_set_default_priority (struct ucl_parser *parser,
922 		unsigned prio);
923 /**
924  * Register new handler for a macro
925  * @param parser parser object
926  * @param macro macro name (without leading dot)
927  * @param handler handler (it is called immediately after macro is parsed)
928  * @param ud opaque user data for a handler
929  */
930 UCL_EXTERN void ucl_parser_register_macro (struct ucl_parser *parser,
931 		const char *macro,
932 		ucl_macro_handler handler, void* ud);
933 
934 /**
935  * Register new context dependent handler for a macro
936  * @param parser parser object
937  * @param macro macro name (without leading dot)
938  * @param handler handler (it is called immediately after macro is parsed)
939  * @param ud opaque user data for a handler
940  */
941 UCL_EXTERN void ucl_parser_register_context_macro (struct ucl_parser *parser,
942 		const char *macro,
943 		ucl_context_macro_handler handler,
944 		void* ud);
945 
946 /**
947  * Handler to detect unregistered variables
948  * @param data variable data
949  * @param len length of variable
950  * @param replace (out) replace value for variable
951  * @param replace_len (out) replace length for variable
952  * @param need_free (out) UCL will free `dest` after usage
953  * @param ud opaque userdata
954  * @return true if variable
955  */
956 typedef bool (*ucl_variable_handler) (const unsigned char *data, size_t len,
957 		unsigned char **replace, size_t *replace_len, bool *need_free, void* ud);
958 
959 /**
960  * Register new parser variable
961  * @param parser parser object
962  * @param var variable name
963  * @param value variable value
964  */
965 UCL_EXTERN void ucl_parser_register_variable (struct ucl_parser *parser, const char *var,
966 		const char *value);
967 
968 /**
969  * Set handler for unknown variables
970  * @param parser parser structure
971  * @param handler desired handler
972  * @param ud opaque data for the handler
973  */
974 UCL_EXTERN void ucl_parser_set_variables_handler (struct ucl_parser *parser,
975 		ucl_variable_handler handler, void *ud);
976 
977 /**
978  * Load new chunk to a parser
979  * @param parser parser structure
980  * @param data the pointer to the beginning of a chunk
981  * @param len the length of a chunk
982  * @return true if chunk has been added and false in case of error
983  */
984 UCL_EXTERN bool ucl_parser_add_chunk (struct ucl_parser *parser,
985 		const unsigned char *data, size_t len);
986 
987 /**
988  * Load new chunk to a parser with the specified priority
989  * @param parser parser structure
990  * @param data the pointer to the beginning of a chunk
991  * @param len the length of a chunk
992  * @param priority the desired priority of a chunk (only 4 least significant bits
993  * are considered for this parameter)
994  * @return true if chunk has been added and false in case of error
995  */
996 UCL_EXTERN bool ucl_parser_add_chunk_priority (struct ucl_parser *parser,
997 		const unsigned char *data, size_t len, unsigned priority);
998 
999 /**
1000  * Full version of ucl_add_chunk with priority and duplicate strategy
1001  * @param parser parser structure
1002  * @param data the pointer to the beginning of a chunk
1003  * @param len the length of a chunk
1004  * @param priority the desired priority of a chunk (only 4 least significant bits
1005  * are considered for this parameter)
1006  * @param strat duplicates merging strategy
1007  * @param parse_type input format
1008  * @return true if chunk has been added and false in case of error
1009  */
1010 UCL_EXTERN bool ucl_parser_add_chunk_full (struct ucl_parser *parser,
1011 		const unsigned char *data, size_t len, unsigned priority,
1012 		enum ucl_duplicate_strategy strat, enum ucl_parse_type parse_type);
1013 
1014 /**
1015  * Load ucl object from a string
1016  * @param parser parser structure
1017  * @param data the pointer to the string
1018  * @param len the length of the string, if `len` is 0 then `data` must be zero-terminated string
1019  * @return true if string has been added and false in case of error
1020  */
1021 UCL_EXTERN bool ucl_parser_add_string (struct ucl_parser *parser,
1022 		const char *data,size_t len);
1023 
1024 /**
1025  * Load ucl object from a string
1026  * @param parser parser structure
1027  * @param data the pointer to the string
1028  * @param len the length of the string, if `len` is 0 then `data` must be zero-terminated string
1029  * @param priority the desired priority of a chunk (only 4 least significant bits
1030  * are considered for this parameter)
1031  * @return true if string has been added and false in case of error
1032  */
1033 UCL_EXTERN bool ucl_parser_add_string_priority (struct ucl_parser *parser,
1034 		const char *data, size_t len, unsigned priority);
1035 
1036 /**
1037  * Load and add data from a file
1038  * @param parser parser structure
1039  * @param filename the name of file
1040  * @param err if *err is NULL it is set to parser error
1041  * @return true if chunk has been added and false in case of error
1042  */
1043 UCL_EXTERN bool ucl_parser_add_file (struct ucl_parser *parser,
1044 		const char *filename);
1045 
1046 /**
1047  * Load and add data from a file
1048  * @param parser parser structure
1049  * @param filename the name of file
1050  * @param err if *err is NULL it is set to parser error
1051  * @param priority the desired priority of a chunk (only 4 least significant bits
1052  * are considered for this parameter)
1053  * @return true if chunk has been added and false in case of error
1054  */
1055 UCL_EXTERN bool ucl_parser_add_file_priority (struct ucl_parser *parser,
1056 		const char *filename, unsigned priority);
1057 
1058 /**
1059  * Load and add data from a file
1060  * @param parser parser structure
1061  * @param filename the name of file
1062  * @param priority the desired priority of a chunk (only 4 least significant bits
1063  * are considered for this parameter)
1064  * @param strat Merge strategy to use while parsing this file
1065  * @param parse_type Parser type to use while parsing this file
1066  * @return true if chunk has been added and false in case of error
1067  */
1068 UCL_EXTERN bool ucl_parser_add_file_full (struct ucl_parser *parser, const char *filename,
1069 		unsigned priority, enum ucl_duplicate_strategy strat,
1070 		enum ucl_parse_type parse_type);
1071 
1072 /**
1073  * Load and add data from a file descriptor
1074  * @param parser parser structure
1075  * @param filename the name of file
1076  * @param err if *err is NULL it is set to parser error
1077  * @return true if chunk has been added and false in case of error
1078  */
1079 UCL_EXTERN bool ucl_parser_add_fd (struct ucl_parser *parser,
1080 		int fd);
1081 
1082 /**
1083  * Load and add data from a file descriptor
1084  * @param parser parser structure
1085  * @param filename the name of file
1086  * @param err if *err is NULL it is set to parser error
1087  * @param priority the desired priority of a chunk (only 4 least significant bits
1088  * are considered for this parameter)
1089  * @return true if chunk has been added and false in case of error
1090  */
1091 UCL_EXTERN bool ucl_parser_add_fd_priority (struct ucl_parser *parser,
1092 		int fd, unsigned priority);
1093 
1094 /**
1095  * Load and add data from a file descriptor
1096  * @param parser parser structure
1097  * @param filename the name of file
1098  * @param err if *err is NULL it is set to parser error
1099  * @param priority the desired priority of a chunk (only 4 least significant bits
1100  * are considered for this parameter)
1101  * @param strat Merge strategy to use while parsing this file
1102  * @param parse_type Parser type to use while parsing this file
1103  * @return true if chunk has been added and false in case of error
1104  */
1105 UCL_EXTERN bool ucl_parser_add_fd_full (struct ucl_parser *parser, int fd,
1106 		unsigned priority, enum ucl_duplicate_strategy strat,
1107 		enum ucl_parse_type parse_type);
1108 
1109 /**
1110  * Provide a UCL_ARRAY of paths to search for include files. The object is
1111  * copied so caller must unref the object.
1112  * @param parser parser structure
1113  * @param paths UCL_ARRAY of paths to search
1114  * @return true if the path search array was replaced in the parser
1115  */
1116 UCL_EXTERN bool ucl_set_include_path (struct ucl_parser *parser,
1117 		ucl_object_t *paths);
1118 
1119 /**
1120  * Get a top object for a parser (refcount is increased)
1121  * @param parser parser structure
1122  * @param err if *err is NULL it is set to parser error
1123  * @return top parser object or NULL
1124  */
1125 UCL_EXTERN ucl_object_t* ucl_parser_get_object (struct ucl_parser *parser);
1126 
1127 /**
1128  * Get the error string if parsing has been failed
1129  * @param parser parser object
1130  * @return error description
1131  */
1132 UCL_EXTERN const char *ucl_parser_get_error (struct ucl_parser *parser);
1133 
1134 /**
1135  * Get the code of the last error
1136  * @param parser parser object
1137  * @return error code
1138  */
1139 UCL_EXTERN int ucl_parser_get_error_code (struct ucl_parser *parser);
1140 
1141 /**
1142  * Get the current column number within parser
1143  * @param parser parser object
1144  * @return current column number
1145  */
1146 UCL_EXTERN unsigned ucl_parser_get_column (struct ucl_parser *parser);
1147 
1148 /**
1149  * Get the current line number within parser
1150  * @param parser parser object
1151  * @return current line number
1152  */
1153 UCL_EXTERN unsigned ucl_parser_get_linenum (struct ucl_parser *parser);
1154 
1155 /**
1156  * Clear the error in the parser
1157  * @param parser parser object
1158  */
1159 UCL_EXTERN void ucl_parser_clear_error (struct ucl_parser *parser);
1160 
1161 /**
1162  * Free ucl parser object
1163  * @param parser parser object
1164  */
1165 UCL_EXTERN void ucl_parser_free (struct ucl_parser *parser);
1166 
1167 /**
1168  * Get constant opaque pointer to comments structure for this parser. Increase
1169  * refcount to prevent this object to be destroyed on parser's destruction
1170  * @param parser parser structure
1171  * @return ucl comments pointer or NULL
1172  */
1173 UCL_EXTERN const ucl_object_t * ucl_parser_get_comments (struct ucl_parser *parser);
1174 
1175 /**
1176  * Utility function to find a comment object for the specified object in the input
1177  * @param comments comments object
1178  * @param srch search object
1179  * @return string comment enclosed in ucl_object_t
1180  */
1181 UCL_EXTERN const ucl_object_t * ucl_comments_find (const ucl_object_t *comments,
1182 		const ucl_object_t *srch);
1183 
1184 /**
1185  * Move comment from `from` object to `to` object
1186  * @param comments comments object
1187  * @param what source object
1188  * @param whith destination object
1189  * @return `true` if `from` has comment and it has been moved to `to`
1190  */
1191 UCL_EXTERN bool ucl_comments_move (ucl_object_t *comments,
1192 		const ucl_object_t *from, const ucl_object_t *to);
1193 
1194 /**
1195  * Adds a new comment for an object
1196  * @param comments comments object
1197  * @param obj object to add comment to
1198  * @param comment string representation of a comment
1199  */
1200 UCL_EXTERN void ucl_comments_add (ucl_object_t *comments,
1201 		const ucl_object_t *obj, const char *comment);
1202 
1203 /**
1204  * Add new public key to parser for signatures check
1205  * @param parser parser object
1206  * @param key PEM representation of a key
1207  * @param len length of the key
1208  * @param err if *err is NULL it is set to parser error
1209  * @return true if a key has been successfully added
1210  */
1211 UCL_EXTERN bool ucl_parser_pubkey_add (struct ucl_parser *parser,
1212 		const unsigned char *key, size_t len);
1213 
1214 /**
1215  * Set FILENAME and CURDIR variables in parser
1216  * @param parser parser object
1217  * @param filename filename to set or NULL to set FILENAME to "undef" and CURDIR to getcwd()
1218  * @param need_expand perform realpath() if this variable is true and filename is not NULL
1219  * @return true if variables has been set
1220  */
1221 UCL_EXTERN bool ucl_parser_set_filevars (struct ucl_parser *parser, const char *filename,
1222 		bool need_expand);
1223 
1224 /** @} */
1225 
1226 /**
1227  * @defgroup emitter Emitting functions
1228  * These functions are used to serialise UCL objects to some string representation.
1229  *
1230  * @{
1231  */
1232 
1233 struct ucl_emitter_context;
1234 /**
1235  * Structure using for emitter callbacks
1236  */
1237 struct ucl_emitter_functions {
1238 	/** Append a single character */
1239 	int (*ucl_emitter_append_character) (unsigned char c, size_t nchars, void *ud);
1240 	/** Append a string of a specified length */
1241 	int (*ucl_emitter_append_len) (unsigned const char *str, size_t len, void *ud);
1242 	/** Append a 64 bit integer */
1243 	int (*ucl_emitter_append_int) (int64_t elt, void *ud);
1244 	/** Append floating point element */
1245 	int (*ucl_emitter_append_double) (double elt, void *ud);
1246 	/** Free userdata */
1247 	void (*ucl_emitter_free_func)(void *ud);
1248 	/** Opaque userdata pointer */
1249 	void *ud;
1250 };
1251 
1252 struct ucl_emitter_operations {
1253 	/** Write a primitive element */
1254 	void (*ucl_emitter_write_elt) (struct ucl_emitter_context *ctx,
1255 		const ucl_object_t *obj, bool first, bool print_key);
1256 	/** Start ucl object */
1257 	void (*ucl_emitter_start_object) (struct ucl_emitter_context *ctx,
1258 		const ucl_object_t *obj, bool print_key);
1259 	/** End ucl object */
1260 	void (*ucl_emitter_end_object) (struct ucl_emitter_context *ctx,
1261 		const ucl_object_t *obj);
1262 	/** Start ucl array */
1263 	void (*ucl_emitter_start_array) (struct ucl_emitter_context *ctx,
1264 		const ucl_object_t *obj, bool print_key);
1265 	void (*ucl_emitter_end_array) (struct ucl_emitter_context *ctx,
1266 		const ucl_object_t *obj);
1267 };
1268 
1269 /**
1270  * Structure that defines emitter functions
1271  */
1272 struct ucl_emitter_context {
1273 	/** Name of emitter (e.g. json, compact_json) */
1274 	const char *name;
1275 	/** Unique id (e.g. UCL_EMIT_JSON for standard emitters */
1276 	int id;
1277 	/** A set of output functions */
1278 	const struct ucl_emitter_functions *func;
1279 	/** A set of output operations */
1280 	const struct ucl_emitter_operations *ops;
1281 	/** Current amount of indent tabs */
1282 	unsigned int indent;
1283 	/** Top level object */
1284 	const ucl_object_t *top;
1285 	/** Optional comments */
1286 	const ucl_object_t *comments;
1287 };
1288 
1289 /**
1290  * Emit object to a string
1291  * @param obj object
1292  * @param emit_type if type is #UCL_EMIT_JSON then emit json, if type is
1293  * #UCL_EMIT_CONFIG then emit config like object
1294  * @return dump of an object (must be freed after using) or NULL in case of error
1295  */
1296 UCL_EXTERN unsigned char *ucl_object_emit (const ucl_object_t *obj,
1297 		enum ucl_emitter emit_type);
1298 
1299 /**
1300  * Emit object to a string that can contain `\0` inside
1301  * @param obj object
1302  * @param emit_type if type is #UCL_EMIT_JSON then emit json, if type is
1303  * #UCL_EMIT_CONFIG then emit config like object
1304  * @param len the resulting length
1305  * @return dump of an object (must be freed after using) or NULL in case of error
1306  */
1307 UCL_EXTERN unsigned char *ucl_object_emit_len (const ucl_object_t *obj,
1308 		enum ucl_emitter emit_type, size_t *len);
1309 
1310 /**
1311  * Emit object to a string
1312  * @param obj object
1313  * @param emit_type if type is #UCL_EMIT_JSON then emit json, if type is
1314  * #UCL_EMIT_CONFIG then emit config like object
1315  * @param emitter a set of emitter functions
1316  * @param comments optional comments for the parser
1317  * @return dump of an object (must be freed after using) or NULL in case of error
1318  */
1319 UCL_EXTERN bool ucl_object_emit_full (const ucl_object_t *obj,
1320 		enum ucl_emitter emit_type,
1321 		struct ucl_emitter_functions *emitter,
1322 		const ucl_object_t *comments);
1323 
1324 /**
1325  * Start streamlined UCL object emitter
1326  * @param obj top UCL object
1327  * @param emit_type emit type
1328  * @param emitter a set of emitter functions
1329  * @return new streamlined context that should be freed by
1330  * `ucl_object_emit_streamline_finish`
1331  */
1332 UCL_EXTERN struct ucl_emitter_context* ucl_object_emit_streamline_new (
1333 		const ucl_object_t *obj, enum ucl_emitter emit_type,
1334 		struct ucl_emitter_functions *emitter);
1335 
1336 /**
1337  * Start object or array container for the streamlined output
1338  * @param ctx streamlined context
1339  * @param obj container object
1340  */
1341 UCL_EXTERN void ucl_object_emit_streamline_start_container (
1342 		struct ucl_emitter_context *ctx, const ucl_object_t *obj);
1343 /**
1344  * Add a complete UCL object to streamlined output
1345  * @param ctx streamlined context
1346  * @param obj object to output
1347  */
1348 UCL_EXTERN void ucl_object_emit_streamline_add_object (
1349 		struct ucl_emitter_context *ctx, const ucl_object_t *obj);
1350 /**
1351  * End previously added container
1352  * @param ctx streamlined context
1353  */
1354 UCL_EXTERN void ucl_object_emit_streamline_end_container (
1355 		struct ucl_emitter_context *ctx);
1356 /**
1357  * Terminate streamlined container finishing all containers in it
1358  * @param ctx streamlined context
1359  */
1360 UCL_EXTERN void ucl_object_emit_streamline_finish (
1361 		struct ucl_emitter_context *ctx);
1362 
1363 /**
1364  * Returns functions to emit object to memory
1365  * @param pmem target pointer (should be freed by caller)
1366  * @return emitter functions structure
1367  */
1368 UCL_EXTERN struct ucl_emitter_functions* ucl_object_emit_memory_funcs (
1369 		void **pmem);
1370 
1371 /**
1372  * Returns functions to emit object to FILE *
1373  * @param fp FILE * object
1374  * @return emitter functions structure
1375  */
1376 UCL_EXTERN struct ucl_emitter_functions* ucl_object_emit_file_funcs (
1377 		FILE *fp);
1378 /**
1379  * Returns functions to emit object to a file descriptor
1380  * @param fd file descriptor
1381  * @return emitter functions structure
1382  */
1383 UCL_EXTERN struct ucl_emitter_functions* ucl_object_emit_fd_funcs (
1384 		int fd);
1385 
1386 /**
1387  * Free emitter functions
1388  * @param f pointer to functions
1389  */
1390 UCL_EXTERN void ucl_object_emit_funcs_free (struct ucl_emitter_functions *f);
1391 
1392 /** @} */
1393 
1394 /**
1395  * @defgroup schema Schema functions
1396  * These functions are used to validate UCL objects using json schema format
1397  *
1398  * @{
1399  */
1400 
1401 /**
1402  * Used to define UCL schema error
1403  */
1404 enum ucl_schema_error_code {
1405 	UCL_SCHEMA_OK = 0,          /**< no error */
1406 	UCL_SCHEMA_TYPE_MISMATCH,   /**< type of object is incorrect */
1407 	UCL_SCHEMA_INVALID_SCHEMA,  /**< schema is invalid */
1408 	UCL_SCHEMA_MISSING_PROPERTY,/**< one or more missing properties */
1409 	UCL_SCHEMA_CONSTRAINT,      /**< constraint found */
1410 	UCL_SCHEMA_MISSING_DEPENDENCY, /**< missing dependency */
1411 	UCL_SCHEMA_EXTERNAL_REF_MISSING, /**< cannot fetch external ref */
1412 	UCL_SCHEMA_EXTERNAL_REF_INVALID, /**< invalid external ref */
1413 	UCL_SCHEMA_INTERNAL_ERROR, /**< something bad happened */
1414 	UCL_SCHEMA_UNKNOWN          /**< generic error */
1415 };
1416 
1417 /**
1418  * Generic ucl schema error
1419  */
1420 struct ucl_schema_error {
1421 	enum ucl_schema_error_code code;	/**< error code */
1422 	char msg[128];						/**< error message */
1423 	const ucl_object_t *obj;			/**< object where error occured */
1424 };
1425 
1426 /**
1427  * Validate object `obj` using schema object `schema`.
1428  * @param schema schema object
1429  * @param obj object to validate
1430  * @param err error pointer, if this parameter is not NULL and error has been
1431  * occured, then `err` is filled with the exact error definition.
1432  * @return true if `obj` is valid using `schema`
1433  */
1434 UCL_EXTERN bool ucl_object_validate (const ucl_object_t *schema,
1435 		const ucl_object_t *obj, struct ucl_schema_error *err);
1436 
1437 /**
1438  * Validate object `obj` using schema object `schema` and root schema at `root`.
1439  * @param schema schema object
1440  * @param obj object to validate
1441  * @param root root schema object
1442  * @param err error pointer, if this parameter is not NULL and error has been
1443  * occured, then `err` is filled with the exact error definition.
1444  * @return true if `obj` is valid using `schema`
1445  */
1446 UCL_EXTERN bool ucl_object_validate_root (const ucl_object_t *schema,
1447 		const ucl_object_t *obj,
1448 		const ucl_object_t *root,
1449 		struct ucl_schema_error *err);
1450 
1451 /**
1452  * Validate object `obj` using schema object `schema` and root schema at `root`
1453  * using some external references provided.
1454  * @param schema schema object
1455  * @param obj object to validate
1456  * @param root root schema object
1457  * @param ext_refs external references (might be modified during validation)
1458  * @param err error pointer, if this parameter is not NULL and error has been
1459  * occured, then `err` is filled with the exact error definition.
1460  * @return true if `obj` is valid using `schema`
1461  */
1462 UCL_EXTERN bool ucl_object_validate_root_ext (const ucl_object_t *schema,
1463 		const ucl_object_t *obj,
1464 		const ucl_object_t *root,
1465 		ucl_object_t *ext_refs,
1466 		struct ucl_schema_error *err);
1467 
1468 /** @} */
1469 
1470 #ifdef  __cplusplus
1471 }
1472 #endif
1473 /*
1474  * XXX: Poorly named API functions, need to replace them with the appropriate
1475  * named function. All API functions *must* use naming ucl_object_*. Usage of
1476  * ucl_obj* should be avoided.
1477  */
1478 #define ucl_obj_todouble_safe ucl_object_todouble_safe
1479 #define ucl_obj_todouble ucl_object_todouble
1480 #define ucl_obj_tostring ucl_object_tostring
1481 #define ucl_obj_tostring_safe ucl_object_tostring_safe
1482 #define ucl_obj_tolstring ucl_object_tolstring
1483 #define ucl_obj_tolstring_safe ucl_object_tolstring_safe
1484 #define ucl_obj_toint ucl_object_toint
1485 #define ucl_obj_toint_safe ucl_object_toint_safe
1486 #define ucl_obj_toboolean ucl_object_toboolean
1487 #define ucl_obj_toboolean_safe ucl_object_toboolean_safe
1488 #define ucl_obj_get_key ucl_object_find_key
1489 #define ucl_obj_get_keyl ucl_object_find_keyl
1490 #define ucl_obj_unref ucl_object_unref
1491 #define ucl_obj_ref ucl_object_ref
1492 #define ucl_obj_free ucl_object_free
1493 
1494 #endif /* UCL_H_ */
1495