You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							95 lines
						
					
					
						
							3.5 KiB
						
					
					
				
			
		
		
	
	
							95 lines
						
					
					
						
							3.5 KiB
						
					
					
				| # Flatted Specifications
 | |
| 
 | |
| This document describes operations performed to produce, or parse, the flatted output.
 | |
| 
 | |
| ## stringify(any) => flattedString
 | |
| 
 | |
| The output is always an `Array` that contains at index `0` the given value.
 | |
| 
 | |
| If the value is an `Array` or an `Object`, per each property value passed through the callback, return the value as is if it's not an `Array`, an `Object`, or a `string`.
 | |
| 
 | |
| In case it's an `Array`, an `Object`, or a `string`, return the index as `string`, associated through a `Map`.
 | |
| 
 | |
| Giving the following example:
 | |
| 
 | |
| ```js
 | |
| flatted.stringify('a');                     // ["a"]
 | |
| flatted.stringify(['a']);                   // [["1"],"a"]
 | |
| flatted.stringify(['a', 1, 'b']);           // [["1",1,"2"],"a","b"]
 | |
| ```
 | |
| 
 | |
| There is an `input` containing `[array, "a", "b"]`, where the `array` has indexes `"1"` and `"2"` as strings, indexes that point respectively at `"a"` and `"b"` within the input `[array, "a", "b"]`.
 | |
| 
 | |
| The exact same happens for objects.
 | |
| 
 | |
| ```js
 | |
| flatted.stringify('a');                     // ["a"]
 | |
| flatted.stringify({a: 'a'});                // [{"a":"1"},"a"]
 | |
| flatted.stringify({a: 'a', n: 1, b: 'b'});  // [{"a":"1","n":1,"b":"2"},"a","b"]
 | |
| ```
 | |
| 
 | |
| Every object, string, or array, encountered during serialization will be stored once as stringified index.
 | |
| 
 | |
| ```js
 | |
| // per each property/value of the object/array
 | |
| if (any == null || !/object|string/.test(typeof any))
 | |
|   return any;
 | |
| if (!map.has(any)) {
 | |
|   const index = String(arr.length);
 | |
|   arr.push(any);
 | |
|   map.set(any, index);
 | |
| }
 | |
| return map.get(any);
 | |
| ```
 | |
| 
 | |
| This, performed before going through all properties, grants unique indexes per reference.
 | |
| 
 | |
| The stringified indexes ensure there won't be conflicts with regularly stored numbers.
 | |
| 
 | |
| ## parse(flattedString) => any
 | |
| 
 | |
| Everything that is a `string` is wrapped as `new String`, but strings in the array, from index `1` on, is kept as regular `string`.
 | |
| 
 | |
| ```js
 | |
| const input = JSON.parse('[{"a":"1"},"b"]', Strings).map(strings);
 | |
| // convert strings primitives into String instances
 | |
| function Strings(key, value) {
 | |
|   return typeof value === 'string' ? new String(value) : value;
 | |
| }
 | |
| // converts String instances into strings primitives
 | |
| function strings(value) {
 | |
|   return value instanceof String ? String(value) : value;
 | |
| }
 | |
| ```
 | |
| 
 | |
| The `input` array will have a regular `string` at index `1`, but its object at index `0` will have an `instanceof String` as `.a` property.
 | |
| 
 | |
| That is the key to place back values from the rest of the array, so that per each property of the object at index `0`, if the value is an `instanceof` String, something not serializable via JSON, it means it can be used to retrieve the position of its value from the `input` array.
 | |
| 
 | |
| If such `value` is an object and it hasn't been parsed yet, add it as parsed and go through all its properties/values.
 | |
| 
 | |
| ```js
 | |
| // outside any loop ...
 | |
| const parsed = new Set;
 | |
| 
 | |
| // ... per each property/value ...
 | |
| if (value instanceof Primitive) {
 | |
|   const tmp = input[parseInt(value)];
 | |
|   if (typeof tmp === 'object' && !parsed.has(tmp)) {
 | |
|     parsed.add(tmp);
 | |
|     output[key] = tmp;
 | |
|     if (typeof tmp === 'object' && tmp != null) {
 | |
|       // perform this same logic per
 | |
|       // each nested property/value ...
 | |
|     }
 | |
|   } else {
 | |
|     output[key] = tmp;
 | |
|   }
 | |
| } else
 | |
|   output[key] = tmp;
 | |
| ```
 | |
| 
 | |
| As summary, the whole logic is based on polluting the de-serialization with a kind of variable that is unexpected, hence secure to use as directive to retrieve an index with a value.
 | |
| 
 | |
| The usage of a `Map` and a `Set` to flag known references/strings as visited/stored makes **flatted** a rock solid, fast, and compact, solution.
 |