mirror of
				git://git.psyc.eu/libpsyc
				synced 2024-08-15 03:19:02 +00:00 
			
		
		
		
	dict syntax; index & update mod parser; psyc-mode for emacs
This commit is contained in:
		
							parent
							
								
									3ee2f0e392
								
							
						
					
					
						commit
						d3aa4508b9
					
				
					 31 changed files with 2063 additions and 691 deletions
				
			
		
							
								
								
									
										4
									
								
								.gitignore
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.gitignore
									
										
									
									
										vendored
									
									
								
							|  | @ -19,7 +19,9 @@ test/test_strlen | ||||||
| test/test_text | test/test_text | ||||||
| test/test_table | test/test_table | ||||||
| test/test_packet_id | test/test_packet_id | ||||||
| test/var_is_routing | test/test_index | ||||||
|  | test/test_update | ||||||
|  | test/var_routing | ||||||
| test/var_type | test/var_type | ||||||
| test/uniform_parse | test/uniform_parse | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
							
								
								
									
										124
									
								
								emacs/psyc.el
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										124
									
								
								emacs/psyc.el
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,124 @@ | ||||||
|  | (require 'font-lock) | ||||||
|  | 
 | ||||||
|  | (setq | ||||||
|  |  psyc-op "\\([:!?=$@+-]\\)" | ||||||
|  |  psyc-routing-op "^\\([:=]\\)" | ||||||
|  |  psyc-state-op "^\\([?=]\\)$" | ||||||
|  | 
 | ||||||
|  |  psyc-num "\\([0-9]+\\)" | ||||||
|  |  psyc-kw "\\([a-z_][a-z0-9_]*\\)" | ||||||
|  | 
 | ||||||
|  |  psyc-length "^[0-9]+$" | ||||||
|  | 
 | ||||||
|  |  psyc-routing-var-names (regexp-opt '("_amount_fragments" "_context" "_counter" | ||||||
|  |                                       "_fragment" "_source" "_source_relay" | ||||||
|  |                                       "_target" "_target_relay" | ||||||
|  |                                       "_tag" "_tag_relay") 'word) | ||||||
|  |  psyc-routing-var (concat psyc-routing-op psyc-routing-var-names) | ||||||
|  | 
 | ||||||
|  |  psyc-types (concat "\\([a-z][a-z0-9]+\\|" | ||||||
|  |                     (regexp-opt '("_dict" "_list" "_struct" | ||||||
|  |                                   "_int" "_uint" "_num" "_unum" "_flag" "_amount" | ||||||
|  |                                   "_date" "_time" "_uniform") nil) | ||||||
|  |                     "\\)") | ||||||
|  | 
 | ||||||
|  |  psyc-mod-op (concat "^" psyc-op) | ||||||
|  |  psyc-mod-name (concat psyc-mod-op psyc-kw) ;2 | ||||||
|  |  psyc-mod-type (concat psyc-mod-op psyc-types "\\b") ;2 | ||||||
|  |  psyc-mod-len (concat psyc-mod-op psyc-kw " " psyc-num) ;3 | ||||||
|  |  psyc-mod-delim (concat "\\(?: " psyc-num "\\)?[\t ]+") | ||||||
|  |  psyc-mod (concat psyc-mod-op psyc-kw psyc-mod-delim) ;3 | ||||||
|  | 
 | ||||||
|  |  psyc-default-name (concat psyc-mod psyc-kw "[|{]") ;4 | ||||||
|  |  psyc-default-type (concat psyc-mod psyc-types "[_|{]") ;4 | ||||||
|  | ; psyc-default-name (concat "[\t ]" psyc-kw "[|{]") | ||||||
|  | ; psyc-default-type (concat "[\t ]" psyc-types "[_|{]") | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |  psyc-elem-delim "[|{}]" | ||||||
|  |  psyc-elem-start "[|}]" | ||||||
|  |  psyc-elem-op (concat psyc-elem-start "\\(=\\)") | ||||||
|  |  psyc-elem-name (concat psyc-elem-start "=" psyc-kw) | ||||||
|  |  psyc-elem-type (concat psyc-elem-start "=" psyc-types "\\b") | ||||||
|  |  psyc-elem-name-delim (concat psyc-elem-name "\\(:\\)") ;2 | ||||||
|  |  psyc-elem-len (concat "\\(?:" psyc-elem-name-delim "\\)?" psyc-num) ;3 | ||||||
|  | 
 | ||||||
|  |  psyc-update-op (concat "^@" psyc-kw psyc-mod-delim ".+ " psyc-op) ;3 | ||||||
|  |  psyc-update-name (concat "^@" psyc-kw psyc-mod-delim ".+ " | ||||||
|  |                           psyc-op psyc-kw) ;4 | ||||||
|  |  psyc-update-type (concat "^@" psyc-kw psyc-mod-delim ".+ " | ||||||
|  |                           psyc-op psyc-types "\\b") ;4 | ||||||
|  |  psyc-update-len-delim (concat "^@" psyc-kw psyc-mod-delim ".+ " | ||||||
|  |                                psyc-op psyc-kw "?\\(:\\)") ;5 | ||||||
|  |  psyc-update-len (concat "^@" psyc-kw psyc-mod-delim ".+ " | ||||||
|  |                          psyc-op psyc-kw "?:" psyc-num) ;5 | ||||||
|  | 
 | ||||||
|  |  psyc-index-delim "\\(#\\)-?[0-9]+" | ||||||
|  | 
 | ||||||
|  |  psyc-struct-delim "[\t0-9}]\\(\\.\\)" | ||||||
|  |  psyc-struct-name (concat psyc-struct-delim psyc-kw) ;2 | ||||||
|  |  psyc-struct-type (concat psyc-struct-delim psyc-types "\\b") ;2 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |  psyc-dict-key-len (concat "{" psyc-num) | ||||||
|  |  psyc-dict-key (concat "{\\(?:" psyc-num " \\)?\\([^}]+\\)") ;2 | ||||||
|  | 
 | ||||||
|  |  psyc-method (concat "^_" psyc-kw) | ||||||
|  |  ;psyc-body "^[^_:!?=$@+-].*$" | ||||||
|  | 
 | ||||||
|  |  psyc-tmpl-start (concat "\\(\\[\\)" psyc-kw ".+?\\]") | ||||||
|  |  psyc-tmpl-end (concat "\\[" psyc-kw ".+?\\(\\]\\)") | ||||||
|  | 
 | ||||||
|  |  psyc-packet-delim "^|$" | ||||||
|  | 
 | ||||||
|  |  psyc-font-lock-keywords | ||||||
|  |  `( | ||||||
|  |    (,psyc-routing-var	. (2 font-lock-keyword-face t)) | ||||||
|  |    (,psyc-length	. (0 font-lock-constant-face)) | ||||||
|  |    (,psyc-state-op	. (0 font-lock-preprocessor-face)) | ||||||
|  |    (,psyc-mod-op	. (1 font-lock-preprocessor-face)) | ||||||
|  |    (,psyc-mod-name	. (2 font-lock-variable-name-face)) | ||||||
|  |    (,psyc-mod-type	. (2 font-lock-type-face t)) | ||||||
|  |    (,psyc-mod-len	. (3 font-lock-constant-face)) | ||||||
|  | 
 | ||||||
|  |    (,psyc-default-name	. (4 font-lock-variable-name-face)) | ||||||
|  |    (,psyc-default-type	. (4 font-lock-type-face t)) | ||||||
|  | 
 | ||||||
|  |    (,psyc-dict-key-len	. (1 font-lock-constant-face)) | ||||||
|  |    (,psyc-dict-key	. (2 font-lock-string-face)) | ||||||
|  | 
 | ||||||
|  |    (,psyc-elem-op	. (1 font-lock-preprocessor-face)) | ||||||
|  |    (,psyc-elem-name	. (1 font-lock-variable-name-face)) | ||||||
|  |    (,psyc-elem-type	. (1 font-lock-type-face t)) | ||||||
|  |    (,psyc-elem-name-delim . (2 font-lock-comment-face)) | ||||||
|  |    (,psyc-elem-len	. (3 font-lock-constant-face)) | ||||||
|  |    (,psyc-elem-delim	. (0 font-lock-keyword-face)) | ||||||
|  | 
 | ||||||
|  |    (,psyc-update-op	. (3 font-lock-preprocessor-face)) | ||||||
|  |    (,psyc-update-name	. (4 font-lock-variable-name-face)) | ||||||
|  |    (,psyc-update-type	. (4 font-lock-type-face t)) | ||||||
|  |    (,psyc-update-len-delim . (5 font-lock-comment-face)) | ||||||
|  |    (,psyc-update-len	. (5 font-lock-constant-face)) | ||||||
|  | 
 | ||||||
|  |    (,psyc-method	. (0 font-lock-function-name-face)) | ||||||
|  |    ;(,psyc-body		. (0 font-lock-comment-face)) | ||||||
|  | 
 | ||||||
|  |    (,psyc-index-delim	. (1 font-lock-comment-face)) | ||||||
|  |    (,psyc-struct-name	. (1 font-lock-comment-face)) | ||||||
|  |    (,psyc-struct-name	. (2 font-lock-variable-name-face)) | ||||||
|  |    (,psyc-struct-type	. (2 font-lock-type-face t)) | ||||||
|  | 
 | ||||||
|  |    (,psyc-tmpl-start	. (1 font-lock-keyword-face)) | ||||||
|  |    (,psyc-tmpl-end	. (2 font-lock-keyword-face)) | ||||||
|  | 
 | ||||||
|  |    (,psyc-packet-delim	. (0 font-lock-preprocessor-face)) | ||||||
|  |    )) | ||||||
|  | 
 | ||||||
|  | (define-derived-mode psyc-mode fundamental-mode | ||||||
|  |   "PSYC" | ||||||
|  |   "Major mode for editing PSYC packets" | ||||||
|  | 
 | ||||||
|  |   (setq font-lock-defaults '((psyc-font-lock-keywords))) | ||||||
|  |   (setq show-trailing-whitespace t)) | ||||||
|  | 
 | ||||||
|  | (provide 'psyc) | ||||||
|  | @ -24,7 +24,6 @@ | ||||||
| #define PSYC_VERSION 1 | #define PSYC_VERSION 1 | ||||||
| #define PSYC_EPOCH 1440444041 // 2015-08-24 21:20:41 CET (Monday)
 | #define PSYC_EPOCH 1440444041 // 2015-08-24 21:20:41 CET (Monday)
 | ||||||
| 
 | 
 | ||||||
| #define PSYC_STRING(data, len) (PsycString) {len, data} |  | ||||||
| #define PSYC_C2STR(str)  (PsycString) {sizeof(str)-1, str} | #define PSYC_C2STR(str)  (PsycString) {sizeof(str)-1, str} | ||||||
| #define PSYC_C2STRI(str) {sizeof(str)-1, str} | #define PSYC_C2STRI(str) {sizeof(str)-1, str} | ||||||
| #define PSYC_C2ARG(str)  str, sizeof(str)-1 | #define PSYC_C2ARG(str)  str, sizeof(str)-1 | ||||||
|  | @ -50,53 +49,6 @@ typedef enum { | ||||||
|     PSYC_ERROR = -1, |     PSYC_ERROR = -1, | ||||||
| } PsycRC; | } PsycRC; | ||||||
| 
 | 
 | ||||||
| /// PSYC packet parts.
 |  | ||||||
| typedef enum { |  | ||||||
|     PSYC_PART_RESET = -1, |  | ||||||
|     PSYC_PART_ROUTING = 0, |  | ||||||
|     PSYC_PART_LENGTH = 1, |  | ||||||
|     PSYC_PART_CONTENT = 2, |  | ||||||
|     PSYC_PART_METHOD = 3, |  | ||||||
|     PSYC_PART_DATA = 4, |  | ||||||
|     PSYC_PART_END = 5, |  | ||||||
| } PsycPart; |  | ||||||
| 
 |  | ||||||
| /**
 |  | ||||||
|  * Different types that a variable can have. |  | ||||||
|  * |  | ||||||
|  * This enum lists PSYC variable types that |  | ||||||
|  * this library is capable of checking for |  | ||||||
|  * validity. Other variable types are treated |  | ||||||
|  * as opaque data. |  | ||||||
|  */ |  | ||||||
| typedef enum { |  | ||||||
|     PSYC_TYPE_UNKNOWN, |  | ||||||
|     PSYC_TYPE_AMOUNT, |  | ||||||
|     PSYC_TYPE_COLOR, |  | ||||||
|     PSYC_TYPE_COUNTER, |  | ||||||
|     PSYC_TYPE_DEF, |  | ||||||
|     PSYC_TYPE_DATE, |  | ||||||
|     PSYC_TYPE_DEGREE, |  | ||||||
|     PSYC_TYPE_ENTITY, |  | ||||||
|     PSYC_TYPE_FLAG, |  | ||||||
|     PSYC_TYPE_LANGUAGE, |  | ||||||
|     PSYC_TYPE_LIST, |  | ||||||
|     PSYC_TYPE_NICK, |  | ||||||
|     PSYC_TYPE_PAGE, |  | ||||||
|     PSYC_TYPE_TABLE, |  | ||||||
|     PSYC_TYPE_TIME, |  | ||||||
|     PSYC_TYPE_UNIFORM, |  | ||||||
| } PsycType; |  | ||||||
| 
 |  | ||||||
| /**
 |  | ||||||
|  * List types. |  | ||||||
|  * Possible types are text and binary. |  | ||||||
|  */ |  | ||||||
| typedef enum { |  | ||||||
|     PSYC_LIST_TEXT = 1, |  | ||||||
|     PSYC_LIST_BINARY = 2, |  | ||||||
| } PsycListType; |  | ||||||
| 
 |  | ||||||
| /**
 | /**
 | ||||||
|  * String struct. |  * String struct. | ||||||
|  * |  * | ||||||
|  | @ -109,14 +61,15 @@ typedef struct { | ||||||
|     char *data; |     char *data; | ||||||
| } PsycString; | } PsycString; | ||||||
| 
 | 
 | ||||||
| #include "psyc/syntax.h" | #define PSYC_STRING(data, len) (PsycString) {len, data} | ||||||
|  | 
 | ||||||
| #include "psyc/match.h" | #include "psyc/match.h" | ||||||
| #include "psyc/method.h" | #include "psyc/method.h" | ||||||
| #include "psyc/packet.h" | #include "psyc/packet.h" | ||||||
|  | #include "psyc/variable.h" | ||||||
| #include "psyc/parse.h" | #include "psyc/parse.h" | ||||||
| #include "psyc/render.h" | #include "psyc/render.h" | ||||||
| #include "psyc/text.h" | #include "psyc/text.h" | ||||||
| #include "psyc/uniform.h" | #include "psyc/uniform.h" | ||||||
| #include "psyc/variable.h" |  | ||||||
| 
 | 
 | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
|  | @ -4,7 +4,7 @@ prefix = /usr | ||||||
| includedir = ${prefix}/include | includedir = ${prefix}/include | ||||||
| 
 | 
 | ||||||
| INSTALL = install | INSTALL = install | ||||||
| HEADERS = match.h method.h packet.h parse.h render.h syntax.h text.h uniform.h variable.h | HEADERS = match.h method.h packet.h parse.h render.h text.h uniform.h variable.h | ||||||
| 
 | 
 | ||||||
| install: ${HEADERS} | install: ${HEADERS} | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1,12 +1,15 @@ | ||||||
|  | #ifndef PSYC_MATCH_H | ||||||
|  | #define PSYC_MATCH_H | ||||||
|  | 
 | ||||||
| typedef struct { | typedef struct { | ||||||
|     PsycString key; |     PsycString key; | ||||||
|     void *value; |     void *value; | ||||||
| } PsycDict; | } PsycMap; | ||||||
| 
 | 
 | ||||||
| typedef struct { | typedef struct { | ||||||
|     PsycString key; |     PsycString key; | ||||||
|     intptr_t value; |     intptr_t value; | ||||||
| } PsycDictInt; | } PsycMapInt; | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  * Checks if long keyword string inherits from short keyword string. |  * Checks if long keyword string inherits from short keyword string. | ||||||
|  | @ -21,10 +24,10 @@ int | ||||||
| psyc_matches (char *sho, size_t slen, char *lon, size_t llen); | psyc_matches (char *sho, size_t slen, char *lon, size_t llen); | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  * Look up value associated with a key in a dictionary. |  * Look up value associated with a key in a map. | ||||||
|  * |  * | ||||||
|  * @param dict The dictionary to search, should be ordered alphabetically. |  * @param map The dictionary to search, should be ordered alphabetically. | ||||||
|  * @param size Size of dict. |  * @param size Size of map. | ||||||
|  * @param key Key to look for. |  * @param key Key to look for. | ||||||
|  * @param keylen Length of key. |  * @param keylen Length of key. | ||||||
|  * @param inherit If true, also look for anything inheriting from key, |  * @param inherit If true, also look for anything inheriting from key, | ||||||
|  | @ -34,16 +37,18 @@ psyc_matches (char *sho, size_t slen, char *lon, size_t llen); | ||||||
|  */ |  */ | ||||||
| 
 | 
 | ||||||
| void * | void * | ||||||
| psyc_dict_lookup (const PsycDict *dict, size_t size, | psyc_map_lookup (const PsycMap *map, size_t size, | ||||||
| 		  const char *key, size_t keylen, PsycBool inherit); | 		 const char *key, size_t keylen, PsycBool inherit); | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  * Look up value associated with a key in a dictionary of integers. |  * Look up value associated with a key in a map with integer values. | ||||||
|  * @see psyc_dict_lookup |  * @see psyc_map_lookup | ||||||
|  */ |  */ | ||||||
| static inline intptr_t | static inline intptr_t | ||||||
| psyc_dict_lookup_int (const PsycDictInt * dict, size_t size, | psyc_map_lookup_int (const PsycMapInt * map, size_t size, | ||||||
| 		      const char *key, size_t keylen, PsycBool inherit) | 		      const char *key, size_t keylen, PsycBool inherit) | ||||||
| { | { | ||||||
|     return (intptr_t) psyc_dict_lookup((PsycDict *) dict, size, key, keylen, inherit); |     return (intptr_t) psyc_map_lookup((PsycMap *) map, size, key, keylen, inherit); | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | @ -18,9 +18,53 @@ | ||||||
| 
 | 
 | ||||||
| #include <math.h> | #include <math.h> | ||||||
| 
 | 
 | ||||||
| #include "syntax.h" |  | ||||||
| #include "method.h" | #include "method.h" | ||||||
| 
 | 
 | ||||||
|  | /**
 | ||||||
|  |  * Maximum allowed content size without length. | ||||||
|  |  * Parser stops after reaching this limit. | ||||||
|  |  */ | ||||||
|  | #define PSYC_CONTENT_SIZE_MAX 512 | ||||||
|  | /**
 | ||||||
|  |  * Content size after which a length is added when rendering. | ||||||
|  |  */ | ||||||
|  | #ifndef PSYC_CONTENT_SIZE_THRESHOLD | ||||||
|  | # define PSYC_CONTENT_SIZE_THRESHOLD 9 | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Maximum allowed modifier size without length. | ||||||
|  |  * Parser stops after reaching this limit. | ||||||
|  |  */ | ||||||
|  | #define PSYC_MODIFIER_SIZE_MAX 256 | ||||||
|  | /**
 | ||||||
|  |  * Modifier size after which a length is added when rendering. | ||||||
|  |  */ | ||||||
|  | #ifndef PSYC_MODIFIER_SIZE_THRESHOLD | ||||||
|  | # define PSYC_MODIFIER_SIZE_THRESHOLD 9 | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Maximum allowed element size without length. | ||||||
|  |  * Parser stops after reaching this limit. | ||||||
|  |  */ | ||||||
|  | #define PSYC_ELEM_SIZE_MAX 128 | ||||||
|  | /**
 | ||||||
|  |  * Element size after which a length is added when rendering. | ||||||
|  |  */ | ||||||
|  | #ifndef PSYC_ELEM_SIZE_THRESHOLD | ||||||
|  | # define PSYC_ELEM_SIZE_THRESHOLD 9 | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | #define PSYC_PACKET_DELIMITER_CHAR '|' | ||||||
|  | #define PSYC_PACKET_DELIMITER	   "\n|\n" | ||||||
|  | 
 | ||||||
|  | #define PSYC_LIST_ELEM_START '|' | ||||||
|  | #define PSYC_DICT_KEY_START '{' | ||||||
|  | #define PSYC_DICT_KEY_END '}' | ||||||
|  | #define PSYC_DICT_VALUE_START PSYC_DICT_KEY_END | ||||||
|  | #define PSYC_DICT_VALUE_END PSYC_DICT_KEY_START | ||||||
|  | 
 | ||||||
| /** Modifier flags. */ | /** Modifier flags. */ | ||||||
| typedef enum { | typedef enum { | ||||||
|     /// Modifier needs to be checked if it needs length.
 |     /// Modifier needs to be checked if it needs length.
 | ||||||
|  | @ -30,18 +74,18 @@ typedef enum { | ||||||
|     /// Modifier doesn't need length.
 |     /// Modifier doesn't need length.
 | ||||||
|     PSYC_MODIFIER_NO_LENGTH = 2, |     PSYC_MODIFIER_NO_LENGTH = 2, | ||||||
|     /// Routing modifier, which implies that it doesn't need length.
 |     /// Routing modifier, which implies that it doesn't need length.
 | ||||||
|     PSYC_MODIFIER_ROUTING = 3, |     PSYC_MODIFIER_ROUTING = 4, | ||||||
| } PsycModifierFlag; | } PsycModifierFlag; | ||||||
| 
 | 
 | ||||||
| /** List flags. */ | /** List/dict element flags. */ | ||||||
| typedef enum { | typedef enum { | ||||||
|     /// List needs to be checked if it needs length.
 |     /// Element needs to be checked if it needs length.
 | ||||||
|     PSYC_LIST_CHECK_LENGTH = 0, |     PSYC_ELEM_CHECK_LENGTH = 0, | ||||||
|     /// List needs length.
 |     /// Element needs length.
 | ||||||
|     PSYC_LIST_NEED_LENGTH = 1, |     PSYC_ELEM_NEED_LENGTH = 1, | ||||||
|     /// List doesn't need length.
 |     /// Element doesn't need length.
 | ||||||
|     PSYC_LIST_NO_LENGTH = 2, |     PSYC_ELEM_NO_LENGTH = 2, | ||||||
| } PsycListFlag; | } PsycElemFlag; | ||||||
| 
 | 
 | ||||||
| /** Packet flags. */ | /** Packet flags. */ | ||||||
| typedef enum { | typedef enum { | ||||||
|  | @ -77,35 +121,90 @@ typedef enum { | ||||||
|     PSYC_PACKET_ID_ELEMS = 5, |     PSYC_PACKET_ID_ELEMS = 5, | ||||||
| } PsycPacketId; | } PsycPacketId; | ||||||
| 
 | 
 | ||||||
| /** Structure for a modifier. */ |  | ||||||
| typedef struct { | typedef struct { | ||||||
|     char oper; |     PsycString type; | ||||||
|  |     PsycString value; | ||||||
|  |     size_t length; | ||||||
|  |     PsycElemFlag flag; | ||||||
|  | } PsycElem; | ||||||
|  | 
 | ||||||
|  | #define PSYC_ELEM(typ, typlen, val, vallen, flg)	\ | ||||||
|  |     (PsycElem) {					\ | ||||||
|  | 	.type = PSYC_STRING(typ, typlen),		\ | ||||||
|  | 	.value = PSYC_STRING(val, vallen),		\ | ||||||
|  | 	.flag = flg,					\ | ||||||
|  |      } | ||||||
|  | #define PSYC_ELEM_TV(typ, typlen, val, vallen)		\ | ||||||
|  |     (PsycElem) {					\ | ||||||
|  | 	.type = PSYC_STRING(typ, typlen),		\ | ||||||
|  | 	.value = PSYC_STRING(val, vallen),		\ | ||||||
|  |      } | ||||||
|  | #define PSYC_ELEM_VF(val, vallen, flg)			\ | ||||||
|  |     (PsycElem) {					\ | ||||||
|  | 	.value = PSYC_STRING(val, vallen),		\ | ||||||
|  | 	.flag = flg,					\ | ||||||
|  |     } | ||||||
|  | #define PSYC_ELEM_V(val, vallen)			\ | ||||||
|  |     (PsycElem) {					\ | ||||||
|  | 	.value = PSYC_STRING(val, vallen),		\ | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | /** Dict key */ | ||||||
|  | typedef struct { | ||||||
|  |     PsycString value; | ||||||
|  |     size_t length; | ||||||
|  |     PsycElemFlag flag; | ||||||
|  | } PsycDictKey; | ||||||
|  | 
 | ||||||
|  | #define PSYC_DICT_KEY(k, klen, flg)			\ | ||||||
|  |     (PsycDictKey) {					\ | ||||||
|  | 	.value = PSYC_STRING(k, klen),			\ | ||||||
|  | 	.flag = flg,					\ | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | /** Dict key/value */ | ||||||
|  | typedef struct { | ||||||
|  |     PsycElem value; | ||||||
|  |     PsycDictKey key; | ||||||
|  | } PsycDictElem; | ||||||
|  | 
 | ||||||
|  | #define PSYC_DICT_ELEM(k, v)				\ | ||||||
|  |     (PsycDictElem) {					\ | ||||||
|  | 	.key = k,					\ | ||||||
|  | 	.value = v,					\ | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | /** Dictionary */ | ||||||
|  | typedef struct { | ||||||
|  |     PsycString type; | ||||||
|  |     PsycDictElem *elems; | ||||||
|  |     size_t num_elems; | ||||||
|  |     size_t length; | ||||||
|  | } PsycDict; | ||||||
|  | 
 | ||||||
|  | /** List */ | ||||||
|  | typedef struct { | ||||||
|  |     PsycString type; | ||||||
|  |     PsycElem *elems; | ||||||
|  |     size_t num_elems; | ||||||
|  |     size_t length; | ||||||
|  | } PsycList; | ||||||
|  | 
 | ||||||
|  | /** Modifier */ | ||||||
|  | typedef struct { | ||||||
|     PsycString name; |     PsycString name; | ||||||
|     PsycString value; |     PsycString value; | ||||||
|     PsycModifierFlag flag; |     PsycModifierFlag flag; | ||||||
|  |     char oper; | ||||||
| } PsycModifier; | } PsycModifier; | ||||||
| 
 | 
 | ||||||
| /** Structure for an entity or routing header. */ | /** Entity or routing header */ | ||||||
| typedef struct { | typedef struct { | ||||||
|     size_t lines; |     size_t lines; | ||||||
|     PsycModifier *modifiers; |     PsycModifier *modifiers; | ||||||
| } PsycHeader; | } PsycHeader; | ||||||
| 
 | 
 | ||||||
| /** Structure for a list. */ | /** Intermediate struct for a PSYC packet. */ | ||||||
| typedef struct { |  | ||||||
|     size_t num_elems; |  | ||||||
|     PsycString *elems; |  | ||||||
|     size_t length; |  | ||||||
|     PsycListFlag flag; |  | ||||||
| } PsycList; |  | ||||||
| 
 |  | ||||||
| typedef struct { |  | ||||||
|     PsycList *list; |  | ||||||
|     size_t width; |  | ||||||
|     size_t length; |  | ||||||
| } PsycTable; |  | ||||||
| 
 |  | ||||||
| /** Intermediate struct for a PSYC packet */ |  | ||||||
| typedef struct { | typedef struct { | ||||||
|     PsycHeader routing;		///< Routing header.
 |     PsycHeader routing;		///< Routing header.
 | ||||||
|     PsycHeader entity;		///< Entity header.
 |     PsycHeader entity;		///< Entity header.
 | ||||||
|  | @ -135,16 +234,12 @@ psyc_num_length (size_t n) | ||||||
| static inline PsycModifierFlag | static inline PsycModifierFlag | ||||||
| psyc_modifier_length_check (PsycModifier *m) | psyc_modifier_length_check (PsycModifier *m) | ||||||
| { | { | ||||||
|     PsycModifierFlag flag; |     if (m->value.length > 0 | ||||||
|  | 	&& (m->value.length > PSYC_MODIFIER_SIZE_THRESHOLD | ||||||
|  | 	    || memchr(m->value.data, (int) '\n', m->value.length))) | ||||||
|  | 	return PSYC_MODIFIER_NEED_LENGTH; | ||||||
| 
 | 
 | ||||||
|     if (m->value.length > PSYC_MODIFIER_SIZE_THRESHOLD) |     return PSYC_MODIFIER_NO_LENGTH; | ||||||
| 	flag = PSYC_MODIFIER_NEED_LENGTH; |  | ||||||
|     else if (memchr(m->value.data, (int) '\n', m->value.length)) |  | ||||||
| 	flag = PSYC_MODIFIER_NEED_LENGTH; |  | ||||||
|     else |  | ||||||
| 	flag = PSYC_MODIFIER_NO_LENGTH; |  | ||||||
| 
 |  | ||||||
|     return flag; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /** Initialize modifier */ | /** Initialize modifier */ | ||||||
|  | @ -153,33 +248,61 @@ psyc_modifier_init (PsycModifier *m, PsycOperator oper, | ||||||
| 		    char *name, size_t namelen, | 		    char *name, size_t namelen, | ||||||
| 		    char *value, size_t valuelen, PsycModifierFlag flag) | 		    char *value, size_t valuelen, PsycModifierFlag flag) | ||||||
| { | { | ||||||
|     *m = (PsycModifier) {oper, {namelen, name}, {valuelen, value}, flag}; |     *m = (PsycModifier) { | ||||||
|  | 	.oper = oper, | ||||||
|  | 	.name = {namelen, name}, | ||||||
|  | 	.value = {valuelen, value}, | ||||||
|  | 	.flag = flag | ||||||
|  |     }; | ||||||
| 
 | 
 | ||||||
|     if (flag == PSYC_MODIFIER_CHECK_LENGTH) // find out if it needs a length
 |     if (flag == PSYC_MODIFIER_CHECK_LENGTH) // find out if it needs a length
 | ||||||
| 	m->flag = psyc_modifier_length_check(m); | 	m->flag = psyc_modifier_length_check(m); | ||||||
|  |     else if (flag & PSYC_MODIFIER_ROUTING) | ||||||
|  | 	m->flag |= PSYC_MODIFIER_NO_LENGTH; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  * \internal |  * \internal | ||||||
|  * Get the total length of a modifier when rendered. |  * Check if a list/dict element needs length. | ||||||
|  |  */ | ||||||
|  | PsycElemFlag | ||||||
|  | psyc_elem_length_check (PsycString *value, const char end); | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * \internal | ||||||
|  |  * Get the rendered length of a list/dict element. | ||||||
|  |  */ | ||||||
|  | size_t | ||||||
|  | psyc_elem_length (PsycElem *elem); | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * \internal | ||||||
|  |  * Get the rendered length of a dict key. | ||||||
|  |  */ | ||||||
|  | size_t | ||||||
|  | psyc_dict_key_length (PsycDictKey *elem); | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * \internal | ||||||
|  |  * Get the rendered length of a list. | ||||||
|  |  */ | ||||||
|  | size_t | ||||||
|  | psyc_list_length_set (PsycList *list); | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * \internal | ||||||
|  |  * Get the rendered length of a dict. | ||||||
|  |  */ | ||||||
|  | size_t | ||||||
|  | psyc_dict_length_set (PsycDict *dict); | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * \internal | ||||||
|  |  * Get the rendered length of a modifier. | ||||||
|  */ |  */ | ||||||
| size_t | size_t | ||||||
| psyc_modifier_length (PsycModifier *m); | psyc_modifier_length (PsycModifier *m); | ||||||
| 
 | 
 | ||||||
| /**
 |  | ||||||
|  * \internal |  | ||||||
|  * Check if a list needs length. |  | ||||||
|  */ |  | ||||||
| PsycListFlag |  | ||||||
| psyc_list_length_check (PsycList *list); |  | ||||||
| 
 |  | ||||||
| /**
 |  | ||||||
|  * \internal |  | ||||||
|  * Get the total length of a list when rendered. |  | ||||||
|  */ |  | ||||||
| PsycListFlag |  | ||||||
| psyc_list_length (PsycList *list); |  | ||||||
| 
 |  | ||||||
| /**
 | /**
 | ||||||
|  * \internal |  * \internal | ||||||
|  * Check if a packet needs length. |  * Check if a packet needs length. | ||||||
|  | @ -195,12 +318,11 @@ psyc_packet_length_set (PsycPacket *p); | ||||||
| 
 | 
 | ||||||
| /** Initialize a list. */ | /** Initialize a list. */ | ||||||
| void | void | ||||||
| psyc_list_init (PsycList *list, PsycString *elems, size_t num_elems, | psyc_list_init (PsycList *list, PsycElem *elems, size_t num_elems); | ||||||
| 		PsycListFlag flag); |  | ||||||
| 
 | 
 | ||||||
| /** Initialize a table. */ | /** Initialize a dict. */ | ||||||
| void | void | ||||||
| psyc_table_init (PsycTable *table, size_t width, PsycList *list); | psyc_dict_init (PsycDict *dict, PsycDictElem *elems, size_t num_elems); | ||||||
| 
 | 
 | ||||||
| /** Initialize a packet. */ | /** Initialize a packet. */ | ||||||
| void | void | ||||||
|  | @ -218,10 +340,13 @@ psyc_packet_init_raw (PsycPacket *packet, | ||||||
| 		      char *content, size_t contentlen, | 		      char *content, size_t contentlen, | ||||||
| 		      PsycPacketFlag flag); | 		      PsycPacketFlag flag); | ||||||
| 
 | 
 | ||||||
| /** Get the total length of a packet ID when rendered. */ | void | ||||||
| size_t | psyc_packet_id (PsycList *list, PsycElem *elems, | ||||||
| psyc_packet_id_length (size_t contextlen, size_t sourcelen, size_t targetelen, | 		char *context, size_t contextlen, | ||||||
| 		       size_t counterlen, size_t fragmentlen); | 		char *source, size_t sourcelen, | ||||||
|  | 		char *target, size_t targetlen, | ||||||
|  | 		char *counter, size_t counterlen, | ||||||
|  | 		char *fragment, size_t fragmentlen);  | ||||||
| 
 | 
 | ||||||
| /** @} */ // end of packet group
 | /** @} */ // end of packet group
 | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -132,6 +132,10 @@ typedef enum { | ||||||
|  * @see psyc_parse() |  * @see psyc_parse() | ||||||
|  */ |  */ | ||||||
| typedef enum { | typedef enum { | ||||||
|  |     /// Error, no length is set for a modifier which is longer than PSYC_MODIFIER_SIZE_THRESHOLD.
 | ||||||
|  |     PSYC_PARSE_ERROR_MOD_NO_LEN = -10, | ||||||
|  |     /// Error, no length is set for the content but it is longer than PSYC_CONTENT_SIZE_THRESHOLD.
 | ||||||
|  |     PSYC_PARSE_ERROR_NO_LEN = -9, | ||||||
|     /// Error, packet is not ending with a valid delimiter.
 |     /// Error, packet is not ending with a valid delimiter.
 | ||||||
|     PSYC_PARSE_ERROR_END = -8, |     PSYC_PARSE_ERROR_END = -8, | ||||||
|     /// Error, expected NL after the method.
 |     /// Error, expected NL after the method.
 | ||||||
|  | @ -197,106 +201,286 @@ typedef enum { | ||||||
|     PSYC_PARSE_COMPLETE = 13, |     PSYC_PARSE_COMPLETE = 13, | ||||||
| } PsycParseRC; | } PsycParseRC; | ||||||
| 
 | 
 | ||||||
|  | /// PSYC packet parts.
 | ||||||
|  | typedef enum { | ||||||
|  |     PSYC_PART_RESET = -1, | ||||||
|  |     PSYC_PART_ROUTING = 0, | ||||||
|  |     PSYC_PART_LENGTH = 1, | ||||||
|  |     PSYC_PART_CONTENT = 2, | ||||||
|  |     PSYC_PART_METHOD = 3, | ||||||
|  |     PSYC_PART_DATA = 4, | ||||||
|  |     PSYC_PART_END = 5, | ||||||
|  | } PsycPart; | ||||||
|  | 
 | ||||||
| /**
 | /**
 | ||||||
|  * The return value definitions for the list parsing function. |  * The return value definitions for the list parsing function. | ||||||
|  * @see psyc_parse_list() |  * @see psyc_parse_list() | ||||||
|  */ |  */ | ||||||
| typedef enum { | typedef enum { | ||||||
|     PSYC_PARSE_LIST_ERROR_DELIM = -4, |     /// Error, no length is set for an element which is longer than PSYC_ELEM_SIZE_THRESHOLD.
 | ||||||
|     PSYC_PARSE_LIST_ERROR_LEN = -3, |     PSYC_PARSE_LIST_ERROR_ELEM_NO_LEN = -6, | ||||||
|  |     PSYC_PARSE_LIST_ERROR_ELEM_LENGTH = -5, | ||||||
|  |     PSYC_PARSE_LIST_ERROR_ELEM_TYPE = -4, | ||||||
|  |     PSYC_PARSE_LIST_ERROR_ELEM_START = -3, | ||||||
|     PSYC_PARSE_LIST_ERROR_TYPE = -2, |     PSYC_PARSE_LIST_ERROR_TYPE = -2, | ||||||
|     PSYC_PARSE_LIST_ERROR = -1, |     PSYC_PARSE_LIST_ERROR = -1, | ||||||
|     /// Completed parsing a list element.
 |  | ||||||
|     PSYC_PARSE_LIST_ELEM = 1, |  | ||||||
|     /// Reached end of buffer.
 |     /// Reached end of buffer.
 | ||||||
|     PSYC_PARSE_LIST_END = 2, |     /// Buffer contains insufficient amount of data.
 | ||||||
|     /// Binary list is incomplete.
 |     /// Fill another buffer and concatenate it with the end of the current buffer,
 | ||||||
|     PSYC_PARSE_LIST_INCOMPLETE = 3, |     /// from the cursor position to the end.
 | ||||||
|  |     PSYC_PARSE_LIST_INSUFFICIENT = 1, | ||||||
|  |     /// Completed parsing the default type of the list.
 | ||||||
|  |     PSYC_PARSE_LIST_TYPE = 2, | ||||||
|  |     /// Start of an element is parsed but still not complete.
 | ||||||
|  |     PSYC_PARSE_LIST_ELEM_START = 3, | ||||||
|  |     /// Continuation of an incomplete element.
 | ||||||
|  |     PSYC_PARSE_LIST_ELEM_CONT = 4, | ||||||
|  |     /// Element parsing completed.
 | ||||||
|  |     PSYC_PARSE_LIST_ELEM_END = 5, | ||||||
|  |     /// Completed parsing a list element.
 | ||||||
|  |     PSYC_PARSE_LIST_ELEM = 6, | ||||||
|  |     /// Completed parsing the last element in the list.
 | ||||||
|  |     PSYC_PARSE_LIST_ELEM_LAST = 7, | ||||||
|  |     /// Reached end of buffer.
 | ||||||
|  |     PSYC_PARSE_LIST_END = 8, | ||||||
| } PsycParseListRC; | } PsycParseListRC; | ||||||
| 
 | 
 | ||||||
| typedef enum { | typedef enum { | ||||||
|     PSYC_PARSE_TABLE_ERROR_BODY = -5, |     PSYC_LIST_PART_START = 0, | ||||||
|     PSYC_PARSE_TABLE_ERROR_DELIM = -4, |     PSYC_LIST_PART_TYPE = 1, | ||||||
|     PSYC_PARSE_TABLE_ERROR_HEAD = -3, |     PSYC_LIST_PART_ELEM_START = 2, | ||||||
|     PSYC_PARSE_TABLE_ERROR_WIDTH = -2, |     PSYC_LIST_PART_ELEM_TYPE = 3, | ||||||
|     PSYC_PARSE_TABLE_ERROR = -1, |     PSYC_LIST_PART_ELEM_LENGTH = 4, | ||||||
|     /// Completed parsing the width of the table.
 |     PSYC_LIST_PART_ELEM = 5, | ||||||
|     PSYC_PARSE_TABLE_WIDTH = 1, | } PsycListPart; | ||||||
| #ifdef PSYC_PARSE_TABLE_HEAD |  | ||||||
|     /// Completed parsing the name of the key column.
 |  | ||||||
|     PSYC_PARSE_TABLE_NAME_KEY = 2, |  | ||||||
|     /// Completed parsing the name of a value column.
 |  | ||||||
|     PSYC_PARSE_TABLE_NAME_VALUE = 3, |  | ||||||
| #endif |  | ||||||
|     /// Completed parsing a key.
 |  | ||||||
|     PSYC_PARSE_TABLE_KEY = 4, |  | ||||||
|     /// Completed parsing a value.
 |  | ||||||
|     PSYC_PARSE_TABLE_VALUE = 5, |  | ||||||
|     /// Completed parsing a key and reached end of buffer.
 |  | ||||||
|     PSYC_PARSE_TABLE_KEY_END = 6, |  | ||||||
|     /// Completed parsing a value and reached end of buffer.
 |  | ||||||
|     PSYC_PARSE_TABLE_VALUE_END = 7, |  | ||||||
|     /// Binary table is incomplete.
 |  | ||||||
|     PSYC_PARSE_TABLE_INCOMPLETE = 8, |  | ||||||
| } PsycParseTableRC; |  | ||||||
| 
 | 
 | ||||||
| typedef enum { | typedef enum { | ||||||
|     PSYC_TABLE_PART_START = 0, |     PSYC_PARSE_DICT_ERROR_VALUE = -9, | ||||||
|     PSYC_TABLE_PART_WIDTH = 1, |     PSYC_PARSE_DICT_ERROR_VALUE_LENGTH = -8, | ||||||
| #ifdef PSYC_PARSE_TABLE_HEAD |     PSYC_PARSE_DICT_ERROR_VALUE_TYPE = -7, | ||||||
|     PSYC_TABLE_PART_HEAD_START = 2, |     PSYC_PARSE_DICT_ERROR_VALUE_START = -6, | ||||||
|     PSYC_TABLE_PART_HEAD = 3, |     PSYC_PARSE_DICT_ERROR_KEY = -5, | ||||||
| #endif |     PSYC_PARSE_DICT_ERROR_KEY_LENGTH = -4, | ||||||
|     PSYC_TABLE_PART_BODY_START = 4, |     PSYC_PARSE_DICT_ERROR_KEY_START = -3, | ||||||
|     PSYC_TABLE_PART_BODY = 5, |     PSYC_PARSE_DICT_ERROR_TYPE = -2, | ||||||
| } PsycTablePart; |     PSYC_PARSE_DICT_ERROR = -1, | ||||||
|  |     /// Reached end of buffer.
 | ||||||
|  |     /// Buffer contains insufficient amount of data.
 | ||||||
|  |     /// Fill another buffer and concatenate it with the end of the current buffer,
 | ||||||
|  |     /// from the cursor position to the end.
 | ||||||
|  |     PSYC_PARSE_DICT_INSUFFICIENT = 1, | ||||||
|  |     /// Completed parsing the default type of the dict.
 | ||||||
|  |     PSYC_PARSE_DICT_TYPE = 2, | ||||||
|  |     /// Start of a key is parsed but still not complete.
 | ||||||
|  |     PSYC_PARSE_DICT_KEY_START = 3, | ||||||
|  |     /// Continuation of an incomplete key.
 | ||||||
|  |     PSYC_PARSE_DICT_KEY_CONT = 4, | ||||||
|  |     /// Last continuation of a key.
 | ||||||
|  |     PSYC_PARSE_DICT_KEY_END = 5, | ||||||
|  |     /// Completed parsing a key in one go.
 | ||||||
|  |     PSYC_PARSE_DICT_KEY = 6, | ||||||
|  |     /// Start of a value is parsed but still not complete.
 | ||||||
|  |     PSYC_PARSE_DICT_VALUE_START = 7, | ||||||
|  |     /// Continuation of an incomplete value.
 | ||||||
|  |     PSYC_PARSE_DICT_VALUE_CONT = 8, | ||||||
|  |     /// Last continuation of a value.
 | ||||||
|  |     PSYC_PARSE_DICT_VALUE_END = 9, | ||||||
|  |     /// Completed parsing a value.
 | ||||||
|  |     PSYC_PARSE_DICT_VALUE = 10, | ||||||
|  |     /// Completed parsing the last value.
 | ||||||
|  |     PSYC_PARSE_DICT_VALUE_LAST = 11, | ||||||
|  |     /// Reached end of buffer.
 | ||||||
|  |     PSYC_PARSE_DICT_END = 12, | ||||||
|  | } PsycParseDictRC; | ||||||
|  | 
 | ||||||
|  | typedef enum { | ||||||
|  |     PSYC_DICT_PART_START = 0, | ||||||
|  |     PSYC_DICT_PART_TYPE = 1, | ||||||
|  |     PSYC_DICT_PART_KEY_START = 2, | ||||||
|  |     PSYC_DICT_PART_KEY_LENGTH = 3, | ||||||
|  |     PSYC_DICT_PART_KEY = 4, | ||||||
|  |     PSYC_DICT_PART_VALUE_START = 5, | ||||||
|  |     PSYC_DICT_PART_VALUE_TYPE = 6, | ||||||
|  |     PSYC_DICT_PART_VALUE_LENGTH = 7, | ||||||
|  |     PSYC_DICT_PART_VALUE = 8, | ||||||
|  | } PsycDictPart; | ||||||
|  | 
 | ||||||
|  | typedef enum { | ||||||
|  |     PSYC_PARSE_INDEX_ERROR_DICT = -6, | ||||||
|  |     PSYC_PARSE_INDEX_ERROR_DICT_LENGTH = -5, | ||||||
|  |     PSYC_PARSE_INDEX_ERROR_STRUCT = -4, | ||||||
|  |     PSYC_PARSE_INDEX_ERROR_LIST = -3, | ||||||
|  |     PSYC_PARSE_INDEX_ERROR_TYPE = -2, | ||||||
|  |     PSYC_PARSE_INDEX_ERROR = -1, | ||||||
|  |     /// Reached end of buffer.
 | ||||||
|  |     /// Buffer contains insufficient amount of data.
 | ||||||
|  |     /// Fill another buffer and concatenate it with the end of the current buffer,
 | ||||||
|  |     /// from the cursor position to the end.
 | ||||||
|  |     PSYC_PARSE_INDEX_INSUFFICIENT = 1, | ||||||
|  |     // Completed parsing a list index.
 | ||||||
|  |     PSYC_PARSE_INDEX_LIST = 3, | ||||||
|  |     // Completed parsing a list index at the end of the buffer.
 | ||||||
|  |     PSYC_PARSE_INDEX_LIST_LAST = 4, | ||||||
|  |     // Completed parsing a struct element name.
 | ||||||
|  |     PSYC_PARSE_INDEX_STRUCT = 5, | ||||||
|  |     // Completed parsing a struct element name at the end of the buffer.
 | ||||||
|  |     PSYC_PARSE_INDEX_STRUCT_LAST = 6, | ||||||
|  |     /// Start of a dict key is parsed but still not complete.
 | ||||||
|  |     PSYC_PARSE_INDEX_DICT_START = 7, | ||||||
|  |     /// Continuation of an incomplete dict key.
 | ||||||
|  |     PSYC_PARSE_INDEX_DICT_CONT = 8, | ||||||
|  |     /// Last continuation of a dict key.
 | ||||||
|  |     PSYC_PARSE_INDEX_DICT_END = 9, | ||||||
|  |     /// Completed parsing a dict key in one go.
 | ||||||
|  |     PSYC_PARSE_INDEX_DICT = 10, | ||||||
|  |     /// Reached end of buffer.
 | ||||||
|  |     PSYC_PARSE_INDEX_END = 11, | ||||||
|  | } PsycParseIndexRC; | ||||||
|  | 
 | ||||||
|  | typedef enum { | ||||||
|  |     PSYC_INDEX_PART_START = 0, | ||||||
|  |     PSYC_INDEX_PART_TYPE = 1, | ||||||
|  |     PSYC_INDEX_PART_LIST = 2, | ||||||
|  |     PSYC_INDEX_PART_STRUCT = 3, | ||||||
|  |     PSYC_INDEX_PART_DICT_LENGTH = 4, | ||||||
|  |     PSYC_INDEX_PART_DICT = 5, | ||||||
|  | } PsycIndexPart; | ||||||
|  | 
 | ||||||
|  | typedef enum { | ||||||
|  |     PSYC_PARSE_UPDATE_ERROR_VALUE = -24, | ||||||
|  |     PSYC_PARSE_UPDATE_ERROR_LENGTH = -23, | ||||||
|  |     PSYC_PARSE_UPDATE_ERROR_TYPE = -22, | ||||||
|  |     PSYC_PARSE_UPDATE_ERROR_OPER = -21, | ||||||
|  |     PSYC_PARSE_UPDATE_ERROR = -1, | ||||||
|  |     /// Reached end of buffer.
 | ||||||
|  |     /// Buffer contains insufficient amount of data.
 | ||||||
|  |     /// Fill another buffer and concatenate it with the end of the current buffer,
 | ||||||
|  |     /// from the cursor position to the end.
 | ||||||
|  |     PSYC_PARSE_UPDATE_INSUFFICIENT = 1, | ||||||
|  | 
 | ||||||
|  |     // Completed parsing a list index.
 | ||||||
|  |     PSYC_PARSE_UPDATE_INDEX_LIST = 3, | ||||||
|  |     // Completed parsing a struct element name.
 | ||||||
|  |     PSYC_PARSE_UPDATE_INDEX_STRUCT = 5, | ||||||
|  |     /// Start of a dict key is parsed but still not complete.
 | ||||||
|  |     PSYC_PARSE_UPDATE_INDEX_DICT_START = 7, | ||||||
|  |     /// Continuation of an incomplete dict key.
 | ||||||
|  |     PSYC_PARSE_UPDATE_INDEX_DICT_CONT = 8, | ||||||
|  |     /// Last continuation of a dict key.
 | ||||||
|  |     PSYC_PARSE_UPDATE_INDEX_DICT_END = 9, | ||||||
|  |     /// Completed parsing a dict key in one go.
 | ||||||
|  |     PSYC_PARSE_UPDATE_INDEX_DICT = 10, | ||||||
|  | 
 | ||||||
|  |     /// Completed parsing the type.
 | ||||||
|  |     PSYC_PARSE_UPDATE_TYPE = 21, | ||||||
|  |     /// Completed parsing the type and reached end of buffer.
 | ||||||
|  |     PSYC_PARSE_UPDATE_TYPE_END = 22, | ||||||
|  |     /// Start of the value is parsed but still not complete.
 | ||||||
|  |     PSYC_PARSE_UPDATE_VALUE_START = 23, | ||||||
|  |     /// Continuation of incomplete value.
 | ||||||
|  |     PSYC_PARSE_UPDATE_VALUE_CONT = 24, | ||||||
|  |     /// Last continuation of the value.
 | ||||||
|  |     PSYC_PARSE_UPDATE_VALUE_END = 25, | ||||||
|  |     /// Completed parsing the value in one go.
 | ||||||
|  |     PSYC_PARSE_UPDATE_VALUE = 26, | ||||||
|  |     /// Reached end of buffer.
 | ||||||
|  |     PSYC_PARSE_UPDATE_END = 27, | ||||||
|  | } PsycParseUpdateRC; | ||||||
|  | 
 | ||||||
|  | typedef enum { | ||||||
|  |     PSYC_UPDATE_PART_START = 0, | ||||||
|  | 
 | ||||||
|  |     PSYC_UPDATE_INDEX_PART_TYPE = 1, | ||||||
|  |     PSYC_UPDATE_INDEX_PART_LIST = 2, | ||||||
|  |     PSYC_UPDATE_INDEX_PART_STRUCT = 3, | ||||||
|  |     PSYC_UPDATE_INDEX_PART_DICT_LENGTH = 4, | ||||||
|  |     PSYC_UPDATE_INDEX_PART_DICT = 5, | ||||||
|  | 
 | ||||||
|  |     PSYC_UPDATE_PART_TYPE = 12, | ||||||
|  |     PSYC_UPDATE_PART_LENGTH = 13, | ||||||
|  |     PSYC_UPDATE_PART_VALUE = 14, | ||||||
|  | } PsycUpdatePart; | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  * Struct for keeping parser state. |  * Struct for keeping parser state. | ||||||
|  */ |  */ | ||||||
| typedef struct { | typedef struct { | ||||||
|  |     PsycString buffer;		///< Buffer with data to be parsed.
 | ||||||
|     size_t cursor;		///< Current position in buffer.
 |     size_t cursor;		///< Current position in buffer.
 | ||||||
|     size_t startc;		///< Position where the parsing would be resumed.
 |     size_t startc;		///< Position where the parsing would be resumed.
 | ||||||
|     PsycString buffer;		///< Buffer with data to be parsed.
 |  | ||||||
|     uint8_t flags;		///< Flags for the parser, see PsycParseFlag.
 |  | ||||||
|     PsycPart part;		///< Part of the packet being parsed currently.
 |  | ||||||
| 
 | 
 | ||||||
|     size_t routinglen;		///< Length of routing part parsed so far.
 |     size_t routinglen;		///< Length of routing part parsed so far.
 | ||||||
|     size_t content_parsed;	///< Number of bytes parsed from the content so far.
 |  | ||||||
|     size_t contentlen;		///< Expected length of the content.
 |     size_t contentlen;		///< Expected length of the content.
 | ||||||
|     PsycBool contentlen_found;	///< Is there a length given for this packet?
 |     size_t content_parsed;	///< Number of bytes parsed from the content so far.
 | ||||||
|     size_t value_parsed;	///< Number of bytes parsed from the value so far.
 |  | ||||||
|     size_t valuelen;		///< Expected length of the value.
 |     size_t valuelen;		///< Expected length of the value.
 | ||||||
|     PsycBool valuelen_found;	///< Is there a length given for this modifier?
 |     size_t value_parsed;	///< Number of bytes parsed from the value so far.
 | ||||||
|  | 
 | ||||||
|  |     PsycPart part;		///< Part of the packet being parsed currently.
 | ||||||
|  |     uint8_t flags;		///< Flags for the parser, see PsycParseFlag.
 | ||||||
|  |     uint8_t contentlen_found;	///< Is there a length given for this packet?
 | ||||||
|  |     uint8_t valuelen_found;	///< Is there a length given for this modifier?
 | ||||||
| } PsycParseState; | } PsycParseState; | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  * Struct for keeping list parser state. |  * Struct for keeping list parser state. | ||||||
|  */ |  */ | ||||||
| typedef struct { | typedef struct { | ||||||
|  |     PsycString buffer;		///< Buffer with data to be parsed.
 | ||||||
|     size_t cursor;		///< Current position in buffer.
 |     size_t cursor;		///< Current position in buffer.
 | ||||||
|     size_t startc;		///< Line start position.
 |     size_t startc;		///< Line start position.
 | ||||||
|     PsycString buffer;		///< Buffer with data to be parsed.
 |  | ||||||
|     PsycListType type;		///< List type.
 |  | ||||||
|     char term;			///< Terminator character at the end.
 |  | ||||||
|     uint8_t term_set;		///< Look for terminator.
 |  | ||||||
| 
 | 
 | ||||||
|     size_t elem_parsed;		///< Number of bytes parsed from the elem so far.
 |     PsycString type;		///< List type.
 | ||||||
|     size_t elemlen;		///< Expected length of the elem.
 |     size_t elemlen;		///< Expected length of the elem.
 | ||||||
|  |     size_t elem_parsed;		///< Number of bytes parsed from the elem so far.
 | ||||||
|  | 
 | ||||||
|  |     PsycListPart part;		///< Part of the list being parsed currently.
 | ||||||
|  |     uint8_t elemlen_found;	///< Is there a length given for this element?
 | ||||||
| } PsycParseListState; | } PsycParseListState; | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  * Struct for keeping table parser state. |  * Struct for keeping dict parser state. | ||||||
|  */ |  */ | ||||||
| typedef struct { | typedef struct { | ||||||
|  |     PsycString buffer;		///< Buffer with data to be parsed.
 | ||||||
|     size_t cursor;		///< Current position in buffer.
 |     size_t cursor;		///< Current position in buffer.
 | ||||||
|     size_t startc;		///< Line start position.
 |     size_t startc;		///< Line start position.
 | ||||||
|  | 
 | ||||||
|  |     size_t elemlen;		///< Expected length of the key/value.
 | ||||||
|  |     size_t elem_parsed;		///< Number of bytes parsed from the key/value so far.
 | ||||||
|  | 
 | ||||||
|  |     PsycDictPart part;		///< Part of the dict being parsed currently.
 | ||||||
|  |     uint8_t elemlen_found;	///< Is there a length given for this key/value?
 | ||||||
|  | } PsycParseDictState; | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Struct for keeping index parser state. | ||||||
|  |  */ | ||||||
|  | typedef struct { | ||||||
|     PsycString buffer;		///< Buffer with data to be parsed.
 |     PsycString buffer;		///< Buffer with data to be parsed.
 | ||||||
|     PsycTablePart part;		///< Table type.
 |     size_t cursor;		///< Current position in buffer.
 | ||||||
|     size_t width;		///< Width of table.
 |     size_t startc;		///< Position where the parsing would be resumed.
 | ||||||
|     size_t elems;               ///< Elems parsed so far in the table.
 | 
 | ||||||
|     PsycParseListState list; |     size_t elemlen;		///< Expected length of an element.
 | ||||||
| } PsycParseTableState; |     size_t elem_parsed;		///< Number of bytes parsed from the elem so far.
 | ||||||
|  | 
 | ||||||
|  |     PsycIndexPart part;		///< Part of the packet being parsed currently.
 | ||||||
|  |     uint8_t elemlen_found;	///< Is there a length given for this element?
 | ||||||
|  | } PsycParseIndexState; | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Struct for keeping update modifier parser state. | ||||||
|  |  */ | ||||||
|  | typedef struct { | ||||||
|  |     PsycString buffer;		///< Buffer with data to be parsed.
 | ||||||
|  |     size_t cursor;		///< Current position in buffer.
 | ||||||
|  |     size_t startc;		///< Position where the parsing would be resumed.
 | ||||||
|  | 
 | ||||||
|  |     size_t elemlen;		///< Expected length of an element.
 | ||||||
|  |     size_t elem_parsed;		///< Number of bytes parsed from the elem so far.
 | ||||||
|  | 
 | ||||||
|  |     PsycUpdatePart part;	///< Part of the packet being parsed currently.
 | ||||||
|  |     uint8_t elemlen_found;	///< Is there a length given for this element?
 | ||||||
|  | } PsycParseUpdateState; | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  * Initializes the state struct. |  * Initializes the state struct. | ||||||
|  | @ -339,7 +523,7 @@ psyc_parse_buffer_set (PsycParseState *state, const char *buffer, size_t length) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  * Initializes the list state. |  * Initializes the list parser state. | ||||||
|  */ |  */ | ||||||
| static inline void | static inline void | ||||||
| psyc_parse_list_state_init (PsycParseListState *state) | psyc_parse_list_state_init (PsycParseListState *state) | ||||||
|  | @ -358,27 +542,60 @@ psyc_parse_list_buffer_set (PsycParseListState *state, | ||||||
|     state->cursor = 0; |     state->cursor = 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /**
 | ||||||
|  |  * Initializes the dict parser state. | ||||||
|  |  */ | ||||||
| static inline void | static inline void | ||||||
| psyc_parse_list_term_set (PsycParseListState *state, char term) | psyc_parse_dict_state_init (PsycParseDictState *state) | ||||||
| { | { | ||||||
|     state->term = term; |     memset(state, 0, sizeof(PsycParseDictState)); | ||||||
|     state->term_set = PSYC_TRUE; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  * Initializes the table state. |  * Sets a new buffer in the dict parser state struct with data to be parsed. | ||||||
|  */ |  */ | ||||||
| static inline void | static inline void | ||||||
| psyc_parse_table_state_init (PsycParseTableState *state) | psyc_parse_dict_buffer_set (PsycParseDictState *state, | ||||||
|  | 			     const char *buffer, size_t length) | ||||||
| { | { | ||||||
|     memset(state, 0, sizeof(PsycParseTableState)); |     state->buffer = (PsycString) {length, (char*)buffer}; | ||||||
|  |     state->cursor = 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  * Sets a new buffer in the list parser state struct with data to be parsed. |  * Initializes the index parser state. | ||||||
|  */ |  */ | ||||||
| static inline void | static inline void | ||||||
| psyc_parse_table_buffer_set (PsycParseTableState *state, | psyc_parse_index_state_init (PsycParseIndexState *state) | ||||||
|  | { | ||||||
|  |     memset(state, 0, sizeof(PsycParseIndexState)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Sets a new buffer in the index parser state struct with data to be parsed. | ||||||
|  |  */ | ||||||
|  | static inline void | ||||||
|  | psyc_parse_index_buffer_set (PsycParseIndexState *state, | ||||||
|  | 			     const char *buffer, size_t length) | ||||||
|  | { | ||||||
|  |     state->buffer = (PsycString) {length, (char*)buffer}; | ||||||
|  |     state->cursor = 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Initializes the update modifier parser state. | ||||||
|  |  */ | ||||||
|  | static inline void | ||||||
|  | psyc_parse_update_state_init (PsycParseUpdateState *state) | ||||||
|  | { | ||||||
|  |     memset(state, 0, sizeof(PsycParseUpdateState)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Sets a new buffer in the update modifier parser state struct with data to be parsed. | ||||||
|  |  */ | ||||||
|  | static inline void | ||||||
|  | psyc_parse_update_buffer_set (PsycParseUpdateState *state, | ||||||
| 			     const char *buffer, size_t length) | 			     const char *buffer, size_t length) | ||||||
| { | { | ||||||
|     state->buffer = (PsycString) {length, (char*)buffer}; |     state->buffer = (PsycString) {length, (char*)buffer}; | ||||||
|  | @ -471,10 +688,25 @@ psyc_parse (PsycParseState *state, char *oper, | ||||||
| static inline | static inline | ||||||
| #endif | #endif | ||||||
| PsycParseListRC | PsycParseListRC | ||||||
| psyc_parse_list (PsycParseListState *state, PsycString *elem); | psyc_parse_list (PsycParseListState *state, PsycString *type, PsycString *elem); | ||||||
| 
 | 
 | ||||||
| PsycParseTableRC | #ifdef __INLINE_PSYC_PARSE | ||||||
| psyc_parse_table (PsycParseTableState *state, PsycString *elem); | static inline | ||||||
|  | #endif | ||||||
|  | PsycParseDictRC | ||||||
|  | psyc_parse_dict (PsycParseDictState *state, PsycString *type, PsycString *elem); | ||||||
|  | 
 | ||||||
|  | #ifdef __INLINE_PSYC_PARSE | ||||||
|  | static inline | ||||||
|  | #endif | ||||||
|  | PsycParseIndexRC | ||||||
|  | psyc_parse_index (PsycParseIndexState *state, PsycString *idx); | ||||||
|  | 
 | ||||||
|  | #ifdef __INLINE_PSYC_PARSE | ||||||
|  | static inline | ||||||
|  | #endif | ||||||
|  | PsycParseUpdateRC | ||||||
|  | psyc_parse_update (PsycParseUpdateState *state, char *oper, PsycString *value); | ||||||
| 
 | 
 | ||||||
| static inline size_t | static inline size_t | ||||||
| psyc_parse_int (const char *value, size_t len, int64_t *n) | psyc_parse_int (const char *value, size_t len, int64_t *n) | ||||||
|  | @ -516,7 +748,7 @@ psyc_parse_uint (const char *value, size_t len, uint64_t *n) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static inline size_t | static inline size_t | ||||||
| psyc_parse_index (const char *value, size_t len, int64_t *n) | psyc_parse_list_index (const char *value, size_t len, int64_t *n) | ||||||
| { | { | ||||||
|     if (!value || len == 0 || value[0] != '#') |     if (!value || len == 0 || value[0] != '#') | ||||||
| 	return 0; | 	return 0; | ||||||
|  |  | ||||||
|  | @ -51,6 +51,12 @@ static inline | ||||||
| PsycRenderRC | PsycRenderRC | ||||||
| psyc_render (PsycPacket *packet, char *buffer, size_t buflen); | psyc_render (PsycPacket *packet, char *buffer, size_t buflen); | ||||||
| 
 | 
 | ||||||
|  | PsycRenderRC | ||||||
|  | psyc_render_elem (PsycElem *elem, char *buffer, size_t buflen); | ||||||
|  | 
 | ||||||
|  | PsycRenderRC | ||||||
|  | psyc_render_dict_key (PsycDictKey *elem, char *buffer, size_t buflen); | ||||||
|  | 
 | ||||||
| /**
 | /**
 | ||||||
|  * Render a PSYC list into a buffer. |  * Render a PSYC list into a buffer. | ||||||
|  */ |  */ | ||||||
|  | @ -61,15 +67,7 @@ PsycRenderRC | ||||||
| psyc_render_list (PsycList *list, char *buffer, size_t buflen); | psyc_render_list (PsycList *list, char *buffer, size_t buflen); | ||||||
| 
 | 
 | ||||||
| PsycRenderRC | PsycRenderRC | ||||||
| psyc_render_table (PsycTable *table, char *buffer, size_t buflen); | psyc_render_dict (PsycDict *dict, char *buffer, size_t buflen); | ||||||
| 
 |  | ||||||
| PsycRenderRC |  | ||||||
| psyc_render_packet_id (char *context, size_t contextlen, |  | ||||||
| 		       char *source, size_t sourcelen, |  | ||||||
| 		       char *target, size_t targetlen, |  | ||||||
| 		       char *counter, size_t counterlen, |  | ||||||
| 		       char *fragment, size_t fragmentlen, |  | ||||||
| 		       char *buffer, size_t buflen); |  | ||||||
| 
 | 
 | ||||||
| /** @} */ // end of render group
 | /** @} */ // end of render group
 | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1,21 +0,0 @@ | ||||||
| #ifndef PSYC_SYNTAX_H |  | ||||||
| 
 |  | ||||||
| #ifndef PSYC_LIST_SIZE_LIMIT |  | ||||||
| # define PSYC_LIST_SIZE_LIMIT 404 |  | ||||||
| #endif |  | ||||||
| 
 |  | ||||||
| /* beyond this a content length must be provided */ |  | ||||||
| #ifndef PSYC_CONTENT_SIZE_THRESHOLD |  | ||||||
| # define PSYC_CONTENT_SIZE_THRESHOLD 444 |  | ||||||
| #endif |  | ||||||
| 
 |  | ||||||
| /* beyond this a modifier value length must be provided */ |  | ||||||
| #ifndef PSYC_MODIFIER_SIZE_THRESHOLD |  | ||||||
| # define PSYC_MODIFIER_SIZE_THRESHOLD 404 |  | ||||||
| #endif |  | ||||||
| 
 |  | ||||||
| #define PSYC_PACKET_DELIMITER_CHAR '|' |  | ||||||
| #define PSYC_PACKET_DELIMITER	   "\n|\n" |  | ||||||
| 
 |  | ||||||
| #define PSYC_SYNTAX_H |  | ||||||
| #endif |  | ||||||
|  | @ -5,16 +5,17 @@ | ||||||
| #ifndef PSYC_VARIABLE_H | #ifndef PSYC_VARIABLE_H | ||||||
| #define PSYC_VARIABLE_H | #define PSYC_VARIABLE_H | ||||||
| 
 | 
 | ||||||
|  | #include "match.h" | ||||||
| #include "packet.h" | #include "packet.h" | ||||||
| 
 | 
 | ||||||
| /// Routing variables in alphabetical order.
 | /// Routing variables in alphabetical order.
 | ||||||
| extern const PsycDictInt psyc_rvars[]; | extern const PsycMapInt psyc_rvars[]; | ||||||
| 
 | 
 | ||||||
| // Variable types in alphabetical order.
 | // Variable types in alphabetical order.
 | ||||||
| extern const PsycDictInt psyc_var_types[]; | extern const PsycMapInt psyc_var_types[]; | ||||||
| 
 | 
 | ||||||
| /// Method names in alphabetical order.
 | /// Method names in alphabetical order.
 | ||||||
| extern const PsycDictInt psyc_methods[]; | extern const PsycMapInt psyc_methods[]; | ||||||
| 
 | 
 | ||||||
| extern const size_t psyc_rvars_num; | extern const size_t psyc_rvars_num; | ||||||
| extern const size_t psyc_var_types_num; | extern const size_t psyc_var_types_num; | ||||||
|  | @ -37,14 +38,41 @@ typedef enum { | ||||||
|     PSYC_RVARS_NUM, |     PSYC_RVARS_NUM, | ||||||
| } PsycRoutingVar; | } PsycRoutingVar; | ||||||
| 
 | 
 | ||||||
|  | /**
 | ||||||
|  |  * Variable types. | ||||||
|  |  * | ||||||
|  |  * This enum lists PSYC variable types that | ||||||
|  |  * this library is capable of checking for | ||||||
|  |  * validity. Other variable types are treated | ||||||
|  |  * as opaque data. | ||||||
|  |  */ | ||||||
|  | typedef enum { | ||||||
|  |     PSYC_TYPE_UNKNOWN, | ||||||
|  |     PSYC_TYPE_AMOUNT, | ||||||
|  |     PSYC_TYPE_COLOR, | ||||||
|  |     PSYC_TYPE_COUNTER, | ||||||
|  |     PSYC_TYPE_DATE, | ||||||
|  |     PSYC_TYPE_DEGREE, | ||||||
|  |     PSYC_TYPE_DICT, | ||||||
|  |     PSYC_TYPE_ENTITY, | ||||||
|  |     PSYC_TYPE_FLAG, | ||||||
|  |     PSYC_TYPE_LANGUAGE, | ||||||
|  |     PSYC_TYPE_LIST, | ||||||
|  |     PSYC_TYPE_NICK, | ||||||
|  |     PSYC_TYPE_PAGE, | ||||||
|  |     PSYC_TYPE_STRUCT, | ||||||
|  |     PSYC_TYPE_TIME, | ||||||
|  |     PSYC_TYPE_UNIFORM, | ||||||
|  | } PsycType; | ||||||
|  | 
 | ||||||
| /**
 | /**
 | ||||||
|  * Look up routing variable. |  * Look up routing variable. | ||||||
|  */ |  */ | ||||||
| static inline PsycRoutingVar | static inline PsycRoutingVar | ||||||
| psyc_var_routing (const char *name, size_t len) | psyc_var_routing (const char *name, size_t len) | ||||||
| { | { | ||||||
|     return (PsycRoutingVar) psyc_dict_lookup((PsycDict *)psyc_rvars, |     return (PsycRoutingVar) | ||||||
| 				       psyc_rvars_num, name, len, PSYC_NO); | 	psyc_map_lookup((PsycMap*)psyc_rvars, psyc_rvars_num, name, len, PSYC_NO); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  | @ -53,8 +81,9 @@ psyc_var_routing (const char *name, size_t len) | ||||||
| static inline PsycType | static inline PsycType | ||||||
| psyc_var_type (const char *name, size_t len) | psyc_var_type (const char *name, size_t len) | ||||||
| { | { | ||||||
|     return (PsycType) psyc_dict_lookup((PsycDict *)psyc_var_types, |     return (PsycType) | ||||||
| 				       psyc_var_types_num, name, len, PSYC_YES); | 	psyc_map_lookup((PsycMap*)psyc_var_types, psyc_var_types_num, | ||||||
|  | 			name, len, PSYC_YES); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  |  | ||||||
|  | @ -4,8 +4,10 @@ | ||||||
| #ifdef DEBUG | #ifdef DEBUG | ||||||
| # include <stdio.h> | # include <stdio.h> | ||||||
| # define PP(args) printf args; | # define PP(args) printf args; | ||||||
|  | # define ASSERT(cond) assert(cond) | ||||||
| #else | #else | ||||||
| # define PP(args) | # define PP(args) | ||||||
|  | # define ASSERT(cond) | ||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
| #ifdef TEST | #ifdef TEST | ||||||
|  |  | ||||||
							
								
								
									
										22
									
								
								src/match.c
									
										
									
									
									
								
							
							
						
						
									
										22
									
								
								src/match.c
									
										
									
									
									
								
							|  | @ -107,11 +107,11 @@ psyc_matches(char *sho, size_t slen, char *lon, size_t llen) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  * Look up value associated with a key in a dictionary. |  * Look up value associated with a key in a map. | ||||||
|  */ |  */ | ||||||
| void * | void * | ||||||
| psyc_dict_lookup(const PsycDict * dict, size_t size, | psyc_map_lookup(const PsycMap * map, size_t size, | ||||||
| 		 const char *key, size_t keylen, PsycBool inherit) | 		const char *key, size_t keylen, PsycBool inherit) | ||||||
| { | { | ||||||
|     //size_t cursor = 1;
 |     //size_t cursor = 1;
 | ||||||
|     size_t c = 0; |     size_t c = 0; | ||||||
|  | @ -122,25 +122,25 @@ psyc_dict_lookup(const PsycDict * dict, size_t size, | ||||||
| 
 | 
 | ||||||
|     //for (c = 0, i = 0; c < keylen && i < size; c++) {
 |     //for (c = 0, i = 0; c < keylen && i < size; c++) {
 | ||||||
|     for (i = 0; i < size; i++) { |     for (i = 0; i < size; i++) { | ||||||
| 	if (!(keylen == dict[i].key.length | 	if (!(keylen == map[i].key.length | ||||||
| 	      || (inherit && keylen > dict[i].key.length | 	      || (inherit && keylen > map[i].key.length | ||||||
| 		  && key[dict[i].key.length] == '_'))) | 		  && key[map[i].key.length] == '_'))) | ||||||
| 	    continue; | 	    continue; | ||||||
| 
 | 
 | ||||||
| 	match = 1; | 	match = 1; | ||||||
| 	for (c = 0; c < keylen; c++) { | 	for (c = 0; c < keylen; c++) { | ||||||
| 	    if (c < dict[i].key.length && dict[i].key.data[c] == key[c]) | 	    if (c < map[i].key.length && map[i].key.data[c] == key[c]) | ||||||
| 		continue; | 		continue; | ||||||
| 	    else if (c == dict[i].key.length && key[c] == '_') | 	    else if (c == map[i].key.length && key[c] == '_') | ||||||
| 		return dict[i].value;	// after the end of a matching prefix
 | 		return map[i].value;	// after the end of a matching prefix
 | ||||||
| 	    else if (dict[i].key.data[c] > key[c]) | 	    else if (map[i].key.data[c] > key[c]) | ||||||
| 		return NULL; | 		return NULL; | ||||||
| 
 | 
 | ||||||
| 	    match = 0; | 	    match = 0; | ||||||
| 	    break; | 	    break; | ||||||
| 	} | 	} | ||||||
| 	if (match) | 	if (match) | ||||||
| 	    return dict[i].value; | 	    return map[i].value; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     return NULL; |     return NULL; | ||||||
|  |  | ||||||
							
								
								
									
										159
									
								
								src/packet.c
									
										
									
									
									
								
							
							
						
						
									
										159
									
								
								src/packet.c
									
										
									
									
									
								
							|  | @ -1,73 +1,99 @@ | ||||||
| #include "lib.h" | #include "lib.h" | ||||||
| #include <psyc/syntax.h> |  | ||||||
| #include <psyc/packet.h> | #include <psyc/packet.h> | ||||||
| 
 | 
 | ||||||
| inline PsycListFlag | inline PsycElemFlag | ||||||
| psyc_list_length_check (PsycList *list) | psyc_elem_length_check (PsycString *value, const char end) | ||||||
| { | { | ||||||
|     PsycListFlag flag = PSYC_LIST_NO_LENGTH; |     if (value->length > PSYC_ELEM_SIZE_THRESHOLD | ||||||
|     size_t i, length = 0; | 	|| memchr(value->data, (int)end, value->length)) | ||||||
|  | 	return PSYC_ELEM_NEED_LENGTH; | ||||||
| 
 | 
 | ||||||
|     for (i = 0; i < list->num_elems; i++) { |     return PSYC_ELEM_NO_LENGTH;; | ||||||
| 	PsycString *elem = &list->elems[i]; |  | ||||||
| 	length += 1 + elem->length; // |elem
 |  | ||||||
| 	if (length > PSYC_MODIFIER_SIZE_THRESHOLD || |  | ||||||
| 	    memchr(elem->data, (int) '|', elem->length) || |  | ||||||
| 	    memchr(elem->data, (int) '\n', elem->length)) { |  | ||||||
| 	    flag = PSYC_LIST_NEED_LENGTH; |  | ||||||
| 	    break; |  | ||||||
| 	} |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     return flag; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| inline PsycListFlag | inline size_t | ||||||
| psyc_list_length (PsycList *list) | psyc_elem_length (PsycElem *elem) | ||||||
| { | { | ||||||
|     size_t i, length = 0; |     return | ||||||
|  | 	(elem->type.length ? 1 + elem->type.length : 0) | ||||||
|  | 	+ (elem->value.length && elem->flag != PSYC_ELEM_NO_LENGTH | ||||||
|  | 	   ? (elem->type.length ? 1 : 0) + psyc_num_length(elem->value.length) : 0) | ||||||
|  | 	+ (elem->value.length ? 1 + elem->value.length : 0); | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
|     if (list->flag == PSYC_LIST_NEED_LENGTH) { | inline size_t | ||||||
| 	for (i = 0; i < list->num_elems; i++) { | psyc_dict_key_length (PsycDictKey *elem) | ||||||
| 	    if (i > 0) | { | ||||||
| 		length++;	// |
 |     return | ||||||
| 	    length +=		// length SP elem
 | 	(elem->flag != PSYC_ELEM_NO_LENGTH | ||||||
| 		psyc_num_length(list->elems[i].length) + 1 + list->elems[i].length; | 	 ? psyc_num_length(elem->value.length) : 0) | ||||||
| 	} | 	+ (elem->value.length ? 1 + elem->value.length : 0); | ||||||
|     } else { | } | ||||||
| 	for (i = 0; i < list->num_elems; i++) | 
 | ||||||
| 	    length += 1 + list->elems[i].length; // |elem
 | inline size_t | ||||||
|  | psyc_list_length_set (PsycList *list) | ||||||
|  | { | ||||||
|  |     size_t i; | ||||||
|  |     PsycElem *elem; | ||||||
|  | 
 | ||||||
|  |     list->length = list->type.length; | ||||||
|  | 
 | ||||||
|  |     for (i = 0; i < list->num_elems; i++) { | ||||||
|  | 	elem = &list->elems[i]; | ||||||
|  | 	if (elem->flag == PSYC_ELEM_CHECK_LENGTH) | ||||||
|  | 	    elem->flag = psyc_elem_length_check(&elem->value, '|'); | ||||||
|  | 	elem->length = psyc_elem_length(elem); | ||||||
|  | 	list->length += 1 + elem->length; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     return length; |     return list->length; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | inline size_t | ||||||
|  | psyc_dict_length_set (PsycDict *dict) | ||||||
|  | { | ||||||
|  |     size_t i; | ||||||
|  |     PsycDictKey *key; | ||||||
|  |     PsycElem *value; | ||||||
|  | 
 | ||||||
|  |     dict->length = dict->type.length; | ||||||
|  | 
 | ||||||
|  |     for (i = 0; i < dict->num_elems; i++) { | ||||||
|  | 	key = &dict->elems[i].key; | ||||||
|  | 	value = &dict->elems[i].value; | ||||||
|  | 
 | ||||||
|  | 	if (key->flag == PSYC_ELEM_CHECK_LENGTH) | ||||||
|  | 	    key->flag = psyc_elem_length_check(&key->value, '}'); | ||||||
|  | 	if (value->flag == PSYC_ELEM_CHECK_LENGTH) | ||||||
|  | 	    value->flag = psyc_elem_length_check(&value->value, '{'); | ||||||
|  | 
 | ||||||
|  | 	key->length = psyc_dict_key_length(key); | ||||||
|  | 	value->length = psyc_elem_length(value); | ||||||
|  | 
 | ||||||
|  | 	dict->length += 1 + key->length + 1 + value->length; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return dict->length; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void | void | ||||||
| psyc_list_init (PsycList *list, PsycString *elems, size_t num_elems, | psyc_list_init (PsycList *list, PsycElem *elems, size_t num_elems) | ||||||
| 		PsycListFlag flag) |  | ||||||
| { | { | ||||||
|     *list = (PsycList) { |     *list = (PsycList) { | ||||||
| 	.num_elems = num_elems, | 	.num_elems = num_elems, | ||||||
| 	.elems = elems, | 	.elems = elems, | ||||||
| 	.length = 0, |  | ||||||
| 	.flag = flag, |  | ||||||
|     }; |     }; | ||||||
| 
 |     psyc_list_length_set(list); | ||||||
|     if (flag == PSYC_LIST_CHECK_LENGTH) // check if list elements need length
 |  | ||||||
| 	list->flag = psyc_list_length_check(list); |  | ||||||
| 
 |  | ||||||
|     list->length = psyc_list_length(list); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void | void | ||||||
| psyc_table_init (PsycTable *table, size_t width, PsycList *list) | psyc_dict_init (PsycDict *dict, PsycDictElem *elems, size_t num_elems) | ||||||
| { | { | ||||||
|     *table = (PsycTable) { |     *dict = (PsycDict) { | ||||||
| 	.width = width, | 	.num_elems = num_elems, | ||||||
| 	.list = list, | 	.elems = elems, | ||||||
|     }; |     }; | ||||||
| 
 |     psyc_dict_length_set(dict); | ||||||
|     table->length = (width > 0 ? psyc_num_length(width) + 2 : 0) + list->length; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| inline size_t | inline size_t | ||||||
|  | @ -77,7 +103,10 @@ psyc_modifier_length (PsycModifier *m) | ||||||
|     if (m->name.length > 0) |     if (m->name.length > 0) | ||||||
| 	length += m->name.length + 1 + m->value.length; // name\tvalue
 | 	length += m->name.length + 1 + m->value.length; // name\tvalue
 | ||||||
| 
 | 
 | ||||||
|     if (m->flag == PSYC_MODIFIER_NEED_LENGTH) // add length of length if needed
 |     // add length of length if needed
 | ||||||
|  |     if (m->value.length | ||||||
|  | 	&& (m->flag & PSYC_MODIFIER_NEED_LENGTH | ||||||
|  | 	    || m->flag == PSYC_MODIFIER_CHECK_LENGTH)) | ||||||
| 	length += psyc_num_length(m->value.length) + 1; // SP length
 | 	length += psyc_num_length(m->value.length) + 1; // SP length
 | ||||||
| 
 | 
 | ||||||
|     return length; |     return length; | ||||||
|  | @ -96,7 +125,8 @@ psyc_packet_length_check (PsycPacket *p) | ||||||
|     // If any entity modifiers need length, it is possible they contain
 |     // If any entity modifiers need length, it is possible they contain
 | ||||||
|     // a packet terminator, thus the content should have a length as well.
 |     // a packet terminator, thus the content should have a length as well.
 | ||||||
|     for (i = 0; i < p->entity.lines; i++) |     for (i = 0; i < p->entity.lines; i++) | ||||||
| 	if (p->entity.modifiers[i].flag == PSYC_MODIFIER_NEED_LENGTH) | 	if (p->entity.modifiers[i].flag & PSYC_MODIFIER_NEED_LENGTH | ||||||
|  | 	    || p->entity.modifiers[i].flag == PSYC_MODIFIER_CHECK_LENGTH) | ||||||
| 	    return PSYC_PACKET_NEED_LENGTH; | 	    return PSYC_PACKET_NEED_LENGTH; | ||||||
| 
 | 
 | ||||||
|     if (memmem(p->data.data, p->data.length, PSYC_C2ARG(PSYC_PACKET_DELIMITER))) |     if (memmem(p->data.data, p->data.length, PSYC_C2ARG(PSYC_PACKET_DELIMITER))) | ||||||
|  | @ -137,10 +167,11 @@ psyc_packet_length_set (PsycPacket *p) | ||||||
|     // set total length: routing-header content |\n
 |     // set total length: routing-header content |\n
 | ||||||
|     p->length = p->routinglen + p->contentlen + 2; |     p->length = p->routinglen + p->contentlen + 2; | ||||||
| 
 | 
 | ||||||
|     if (p->contentlen > 0 || p->flag == PSYC_PACKET_NEED_LENGTH) |     if (p->contentlen) | ||||||
| 	p->length++; // add \n at the start of the content part
 | 	p->length++; // add \n at the start of the content part
 | ||||||
| 
 | 
 | ||||||
|     if (p->flag == PSYC_PACKET_NEED_LENGTH) // add length of length if needed
 |     // add length of length if needed
 | ||||||
|  |     if (p->contentlen && !(p->flag & PSYC_PACKET_NO_LENGTH)) | ||||||
| 	p->length += psyc_num_length(p->contentlen); | 	p->length += psyc_num_length(p->contentlen); | ||||||
| 
 | 
 | ||||||
|     return p->length; |     return p->length; | ||||||
|  | @ -198,9 +229,29 @@ psyc_packet_init_raw (PsycPacket *p, | ||||||
|     psyc_packet_length_set(p); |     psyc_packet_length_set(p); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| size_t | void | ||||||
| psyc_packet_id_length (size_t contextlen, size_t sourcelen, size_t targetlen, | psyc_packet_id (PsycList *list, PsycElem *elems, | ||||||
| 		       size_t counterlen, size_t fragmentlen) | 		char *context, size_t contextlen, | ||||||
|  | 		char *source, size_t sourcelen, | ||||||
|  | 		char *target, size_t targetlen, | ||||||
|  | 		char *counter, size_t counterlen, | ||||||
|  | 		char *fragment, size_t fragmentlen) | ||||||
| { | { | ||||||
|     return contextlen + sourcelen + targetlen + counterlen + fragmentlen + 5; |     if (contextlen) | ||||||
|  | 	elems[PSYC_PACKET_ID_CONTEXT] = | ||||||
|  | 	    PSYC_ELEM_VF(context, contextlen, PSYC_ELEM_NO_LENGTH); | ||||||
|  |     if (sourcelen) | ||||||
|  | 	elems[PSYC_PACKET_ID_SOURCE] = | ||||||
|  | 	    PSYC_ELEM_VF(source, sourcelen, PSYC_ELEM_NO_LENGTH); | ||||||
|  |     if (targetlen) | ||||||
|  | 	elems[PSYC_PACKET_ID_TARGET] = | ||||||
|  | 	    PSYC_ELEM_VF(target, targetlen, PSYC_ELEM_NO_LENGTH); | ||||||
|  |     if (counterlen) | ||||||
|  | 	elems[PSYC_PACKET_ID_COUNTER] = | ||||||
|  | 	    PSYC_ELEM_VF(counter, counterlen, PSYC_ELEM_NO_LENGTH); | ||||||
|  |     if (fragmentlen) | ||||||
|  | 	elems[PSYC_PACKET_ID_FRAGMENT] = | ||||||
|  | 	    PSYC_ELEM_VF(fragment, fragmentlen, PSYC_ELEM_NO_LENGTH); | ||||||
|  | 
 | ||||||
|  |     psyc_list_init(list, elems, PSYC_PACKET_ID_ELEMS); | ||||||
| } | } | ||||||
|  |  | ||||||
							
								
								
									
										880
									
								
								src/parse.c
									
										
									
									
									
								
							
							
						
						
									
										880
									
								
								src/parse.c
									
										
									
									
									
								
							|  | @ -15,33 +15,67 @@ | ||||||
| 	return ret;							\ | 	return ret;							\ | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | #define ADVANCE_STARTC_OR_RETURN(ret)					\ | ||||||
|  |     state->startc = state->cursor + 1;					\ | ||||||
|  |     if (++(state->cursor) >= state->buffer.length)			\ | ||||||
|  | 	return ret;							\ | ||||||
|  | 
 | ||||||
| typedef enum { | typedef enum { | ||||||
|     PARSE_ERROR = -1, |     PARSE_ERROR = -1, | ||||||
|     PARSE_SUCCESS = 0, |     PARSE_SUCCESS = 0, | ||||||
|     PARSE_INSUFFICIENT = 1, |     PARSE_INSUFFICIENT = 1, | ||||||
|     PARSE_COMPLETE = 100, |     PARSE_INCOMPLETE = 2, | ||||||
|     PARSE_INCOMPLETE = 101, |  | ||||||
| } ParseRC; | } ParseRC; | ||||||
| 
 | 
 | ||||||
|  | typedef struct { | ||||||
|  |     PsycString buffer; | ||||||
|  |     size_t cursor; | ||||||
|  |     size_t startc; | ||||||
|  | } ParseState; | ||||||
|  | 
 | ||||||
| /**
 | /**
 | ||||||
|  * Parse variable name or method name. |  * Parse variable name or method name. | ||||||
|  |  * | ||||||
|  * It should contain one or more keyword characters. |  * It should contain one or more keyword characters. | ||||||
|  |  * | ||||||
|  * @return PARSE_ERROR or PARSE_SUCCESS |  * @return PARSE_ERROR or PARSE_SUCCESS | ||||||
|  */ |  */ | ||||||
| static inline ParseRC | static inline ParseRC | ||||||
| parse_keyword (PsycParseState *state, PsycString *name) | parse_keyword (ParseState *state, PsycString *name) | ||||||
| { | { | ||||||
|     name->data = state->buffer.data + state->cursor; |     name->data = state->buffer.data + state->cursor; | ||||||
|     name->length = 0; |     name->length = 0; | ||||||
| 
 | 
 | ||||||
|     while (psyc_is_kw_char(state->buffer.data[state->cursor])) { |     while (psyc_is_kw_char(state->buffer.data[state->cursor])) { | ||||||
| 	name->length++;	// was a valid char, increase length
 | 	name->length++;	// was a valid char, increase length
 | ||||||
| 	ADVANCE_CURSOR_OR_RETURN(PSYC_PARSE_INSUFFICIENT); | 	ADVANCE_CURSOR_OR_RETURN(PARSE_INSUFFICIENT); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     return name->length > 0 ? PARSE_SUCCESS : PARSE_ERROR; |     return name->length > 0 ? PARSE_SUCCESS : PARSE_ERROR; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /**
 | ||||||
|  |  * Parse length. | ||||||
|  |  * | ||||||
|  |  * @return PARSE_SUCCESS, PARSE_ERROR or PARSE_INSUFFICIENT | ||||||
|  |  */ | ||||||
|  | static inline ParseRC | ||||||
|  | parse_length (ParseState *state, size_t *len) | ||||||
|  | { | ||||||
|  |     ParseRC ret = PARSE_ERROR; | ||||||
|  |     *len = 0; | ||||||
|  | 
 | ||||||
|  |     if (psyc_is_numeric(state->buffer.data[state->cursor])) { | ||||||
|  | 	ret = PARSE_SUCCESS; | ||||||
|  | 	do { | ||||||
|  | 	    *len = 10 * *len + state->buffer.data[state->cursor] - '0'; | ||||||
|  | 	    ADVANCE_CURSOR_OR_RETURN(PARSE_INSUFFICIENT); | ||||||
|  | 	} while (psyc_is_numeric(state->buffer.data[state->cursor])); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| /**
 | /**
 | ||||||
|  * Parse binary data. |  * Parse binary data. | ||||||
|  * |  * | ||||||
|  | @ -50,13 +84,12 @@ parse_keyword (PsycParseState *state, PsycString *name) | ||||||
|  * @param length Expected length of the data. |  * @param length Expected length of the data. | ||||||
|  * @param parsed Number of bytes parsed so far. |  * @param parsed Number of bytes parsed so far. | ||||||
|  * |  * | ||||||
|  * @return PARSE_COMPLETE or PARSE_INCOMPLETE |  * @return PARSE_SUCCESS or PARSE_INCOMPLETE | ||||||
|  */ |  */ | ||||||
| static inline ParseRC | static inline ParseRC | ||||||
| psyc_parse_binary_value (PsycParseState *state, PsycString *value, | parse_binary (ParseState *state, size_t length, PsycString *value, size_t *parsed) | ||||||
| 			 size_t *length, size_t *parsed) |  | ||||||
| { | { | ||||||
|     size_t remaining = *length - *parsed; |     size_t remaining = length - *parsed; | ||||||
|     value->data = state->buffer.data + state->cursor; |     value->data = state->buffer.data + state->cursor; | ||||||
| 
 | 
 | ||||||
|     if (state->cursor + remaining > state->buffer.length) { |     if (state->cursor + remaining > state->buffer.length) { | ||||||
|  | @ -70,9 +103,32 @@ psyc_parse_binary_value (PsycParseState *state, PsycString *value, | ||||||
|     value->length = remaining; |     value->length = remaining; | ||||||
|     state->cursor += remaining; |     state->cursor += remaining; | ||||||
|     *parsed += remaining; |     *parsed += remaining; | ||||||
|     assert(*parsed == *length); |     ASSERT(*parsed == length); | ||||||
| 
 | 
 | ||||||
|     return PARSE_COMPLETE; |     return PARSE_SUCCESS; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Parse data until a given character is found. | ||||||
|  |  * | ||||||
|  |  * @param state  Parser state. | ||||||
|  |  * @param value  Start & length of parsed data is saved here. | ||||||
|  |  * @param end    Parse until this character is found. | ||||||
|  |  * @param parsed Number of bytes parsed so far. | ||||||
|  |  * | ||||||
|  |  * @return PARSE_SUCCESS or PARSE_INSUFFICIENT | ||||||
|  |  */ | ||||||
|  | static inline ParseRC | ||||||
|  | parse_until (ParseState *state, const char end, PsycString *value) | ||||||
|  | { | ||||||
|  |     value->data = state->buffer.data + state->cursor; | ||||||
|  | 
 | ||||||
|  |     while (state->buffer.data[state->cursor] != end) { | ||||||
|  | 	value->length++; | ||||||
|  | 	ADVANCE_CURSOR_OR_RETURN(PARSE_INSUFFICIENT); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return PARSE_SUCCESS; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  | @ -86,7 +142,7 @@ psyc_parse_modifier (PsycParseState *state, char *oper, | ||||||
|     *oper = *(state->buffer.data + state->cursor); |     *oper = *(state->buffer.data + state->cursor); | ||||||
|     ADVANCE_CURSOR_OR_RETURN(PSYC_PARSE_INSUFFICIENT); |     ADVANCE_CURSOR_OR_RETURN(PSYC_PARSE_INSUFFICIENT); | ||||||
| 
 | 
 | ||||||
|     ParseRC ret = parse_keyword(state, name); |     ParseRC ret = parse_keyword((ParseState*)state, name); | ||||||
|     if (ret == PARSE_ERROR) |     if (ret == PARSE_ERROR) | ||||||
| 	return PSYC_PARSE_ERROR_MOD_NAME; | 	return PSYC_PARSE_ERROR_MOD_NAME; | ||||||
|     else if (ret != PARSE_SUCCESS) |     else if (ret != PARSE_SUCCESS) | ||||||
|  | @ -123,9 +179,7 @@ psyc_parse_modifier (PsycParseState *state, char *oper, | ||||||
| 	if (++(state->cursor) >= state->buffer.length) | 	if (++(state->cursor) >= state->buffer.length) | ||||||
| 	    return length ? PARSE_INCOMPLETE : PARSE_SUCCESS; // if length=0 we're done
 | 	    return length ? PARSE_INCOMPLETE : PARSE_SUCCESS; // if length=0 we're done
 | ||||||
| 
 | 
 | ||||||
| 	ret = | 	ret = parse_binary((ParseState*)state, state->valuelen, value, &state->value_parsed); | ||||||
| 	    psyc_parse_binary_value(state, value, &(state->valuelen), |  | ||||||
| 				    &(state->value_parsed)); |  | ||||||
| 	if (ret == PARSE_INCOMPLETE) | 	if (ret == PARSE_INCOMPLETE) | ||||||
| 	    return ret; | 	    return ret; | ||||||
| 
 | 
 | ||||||
|  | @ -244,8 +298,8 @@ psyc_parse (PsycParseState *state, char *oper, | ||||||
|     case PSYC_PART_CONTENT: |     case PSYC_PART_CONTENT: | ||||||
| 	// In case of an incomplete binary variable resume parsing it.
 | 	// In case of an incomplete binary variable resume parsing it.
 | ||||||
| 	if (state->value_parsed < state->valuelen) { | 	if (state->value_parsed < state->valuelen) { | ||||||
| 	    ret = psyc_parse_binary_value(state, value, &(state->valuelen), | 	    ret = parse_binary((ParseState*)state, state->valuelen, value, | ||||||
| 					  &(state->value_parsed)); | 			       &state->value_parsed); | ||||||
| 	    state->content_parsed += value->length; | 	    state->content_parsed += value->length; | ||||||
| 
 | 
 | ||||||
| 	    if (ret == PARSE_INCOMPLETE) | 	    if (ret == PARSE_INCOMPLETE) | ||||||
|  | @ -303,7 +357,7 @@ psyc_parse (PsycParseState *state, char *oper, | ||||||
| 
 | 
 | ||||||
|     case PSYC_PART_METHOD: |     case PSYC_PART_METHOD: | ||||||
| 	pos = state->cursor; | 	pos = state->cursor; | ||||||
| 	ret = parse_keyword(state, name); | 	ret = parse_keyword((ParseState*)state, name); | ||||||
| 
 | 
 | ||||||
| 	if (ret == PARSE_INSUFFICIENT) | 	if (ret == PARSE_INSUFFICIENT) | ||||||
| 	    return ret; | 	    return ret; | ||||||
|  | @ -345,8 +399,8 @@ psyc_parse (PsycParseState *state, char *oper, | ||||||
| 		    state->valuelen--; // \n at the end is not part of data
 | 		    state->valuelen--; // \n at the end is not part of data
 | ||||||
| 	    } | 	    } | ||||||
| 	    if (state->value_parsed < state->valuelen) { | 	    if (state->value_parsed < state->valuelen) { | ||||||
| 		ret = psyc_parse_binary_value(state, value, &(state->valuelen), | 		ret = parse_binary((ParseState*)state, state->valuelen, value, | ||||||
| 					      &(state->value_parsed)); | 				   &state->value_parsed); | ||||||
| 		state->content_parsed += value->length; | 		state->content_parsed += value->length; | ||||||
| 
 | 
 | ||||||
| 		if (ret == PARSE_INCOMPLETE) | 		if (ret == PARSE_INCOMPLETE) | ||||||
|  | @ -422,191 +476,653 @@ psyc_parse (PsycParseState *state, char *oper, | ||||||
|     return PSYC_PARSE_ERROR; // should not be reached
 |     return PSYC_PARSE_ERROR; // should not be reached
 | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /** List parser. */ | /**
 | ||||||
|  |  * Parse list. | ||||||
|  |  * | ||||||
|  |  * list		= [ default-type ] *list-elem | ||||||
|  |  * list-elem	= "|" ( type [ SP list-value ] / [ length ] [ ":" type ] [ SP *OCTET ] ) | ||||||
|  |  * list-value	= %x00-7B / %x7D-FF	; any byte except "|" | ||||||
|  |  */ | ||||||
| #ifdef __INLINE_PSYC_PARSE | #ifdef __INLINE_PSYC_PARSE | ||||||
| static inline | static inline | ||||||
| #endif | #endif | ||||||
| PsycParseListRC | PsycParseListRC | ||||||
| psyc_parse_list (PsycParseListState *state, PsycString *elem) | psyc_parse_list (PsycParseListState *state, PsycString *type, PsycString *elem) | ||||||
| { | { | ||||||
|  |     ParseRC ret; | ||||||
|  | 
 | ||||||
|     if (state->cursor >= state->buffer.length) |     if (state->cursor >= state->buffer.length) | ||||||
| 	return PSYC_PARSE_LIST_INCOMPLETE; | 	return PSYC_PARSE_LIST_END; | ||||||
| 
 |  | ||||||
|     state->startc = state->cursor; |  | ||||||
| 
 |  | ||||||
|     if (!state->type) {	// If type is not set we're at the start.
 |  | ||||||
| 	// First character is either | for text lists, or a number for binary lists
 |  | ||||||
| 	if (state->buffer.data[state->cursor] == '|') { |  | ||||||
| 	    state->type = PSYC_LIST_TEXT; |  | ||||||
| 	    state->cursor++; |  | ||||||
| 	} else if (psyc_is_numeric(state->buffer.data[state->cursor])) |  | ||||||
| 	    state->type = PSYC_LIST_BINARY; |  | ||||||
| 	else |  | ||||||
| 	    return PSYC_PARSE_LIST_ERROR_TYPE; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     if (state->type == PSYC_LIST_TEXT) { |  | ||||||
| 	elem->data = state->buffer.data + state->cursor; |  | ||||||
| 	elem->length = 0; |  | ||||||
| 
 |  | ||||||
| 	if (state->cursor >= state->buffer.length) |  | ||||||
| 	    return PSYC_PARSE_LIST_END; |  | ||||||
| 
 |  | ||||||
| 	if (state->term_set) { |  | ||||||
| 	    while (state->buffer.data[state->cursor] != '|') { |  | ||||||
| 		elem->length++; |  | ||||||
| 		if (state->buffer.data[state->cursor] == state->term) |  | ||||||
| 		    return PSYC_PARSE_LIST_END; |  | ||||||
| 		if (++(state->cursor) >= state->buffer.length) |  | ||||||
| 		    return PSYC_PARSE_LIST_END; |  | ||||||
| 	    } |  | ||||||
| 	} else { |  | ||||||
| 	    while (state->buffer.data[state->cursor] != '|') { |  | ||||||
| 		elem->length++; |  | ||||||
| 		if (++(state->cursor) >= state->buffer.length) |  | ||||||
| 		    return PSYC_PARSE_LIST_END; |  | ||||||
| 	    } |  | ||||||
| 	} |  | ||||||
| 	state->cursor++; |  | ||||||
| 	return PSYC_PARSE_LIST_ELEM; |  | ||||||
|     } else { // binary list
 |  | ||||||
| 	if (!(state->elem_parsed < state->elemlen)) { |  | ||||||
| 	    // Element starts with a number.
 |  | ||||||
| 	    if (psyc_is_numeric(state->buffer.data[state->cursor])) { |  | ||||||
| 		do { |  | ||||||
| 		    state->elemlen = |  | ||||||
| 			10 * state->elemlen + |  | ||||||
| 			state->buffer.data[state->cursor] - '0'; |  | ||||||
| 		    ADVANCE_CURSOR_OR_RETURN(PSYC_PARSE_LIST_INCOMPLETE); |  | ||||||
| 		} while (psyc_is_numeric(state->buffer.data[state->cursor])); |  | ||||||
| 	    } else |  | ||||||
| 		return PSYC_PARSE_LIST_ERROR_LEN; |  | ||||||
| 
 |  | ||||||
| 	    if (state->buffer.data[state->cursor] != ' ') |  | ||||||
| 		return PSYC_PARSE_LIST_ERROR_LEN; |  | ||||||
| 
 |  | ||||||
| 	    state->cursor++; |  | ||||||
| 	    elem->data = state->buffer.data + state->cursor; |  | ||||||
| 	    elem->length = 0; |  | ||||||
| 	    state->elem_parsed = 0; |  | ||||||
| 	} |  | ||||||
| 	// Start or resume parsing the binary data
 |  | ||||||
| 	if (state->elem_parsed < state->elemlen) { |  | ||||||
| 	    if (PARSE_INCOMPLETE == psyc_parse_binary_value((PsycParseState*)state, |  | ||||||
| 							    elem, &state->elemlen, |  | ||||||
| 							    &state->elem_parsed)) |  | ||||||
| 		return PSYC_PARSE_LIST_INCOMPLETE; |  | ||||||
| 
 |  | ||||||
| 	    state->elemlen = 0; |  | ||||||
| 
 |  | ||||||
| 	    if (state->cursor >= state->buffer.length) |  | ||||||
| 		return PSYC_PARSE_LIST_END; |  | ||||||
| 
 |  | ||||||
| 	    if (state->buffer.data[state->cursor] != '|') |  | ||||||
| 		return PSYC_PARSE_LIST_ERROR_DELIM; |  | ||||||
| 
 |  | ||||||
| 	    state->cursor++; |  | ||||||
| 	    return PSYC_PARSE_LIST_ELEM; |  | ||||||
| 	} |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     return PSYC_PARSE_LIST_ERROR; // should not be reached
 |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| PsycParseTableRC |  | ||||||
| psyc_parse_table (PsycParseTableState *state, PsycString *elem) |  | ||||||
| { |  | ||||||
|     if (state->cursor >= state->buffer.length) |  | ||||||
| 	return PSYC_PARSE_TABLE_INCOMPLETE; |  | ||||||
| 
 | 
 | ||||||
|     state->startc = state->cursor; |     state->startc = state->cursor; | ||||||
| 
 | 
 | ||||||
|     switch (state->part) { |     switch (state->part) { | ||||||
|     case PSYC_TABLE_PART_START: |     case PSYC_LIST_PART_START: | ||||||
| 	if (state->buffer.data[state->cursor] != '*') { | 	type->length = elem->length = 0; | ||||||
| 	    state->part = PSYC_TABLE_PART_BODY_START; | 	type->data = elem->data = NULL; | ||||||
| 	    goto PSYC_TABLE_PART_BODY_START; | 
 | ||||||
| 	} else { | 	state->part = PSYC_LIST_PART_TYPE; | ||||||
| 	    state->part = PSYC_TABLE_PART_WIDTH; | 	// fall thru
 | ||||||
| 	    ADVANCE_CURSOR_OR_RETURN(PSYC_PARSE_TABLE_INCOMPLETE); | 
 | ||||||
|  |     case PSYC_LIST_PART_TYPE: | ||||||
|  | 	switch (parse_keyword((ParseState*)state, type)) { | ||||||
|  | 	case PARSE_SUCCESS: // end of keyword
 | ||||||
|  | 	    state->part = PSYC_LIST_PART_ELEM_START; | ||||||
|  | 	    return PSYC_PARSE_LIST_TYPE; | ||||||
|  | 	case PARSE_INSUFFICIENT: // end of buffer
 | ||||||
|  | 	    return PSYC_PARSE_LIST_END; | ||||||
|  | 	case PARSE_ERROR: // no keyword
 | ||||||
|  | 	    state->part = PSYC_LIST_PART_ELEM_START; | ||||||
|  | 	    break; | ||||||
|  | 	default: // should not be reached
 | ||||||
|  | 	    return PSYC_PARSE_LIST_ERROR; | ||||||
| 	} | 	} | ||||||
| 	// fall thru
 | 	// fall thru
 | ||||||
| 
 | 
 | ||||||
|     case PSYC_TABLE_PART_WIDTH: |     case PSYC_LIST_PART_ELEM_START: | ||||||
| 	if (psyc_is_numeric(state->buffer.data[state->cursor])) { | 	if (state->buffer.data[state->cursor] != '|') | ||||||
| 	    do { | 	    return PSYC_PARSE_LIST_ERROR_ELEM_START; | ||||||
| 		state->width = | 
 | ||||||
| 		    10 * state->width + state->buffer.data[state->cursor] - '0'; | 	type->length = elem->length = 0; | ||||||
| 		ADVANCE_CURSOR_OR_RETURN(PSYC_PARSE_TABLE_INCOMPLETE); | 	type->data = elem->data = NULL; | ||||||
| 	    } while (psyc_is_numeric(state->buffer.data[state->cursor])); | 
 | ||||||
| 	} else | 	state->elem_parsed = 0; | ||||||
| 	    return PSYC_PARSE_TABLE_ERROR_WIDTH; | 	state->elemlen_found = 0; | ||||||
|  | 
 | ||||||
|  | 	state->part = PSYC_LIST_PART_ELEM_LENGTH; | ||||||
|  | 	ADVANCE_STARTC_OR_RETURN(PSYC_PARSE_LIST_ELEM_LAST); | ||||||
|  | 	// fall thru
 | ||||||
|  | 
 | ||||||
|  |     case PSYC_LIST_PART_ELEM_TYPE: | ||||||
|  | 	if (state->buffer.data[state->cursor] == '=') { | ||||||
|  | 	    ADVANCE_CURSOR_OR_RETURN(PSYC_PARSE_INSUFFICIENT); | ||||||
|  | 
 | ||||||
|  | 	    switch (parse_keyword((ParseState*)state, type)) { | ||||||
|  | 	    case PARSE_SUCCESS: | ||||||
|  | 		switch (state->buffer.data[state->cursor]) { | ||||||
|  | 		case ':': | ||||||
|  | 		    state->part = PSYC_LIST_PART_ELEM_LENGTH; | ||||||
|  | 		    ADVANCE_STARTC_OR_RETURN(PSYC_PARSE_LIST_ELEM_LAST); | ||||||
|  | 		    break; | ||||||
|  | 		case ' ': | ||||||
|  | 		    state->part = PSYC_LIST_PART_ELEM; | ||||||
|  | 		    ADVANCE_STARTC_OR_RETURN(PSYC_PARSE_LIST_ELEM_LAST); | ||||||
|  | 		    goto PSYC_LIST_PART_ELEM; | ||||||
|  | 		case '|': | ||||||
|  | 		    state->part = PSYC_LIST_PART_ELEM_START; | ||||||
|  | 		    return PSYC_PARSE_LIST_ELEM; | ||||||
|  | 		    break; | ||||||
|  | 		default: | ||||||
|  | 		    return PSYC_PARSE_LIST_ERROR_ELEM_TYPE; | ||||||
|  | 		} | ||||||
|  | 		break; | ||||||
|  | 	    case PARSE_INSUFFICIENT: // end of buffer
 | ||||||
|  | 		return PSYC_PARSE_LIST_ELEM_LAST; | ||||||
|  | 	    case PARSE_ERROR: | ||||||
|  | 		return PSYC_PARSE_LIST_ERROR_ELEM_TYPE; | ||||||
|  | 	    default: // should not be reached
 | ||||||
|  | 		return PSYC_PARSE_LIST_ERROR; | ||||||
|  | 	    } | ||||||
|  | 	} | ||||||
|  | 	// fall thru
 | ||||||
|  | 
 | ||||||
|  |     case PSYC_LIST_PART_ELEM_LENGTH: | ||||||
|  | 	switch (parse_length((ParseState*)state, &state->elemlen)) { | ||||||
|  | 	case PARSE_SUCCESS: // length is complete
 | ||||||
|  | 	    state->elemlen_found = 1; | ||||||
|  | 	    state->elem_parsed = 0; | ||||||
|  | 	    elem->length = state->elemlen; | ||||||
|  | 	    elem->data = NULL; | ||||||
|  | 	    break; | ||||||
|  | 	case PARSE_INSUFFICIENT: // length is incomplete
 | ||||||
|  | 	    return PSYC_PARSE_LIST_INSUFFICIENT; | ||||||
|  | 	case PARSE_ERROR: // no length
 | ||||||
|  | 	    break; | ||||||
|  | 	default: // should not be reached
 | ||||||
|  | 	    return PSYC_PARSE_LIST_ERROR; | ||||||
|  | 	} | ||||||
| 
 | 
 | ||||||
| 	switch (state->buffer.data[state->cursor]) { | 	switch (state->buffer.data[state->cursor]) { | ||||||
| #ifdef PSYC_PARSE_TABLE_HEAD |  | ||||||
| 	case '|': |  | ||||||
| 	    state->part = PSYC_TABLE_PART_HEAD_START; |  | ||||||
| 	    break; |  | ||||||
| #endif |  | ||||||
| 	case ' ': | 	case ' ': | ||||||
| 	    state->part = PSYC_TABLE_PART_BODY_START; | 	    state->part = PSYC_LIST_PART_ELEM; | ||||||
| 	    state->cursor++; | 	    ADVANCE_STARTC_OR_RETURN(PSYC_PARSE_LIST_ELEM_LAST); | ||||||
|  | 	    break; | ||||||
|  | 	case '|': | ||||||
|  | 	    state->part = PSYC_LIST_PART_ELEM_START; | ||||||
|  | 	    return PSYC_PARSE_LIST_ELEM; | ||||||
|  | 	default: | ||||||
|  | 	    return PSYC_PARSE_LIST_ERROR_ELEM_LENGTH; | ||||||
| 	} | 	} | ||||||
| 
 |  | ||||||
| 	elem->length = state->width; |  | ||||||
| 	return PSYC_TABLE_PART_WIDTH; |  | ||||||
| #ifdef PSYC_PARSE_TABLE_HEAD |  | ||||||
|     case PSYC_TABLE_PART_HEAD_START: |  | ||||||
| 	psyc_parse_list_buffer_set(&state->list, state->buffer.data + state->cursor, |  | ||||||
| 				   state->buffer.length - state->cursor); |  | ||||||
| 	psyc_parse_list_term_set(&state->list, ' '); |  | ||||||
| 	state->part = PSYC_TABLE_PART_HEAD; |  | ||||||
| 	// fall thru
 | 	// fall thru
 | ||||||
| 
 | 
 | ||||||
|     case PSYC_TABLE_PART_HEAD: |     case PSYC_LIST_PART_ELEM: | ||||||
| 	switch (psyc_parse_list(&state->list, elem)) { |     PSYC_LIST_PART_ELEM: | ||||||
| 	case PSYC_PARSE_LIST_ELEM: | 	if (state->elemlen_found) { | ||||||
| 	    if (state->elems == 0) { | 	    switch (parse_binary((ParseState*)state, state->elemlen, elem, | ||||||
| 		state->elems++; | 				 &state->elem_parsed)) { | ||||||
| 		return PSYC_PARSE_TABLE_NAME_KEY; | 	    case PARSE_SUCCESS: | ||||||
| 	    } else if (state->elems < state->width) { | 		if (elem->length == state->elem_parsed) | ||||||
| 		state->elems++; | 		    ret = PSYC_PARSE_LIST_ELEM; | ||||||
| 		return PSYC_PARSE_TABLE_NAME_VALUE; | 		else | ||||||
| 	    } else // too many elements
 | 		    ret = PSYC_PARSE_LIST_ELEM_END; | ||||||
| 		return PSYC_PARSE_TABLE_ERROR_HEAD; | 		break; | ||||||
| 
 | 	    case PARSE_INCOMPLETE: | ||||||
| 	case PSYC_PARSE_LIST_END: | 		if (elem->length == state->elem_parsed) | ||||||
| 	    if (state->elems != state->width) | 		    ret = PSYC_PARSE_LIST_ELEM_START; | ||||||
| 		return PSYC_PARSE_TABLE_ERROR_HEAD; | 		else | ||||||
| 
 | 		    ret = PSYC_PARSE_LIST_ELEM_CONT; | ||||||
| 	    state->part = PSYC_TABLE_PART_BODY_START; | 		break; | ||||||
| 	    state->cursor += state->list.cursor + 1; | 	    default: // should not be reached
 | ||||||
| 	    psyc_parse_list_state_init(&state->list); | 		return PSYC_PARSE_LIST_ERROR; | ||||||
| 	    return state->elems++ == 0 | 	    } | ||||||
| 		? PSYC_PARSE_TABLE_NAME_KEY : PSYC_PARSE_TABLE_NAME_VALUE; | 	} else { | ||||||
| 	default: | 	    switch (parse_until((ParseState*)state, '|', elem)) { | ||||||
| 	    return PSYC_PARSE_TABLE_ERROR_HEAD; | 	    case PARSE_SUCCESS: | ||||||
|  | 		ret = PSYC_PARSE_LIST_ELEM; | ||||||
|  | 		break; | ||||||
|  | 	    case PARSE_INSUFFICIENT: | ||||||
|  | 		return PSYC_PARSE_LIST_ELEM_LAST; | ||||||
|  | 	    default: // should not be reached
 | ||||||
|  | 		return PSYC_PARSE_LIST_ERROR; | ||||||
|  | 	    } | ||||||
| 	} | 	} | ||||||
| #endif |  | ||||||
|     case PSYC_TABLE_PART_BODY_START: |  | ||||||
|     PSYC_TABLE_PART_BODY_START: |  | ||||||
| 	psyc_parse_list_buffer_set(&state->list, state->buffer.data + state->cursor, |  | ||||||
| 				   state->buffer.length - state->cursor); |  | ||||||
| 	state->part = PSYC_TABLE_PART_BODY; |  | ||||||
| 	// fall thru
 |  | ||||||
| 
 | 
 | ||||||
|     case PSYC_TABLE_PART_BODY: | 	state->part = PSYC_LIST_PART_ELEM_START; | ||||||
| 	switch (psyc_parse_list(&state->list, elem)) { | 	state->startc = state->cursor; | ||||||
| 	case PSYC_PARSE_LIST_ELEM: | 	return ret; | ||||||
| 	    return state->elems++ % (state->width + 1) == 0 |  | ||||||
| 		? PSYC_PARSE_TABLE_KEY : PSYC_PARSE_TABLE_VALUE; |  | ||||||
| 	case PSYC_PARSE_LIST_END: |  | ||||||
| 	    return state->elems++ % (state->width + 1) == 0 |  | ||||||
| 		? PSYC_PARSE_TABLE_KEY_END : PSYC_PARSE_TABLE_VALUE_END; |  | ||||||
| 	default: |  | ||||||
| 	    return PSYC_PARSE_TABLE_ERROR_BODY; |  | ||||||
| 	} |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     return PSYC_PARSE_LIST_ERROR; // should not be reached
 |     return PSYC_PARSE_LIST_ERROR; // should not be reached
 | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Parse dictionary. | ||||||
|  |  * | ||||||
|  |  * dict		= [ type ] *dict-item | ||||||
|  |  * dict-item	= "{" ( dict-key / length SP OCTET) "}" | ||||||
|  |  *                ( type [ SP dict-value ] / [ length ] [ ":" type ] [ SP *OCTET ] ) | ||||||
|  |  * dict-key	= %x00-7C / %x7E-FF	; any byte except "{" | ||||||
|  |  * dict-value	= %x00-7A / %x7C-FF	; any byte except "}" | ||||||
|  |  */ | ||||||
|  | PsycParseDictRC | ||||||
|  | psyc_parse_dict (PsycParseDictState *state, PsycString *type, PsycString *elem) | ||||||
|  | { | ||||||
|  |     ParseRC ret; | ||||||
|  | 
 | ||||||
|  |     if (state->cursor >= state->buffer.length) | ||||||
|  | 	return PSYC_PARSE_DICT_END; | ||||||
|  | 
 | ||||||
|  |     state->startc = state->cursor; | ||||||
|  | 
 | ||||||
|  |     switch (state->part) { | ||||||
|  |     case PSYC_DICT_PART_START: | ||||||
|  | 	type->length = elem->length = 0; | ||||||
|  | 	type->data = elem->data = NULL; | ||||||
|  | 
 | ||||||
|  | 	state->part = PSYC_DICT_PART_TYPE; | ||||||
|  | 	// fall thru
 | ||||||
|  | 
 | ||||||
|  |     case PSYC_DICT_PART_TYPE: | ||||||
|  | 	switch (parse_keyword((ParseState*)state, type)) { | ||||||
|  | 	case PARSE_SUCCESS: // end of keyword
 | ||||||
|  | 	    state->part = PSYC_DICT_PART_KEY_START; | ||||||
|  | 	    return PSYC_PARSE_DICT_TYPE; | ||||||
|  | 	case PARSE_INSUFFICIENT: // end of buffer
 | ||||||
|  | 	    return PSYC_PARSE_DICT_END; | ||||||
|  | 	case PARSE_ERROR: // no keyword
 | ||||||
|  | 	    state->part = PSYC_DICT_PART_KEY_START; | ||||||
|  | 	    break; | ||||||
|  | 	default: // should not be reached
 | ||||||
|  | 	    return PSYC_PARSE_DICT_ERROR; | ||||||
|  | 	} | ||||||
|  | 	// fall thru
 | ||||||
|  | 
 | ||||||
|  |     case PSYC_DICT_PART_KEY_START: | ||||||
|  | 	if (state->buffer.data[state->cursor] != '{') | ||||||
|  | 	    return PSYC_PARSE_DICT_ERROR_KEY_START; | ||||||
|  | 
 | ||||||
|  | 	type->length = elem->length = 0; | ||||||
|  | 	type->data = elem->data = NULL; | ||||||
|  | 
 | ||||||
|  | 	state->elem_parsed = 0; | ||||||
|  | 	state->elemlen_found = 0; | ||||||
|  | 
 | ||||||
|  | 	state->part = PSYC_DICT_PART_KEY_LENGTH; | ||||||
|  | 	ADVANCE_STARTC_OR_RETURN(PSYC_PARSE_DICT_INSUFFICIENT); | ||||||
|  | 	// fall thru
 | ||||||
|  | 
 | ||||||
|  |     case PSYC_DICT_PART_KEY_LENGTH: | ||||||
|  | 	switch (parse_length((ParseState*)state, &state->elemlen)) { | ||||||
|  | 	case PARSE_SUCCESS: // length is complete
 | ||||||
|  | 	    state->elemlen_found = 1; | ||||||
|  | 	    state->elem_parsed = 0; | ||||||
|  | 	    elem->length = state->elemlen; | ||||||
|  | 	    elem->data = NULL; | ||||||
|  | 
 | ||||||
|  | 	    if (state->buffer.data[state->cursor] != ' ') | ||||||
|  | 		return PSYC_PARSE_DICT_ERROR_KEY_LENGTH; | ||||||
|  | 
 | ||||||
|  | 	    state->part = PSYC_DICT_PART_KEY; | ||||||
|  | 	    ADVANCE_STARTC_OR_RETURN(PSYC_PARSE_INSUFFICIENT); | ||||||
|  | 	    break; | ||||||
|  | 	case PARSE_INSUFFICIENT: // length is incomplete
 | ||||||
|  | 	    return PSYC_PARSE_DICT_INSUFFICIENT; | ||||||
|  | 	case PARSE_ERROR: // no length
 | ||||||
|  | 	    state->part = PSYC_DICT_PART_KEY; | ||||||
|  | 	    break; | ||||||
|  | 	default: // should not be reached
 | ||||||
|  | 	    return PSYC_PARSE_DICT_ERROR; | ||||||
|  | 	} | ||||||
|  | 	// fall thru
 | ||||||
|  | 
 | ||||||
|  |     case PSYC_DICT_PART_KEY: | ||||||
|  | 	if (state->elemlen_found) { | ||||||
|  | 	    switch (parse_binary((ParseState*)state, state->elemlen, elem, | ||||||
|  | 				 &state->elem_parsed)) { | ||||||
|  | 	    case PARSE_SUCCESS: | ||||||
|  | 		if (elem->length == state->elem_parsed) | ||||||
|  | 		    ret = PSYC_PARSE_DICT_KEY; | ||||||
|  | 		else | ||||||
|  | 		    ret = PSYC_PARSE_DICT_KEY_END; | ||||||
|  | 		break; | ||||||
|  | 	    case PARSE_INCOMPLETE: | ||||||
|  | 		if (elem->length == state->elem_parsed) | ||||||
|  | 		    ret = PSYC_PARSE_DICT_KEY_START; | ||||||
|  | 		else | ||||||
|  | 		    ret = PSYC_PARSE_DICT_KEY_CONT; | ||||||
|  | 		break; | ||||||
|  | 	    default: // should not be reached
 | ||||||
|  | 		return PSYC_PARSE_DICT_ERROR; | ||||||
|  | 	    } | ||||||
|  | 	} else { | ||||||
|  | 	    switch (parse_until((ParseState*)state, '}', elem)) { | ||||||
|  | 	    case PARSE_SUCCESS: | ||||||
|  | 		ret = PSYC_PARSE_DICT_KEY; | ||||||
|  | 		break; | ||||||
|  | 	    case PARSE_INSUFFICIENT: | ||||||
|  | 		return PSYC_PARSE_DICT_INSUFFICIENT; | ||||||
|  | 	    default: // should not be reached
 | ||||||
|  | 		return PSYC_PARSE_DICT_ERROR; | ||||||
|  | 	    } | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	state->part = PSYC_DICT_PART_VALUE_START; | ||||||
|  | 	state->startc = state->cursor; | ||||||
|  | 	return ret; | ||||||
|  | 
 | ||||||
|  |     case PSYC_DICT_PART_VALUE_START: | ||||||
|  | 	switch (state->buffer.data[state->cursor] != '}') | ||||||
|  | 	    return PSYC_PARSE_DICT_ERROR_VALUE_START; | ||||||
|  | 
 | ||||||
|  | 	type->length = elem->length = 0; | ||||||
|  | 	type->data = elem->data = NULL; | ||||||
|  | 
 | ||||||
|  | 	state->elem_parsed = 0; | ||||||
|  | 	state->elemlen_found = 0; | ||||||
|  | 
 | ||||||
|  | 	state->part = PSYC_DICT_PART_VALUE_TYPE; | ||||||
|  | 	ADVANCE_STARTC_OR_RETURN(PSYC_PARSE_DICT_VALUE_LAST); | ||||||
|  | 	// fall thru
 | ||||||
|  | 
 | ||||||
|  |     case PSYC_DICT_PART_VALUE_TYPE: | ||||||
|  | 	if (state->buffer.data[state->cursor] == '=') { | ||||||
|  | 	    ADVANCE_CURSOR_OR_RETURN(PSYC_PARSE_INSUFFICIENT); | ||||||
|  | 
 | ||||||
|  | 	    switch (parse_keyword((ParseState*)state, type)) { | ||||||
|  | 	    case PARSE_SUCCESS: | ||||||
|  | 		switch (state->buffer.data[state->cursor]) { | ||||||
|  | 		case ':': | ||||||
|  | 		    state->part = PSYC_DICT_PART_VALUE_LENGTH; | ||||||
|  | 		    ADVANCE_STARTC_OR_RETURN(PSYC_PARSE_DICT_VALUE_LAST); | ||||||
|  | 		    break; | ||||||
|  | 		case ' ': | ||||||
|  | 		    state->part = PSYC_DICT_PART_VALUE; | ||||||
|  | 		    ADVANCE_STARTC_OR_RETURN(PSYC_PARSE_DICT_VALUE_LAST); | ||||||
|  | 		    goto PSYC_DICT_PART_VALUE; | ||||||
|  | 		case '{': | ||||||
|  | 		    state->part = PSYC_DICT_PART_KEY_START; | ||||||
|  | 		    return PSYC_PARSE_DICT_VALUE; | ||||||
|  | 		    break; | ||||||
|  | 		default: | ||||||
|  | 		    return PSYC_PARSE_DICT_ERROR_VALUE_TYPE; | ||||||
|  | 		} | ||||||
|  | 		break; | ||||||
|  | 	    case PARSE_INSUFFICIENT: // end of buffer
 | ||||||
|  | 		return PSYC_PARSE_DICT_VALUE_LAST; | ||||||
|  | 	    case PARSE_ERROR: | ||||||
|  | 		return PSYC_PARSE_DICT_ERROR_VALUE_TYPE; | ||||||
|  | 	    default: // should not be reached
 | ||||||
|  | 		return PSYC_PARSE_DICT_ERROR; | ||||||
|  | 	    } | ||||||
|  | 	} | ||||||
|  | 	// fall thru
 | ||||||
|  | 
 | ||||||
|  |     case PSYC_DICT_PART_VALUE_LENGTH: | ||||||
|  | 	switch (parse_length((ParseState*)state, &state->elemlen)) { | ||||||
|  | 	case PARSE_SUCCESS: // length is complete
 | ||||||
|  | 	    state->elemlen_found = 1; | ||||||
|  | 	    state->elem_parsed = 0; | ||||||
|  | 	    elem->length = state->elemlen; | ||||||
|  | 	    elem->data = NULL; | ||||||
|  | 	    break; | ||||||
|  | 	case PARSE_INSUFFICIENT: // length is incomplete
 | ||||||
|  | 	    return PSYC_PARSE_DICT_INSUFFICIENT; | ||||||
|  | 	case PARSE_ERROR: // no length
 | ||||||
|  | 	    break; | ||||||
|  | 	default: // should not be reached
 | ||||||
|  | 	    return PSYC_PARSE_DICT_ERROR; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	switch (state->buffer.data[state->cursor]) { | ||||||
|  | 	case ' ': | ||||||
|  | 	    state->part = PSYC_DICT_PART_VALUE; | ||||||
|  | 	    ADVANCE_STARTC_OR_RETURN(PSYC_PARSE_DICT_VALUE_LAST); | ||||||
|  | 	    break; | ||||||
|  | 	case '{': | ||||||
|  | 	    state->part = PSYC_DICT_PART_KEY_START; | ||||||
|  | 	    return PSYC_PARSE_DICT_VALUE; | ||||||
|  | 	default: | ||||||
|  | 	    return PSYC_PARSE_DICT_ERROR_VALUE_LENGTH; | ||||||
|  | 	} | ||||||
|  | 	// fall thru
 | ||||||
|  | 
 | ||||||
|  |     case PSYC_DICT_PART_VALUE: | ||||||
|  |     PSYC_DICT_PART_VALUE: | ||||||
|  | 	if (state->elemlen_found) { | ||||||
|  | 	    switch (parse_binary((ParseState*)state, state->elemlen, elem, | ||||||
|  | 				 &state->elem_parsed)) { | ||||||
|  | 	    case PARSE_SUCCESS: | ||||||
|  | 		if (elem->length == state->elem_parsed) | ||||||
|  | 		    ret = PSYC_PARSE_DICT_VALUE; | ||||||
|  | 		else | ||||||
|  | 		    ret = PSYC_PARSE_DICT_VALUE_END; | ||||||
|  | 		break; | ||||||
|  | 	    case PARSE_INCOMPLETE: | ||||||
|  | 		if (elem->length == state->elem_parsed) | ||||||
|  | 		    ret = PSYC_PARSE_DICT_VALUE_START; | ||||||
|  | 		else | ||||||
|  | 		    ret = PSYC_PARSE_DICT_VALUE_CONT; | ||||||
|  | 		break; | ||||||
|  | 	    default: // should not be reached
 | ||||||
|  | 		return PSYC_PARSE_DICT_ERROR; | ||||||
|  | 	    } | ||||||
|  | 	} else { | ||||||
|  | 	    switch (parse_until((ParseState*)state, '{', elem)) { | ||||||
|  | 	    case PARSE_SUCCESS: | ||||||
|  | 		ret = PSYC_PARSE_DICT_VALUE; | ||||||
|  | 		break; | ||||||
|  | 	    case PARSE_INSUFFICIENT: | ||||||
|  | 		return PSYC_PARSE_DICT_VALUE_LAST; | ||||||
|  | 	    default: // should not be reached
 | ||||||
|  | 		return PSYC_PARSE_DICT_ERROR; | ||||||
|  | 	    } | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	state->part = PSYC_DICT_PART_KEY_START; | ||||||
|  | 	return ret; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return PSYC_PARSE_DICT_ERROR; // should not be reached
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #ifdef __INLINE_PSYC_PARSE | ||||||
|  | static inline | ||||||
|  | #endif | ||||||
|  | PsycParseIndexRC | ||||||
|  | psyc_parse_index (PsycParseIndexState *state, PsycString *idx) | ||||||
|  | { | ||||||
|  |     ParseRC ret; | ||||||
|  | 
 | ||||||
|  |     if (state->cursor >= state->buffer.length) | ||||||
|  | 	return PSYC_PARSE_INDEX_END; | ||||||
|  | 
 | ||||||
|  |     state->startc = state->cursor; | ||||||
|  | 
 | ||||||
|  |     switch (state->part) { | ||||||
|  |     case PSYC_INDEX_PART_START: | ||||||
|  |     case PSYC_INDEX_PART_TYPE: | ||||||
|  | 	idx->length = 0; | ||||||
|  | 	idx->data = NULL; | ||||||
|  | 
 | ||||||
|  | 	switch (state->buffer.data[state->cursor]) { | ||||||
|  | 	case '#': | ||||||
|  | 	    state->part = PSYC_INDEX_PART_LIST; | ||||||
|  | 	    ADVANCE_STARTC_OR_RETURN(PSYC_PARSE_INSUFFICIENT); | ||||||
|  | 	    goto PSYC_INDEX_PART_LIST; | ||||||
|  | 	case '.': | ||||||
|  | 	    state->part = PSYC_INDEX_PART_DICT; | ||||||
|  | 	    ADVANCE_STARTC_OR_RETURN(PSYC_PARSE_INSUFFICIENT); | ||||||
|  | 	    goto PSYC_INDEX_PART_STRUCT; | ||||||
|  | 	case '{': | ||||||
|  | 	    state->part = PSYC_INDEX_PART_DICT; | ||||||
|  | 	    ADVANCE_STARTC_OR_RETURN(PSYC_PARSE_INSUFFICIENT); | ||||||
|  | 	    goto PSYC_INDEX_PART_DICT_LENGTH; | ||||||
|  | 	default: | ||||||
|  | 	    return PSYC_PARSE_INDEX_ERROR_TYPE; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  |     case PSYC_INDEX_PART_LIST: | ||||||
|  |     PSYC_INDEX_PART_LIST: | ||||||
|  | 	switch (parse_length((ParseState*)state, &idx->length)) { | ||||||
|  | 	case PARSE_SUCCESS: // list index is complete
 | ||||||
|  | 	    state->part = PSYC_INDEX_PART_TYPE; | ||||||
|  | 	    return PSYC_PARSE_INDEX_LIST; | ||||||
|  | 	case PARSE_INSUFFICIENT: // list index at the end of buffer
 | ||||||
|  | 	    return PSYC_PARSE_INDEX_LIST_LAST; | ||||||
|  | 	case PARSE_ERROR: // no index
 | ||||||
|  | 	    return PSYC_PARSE_INDEX_ERROR_LIST; | ||||||
|  | 	default: // should not be reached
 | ||||||
|  | 	    return PSYC_PARSE_INDEX_ERROR; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  |     case PSYC_INDEX_PART_STRUCT: | ||||||
|  |     PSYC_INDEX_PART_STRUCT: | ||||||
|  | 	switch (parse_keyword((ParseState*)state, idx)) { | ||||||
|  | 	case PARSE_SUCCESS: // end of keyword
 | ||||||
|  | 	    state->part = PSYC_INDEX_PART_TYPE; | ||||||
|  | 	    return PSYC_PARSE_INDEX_STRUCT; | ||||||
|  | 	case PARSE_INSUFFICIENT: // end of buffer
 | ||||||
|  | 	    return PSYC_PARSE_INDEX_STRUCT_LAST; | ||||||
|  | 	case PARSE_ERROR: // no keyword
 | ||||||
|  | 	    return PSYC_PARSE_INDEX_ERROR_STRUCT; | ||||||
|  | 	default: // should not be reached
 | ||||||
|  | 	    return PSYC_PARSE_INDEX_ERROR; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  |     case PSYC_INDEX_PART_DICT_LENGTH: | ||||||
|  |     PSYC_INDEX_PART_DICT_LENGTH: | ||||||
|  | 	switch (parse_length((ParseState*)state, &state->elemlen)) { | ||||||
|  | 	case PARSE_SUCCESS: // length is complete
 | ||||||
|  | 	    state->elemlen_found = 1; | ||||||
|  | 	    state->elem_parsed = 0; | ||||||
|  | 	    idx->length = state->elemlen; | ||||||
|  | 	    idx->data = NULL; | ||||||
|  | 
 | ||||||
|  | 	    if (state->buffer.data[state->cursor] != ' ') | ||||||
|  | 		return PSYC_PARSE_INDEX_ERROR_DICT_LENGTH; | ||||||
|  | 
 | ||||||
|  | 	    state->part = PSYC_INDEX_PART_DICT; | ||||||
|  | 	    ADVANCE_STARTC_OR_RETURN(PSYC_PARSE_INSUFFICIENT); | ||||||
|  | 	    break; | ||||||
|  | 	case PARSE_INSUFFICIENT: // length is incomplete
 | ||||||
|  | 	    return PSYC_PARSE_DICT_INSUFFICIENT; | ||||||
|  | 	case PARSE_ERROR: // no length
 | ||||||
|  | 	    state->part = PSYC_INDEX_PART_DICT; | ||||||
|  | 	    break; | ||||||
|  | 	default: // should not be reached
 | ||||||
|  | 	    return PSYC_PARSE_INDEX_ERROR; | ||||||
|  | 	} | ||||||
|  | 	// fall thru
 | ||||||
|  | 
 | ||||||
|  |     case PSYC_INDEX_PART_DICT: | ||||||
|  | 	if (state->elemlen_found) { | ||||||
|  | 	    switch (parse_binary((ParseState*)state, state->elemlen, idx, | ||||||
|  | 				 &state->elem_parsed)) { | ||||||
|  | 	    case PARSE_SUCCESS: | ||||||
|  | 		if (idx->length == state->elem_parsed) | ||||||
|  | 		    ret = PSYC_PARSE_INDEX_DICT; | ||||||
|  | 		else | ||||||
|  | 		    ret = PSYC_PARSE_INDEX_DICT_END; | ||||||
|  | 		break; | ||||||
|  | 	    case PARSE_INCOMPLETE: | ||||||
|  | 		if (idx->length == state->elem_parsed) | ||||||
|  | 		    ret = PSYC_PARSE_INDEX_DICT_START; | ||||||
|  | 		else | ||||||
|  | 		    ret = PSYC_PARSE_INDEX_DICT_CONT; | ||||||
|  | 		break; | ||||||
|  | 	    default: // should not be reached
 | ||||||
|  | 		return PSYC_PARSE_INDEX_ERROR_DICT; | ||||||
|  | 	    } | ||||||
|  | 	} else { | ||||||
|  | 	    switch (parse_until((ParseState*)state, '}', idx)) { | ||||||
|  | 	    case PARSE_SUCCESS: | ||||||
|  | 		ret = PSYC_PARSE_INDEX_DICT; | ||||||
|  | 		break; | ||||||
|  | 	    case PARSE_INSUFFICIENT: | ||||||
|  | 		return PSYC_PARSE_INDEX_INSUFFICIENT; | ||||||
|  | 	    default: // should not be reached
 | ||||||
|  | 		return PSYC_PARSE_INDEX_ERROR_DICT; | ||||||
|  | 	    } | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	state->part = PSYC_INDEX_PART_TYPE; | ||||||
|  | 	state->cursor++; | ||||||
|  | 	return ret; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return PSYC_PARSE_INDEX_ERROR; // should not be reached
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #ifdef __INLINE_PSYC_PARSE | ||||||
|  | static inline | ||||||
|  | #endif | ||||||
|  | PsycParseUpdateRC | ||||||
|  | psyc_parse_update (PsycParseUpdateState *state, char *oper, PsycString *value) | ||||||
|  | { | ||||||
|  |     PsycParseIndexRC ret; | ||||||
|  | 
 | ||||||
|  |     if (state->cursor >= state->buffer.length) | ||||||
|  | 	return PSYC_PARSE_UPDATE_END; | ||||||
|  | 
 | ||||||
|  |     state->startc = state->cursor; | ||||||
|  | 
 | ||||||
|  |     switch (state->part) { | ||||||
|  |     case PSYC_UPDATE_PART_START: | ||||||
|  | 	value->length = 0; | ||||||
|  | 	value->data = NULL; | ||||||
|  | 	// fall thru
 | ||||||
|  | 
 | ||||||
|  |     case PSYC_INDEX_PART_TYPE: | ||||||
|  |     case PSYC_INDEX_PART_LIST: | ||||||
|  |     case PSYC_INDEX_PART_STRUCT: | ||||||
|  |     case PSYC_INDEX_PART_DICT_LENGTH: | ||||||
|  |     case PSYC_INDEX_PART_DICT: | ||||||
|  | 	ret = psyc_parse_index((PsycParseIndexState*)state, value); | ||||||
|  | 
 | ||||||
|  | 	switch (ret) { | ||||||
|  | 	case PSYC_PARSE_INDEX_INSUFFICIENT: | ||||||
|  | 	case PSYC_PARSE_INDEX_LIST_LAST: | ||||||
|  | 	case PSYC_PARSE_INDEX_STRUCT_LAST: | ||||||
|  | 	case PSYC_PARSE_INDEX_END: | ||||||
|  | 	    return PSYC_PARSE_UPDATE_INSUFFICIENT; | ||||||
|  | 	case PSYC_PARSE_INDEX_ERROR_TYPE: | ||||||
|  | 	    if (state->buffer.data[state->cursor] != ' ') | ||||||
|  | 		return ret; | ||||||
|  | 	    state->part = PSYC_PARSE_UPDATE_TYPE; | ||||||
|  | 	    value->length = 0; | ||||||
|  | 	    value->data = NULL; | ||||||
|  | 	    ADVANCE_STARTC_OR_RETURN(PSYC_PARSE_UPDATE_INSUFFICIENT); | ||||||
|  | 	    break; | ||||||
|  | 	default: | ||||||
|  | 	    return ret; | ||||||
|  | 	} | ||||||
|  |     case PSYC_UPDATE_PART_TYPE: | ||||||
|  | 	if (!psyc_is_oper(state->buffer.data[state->cursor])) | ||||||
|  | 	    return PSYC_PARSE_UPDATE_ERROR_OPER; | ||||||
|  | 
 | ||||||
|  | 	*oper = state->buffer.data[state->cursor]; | ||||||
|  | 	ADVANCE_CURSOR_OR_RETURN(PSYC_PARSE_UPDATE_END); | ||||||
|  | 
 | ||||||
|  | 	switch (parse_keyword((ParseState*)state, value)) { | ||||||
|  | 	case PARSE_SUCCESS: // end of keyword
 | ||||||
|  | 	case PARSE_ERROR: // no keyword
 | ||||||
|  | 	    switch (state->buffer.data[state->cursor]) { | ||||||
|  | 	    case ':': | ||||||
|  | 		state->part = PSYC_UPDATE_PART_LENGTH; | ||||||
|  | 		break; | ||||||
|  | 	    case ' ': | ||||||
|  | 		state->part = PSYC_UPDATE_PART_VALUE; | ||||||
|  | 		break; | ||||||
|  | 	    default: | ||||||
|  | 		return PSYC_PARSE_UPDATE_ERROR_TYPE; | ||||||
|  | 	    } | ||||||
|  | 
 | ||||||
|  | 	    state->cursor++; | ||||||
|  | 	    return PSYC_PARSE_UPDATE_TYPE; | ||||||
|  | 	    break; | ||||||
|  | 	case PARSE_INSUFFICIENT: // end of buffer
 | ||||||
|  | 	    return PSYC_PARSE_UPDATE_TYPE_END; | ||||||
|  | 	default: // should not be reached
 | ||||||
|  | 	    return PSYC_PARSE_UPDATE_ERROR; | ||||||
|  | 	} | ||||||
|  | 	break; | ||||||
|  | 
 | ||||||
|  |     case PSYC_UPDATE_PART_LENGTH: | ||||||
|  | 	switch (parse_length((ParseState*)state, &state->elemlen)) { | ||||||
|  | 	case PARSE_SUCCESS: // length is complete
 | ||||||
|  | 	    state->elemlen_found = 1; | ||||||
|  | 	    state->elem_parsed = 0; | ||||||
|  | 	    value->length = state->elemlen; | ||||||
|  | 	    value->data = NULL; | ||||||
|  | 
 | ||||||
|  | 	    if (state->buffer.data[state->cursor] != ' ') | ||||||
|  | 		return PSYC_PARSE_UPDATE_ERROR_LENGTH; | ||||||
|  | 
 | ||||||
|  | 	    state->part = PSYC_UPDATE_PART_VALUE; | ||||||
|  | 	    if (value->length == 0) | ||||||
|  | 		return PSYC_PARSE_UPDATE_END; | ||||||
|  | 	    ADVANCE_STARTC_OR_RETURN(PSYC_PARSE_UPDATE_INSUFFICIENT); | ||||||
|  | 	    break; | ||||||
|  | 	case PARSE_INSUFFICIENT: // length is incomplete
 | ||||||
|  | 	    if (value->length == 0) | ||||||
|  | 		return PSYC_PARSE_UPDATE_END; | ||||||
|  | 	    return PSYC_PARSE_UPDATE_INSUFFICIENT; | ||||||
|  | 	case PARSE_ERROR: // no length after :
 | ||||||
|  | 	    return PSYC_PARSE_UPDATE_ERROR_LENGTH; | ||||||
|  | 	default: // should not be reached
 | ||||||
|  | 	    return PSYC_PARSE_UPDATE_ERROR; | ||||||
|  | 	} | ||||||
|  | 	// fall thru
 | ||||||
|  | 
 | ||||||
|  |     case PSYC_UPDATE_PART_VALUE: | ||||||
|  | 	if (state->elemlen_found) { | ||||||
|  | 	    switch (parse_binary((ParseState*)state, state->elemlen, value, | ||||||
|  | 				 &state->elem_parsed)) { | ||||||
|  | 	    case PARSE_SUCCESS: | ||||||
|  | 		if (value->length == state->elem_parsed) | ||||||
|  | 		    ret = PSYC_PARSE_UPDATE_VALUE; | ||||||
|  | 		else | ||||||
|  | 		    ret = PSYC_PARSE_UPDATE_VALUE_END; | ||||||
|  | 		break; | ||||||
|  | 	    case PARSE_INCOMPLETE: | ||||||
|  | 		if (value->length == state->elem_parsed) | ||||||
|  | 		    ret = PSYC_PARSE_UPDATE_VALUE_START; | ||||||
|  | 		else | ||||||
|  | 		    ret = PSYC_PARSE_UPDATE_VALUE_CONT; | ||||||
|  | 		break; | ||||||
|  | 	    default: // should not be reached
 | ||||||
|  | 		return PSYC_PARSE_UPDATE_ERROR_VALUE; | ||||||
|  | 	    } | ||||||
|  | 	} else { | ||||||
|  | 	    value->data = state->buffer.data + state->cursor; | ||||||
|  | 	    value->length = state->buffer.length - state->cursor; | ||||||
|  | 	    ret = PSYC_PARSE_UPDATE_VALUE; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	state->part = PSYC_INDEX_PART_TYPE; | ||||||
|  | 	state->cursor++; | ||||||
|  | 	return ret; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return PSYC_PARSE_INDEX_ERROR; // should not be reached
 | ||||||
|  | } | ||||||
|  |  | ||||||
							
								
								
									
										190
									
								
								src/render.c
									
										
									
									
									
								
							
							
						
						
									
										190
									
								
								src/render.c
									
										
									
									
									
								
							|  | @ -3,7 +3,59 @@ | ||||||
| #include "lib.h" | #include "lib.h" | ||||||
| #include <psyc/packet.h> | #include <psyc/packet.h> | ||||||
| #include <psyc/render.h> | #include <psyc/render.h> | ||||||
| #include <psyc/syntax.h> | 
 | ||||||
|  | inline PsycRenderRC | ||||||
|  | psyc_render_elem (PsycElem *elem, char *buffer, size_t buflen) | ||||||
|  | { | ||||||
|  |     size_t cur = 0; | ||||||
|  | 
 | ||||||
|  |     if (elem->length > buflen) // return error if element doesn't fit in buffer
 | ||||||
|  | 	return PSYC_RENDER_ERROR; | ||||||
|  | 
 | ||||||
|  |     if (elem->type.length) { | ||||||
|  | 	buffer[cur++] = '='; | ||||||
|  | 	memcpy(buffer + cur, PSYC_S2ARG(elem->type)); | ||||||
|  | 	cur += elem->type.length; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if (elem->value.length && !(elem->flag & PSYC_ELEM_NO_LENGTH)) { | ||||||
|  | 	if (elem->type.length) | ||||||
|  | 	    buffer[cur++] = ':'; | ||||||
|  | 	cur += itoa(elem->value.length, buffer + cur, 10); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if (elem->value.length) { | ||||||
|  | 	buffer[cur++] = ' '; | ||||||
|  | 	memcpy(buffer + cur, PSYC_S2ARG(elem->value)); | ||||||
|  | 	cur += elem->value.length; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // Actual length should be equal to pre-calculated length at this point.
 | ||||||
|  |     ASSERT(cur == elem->length); | ||||||
|  |     return PSYC_RENDER_SUCCESS; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | inline PsycRenderRC | ||||||
|  | psyc_render_dict_key (PsycDictKey *elem, char *buffer, size_t buflen) | ||||||
|  | { | ||||||
|  |     size_t cur = 0; | ||||||
|  | 
 | ||||||
|  |     if (elem->length > buflen) // return error if element doesn't fit in buffer
 | ||||||
|  | 	return PSYC_RENDER_ERROR; | ||||||
|  | 
 | ||||||
|  |     if (elem->value.length && !(elem->flag & PSYC_ELEM_NO_LENGTH)) | ||||||
|  | 	cur += itoa(elem->value.length, buffer + cur, 10); | ||||||
|  | 
 | ||||||
|  |     if (elem->value.length) { | ||||||
|  | 	buffer[cur++] = ' '; | ||||||
|  | 	memcpy(buffer + cur, PSYC_S2ARG(elem->value)); | ||||||
|  | 	cur += elem->value.length; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // Actual length should be equal to pre-calculated length at this point.
 | ||||||
|  |     ASSERT(cur == elem->length); | ||||||
|  |     return PSYC_RENDER_SUCCESS; | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
| #ifdef __INLINE_PSYC_RENDER | #ifdef __INLINE_PSYC_RENDER | ||||||
| static inline | static inline | ||||||
|  | @ -12,51 +64,55 @@ PsycRenderRC | ||||||
| psyc_render_list (PsycList *list, char *buffer, size_t buflen) | psyc_render_list (PsycList *list, char *buffer, size_t buflen) | ||||||
| { | { | ||||||
|     size_t i, cur = 0; |     size_t i, cur = 0; | ||||||
|     PsycString *elem; |  | ||||||
| 
 | 
 | ||||||
|     if (list->length > buflen) // return error if list doesn't fit in buffer
 |     if (list->length > buflen) // return error if list doesn't fit in buffer
 | ||||||
| 	return PSYC_RENDER_ERROR; | 	return PSYC_RENDER_ERROR; | ||||||
| 
 | 
 | ||||||
|     if (list->flag == PSYC_LIST_NEED_LENGTH) { |     if (list->type.length) { | ||||||
| 	for (i = 0; i < list->num_elems; i++) { | 	memcpy(buffer + cur, PSYC_S2ARG(list->type)); | ||||||
| 	    elem = &list->elems[i]; | 	cur += list->type.length; | ||||||
| 	    if (i > 0) |     } | ||||||
| 		buffer[cur++] = '|'; | 
 | ||||||
| 	    cur += itoa(elem->length, buffer + cur, 10); |     for (i = 0; i < list->num_elems; i++) { | ||||||
| 	    buffer[cur++] = ' '; | 	buffer[cur++] = '|'; | ||||||
| 	    memcpy(buffer + cur, elem->data, elem->length); | 	psyc_render_elem(&list->elems[i], buffer + cur, buflen - cur); | ||||||
| 	    cur += elem->length; | 	cur += list->elems[i].length; | ||||||
| 	} |  | ||||||
|     } else { |  | ||||||
| 	for (i = 0; i < list->num_elems; i++) { |  | ||||||
| 	    elem = &list->elems[i]; |  | ||||||
| 	    buffer[cur++] = '|'; |  | ||||||
| 	    memcpy(buffer + cur, elem->data, elem->length); |  | ||||||
| 	    cur += elem->length; |  | ||||||
| 	} |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| #ifdef DEBUG |  | ||||||
|     // Actual length should be equal to pre-calculated length at this point.
 |     // Actual length should be equal to pre-calculated length at this point.
 | ||||||
|     assert(cur == list->length); |     ASSERT(cur == list->length); | ||||||
| #endif |  | ||||||
|     return PSYC_RENDER_SUCCESS; |     return PSYC_RENDER_SUCCESS; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | #ifdef __INLINE_PSYC_RENDER | ||||||
|  | static inline | ||||||
|  | #endif | ||||||
| PsycRenderRC | PsycRenderRC | ||||||
| psyc_render_table (PsycTable *table, char *buffer, size_t buflen) | psyc_render_dict (PsycDict *dict, char *buffer, size_t buflen) | ||||||
| { | { | ||||||
|     size_t cur = 0; |     size_t i, cur = 0; | ||||||
| 
 | 
 | ||||||
|     if (table->length > buflen) // return error if table doesn't fit in buffer
 |     if (dict->length > buflen) // return error if dict doesn't fit in buffer
 | ||||||
| 	return PSYC_RENDER_ERROR; | 	return PSYC_RENDER_ERROR; | ||||||
| 
 | 
 | ||||||
|     if (table->width > 0) { |     if (dict->type.length) { | ||||||
| 	cur = sprintf(buffer, "*%ld", table->width); | 	memcpy(buffer + cur, PSYC_S2ARG(dict->type)); | ||||||
| 	buffer[cur++] = ' '; | 	cur += dict->type.length; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     return psyc_render_list(table->list, buffer + cur, buflen - cur); |     for (i = 0; i < dict->num_elems; i++) { | ||||||
|  | 	buffer[cur++] = '{'; | ||||||
|  | 	psyc_render_dict_key(&dict->elems[i].key, buffer + cur, buflen - cur); | ||||||
|  | 	cur += dict->elems[i].key.length; | ||||||
|  | 
 | ||||||
|  | 	buffer[cur++] = '}'; | ||||||
|  | 	psyc_render_elem(&dict->elems[i].value, buffer + cur, buflen - cur); | ||||||
|  | 	cur += dict->elems[i].value.length; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // Actual length should be equal to pre-calculated length at this point.
 | ||||||
|  |     ASSERT(cur == dict->length); | ||||||
|  |     return PSYC_RENDER_SUCCESS; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static inline size_t | static inline size_t | ||||||
|  | @ -70,7 +126,9 @@ psyc_render_modifier (PsycModifier *mod, char *buffer) | ||||||
|     if (cur == 1) |     if (cur == 1) | ||||||
| 	return cur; // error, name can't be empty
 | 	return cur; // error, name can't be empty
 | ||||||
| 
 | 
 | ||||||
|     if (mod->flag == PSYC_MODIFIER_NEED_LENGTH) { |     if (mod->value.length | ||||||
|  | 	&& (mod->flag & PSYC_MODIFIER_NEED_LENGTH | ||||||
|  | 	    || mod->flag == PSYC_MODIFIER_CHECK_LENGTH)) { | ||||||
| 	buffer[cur++] = ' '; | 	buffer[cur++] = ' '; | ||||||
| 	cur += itoa(mod->value.length, buffer + cur, 10); | 	cur += itoa(mod->value.length, buffer + cur, 10); | ||||||
|     } |     } | ||||||
|  | @ -87,54 +145,52 @@ psyc_render_modifier (PsycModifier *mod, char *buffer) | ||||||
| static inline | static inline | ||||||
| #endif | #endif | ||||||
| PsycRenderRC | PsycRenderRC | ||||||
| psyc_render (PsycPacket *packet, char *buffer, size_t buflen) | psyc_render (PsycPacket *p, char *buffer, size_t buflen) | ||||||
| { | { | ||||||
|     size_t i, cur = 0, len; |     size_t i, cur = 0, len; | ||||||
| 
 | 
 | ||||||
|     if (packet->length > buflen) // return error if packet doesn't fit in buffer
 |     if (p->length > buflen) // return error if packet doesn't fit in buffer
 | ||||||
| 	return PSYC_RENDER_ERROR; | 	return PSYC_RENDER_ERROR; | ||||||
| 
 | 
 | ||||||
|     // render routing modifiers
 |     // render routing modifiers
 | ||||||
|     for (i = 0; i < packet->routing.lines; i++) { |     for (i = 0; i < p->routing.lines; i++) { | ||||||
| 	len = psyc_render_modifier(&packet->routing.modifiers[i], buffer + cur); | 	len = psyc_render_modifier(&p->routing.modifiers[i], buffer + cur); | ||||||
| 	cur += len; | 	cur += len; | ||||||
| 	if (len <= 1) | 	if (len <= 1) | ||||||
| 	    return PSYC_RENDER_ERROR_MODIFIER_NAME_MISSING; | 	    return PSYC_RENDER_ERROR_MODIFIER_NAME_MISSING; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     // add length if needed
 |     // add length if needed
 | ||||||
|     if (packet->flag == PSYC_PACKET_NEED_LENGTH) |     if (p->contentlen && !(p->flag & PSYC_PACKET_NO_LENGTH)) | ||||||
| 	cur += itoa(packet->contentlen, buffer + cur, 10); | 	cur += itoa(p->contentlen, buffer + cur, 10); | ||||||
| 
 | 
 | ||||||
|     if (packet->flag == PSYC_PACKET_NEED_LENGTH || packet->content.length |     if (p->contentlen) | ||||||
| 	|| packet->stateop || packet->entity.lines |  | ||||||
| 	|| packet->method.length || packet->data.length) |  | ||||||
| 	buffer[cur++] = '\n'; // start of content part if there's content or length
 | 	buffer[cur++] = '\n'; // start of content part if there's content or length
 | ||||||
| 
 | 
 | ||||||
|     if (packet->content.length) { // render raw content if present
 |     if (p->content.length) { // render raw content if present
 | ||||||
| 	memcpy(buffer + cur, packet->content.data, packet->content.length); | 	memcpy(buffer + cur, p->content.data, p->content.length); | ||||||
| 	cur += packet->content.length; | 	cur += p->content.length; | ||||||
|     } else { |     } else { | ||||||
| 	if (packet->stateop) { | 	if (p->stateop) { | ||||||
| 	    buffer[cur++] = packet->stateop; | 	    buffer[cur++] = p->stateop; | ||||||
| 	    buffer[cur++] = '\n'; | 	    buffer[cur++] = '\n'; | ||||||
| 	} | 	} | ||||||
| 	// render entity modifiers
 | 	// render entity modifiers
 | ||||||
| 	for (i = 0; i < packet->entity.lines; i++) | 	for (i = 0; i < p->entity.lines; i++) | ||||||
| 	    cur += psyc_render_modifier(&packet->entity.modifiers[i], | 	    cur += psyc_render_modifier(&p->entity.modifiers[i], | ||||||
| 					buffer + cur); | 					buffer + cur); | ||||||
| 
 | 
 | ||||||
| 	if (packet->method.length) { // add method\n
 | 	if (p->method.length) { // add method\n
 | ||||||
| 	    memcpy(buffer + cur, packet->method.data, packet->method.length); | 	    memcpy(buffer + cur, p->method.data, p->method.length); | ||||||
| 	    cur += packet->method.length; | 	    cur += p->method.length; | ||||||
| 	    buffer[cur++] = '\n'; | 	    buffer[cur++] = '\n'; | ||||||
| 
 | 
 | ||||||
| 	    if (packet->data.length) { // add data\n
 | 	    if (p->data.length) { // add data\n
 | ||||||
| 		memcpy(buffer + cur, packet->data.data, packet->data.length); | 		memcpy(buffer + cur, p->data.data, p->data.length); | ||||||
| 		cur += packet->data.length; | 		cur += p->data.length; | ||||||
| 		buffer[cur++] = '\n'; | 		buffer[cur++] = '\n'; | ||||||
| 	    } | 	    } | ||||||
| 	} else if (packet->data.length)	// error, we have data but no modifier
 | 	} else if (p->data.length)	// error, we have data but no modifier
 | ||||||
| 	    return PSYC_RENDER_ERROR_METHOD_MISSING; | 	    return PSYC_RENDER_ERROR_METHOD_MISSING; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -143,32 +199,6 @@ psyc_render (PsycPacket *packet, char *buffer, size_t buflen) | ||||||
|     buffer[cur++] = '\n'; |     buffer[cur++] = '\n'; | ||||||
| 
 | 
 | ||||||
|     // actual length should be equal to pre-calculated length at this point
 |     // actual length should be equal to pre-calculated length at this point
 | ||||||
|     assert(cur == packet->length); |     ASSERT(cur == p->length); | ||||||
|     return PSYC_RENDER_SUCCESS; |     return PSYC_RENDER_SUCCESS; | ||||||
| } | } | ||||||
| 
 |  | ||||||
| PsycRenderRC |  | ||||||
| psyc_render_packet_id (char *context, size_t contextlen, |  | ||||||
| 		       char *source, size_t sourcelen, |  | ||||||
| 		       char *target, size_t targetlen, |  | ||||||
| 		       char *counter, size_t counterlen, |  | ||||||
| 		       char *fragment, size_t fragmentlen, |  | ||||||
| 		       char *buffer, size_t buflen) |  | ||||||
| { |  | ||||||
|     PsycList list; |  | ||||||
|     PsycString elems[PSYC_PACKET_ID_ELEMS] = {}; |  | ||||||
| 
 |  | ||||||
|     if (contextlen) |  | ||||||
| 	elems[PSYC_PACKET_ID_CONTEXT] = PSYC_STRING(context, contextlen); |  | ||||||
|     if (sourcelen) |  | ||||||
| 	elems[PSYC_PACKET_ID_SOURCE] = PSYC_STRING(source, sourcelen); |  | ||||||
|     if (targetlen) |  | ||||||
| 	elems[PSYC_PACKET_ID_TARGET] = PSYC_STRING(target, targetlen); |  | ||||||
|     if (counterlen) |  | ||||||
| 	elems[PSYC_PACKET_ID_COUNTER] = PSYC_STRING(counter, counterlen); |  | ||||||
|     if (fragmentlen) |  | ||||||
| 	elems[PSYC_PACKET_ID_FRAGMENT] = PSYC_STRING(fragment, fragmentlen); |  | ||||||
| 
 |  | ||||||
|     psyc_list_init(&list, elems, PSYC_PACKET_ID_ELEMS, PSYC_LIST_NO_LENGTH); |  | ||||||
|     return psyc_render_list(&list, buffer, buflen); |  | ||||||
| } |  | ||||||
|  |  | ||||||
|  | @ -4,7 +4,7 @@ | ||||||
| #include <psyc/packet.h> | #include <psyc/packet.h> | ||||||
| 
 | 
 | ||||||
| /// Routing variables in alphabetical order.
 | /// Routing variables in alphabetical order.
 | ||||||
| const PsycDictInt psyc_rvars[] = { | const PsycMapInt psyc_rvars[] = { | ||||||
|     { PSYC_C2STRI("_amount_fragments"),	PSYC_RVAR_AMOUNT_FRAGMENTS }, |     { PSYC_C2STRI("_amount_fragments"),	PSYC_RVAR_AMOUNT_FRAGMENTS }, | ||||||
|     { PSYC_C2STRI("_context"),		PSYC_RVAR_CONTEXT }, |     { PSYC_C2STRI("_context"),		PSYC_RVAR_CONTEXT }, | ||||||
|     { PSYC_C2STRI("_counter"),		PSYC_RVAR_COUNTER }, |     { PSYC_C2STRI("_counter"),		PSYC_RVAR_COUNTER }, | ||||||
|  | @ -30,26 +30,26 @@ const PsycDictInt psyc_rvars[] = { | ||||||
| const size_t psyc_rvars_num = PSYC_NUM_ELEM(psyc_rvars); | const size_t psyc_rvars_num = PSYC_NUM_ELEM(psyc_rvars); | ||||||
| 
 | 
 | ||||||
| // Variable types in alphabetical order.
 | // Variable types in alphabetical order.
 | ||||||
| const PsycDictInt psyc_var_types[] = { | const PsycMapInt psyc_var_types[] = { | ||||||
|     { PSYC_C2STRI("_amount"),	PSYC_TYPE_AMOUNT }, |     { PSYC_C2STRI("_amount"),	PSYC_TYPE_AMOUNT }, | ||||||
|     { PSYC_C2STRI("_color"),	PSYC_TYPE_COLOR }, |     { PSYC_C2STRI("_color"),	PSYC_TYPE_COLOR }, | ||||||
|     { PSYC_C2STRI("_date"),	PSYC_TYPE_DATE }, |     { PSYC_C2STRI("_date"),	PSYC_TYPE_DATE }, | ||||||
|     { PSYC_C2STRI("_def"),	PSYC_TYPE_DEF }, |  | ||||||
|     { PSYC_C2STRI("_degree"),	PSYC_TYPE_DEGREE }, |     { PSYC_C2STRI("_degree"),	PSYC_TYPE_DEGREE }, | ||||||
|  |     { PSYC_C2STRI("_dict"),	PSYC_TYPE_DICT }, | ||||||
|     { PSYC_C2STRI("_entity"),	PSYC_TYPE_ENTITY }, |     { PSYC_C2STRI("_entity"),	PSYC_TYPE_ENTITY }, | ||||||
|     { PSYC_C2STRI("_flag"),	PSYC_TYPE_FLAG }, |     { PSYC_C2STRI("_flag"),	PSYC_TYPE_FLAG }, | ||||||
|     { PSYC_C2STRI("_language"),	PSYC_TYPE_LANGUAGE }, |     { PSYC_C2STRI("_language"),	PSYC_TYPE_LANGUAGE }, | ||||||
|     { PSYC_C2STRI("_list"),	PSYC_TYPE_LIST }, |     { PSYC_C2STRI("_list"),	PSYC_TYPE_LIST }, | ||||||
|     { PSYC_C2STRI("_nick"),	PSYC_TYPE_NICK }, |     { PSYC_C2STRI("_nick"),	PSYC_TYPE_NICK }, | ||||||
|     { PSYC_C2STRI("_page"),	PSYC_TYPE_PAGE }, |     { PSYC_C2STRI("_page"),	PSYC_TYPE_PAGE }, | ||||||
|     { PSYC_C2STRI("_table"),	PSYC_TYPE_TABLE }, |     { PSYC_C2STRI("_struct"),	PSYC_TYPE_STRUCT }, | ||||||
|     { PSYC_C2STRI("_time"),	PSYC_TYPE_TIME }, |     { PSYC_C2STRI("_time"),	PSYC_TYPE_TIME }, | ||||||
|     { PSYC_C2STRI("_uniform"),	PSYC_TYPE_UNIFORM }, |     { PSYC_C2STRI("_uniform"),	PSYC_TYPE_UNIFORM }, | ||||||
| }; | }; | ||||||
| const size_t psyc_var_types_num = PSYC_NUM_ELEM(psyc_var_types); | const size_t psyc_var_types_num = PSYC_NUM_ELEM(psyc_var_types); | ||||||
| 
 | 
 | ||||||
| /// Method names in alphabetical order.
 | /// Method names in alphabetical order.
 | ||||||
| const PsycDictInt psyc_methods[] = { | const PsycMapInt psyc_methods[] = { | ||||||
|     { PSYC_C2STRI("_data"),			PSYC_MC_DATA }, |     { PSYC_C2STRI("_data"),			PSYC_MC_DATA }, | ||||||
|     { PSYC_C2STRI("_echo_context_enter"),	PSYC_MC_ECHO_CONTEXT_ENTER }, |     { PSYC_C2STRI("_echo_context_enter"),	PSYC_MC_ECHO_CONTEXT_ENTER }, | ||||||
|     { PSYC_C2STRI("_echo_context_leave"),	PSYC_MC_ECHO_CONTEXT_LEAVE }, |     { PSYC_C2STRI("_echo_context_leave"),	PSYC_MC_ECHO_CONTEXT_LEAVE }, | ||||||
|  | @ -88,7 +88,7 @@ const size_t psyc_methods_num = PSYC_NUM_ELEM(psyc_methods); | ||||||
| PsycMethod | PsycMethod | ||||||
| psyc_method (char *method, size_t methodlen, PsycMethod *family, unsigned int *flag) | psyc_method (char *method, size_t methodlen, PsycMethod *family, unsigned int *flag) | ||||||
| { | { | ||||||
|     int mc = psyc_dict_lookup_int(psyc_methods, psyc_methods_num, |     int mc = psyc_map_lookup_int(psyc_methods, psyc_methods_num, | ||||||
| 				  method, methodlen, PSYC_YES); | 				  method, methodlen, PSYC_YES); | ||||||
| 
 | 
 | ||||||
|     switch (mc) { |     switch (mc) { | ||||||
|  |  | ||||||
|  | @ -3,7 +3,7 @@ DEBUG = 2 | ||||||
| CFLAGS = -I../include -I../src -Wall -std=c99 ${OPT} | CFLAGS = -I../include -I../src -Wall -std=c99 ${OPT} | ||||||
| LDFLAGS = -L../lib | LDFLAGS = -L../lib | ||||||
| LOADLIBES = -lpsyc -lm | LOADLIBES = -lpsyc -lm | ||||||
| TARGETS = test_psyc test_psyc_speed test_parser test_match test_render test_text var_routing var_type uniform_parse test_list test_table test_packet_id method | TARGETS = test_psyc test_psyc_speed test_parser test_match test_render test_text var_routing var_type uniform_parse test_packet_id test_index test_update method | ||||||
| O = test.o | O = test.o | ||||||
| WRAPPER = | WRAPPER = | ||||||
| DIET = diet | DIET = diet | ||||||
|  | @ -47,10 +47,13 @@ test: ${TARGETS} | ||||||
| 	./test_text | 	./test_text | ||||||
| 	./var_routing | 	./var_routing | ||||||
| 	./var_type | 	./var_type | ||||||
|  | 	./method | ||||||
| 	./uniform_parse | 	./uniform_parse | ||||||
| 	./test_list | #	./test_list
 | ||||||
| 	./test_table | #	./test_table
 | ||||||
| 	./test_packet_id | 	./test_packet_id | ||||||
|  | 	./test_index | ||||||
|  | 	./test_update | ||||||
| 	x=0; for f in packets/[0-9]*; do echo ">> $$f"; ./test_psyc -f $$f | ${DIFF} -u $$f -; x=$$((x+$$?)); done; exit $$x | 	x=0; for f in packets/[0-9]*; do echo ">> $$f"; ./test_psyc -f $$f | ${DIFF} -u $$f -; x=$$((x+$$?)); done; exit $$x | ||||||
| 	x=0; for f in packets/[0-9]*; do echo ">> $$f"; ./test_psyc -rf $$f | ${DIFF} -u $$f -; x=$$((x+$$?)); done; exit $$x | 	x=0; for f in packets/[0-9]*; do echo ">> $$f"; ./test_psyc -rf $$f | ${DIFF} -u $$f -; x=$$((x+$$?)); done; exit $$x | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1,4 +0,0 @@ | ||||||
| :_source	psyc://foo.example.com/ |  | ||||||
| :_target	psyc://bar.example.com/ |  | ||||||
| 0 |  | ||||||
| | |  | ||||||
|  | @ -1,7 +0,0 @@ | ||||||
| :_source	psyc://foo.example.com/ |  | ||||||
| :_target	psyc://bar.example.com/ |  | ||||||
| 
 |  | ||||||
| :_foo 0	 |  | ||||||
| _message_private |  | ||||||
| OHAI |  | ||||||
| | |  | ||||||
|  | @ -1,18 +1,17 @@ | ||||||
| =_source	psyc://foo/~bar | =_context	psyc://foo/~bar | ||||||
| :_target	psyc://bar/~baz |  | ||||||
| =_list_foo	|foo|bar|baz |  | ||||||
| :_tag	sch1828hu3r2cm |  | ||||||
| 
 | 
 | ||||||
| =_foo	bar baz | =_list_xxx	|=_foo|=_bar|=_baz|4 abc | ||||||
| =_abc_def 11	ghi jkl | =_list_foo	|=_foo|=_bar ||=_baz| | ||||||
| 
 | =_list_foo	_test|3 foo|=_color:4 blue|=_nick bar | ||||||
| xq | =_list_bar 43	|3 foo|=_color:4 blue|=_nick bar|8  | ||||||
| =_list_bar 36	3 foo|3 bar|7 foo | foo|bar | ||||||
| bar|11 foo | =_list_bar 43	|3 foo|=_color:4 blue|=_nick bar|9  | ||||||
| bar | foo|bar | ||||||
| baz | =_list_a	_type| elem1| elem2| elem3 | ||||||
| :_foo_bar	yay | =_list_b	|=_type1 elem1|=_type2 elem2|=_type3 elem3 | ||||||
| _message_foo_bar | =_list_members	_uniform| psyc://example.net/~alice| psyc://example.org/~bob | ||||||
| ohai there! | =_list_topic	|9 democracy|3 now | ||||||
| \o/ | =_list_c	| foo| bar|6 foobar|3 baz|=_int 234|=_time 1234 | ||||||
|  | =_list_foo	|=_int 234|=_time 1234|=_picture:3 \o/|7 \oXoXo/ | ||||||
|  | _test_list | ||||||
| | | | | ||||||
|  |  | ||||||
							
								
								
									
										7
									
								
								test/packets/02-list2
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								test/packets/02-list2
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,7 @@ | ||||||
|  | :_source	psyc://foo/~bar | ||||||
|  | :_target	psyc://bar/~baz | ||||||
|  | 
 | ||||||
|  | =_list_foo	_test|3 foo|=_color:4 blue|=_nick bar | ||||||
|  | =_list_bar	|3 foo|=_color:4 blue|=_nick bar | ||||||
|  | _test | ||||||
|  | | | ||||||
							
								
								
									
										11
									
								
								test/packets/03-dict
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								test/packets/03-dict
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,11 @@ | ||||||
|  | =_context	psyc://foo/~bar | ||||||
|  | 
 | ||||||
|  | =_dict	_type{4 key1}6 value1{key2} value2{key3}6 value3{key4} value4 | ||||||
|  | =_dict_example	{4 key1}=_type1:6 value1{key2}=_type2 value2{key3}6 value3{key4} value4 | ||||||
|  | =_struct_member	|=_nick|=_picture | ||||||
|  | =_dict_owners	{psyc://example.net/~alice}{psyc://example.org/~bob} | ||||||
|  | =_dict_members	{psyc://example.net/~alice}=_struct_member | alice| \o/{psyc://example.org/~bob}=_struct_member | bob| \oXo/ | ||||||
|  | =_dict_members	{25 psyc://example.net/~alice}=_struct_member:12 | alice| \o/{psyc://example.org/~bob}=_struct_member | bob| \oXo/ | ||||||
|  | =_dict_members	_struct_member{25 psyc://example.net/~alice}12 | alice| \o/{psyc://example.org/~bob} | bob| \oXo/ | ||||||
|  | _test_dict | ||||||
|  | | | ||||||
|  | @ -1,20 +0,0 @@ | ||||||
| =_source	psyc://foo/~bar |  | ||||||
| :_target	psyc://bar/~baz |  | ||||||
| =_list_foo	|foo|bar|baz |  | ||||||
| :_tag	sch1828hu3r2cm |  | ||||||
| 
 |  | ||||||
| ?_test	ignored |  | ||||||
| =_foo	bar baz |  | ||||||
| =_abc_def 11	ghi jkl |  | ||||||
| 
 |  | ||||||
| xq |  | ||||||
| =_list_bar 36	3 foo|3 bar|7 foo |  | ||||||
| bar|11 foo |  | ||||||
| b|r |  | ||||||
| baz |  | ||||||
| :_foo_bar	yay |  | ||||||
| =_amount_x	10 |  | ||||||
| _message_foo_bar |  | ||||||
| ohai there! |  | ||||||
| \o/ |  | ||||||
| | |  | ||||||
|  | @ -1,4 +1,4 @@ | ||||||
| :_list_understand_modules	|_state|_fragments|_context | :_list_understand_modules	| _state| _fragments| _context | ||||||
| 
 | 
 | ||||||
| _request_features | _request_features | ||||||
| | | | | ||||||
|  |  | ||||||
							
								
								
									
										87
									
								
								test/test_index.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								test/test_index.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,87 @@ | ||||||
|  | #include <psyc.h> | ||||||
|  | #include <unistd.h> | ||||||
|  | #include <stdio.h> | ||||||
|  | #include <stdlib.h> | ||||||
|  | #include <lib.h> | ||||||
|  | 
 | ||||||
|  | uint8_t verbose; | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | test_index (const char *buf, size_t buflen) | ||||||
|  | { | ||||||
|  |     char *res = malloc(buflen * 2); | ||||||
|  |     size_t reslen = 0; | ||||||
|  | 
 | ||||||
|  |     int ret, reti, len = 0; | ||||||
|  |     PsycString idx; | ||||||
|  |     PsycParseIndexState state; | ||||||
|  |     psyc_parse_index_state_init(&state); | ||||||
|  |     psyc_parse_index_buffer_set(&state, buf, buflen); | ||||||
|  | 
 | ||||||
|  |     if (verbose) | ||||||
|  | 	printf(">> %.*s\n", (int)buflen, buf); | ||||||
|  | 
 | ||||||
|  |     do { | ||||||
|  | 	ret = reti = psyc_parse_index(&state, &idx); | ||||||
|  | 	len = 0; | ||||||
|  | 
 | ||||||
|  | 	switch (ret) { | ||||||
|  | 	case PSYC_PARSE_INDEX_LIST_LAST: | ||||||
|  | 	    ret = 0; | ||||||
|  | 	case PSYC_PARSE_INDEX_LIST: | ||||||
|  | 	    len = sprintf(res + reslen, "#%ld", idx.length); | ||||||
|  | 	    break; | ||||||
|  | 	case PSYC_PARSE_INDEX_STRUCT_LAST: | ||||||
|  | 	    ret = 0; | ||||||
|  | 	case PSYC_PARSE_INDEX_STRUCT: | ||||||
|  | 	    len = sprintf(res + reslen, ".%.*s", PSYC_S2ARGP(idx)); | ||||||
|  | 	    break; | ||||||
|  | 	case PSYC_PARSE_INDEX_DICT: | ||||||
|  | 	    if (state.elemlen_found) | ||||||
|  | 		len = sprintf(res + reslen, "{%ld %.*s}", | ||||||
|  | 			      idx.length, PSYC_S2ARGP(idx)); | ||||||
|  | 	    else | ||||||
|  | 		len = sprintf(res + reslen, "{%.*s}", PSYC_S2ARGP(idx)); | ||||||
|  | 	    break; | ||||||
|  | 	case PSYC_PARSE_INDEX_END: | ||||||
|  | 	    ret = 0; | ||||||
|  | 	    break; | ||||||
|  | 	default: | ||||||
|  | 	    ret = -1; | ||||||
|  | 	} | ||||||
|  | 	if (verbose) | ||||||
|  | 	    printf("%2d: %.*s\n", reti, len, res + reslen); | ||||||
|  | 	reslen += len; | ||||||
|  |     } while (ret > 0); | ||||||
|  | 
 | ||||||
|  |     if (reslen != buflen || memcmp(res, buf, buflen) != 0) { | ||||||
|  | 	printf("ERROR: got\n\[%.*s] (%ld) instead of\n[%.*s] (%ld)\n", | ||||||
|  | 	       (int)reslen, res, reslen, (int)buflen, buf, buflen); | ||||||
|  | 	ret = 1; | ||||||
|  |     } else | ||||||
|  | 	ret = 0; | ||||||
|  | 
 | ||||||
|  |     free(res); | ||||||
|  |     return ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | main (int argc, char **argv) | ||||||
|  | { | ||||||
|  |     verbose = argc > 1; | ||||||
|  | 
 | ||||||
|  |     if (test_index(PSYC_C2ARG("#1{foo}._bar")) != 0) | ||||||
|  | 	return 1; | ||||||
|  | 
 | ||||||
|  |     if (test_index(PSYC_C2ARG("{3 foo}._bar#0")) != 0) | ||||||
|  | 	return 2; | ||||||
|  | 
 | ||||||
|  |     if (test_index(PSYC_C2ARG("{foo}#2._bar")) != 0) | ||||||
|  | 	return 3; | ||||||
|  | 
 | ||||||
|  |     if (test_index(PSYC_C2ARG("._bar#1{3 foo}")) != 0) | ||||||
|  | 	return 4; | ||||||
|  | 
 | ||||||
|  |     printf("test_index passed all tests.\n"); | ||||||
|  |     return 0; // passed all tests
 | ||||||
|  | } | ||||||
|  | @ -17,21 +17,25 @@ main (int argc, char **argv) | ||||||
| 
 | 
 | ||||||
|     PsycParseListState listState; |     PsycParseListState listState; | ||||||
|     PsycList list_text, list_bin; |     PsycList list_text, list_bin; | ||||||
|     PsycString elems_text[NELEMS], elems_bin[NELEMS], elem; |     PsycElem elems_text[NELEMS], elems_bin[NELEMS]; | ||||||
|  |     PsycString type, value; | ||||||
|     char buf_text[NELEMS * 200], buf_bin[NELEMS * 200], *elems[NELEMS], **elems2 = NULL; |     char buf_text[NELEMS * 200], buf_bin[NELEMS * 200], *elems[NELEMS], **elems2 = NULL; | ||||||
| 
 | 
 | ||||||
|     struct timeval start, end; |     struct timeval start, end; | ||||||
| 
 | 
 | ||||||
|  |     char *text = "1234567890abcdefghijklmnopqrstuvwxyz-._ " | ||||||
|  | 	"1234567890abcdefghijklmnopqrstuvwxyz-._ " | ||||||
|  | 	"1234567890abcdefghijklmnopqrstuvwxyz-._ " | ||||||
|  | 	"1234567890"; | ||||||
|  |     char *bin = "1234567890|abcdefghijklmnopqrstuvwxyz|_\n" | ||||||
|  | 	"1234567890|abcdefghijklmnopqrstuvwxyz|_\n" | ||||||
|  | 	"1234567890|abcdefghijklmnopqrstuvwxyz|_\n" | ||||||
|  | 	"1234567890"; | ||||||
|  | 
 | ||||||
|     for (i=0; i<NELEMS; i++) |     for (i=0; i<NELEMS; i++) | ||||||
| 	elems_text[i] = PSYC_C2STR("1234567890abcdefghijklmnopqrstuvwxyz-._ " | 	elems_text[i] = PSYC_ELEM_V(text, strlen(text)); | ||||||
| 				   "1234567890abcdefghijklmnopqrstuvwxyz-._ " |  | ||||||
| 				   "1234567890abcdefghijklmnopqrstuvwxyz-._ " |  | ||||||
| 				   "1234567890"); |  | ||||||
|     for (i=0; i<NELEMS; i++) |     for (i=0; i<NELEMS; i++) | ||||||
| 	elems_bin[i] = PSYC_C2STR("1234567890|abcdefghijklmnopqrstuvwxyz|_\n" | 	elems_bin[i] = PSYC_ELEM_V(bin, strlen(bin)); | ||||||
| 				  "1234567890|abcdefghijklmnopqrstuvwxyz|_\n" |  | ||||||
| 				  "1234567890|abcdefghijklmnopqrstuvwxyz|_\n" |  | ||||||
| 				  "1234567890"); |  | ||||||
| 
 | 
 | ||||||
|     psyc_list_init(&list_text, elems_text, PSYC_NUM_ELEM(elems_text), |     psyc_list_init(&list_text, elems_text, PSYC_NUM_ELEM(elems_text), | ||||||
| 		   PSYC_LIST_NO_LENGTH); | 		   PSYC_LIST_NO_LENGTH); | ||||||
|  | @ -54,13 +58,14 @@ main (int argc, char **argv) | ||||||
| 	psyc_parse_list_buffer_set(&listState, buf_text, list_text.length); | 	psyc_parse_list_buffer_set(&listState, buf_text, list_text.length); | ||||||
| 	i = 0; | 	i = 0; | ||||||
| 	do { | 	do { | ||||||
| 	    ret = psyc_parse_list(&listState, &elem); | 	    ret = psyc_parse_list(&listState, &type, &value); | ||||||
| 	    switch (ret) { | 	    switch (ret) { | ||||||
| 	    case PSYC_PARSE_LIST_END: | 	    case PSYC_PARSE_LIST_END: | ||||||
| 		ret = 0; | 		ret = 0; | ||||||
| 	    case PSYC_PARSE_LIST_ELEM: | 	    case PSYC_PARSE_LIST_ELEM: | ||||||
| 		if (verbose) | 		if (verbose) | ||||||
| 		    printf("|%d: %.*s... (%ld)\n", i, 10, elem.data, elem.length); | 		    printf("|%ld:%.*s %.*s...\n", value.length, | ||||||
|  | 			   (int)type.length, type.data, 10, value.data); | ||||||
| 		//elems[i] = malloc(elem.length);
 | 		//elems[i] = malloc(elem.length);
 | ||||||
| 		//memcpy(&elems[i++], elem.data, elem.length);
 | 		//memcpy(&elems[i++], elem.data, elem.length);
 | ||||||
| 		break; | 		break; | ||||||
|  |  | ||||||
|  | @ -5,6 +5,8 @@ | ||||||
| #include <psyc/packet.h> | #include <psyc/packet.h> | ||||||
| #include <psyc/render.h> | #include <psyc/render.h> | ||||||
| 
 | 
 | ||||||
|  | uint8_t verbose; | ||||||
|  | 
 | ||||||
| int | int | ||||||
| packet_id (char *context, size_t contextlen, | packet_id (char *context, size_t contextlen, | ||||||
| 	   char *source, size_t sourcelen, | 	   char *source, size_t sourcelen, | ||||||
|  | @ -13,16 +15,22 @@ packet_id (char *context, size_t contextlen, | ||||||
| 	   char *fragment, size_t fragmentlen, | 	   char *fragment, size_t fragmentlen, | ||||||
| 	   char *result, size_t resultlen) | 	   char *result, size_t resultlen) | ||||||
| { | { | ||||||
|     size_t idlen = psyc_packet_id_length(contextlen, sourcelen, targetlen, |     PsycList list; | ||||||
| 					 counterlen, fragmentlen); |     PsycElem elems[PSYC_PACKET_ID_ELEMS]; | ||||||
|  |     memset(&list, 0, sizeof(PsycList)); | ||||||
|  |     memset(elems, 0, sizeof(PsycElem) * PSYC_PACKET_ID_ELEMS); | ||||||
|  | 
 | ||||||
|  |     psyc_packet_id(&list, elems, context, contextlen, | ||||||
|  | 		   source, sourcelen, target, targetlen, | ||||||
|  | 		   counter, counterlen, fragment, fragmentlen); | ||||||
|  | 
 | ||||||
|  |     size_t idlen = list.length; | ||||||
|     char *id = malloc(idlen); |     char *id = malloc(idlen); | ||||||
|     psyc_render_packet_id(context, contextlen, | 
 | ||||||
| 			  source, sourcelen, |     psyc_render_list(&list, id, idlen); | ||||||
| 			  target, targetlen, | 
 | ||||||
| 			  counter, counterlen, |     if (verbose) | ||||||
| 			  fragment, fragmentlen, | 	printf("[%.*s]\n", (int)idlen, id); | ||||||
| 			  id, idlen); |  | ||||||
|     printf("%.*s\n", (int)idlen, id); |  | ||||||
|     int ret = idlen == resultlen && memcmp(result, id, idlen) == 0; |     int ret = idlen == resultlen && memcmp(result, id, idlen) == 0; | ||||||
|     free(id); |     free(id); | ||||||
|     return ret; |     return ret; | ||||||
|  | @ -31,13 +39,16 @@ packet_id (char *context, size_t contextlen, | ||||||
| int | int | ||||||
| main (int argc, char **argv) | main (int argc, char **argv) | ||||||
| { | { | ||||||
|  |     verbose = argc > 1; | ||||||
|  | 
 | ||||||
|     if (!packet_id(PSYC_C2ARG(""), |     if (!packet_id(PSYC_C2ARG(""), | ||||||
| 		   PSYC_C2ARG("psyc://example.net/~alice"), | 		   PSYC_C2ARG("psyc://example.net/~alice"), | ||||||
| 		   PSYC_C2ARG("psyc://example.net/~bob"), | 		   PSYC_C2ARG("psyc://example.net/~bob"), | ||||||
| 		   PSYC_C2ARG("1337"), | 		   PSYC_C2ARG("1337"), | ||||||
| 		   PSYC_C2ARG("42"), | 		   PSYC_C2ARG("42"), | ||||||
| 		   PSYC_C2ARG("||psyc://example.net/~alice|psyc://example.net/~bob" | 		   PSYC_C2ARG("|| psyc://example.net/~alice" | ||||||
| 			      "|1337|42"))) | 			      "| psyc://example.net/~bob" | ||||||
|  | 			      "| 1337| 42"))) | ||||||
| 	return 1; | 	return 1; | ||||||
| 
 | 
 | ||||||
|     if (!packet_id(PSYC_C2ARG("psyc://example.net/@bar"), |     if (!packet_id(PSYC_C2ARG("psyc://example.net/@bar"), | ||||||
|  | @ -45,8 +56,9 @@ main (int argc, char **argv) | ||||||
| 		   PSYC_C2ARG(""), | 		   PSYC_C2ARG(""), | ||||||
| 		   PSYC_C2ARG("1337"), | 		   PSYC_C2ARG("1337"), | ||||||
| 		   PSYC_C2ARG("42"), | 		   PSYC_C2ARG("42"), | ||||||
| 		   PSYC_C2ARG("|psyc://example.net/@bar|psyc://example.net/~alice|" | 		   PSYC_C2ARG("| psyc://example.net/@bar" | ||||||
| 			      "|1337|42"))) | 			      "| psyc://example.net/~alice|" | ||||||
|  | 			      "| 1337| 42"))) | ||||||
| 	return 2; | 	return 2; | ||||||
| 
 | 
 | ||||||
|     if (!packet_id(PSYC_C2ARG("psyc://example.net/@bar"), |     if (!packet_id(PSYC_C2ARG("psyc://example.net/@bar"), | ||||||
|  | @ -54,8 +66,9 @@ main (int argc, char **argv) | ||||||
| 		   PSYC_C2ARG("psyc://example.net/~alice"), | 		   PSYC_C2ARG("psyc://example.net/~alice"), | ||||||
| 		   PSYC_C2ARG("1337"), | 		   PSYC_C2ARG("1337"), | ||||||
| 		   PSYC_C2ARG("42"), | 		   PSYC_C2ARG("42"), | ||||||
| 		   PSYC_C2ARG("|psyc://example.net/@bar||psyc://example.net/~alice" | 		   PSYC_C2ARG("| psyc://example.net/@bar|" | ||||||
| 			      "|1337|42"))) | 			      "| psyc://example.net/~alice" | ||||||
|  | 			      "| 1337| 42"))) | ||||||
| 	return 3; | 	return 3; | ||||||
| 
 | 
 | ||||||
|     if (!packet_id(PSYC_C2ARG("psyc://example.net/@bar"), |     if (!packet_id(PSYC_C2ARG("psyc://example.net/@bar"), | ||||||
|  | @ -63,7 +76,7 @@ main (int argc, char **argv) | ||||||
| 		   PSYC_C2ARG(""), | 		   PSYC_C2ARG(""), | ||||||
| 		   PSYC_C2ARG(""), | 		   PSYC_C2ARG(""), | ||||||
| 		   PSYC_C2ARG(""), | 		   PSYC_C2ARG(""), | ||||||
| 		   PSYC_C2ARG("|psyc://example.net/@bar||||"))) | 		   PSYC_C2ARG("| psyc://example.net/@bar||||"))) | ||||||
| 	return 4; | 	return 4; | ||||||
| 
 | 
 | ||||||
|     return 0; |     return 0; | ||||||
|  |  | ||||||
|  | @ -12,7 +12,7 @@ main (int argc, char **argv) | ||||||
|     uint8_t verbose = argc > 2 && memchr(argv[2], (int)'v', strlen(argv[2])); |     uint8_t verbose = argc > 2 && memchr(argv[2], (int)'v', strlen(argv[2])); | ||||||
|     int idx, ret; |     int idx, ret; | ||||||
|     char buffer[2048], oper; |     char buffer[2048], oper; | ||||||
|     PsycString name, value, elem; |     PsycString name, value, type; | ||||||
|     PsycParseState state; |     PsycParseState state; | ||||||
|     PsycParseListState listState; |     PsycParseListState listState; | ||||||
| 
 | 
 | ||||||
|  | @ -62,12 +62,13 @@ main (int argc, char **argv) | ||||||
| 		psyc_parse_list_state_init(&listState); | 		psyc_parse_list_state_init(&listState); | ||||||
| 		psyc_parse_list_buffer_set(&listState, PSYC_S2ARG(value)); | 		psyc_parse_list_buffer_set(&listState, PSYC_S2ARG(value)); | ||||||
| 
 | 
 | ||||||
| 		while ((ret = psyc_parse_list(&listState, &elem))) { | 		while ((ret = psyc_parse_list(&listState, &type, &value))) { | ||||||
| 		    switch (ret) { | 		    switch (ret) { | ||||||
| 		    case PSYC_PARSE_LIST_END: | 		    case PSYC_PARSE_LIST_END: | ||||||
| 		    case PSYC_PARSE_LIST_ELEM: | 		    case PSYC_PARSE_LIST_ELEM: | ||||||
| 			if (verbose) | 			if (verbose) | ||||||
| 			    printf("|%.*s\n", (int)elem.length, elem.data); | 			    printf("|%.*s %.*s\n", (int)type.length, type.data, | ||||||
|  | 				   (int)value.length, value.data); | ||||||
| 			break; | 			break; | ||||||
| 		    default: | 		    default: | ||||||
| 			printf("Error while parsing list: %i\n", ret); | 			printf("Error while parsing list: %i\n", ret); | ||||||
|  |  | ||||||
							
								
								
									
										128
									
								
								test/test_psyc.c
									
										
									
									
									
								
							
							
						
						
									
										128
									
								
								test/test_psyc.c
									
										
									
									
									
								
							|  | @ -8,9 +8,6 @@ | ||||||
| #include <sys/socket.h> | #include <sys/socket.h> | ||||||
| 
 | 
 | ||||||
| #include <psyc.h> | #include <psyc.h> | ||||||
| #include <psyc/parse.h> |  | ||||||
| #include <psyc/render.h> |  | ||||||
| #include <psyc/syntax.h> |  | ||||||
| 
 | 
 | ||||||
| #include "test.c" | #include "test.c" | ||||||
| 
 | 
 | ||||||
|  | @ -71,10 +68,11 @@ test_input (int i, char *recvbuf, size_t nbytes) | ||||||
|     PsycPacket *packet = &packets[i]; |     PsycPacket *packet = &packets[i]; | ||||||
| 
 | 
 | ||||||
|     char oper; |     char oper; | ||||||
|     PsycString name, value, elem; |     PsycString name, value, type; | ||||||
|     PsycString *pname = NULL, *pvalue = NULL; |     PsycString *pname = NULL, *pvalue = NULL; | ||||||
|     PsycModifier *mod = NULL; |     PsycModifier *mod = NULL; | ||||||
|     PsycParseListState listState; |     PsycParseListState lstate; | ||||||
|  |     PsycParseDictState dstate; | ||||||
|     size_t len; |     size_t len; | ||||||
| 
 | 
 | ||||||
|     // Set buffer with data for the parser.
 |     // Set buffer with data for the parser.
 | ||||||
|  | @ -283,37 +281,131 @@ test_input (int i, char *recvbuf, size_t nbytes) | ||||||
| 	    oper = 0; | 	    oper = 0; | ||||||
| 	    name.length = 0; | 	    name.length = 0; | ||||||
| 	    value.length = 0; | 	    value.length = 0; | ||||||
|  | 	    type.length = 0; | ||||||
| 
 | 
 | ||||||
| 	    if (psyc_var_is_list(PSYC_S2ARG(*pname))) { | 	    switch (psyc_var_type(PSYC_S2ARG(*pname))) { | ||||||
|  | 	    case PSYC_TYPE_LIST: | ||||||
| 		if (verbose >= 2) | 		if (verbose >= 2) | ||||||
| 		    printf("## LIST START\n"); | 		    printf("## LIST START\n"); | ||||||
| 
 | 
 | ||||||
| 		psyc_parse_list_state_init(&listState); | 		psyc_parse_list_state_init(&lstate); | ||||||
| 		psyc_parse_list_buffer_set(&listState, PSYC_S2ARG(*pvalue)); | 		psyc_parse_list_buffer_set(&lstate, PSYC_S2ARG(*pvalue)); | ||||||
| 
 | 
 | ||||||
| 		do { | 		do { | ||||||
| 		    retl = psyc_parse_list(&listState, &elem); | 		    retl = psyc_parse_list(&lstate, &type, &value); | ||||||
| 		    switch (retl) { | 		    switch (retl) { | ||||||
| 		    case PSYC_PARSE_LIST_END: | 		    case PSYC_PARSE_LIST_TYPE: | ||||||
|  | 			if (verbose >= 2) | ||||||
|  | 			    printf("## LIST TYPE: %.*s\n", (int)type.length, type.data); | ||||||
|  | 			break; | ||||||
|  | 		    case PSYC_PARSE_LIST_ELEM_START: | ||||||
|  | 		    case PSYC_PARSE_LIST_ELEM_LAST: | ||||||
| 			retl = 0; | 			retl = 0; | ||||||
| 		    case PSYC_PARSE_LIST_ELEM: | 		    case PSYC_PARSE_LIST_ELEM: | ||||||
| 			if (verbose >= 2) { | 			if (verbose >= 2) { | ||||||
| 			    printf("|%.*s\n", (int)elem.length, elem.data); | 			    printf("|%.*s [%.*s]", (int)type.length, type.data, | ||||||
| 			    if (ret == PSYC_PARSE_LIST_END) | 				   (int)value.length, value.data); | ||||||
| 				printf("## LIST END"); | 			    if (retl == PSYC_PARSE_LIST_ELEM_START) | ||||||
|  | 				printf(" ..."); | ||||||
|  | 			    printf("\n"); | ||||||
|  | 			    if (ret == PSYC_PARSE_LIST_ELEM_LAST) | ||||||
|  | 				printf("## LAST ELEM\n"); | ||||||
| 			} | 			} | ||||||
| 			break; | 			break; | ||||||
| 
 | 		    case PSYC_PARSE_LIST_ELEM_CONT: | ||||||
|  | 			retl = 0; | ||||||
|  | 		    case PSYC_PARSE_LIST_ELEM_END: | ||||||
|  | 			if (verbose >= 2) { | ||||||
|  | 			    printf("... [%.*s]", (int)value.length, value.data); | ||||||
|  | 			    if (retl == PSYC_PARSE_LIST_ELEM_CONT) | ||||||
|  | 				printf(" ..."); | ||||||
|  | 			    printf("\n"); | ||||||
|  | 			} | ||||||
|  | 			break; | ||||||
|  | 		    case PSYC_PARSE_LIST_END: | ||||||
|  | 			retl = 0; | ||||||
|  | 			if (verbose >= 2) | ||||||
|  | 			    printf("## LIST END\n"); | ||||||
|  | 			break; | ||||||
| 		    default: | 		    default: | ||||||
| 			printf("# Error while parsing list: %i\n", retl); | 			printf("# Error while parsing list: %i\n", retl); | ||||||
| 			ret = retl = -1; | 			ret = retl = -1; | ||||||
| 		    } | 		    } | ||||||
| 		} | 		} while (retl > 0); | ||||||
| 		while (retl > 0); | 		break; | ||||||
|  | 	    case PSYC_TYPE_DICT: | ||||||
|  | 		if (verbose >= 2) | ||||||
|  | 		    printf("## DICT START\n"); | ||||||
|  | 
 | ||||||
|  | 		psyc_parse_dict_state_init(&dstate); | ||||||
|  | 		psyc_parse_dict_buffer_set(&dstate, PSYC_S2ARG(*pvalue)); | ||||||
|  | 
 | ||||||
|  | 		do { | ||||||
|  | 		    retl = psyc_parse_dict(&dstate, &type, &value); | ||||||
|  | 		    switch (retl) { | ||||||
|  | 		    case PSYC_PARSE_DICT_TYPE: | ||||||
|  | 			if (verbose >= 2) | ||||||
|  | 			    printf("## DICT TYPE: %.*s\n", (int)type.length, type.data); | ||||||
|  | 			break; | ||||||
|  | 		    case PSYC_PARSE_DICT_KEY_START: | ||||||
|  | 			retl = 0; | ||||||
|  | 		    case PSYC_PARSE_DICT_KEY: | ||||||
|  | 			if (verbose >= 2) { | ||||||
|  | 			    printf("{[%.*s]", (int)value.length, value.data); | ||||||
|  | 			    if (retl == PSYC_PARSE_DICT_KEY_START) | ||||||
|  | 				printf(" ..."); | ||||||
|  | 			    printf("\n"); | ||||||
|  | 			} | ||||||
|  | 			break; | ||||||
|  | 		    case PSYC_PARSE_DICT_KEY_CONT: | ||||||
|  | 			retl = 0; | ||||||
|  | 		    case PSYC_PARSE_DICT_KEY_END: | ||||||
|  | 			if (verbose >= 2) { | ||||||
|  | 			    printf("... [%.*s]", (int)value.length, value.data); | ||||||
|  | 			    if (retl == PSYC_PARSE_DICT_KEY_CONT) | ||||||
|  | 				printf(" ..."); | ||||||
|  | 			    printf("\n"); | ||||||
|  | 			} | ||||||
|  | 			break; | ||||||
|  | 		    case PSYC_PARSE_DICT_VALUE_START: | ||||||
|  | 		    case PSYC_PARSE_DICT_VALUE_LAST: | ||||||
|  | 			retl = 0; | ||||||
|  | 		    case PSYC_PARSE_DICT_VALUE: | ||||||
|  | 			if (verbose >= 2) { | ||||||
|  | 			    printf("}%.*s [%.*s]", (int)type.length, type.data, | ||||||
|  | 				   (int)value.length, value.data); | ||||||
|  | 			    if (retl == PSYC_PARSE_DICT_VALUE_START) | ||||||
|  | 				printf(" ..."); | ||||||
|  | 			    printf("\n"); | ||||||
|  | 			    if (ret == PSYC_PARSE_DICT_VALUE_LAST) | ||||||
|  | 				printf("## LAST VALUE\n"); | ||||||
|  | 			} | ||||||
|  | 			break; | ||||||
|  | 		    case PSYC_PARSE_DICT_VALUE_CONT: | ||||||
|  | 			retl = 0; | ||||||
|  | 		    case PSYC_PARSE_DICT_VALUE_END: | ||||||
|  | 			if (verbose >= 2) { | ||||||
|  | 			    printf("... [%.*s]", (int)value.length, value.data); | ||||||
|  | 			    if (retl == PSYC_PARSE_DICT_VALUE_CONT) | ||||||
|  | 				printf(" ..."); | ||||||
|  | 			    printf("\n"); | ||||||
|  | 			} | ||||||
|  | 			break; | ||||||
|  | 		    case PSYC_PARSE_DICT_END: | ||||||
|  | 			retl = 0; | ||||||
|  | 			printf("## DICT END\n"); | ||||||
|  | 			break; | ||||||
|  | 		    default: | ||||||
|  | 			printf("# Error while parsing dict: %i\n", retl); | ||||||
|  | 			ret = retl = -1; | ||||||
|  | 		    } | ||||||
|  | 		} while (retl > 0); | ||||||
|  | 		break; | ||||||
|  | 	    default: | ||||||
|  | 		break; | ||||||
| 	    } | 	    } | ||||||
| 	} | 	} | ||||||
|     } |     } while (ret > 0); | ||||||
|     while (ret > 0); |  | ||||||
| 
 | 
 | ||||||
|     if (progress) |     if (progress) | ||||||
| 	r = write(1, " ", 1); | 	r = write(1, " ", 1); | ||||||
|  |  | ||||||
|  | @ -1,16 +1,15 @@ | ||||||
| #include <stdio.h> | #include <stdio.h> | ||||||
| 
 | 
 | ||||||
| #include <lib.h> | #include <lib.h> | ||||||
| #include <psyc/render.h> | #include <psyc.h> | ||||||
| #include <psyc/syntax.h> |  | ||||||
| 
 | 
 | ||||||
| #define myUNI	"psyc://10.100.1000/~ludwig"
 | #define myUNI	"psyc://10.100.1000/~ludwig"
 | ||||||
| 
 | 
 | ||||||
| /* example renderer generating a presence packet */ | /* example renderer generating a presence packet */ | ||||||
| int | int | ||||||
| testPresence (char *avail, int availlen, | test_presence (char *avail, int availlen, | ||||||
| 	      char *desc, int desclen, | 	       char *desc, int desclen, | ||||||
| 	      char *rendered, uint8_t verbose) | 	       char *rendered, uint8_t verbose) | ||||||
| { | { | ||||||
|     PsycModifier routing[1]; |     PsycModifier routing[1]; | ||||||
|     psyc_modifier_init(&routing[0], PSYC_OPERATOR_SET, |     psyc_modifier_init(&routing[0], PSYC_OPERATOR_SET, | ||||||
|  | @ -42,7 +41,7 @@ testPresence (char *avail, int availlen, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| int | int | ||||||
| testList (const char *rendered, uint8_t verbose) | test_list (const char *rendered, uint8_t verbose) | ||||||
| { | { | ||||||
|     PsycModifier routing[2]; |     PsycModifier routing[2]; | ||||||
|     psyc_modifier_init(&routing[0], PSYC_OPERATOR_SET, |     psyc_modifier_init(&routing[0], PSYC_OPERATOR_SET, | ||||||
|  | @ -52,23 +51,21 @@ testList (const char *rendered, uint8_t verbose) | ||||||
| 		       PSYC_C2ARG("_context"), | 		       PSYC_C2ARG("_context"), | ||||||
| 		       PSYC_C2ARG(myUNI), PSYC_MODIFIER_ROUTING); | 		       PSYC_C2ARG(myUNI), PSYC_MODIFIER_ROUTING); | ||||||
| 
 | 
 | ||||||
|     PsycString elems_text[] = { |     PsycElem elems_text[] = { | ||||||
| 	PSYC_C2STR("foo"), | 	PSYC_ELEM_V("foo", 3), | ||||||
| 	PSYC_C2STR("bar"), | 	PSYC_ELEM_V("bar", 3), | ||||||
| 	PSYC_C2STR("baz"), | 	PSYC_ELEM_V("baz", 3), | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     PsycString elems_bin[] = { |     PsycElem elems_bin[] = { | ||||||
| 	PSYC_C2STR("foo"), | 	PSYC_ELEM_V("foo", 3), | ||||||
| 	PSYC_C2STR("b|r"), | 	PSYC_ELEM_V("b|r", 3), | ||||||
| 	PSYC_C2STR("baz\nqux"), | 	PSYC_ELEM_V("baz\nqux", 7), | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     PsycList list_text, list_bin; |     PsycList list_text, list_bin; | ||||||
|     psyc_list_init(&list_text, elems_text, |     psyc_list_init(&list_text, elems_text, PSYC_NUM_ELEM(elems_text)); | ||||||
| 		   PSYC_NUM_ELEM(elems_text), PSYC_LIST_CHECK_LENGTH); |     psyc_list_init(&list_bin, elems_bin, PSYC_NUM_ELEM(elems_bin)); | ||||||
|     psyc_list_init(&list_bin, elems_bin, |  | ||||||
| 		   PSYC_NUM_ELEM(elems_bin), PSYC_LIST_CHECK_LENGTH); |  | ||||||
| 
 | 
 | ||||||
|     char buf_text[32], buf_bin[32]; |     char buf_text[32], buf_bin[32]; | ||||||
|     psyc_render_list(&list_text, buf_text, sizeof(buf_text)); |     psyc_render_list(&list_text, buf_text, sizeof(buf_text)); | ||||||
|  | @ -77,10 +74,12 @@ testList (const char *rendered, uint8_t verbose) | ||||||
|     PsycModifier entity[2]; |     PsycModifier entity[2]; | ||||||
|     psyc_modifier_init(&entity[0], PSYC_OPERATOR_SET, |     psyc_modifier_init(&entity[0], PSYC_OPERATOR_SET, | ||||||
| 		       PSYC_C2ARG("_list_text"), | 		       PSYC_C2ARG("_list_text"), | ||||||
| 		       buf_text, list_text.length, list_text.flag); | 		       buf_text, list_text.length, | ||||||
|  | 		       PSYC_MODIFIER_CHECK_LENGTH); | ||||||
|     psyc_modifier_init(&entity[1], PSYC_OPERATOR_SET, |     psyc_modifier_init(&entity[1], PSYC_OPERATOR_SET, | ||||||
| 		       PSYC_C2ARG("_list_binary"), | 		       PSYC_C2ARG("_list_binary"), | ||||||
| 		       buf_bin, list_bin.length, list_bin.flag); | 		       buf_bin, list_bin.length, | ||||||
|  | 		       PSYC_MODIFIER_CHECK_LENGTH); | ||||||
| 
 | 
 | ||||||
|     PsycPacket packet; |     PsycPacket packet; | ||||||
|     psyc_packet_init(&packet, routing, PSYC_NUM_ELEM(routing), |     psyc_packet_init(&packet, routing, PSYC_NUM_ELEM(routing), | ||||||
|  | @ -102,21 +101,21 @@ main (int argc, char **argv) | ||||||
| { | { | ||||||
|     uint8_t verbose = argc > 1; |     uint8_t verbose = argc > 1; | ||||||
| 
 | 
 | ||||||
|     if (testPresence(PSYC_C2ARG("_here"), PSYC_C2ARG("I'm omnipresent right now"), "\
 |     if (test_presence(PSYC_C2ARG("_here"), PSYC_C2ARG("I'm omnipresent right now"), "\
 | ||||||
| :_context\t" myUNI "\n\ | :_context\t" myUNI "\n\ | ||||||
| \n\ | 97\n\ | ||||||
| =_degree_availability\t_here\n\ | =_degree_availability\t_here\n\ | ||||||
| =_description_presence\tI'm omnipresent right now\n\ | =_description_presence 25\tI'm omnipresent right now\n\ | ||||||
| _notice_presence\n\ | _notice_presence\n\ | ||||||
| |\n", verbose)) | |\n", verbose)) | ||||||
| 	return 1; | 	return 1; | ||||||
| 
 | 
 | ||||||
|     if (testList("\
 |     if (test_list("\
 | ||||||
| :_source	psyc://10.100.1000/~ludwig\n\ | :_source	psyc://10.100.1000/~ludwig\n\ | ||||||
| :_context	psyc://10.100.1000/~ludwig\n\ | :_context	psyc://10.100.1000/~ludwig\n\ | ||||||
| 85\n\ | 90\n\ | ||||||
| :_list_text	|foo|bar|baz\n\ | :_list_text 15	| foo| bar| baz\n\ | ||||||
| :_list_binary 21	3 foo|3 b|r|7 baz\n\ | :_list_binary 20	| foo|3 b|r| baz\n\ | ||||||
| qux\n\ | qux\n\ | ||||||
| _test_list\n\ | _test_list\n\ | ||||||
| list test\n\ | list test\n\ | ||||||
|  |  | ||||||
							
								
								
									
										140
									
								
								test/test_update.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										140
									
								
								test/test_update.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,140 @@ | ||||||
|  | #include <psyc.h> | ||||||
|  | #include <unistd.h> | ||||||
|  | #include <stdio.h> | ||||||
|  | #include <stdlib.h> | ||||||
|  | #include <lib.h> | ||||||
|  | 
 | ||||||
|  | uint8_t verbose; | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | test_update (const char *buf, size_t buflen, | ||||||
|  | 	     const char *r_idx, size_t r_idxlen, char r_oper, | ||||||
|  | 	     const char *r_typ, size_t r_typlen, | ||||||
|  | 	     const char *r_val, size_t r_vallen) | ||||||
|  | { | ||||||
|  |     char *idx = malloc(r_idxlen * 2); | ||||||
|  |     char *typ = NULL, *val = NULL; | ||||||
|  |     size_t idxlen = 0, typlen = 0, vallen = 0; | ||||||
|  | 
 | ||||||
|  |     int ret, reti, len = 0; | ||||||
|  |     char oper; | ||||||
|  |     PsycString value; | ||||||
|  |     PsycParseUpdateState state; | ||||||
|  |     psyc_parse_update_state_init(&state); | ||||||
|  |     psyc_parse_update_buffer_set(&state, buf, buflen); | ||||||
|  | 
 | ||||||
|  |     if (verbose) | ||||||
|  | 	printf(">> %.*s\n", (int)buflen, buf); | ||||||
|  | 
 | ||||||
|  |     do { | ||||||
|  | 	ret = reti = psyc_parse_update(&state, &oper, &value); | ||||||
|  | 	len = 0; | ||||||
|  | 
 | ||||||
|  | 	switch (ret) { | ||||||
|  | 	case PSYC_PARSE_INDEX_LIST: | ||||||
|  | 	    len = sprintf(idx + idxlen, "#%ld", value.length); | ||||||
|  | 	    break; | ||||||
|  | 	case PSYC_PARSE_INDEX_STRUCT: | ||||||
|  | 	    len = sprintf(idx + idxlen, ".%.*s", PSYC_S2ARGP(value)); | ||||||
|  | 	    break; | ||||||
|  | 	case PSYC_PARSE_INDEX_DICT: | ||||||
|  | 	    if (state.elemlen_found) | ||||||
|  | 		len = sprintf(idx + idxlen, "{%ld %.*s}", | ||||||
|  | 			      value.length, PSYC_S2ARGP(value)); | ||||||
|  | 	    else | ||||||
|  | 		len = sprintf(idx + idxlen, "{%.*s}", PSYC_S2ARGP(value)); | ||||||
|  | 	    break; | ||||||
|  | 	case PSYC_PARSE_UPDATE_TYPE_END: | ||||||
|  | 	    ret = 0; | ||||||
|  | 	case PSYC_PARSE_UPDATE_TYPE: | ||||||
|  | 	    typ = value.data; | ||||||
|  | 	    typlen = value.length; | ||||||
|  | 	    if (verbose) | ||||||
|  | 		printf("%c[%.*s]\n", oper, PSYC_S2ARGP(value)); | ||||||
|  | 	    break; | ||||||
|  | 	case PSYC_PARSE_UPDATE_VALUE: | ||||||
|  | 	    val = value.data; | ||||||
|  | 	    vallen = value.length; | ||||||
|  | 	    ret = 0; | ||||||
|  | 	    if (verbose) | ||||||
|  | 		printf("[%.*s]\n", PSYC_S2ARGP(value)); | ||||||
|  | 	    break; | ||||||
|  | 	case PSYC_PARSE_UPDATE_END: | ||||||
|  | 	    ret = 0; | ||||||
|  | 	    break; | ||||||
|  | 	default: | ||||||
|  | 	    ret = -1; | ||||||
|  | 	} | ||||||
|  | 	if (verbose && len) | ||||||
|  | 		printf("%2d: %.*s\n", reti, len, idx + idxlen); | ||||||
|  | 	idxlen += len; | ||||||
|  |     } while (ret > 0); | ||||||
|  | 
 | ||||||
|  |     if (ret == 0) { | ||||||
|  | 	if (idxlen != r_idxlen || oper != r_oper | ||||||
|  | 	    || typlen != r_typlen || vallen != r_vallen | ||||||
|  | 	    || memcmp(r_idx, idx, idxlen) != 0 | ||||||
|  | 	    || memcmp(r_typ, typ, typlen) != 0 | ||||||
|  | 	    || memcmp(r_val, val, vallen) != 0) { | ||||||
|  | 	    printf("ERROR: got\n\[%.*s] %c[%.*s]:[%ld] [%.*s] instead of\n[%.*s]\n", | ||||||
|  | 		   (int)idxlen, idx, oper, (int)typlen, typ, vallen, (int)vallen, val, | ||||||
|  | 		   (int)buflen, buf); | ||||||
|  | 	    ret = 1; | ||||||
|  | 	} else | ||||||
|  | 	    ret = 0; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     free(idx); | ||||||
|  |     return ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | main (int argc, char **argv) | ||||||
|  | { | ||||||
|  |     verbose = argc > 1; | ||||||
|  | 
 | ||||||
|  |     if (test_update(PSYC_C2ARG("#1{foo}._bar =_foo:3 bar"), | ||||||
|  | 		    PSYC_C2ARG("#1{foo}._bar"), '=', | ||||||
|  | 		    PSYC_C2ARG("_foo"), | ||||||
|  | 		    PSYC_C2ARG("bar")) != 0) | ||||||
|  | 	return 1; | ||||||
|  | 
 | ||||||
|  |     if (test_update(PSYC_C2ARG("{3 foo}._bar#0 +:3 baz"), | ||||||
|  | 		    PSYC_C2ARG("{3 foo}._bar#0"), '+', | ||||||
|  | 		    PSYC_C2ARG(""), | ||||||
|  | 		    PSYC_C2ARG("baz")) != 0) | ||||||
|  | 	return 2; | ||||||
|  | 
 | ||||||
|  |     if (test_update(PSYC_C2ARG("{foo}#2._bar - 1337"), | ||||||
|  | 		    PSYC_C2ARG("{foo}#2._bar"), '-', | ||||||
|  | 		    PSYC_C2ARG(""), | ||||||
|  | 		    PSYC_C2ARG("1337")) != 0) | ||||||
|  | 	return 3; | ||||||
|  | 
 | ||||||
|  |     if (test_update(PSYC_C2ARG("._bar#1{3 foo} ="), | ||||||
|  | 		    PSYC_C2ARG("._bar#1{3 foo}"), '=', | ||||||
|  | 		    PSYC_C2ARG(""), | ||||||
|  | 		    PSYC_C2ARG("")) != 0) | ||||||
|  | 	return 4; | ||||||
|  | 
 | ||||||
|  |     if (test_update(PSYC_C2ARG("#1{3 foo}._bar =_list"), | ||||||
|  | 		    PSYC_C2ARG("#1{3 foo}._bar"), '=', | ||||||
|  | 		    PSYC_C2ARG("_list"), | ||||||
|  | 		    PSYC_C2ARG("")) != 0) | ||||||
|  | 	return 5; | ||||||
|  | 
 | ||||||
|  |     if (test_update(PSYC_C2ARG("#1{3 foo}._bar =_list "), | ||||||
|  | 		    PSYC_C2ARG("#1{3 foo}._bar"), '=', | ||||||
|  | 		    PSYC_C2ARG("_list"), | ||||||
|  | 		    PSYC_C2ARG("")) != 0) | ||||||
|  | 	return 6; | ||||||
|  | 
 | ||||||
|  |     if (test_update(PSYC_C2ARG("#1{3 foo}._bar =_list"), | ||||||
|  | 		    PSYC_C2ARG("#1{3 foo}._bar"), '=', | ||||||
|  | 		    PSYC_C2ARG("_list"), | ||||||
|  | 		    PSYC_C2ARG("")) != 0) | ||||||
|  | 	return 7; | ||||||
|  | 
 | ||||||
|  |     printf("test_update passed all tests.\n"); | ||||||
|  |     return 0; // passed all tests
 | ||||||
|  | } | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue