325 lines
		
	
	
		
			8.3 KiB
		
	
	
	
		
			Haxe
		
	
	
	
	
	
		
		
			
		
	
	
			325 lines
		
	
	
		
			8.3 KiB
		
	
	
	
		
			Haxe
		
	
	
	
	
	
|  | /* | ||
|  |  * Copyright (C)2005-2019 Haxe Foundation | ||
|  |  * | ||
|  |  * Permission is hereby granted, free of charge, to any person obtaining a | ||
|  |  * copy of this software and associated documentation files (the "Software"), | ||
|  |  * to deal in the Software without restriction, including without limitation | ||
|  |  * the rights to use, copy, modify, merge, publish, distribute, sublicense, | ||
|  |  * and/or sell copies of the Software, and to permit persons to whom the | ||
|  |  * Software is furnished to do so, subject to the following conditions: | ||
|  |  * | ||
|  |  * The above copyright notice and this permission notice shall be included in | ||
|  |  * all copies or substantial portions of the Software. | ||
|  |  * | ||
|  |  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||
|  |  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||
|  |  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||
|  |  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||
|  |  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING | ||
|  |  * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER | ||
|  |  * DEALINGS IN THE SOFTWARE. | ||
|  |  */ | ||
|  | 
 | ||
|  | package haxe.format; | ||
|  | 
 | ||
|  | /** | ||
|  | 	An implementation of JSON parser in Haxe. | ||
|  | 
 | ||
|  | 	This class is used by `haxe.Json` when native JSON implementation | ||
|  | 	is not available. | ||
|  | 
 | ||
|  | 	@see https://haxe.org/manual/std-Json-parsing.html | ||
|  | **/ | ||
|  | class JsonParser { | ||
|  | 	/** | ||
|  | 		Parses given JSON-encoded `str` and returns the resulting object. | ||
|  | 
 | ||
|  | 		JSON objects are parsed into anonymous structures and JSON arrays | ||
|  | 		are parsed into `Array<Dynamic>`. | ||
|  | 
 | ||
|  | 		If given `str` is not valid JSON, an exception will be thrown. | ||
|  | 
 | ||
|  | 		If `str` is null, the result is unspecified. | ||
|  | 	**/ | ||
|  | 	static public inline function parse(str:String):Dynamic { | ||
|  | 		return new JsonParser(str).doParse(); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	var str:String; | ||
|  | 	var pos:Int; | ||
|  | 
 | ||
|  | 	function new(str:String) { | ||
|  | 		this.str = str; | ||
|  | 		this.pos = 0; | ||
|  | 	} | ||
|  | 
 | ||
|  | 	function doParse():Dynamic { | ||
|  | 		var result = parseRec(); | ||
|  | 		var c; | ||
|  | 		while (!StringTools.isEof(c = nextChar())) { | ||
|  | 			switch (c) { | ||
|  | 				case ' '.code, '\r'.code, '\n'.code, '\t'.code: | ||
|  | 				// allow trailing whitespace | ||
|  | 				default: | ||
|  | 					invalidChar(); | ||
|  | 			} | ||
|  | 		} | ||
|  | 		return result; | ||
|  | 	} | ||
|  | 
 | ||
|  | 	function parseRec():Dynamic { | ||
|  | 		while (true) { | ||
|  | 			var c = nextChar(); | ||
|  | 			switch (c) { | ||
|  | 				case ' '.code, '\r'.code, '\n'.code, '\t'.code: | ||
|  | 				// loop | ||
|  | 				case '{'.code: | ||
|  | 					var obj = {}, field = null, comma:Null<Bool> = null; | ||
|  | 					while (true) { | ||
|  | 						var c = nextChar(); | ||
|  | 						switch (c) { | ||
|  | 							case ' '.code, '\r'.code, '\n'.code, '\t'.code: | ||
|  | 							// loop | ||
|  | 							case '}'.code: | ||
|  | 								if (field != null || comma == false) | ||
|  | 									invalidChar(); | ||
|  | 								return obj; | ||
|  | 							case ':'.code: | ||
|  | 								if (field == null) | ||
|  | 									invalidChar(); | ||
|  | 								Reflect.setField(obj, field, parseRec()); | ||
|  | 								field = null; | ||
|  | 								comma = true; | ||
|  | 							case ','.code: | ||
|  | 								if (comma) comma = false else invalidChar(); | ||
|  | 							case '"'.code: | ||
|  | 								if (field != null || comma) invalidChar(); | ||
|  | 								field = parseString(); | ||
|  | 							default: | ||
|  | 								invalidChar(); | ||
|  | 						} | ||
|  | 					} | ||
|  | 				case '['.code: | ||
|  | 					var arr = [], comma:Null<Bool> = null; | ||
|  | 					while (true) { | ||
|  | 						var c = nextChar(); | ||
|  | 						switch (c) { | ||
|  | 							case ' '.code, '\r'.code, '\n'.code, '\t'.code: | ||
|  | 							// loop | ||
|  | 							case ']'.code: | ||
|  | 								if (comma == false) invalidChar(); | ||
|  | 								return arr; | ||
|  | 							case ','.code: | ||
|  | 								if (comma) comma = false else invalidChar(); | ||
|  | 							default: | ||
|  | 								if (comma) invalidChar(); | ||
|  | 								pos--; | ||
|  | 								arr.push(parseRec()); | ||
|  | 								comma = true; | ||
|  | 						} | ||
|  | 					} | ||
|  | 				case 't'.code: | ||
|  | 					var save = pos; | ||
|  | 					if (nextChar() != 'r'.code || nextChar() != 'u'.code || nextChar() != 'e'.code) { | ||
|  | 						pos = save; | ||
|  | 						invalidChar(); | ||
|  | 					} | ||
|  | 					return true; | ||
|  | 				case 'f'.code: | ||
|  | 					var save = pos; | ||
|  | 					if (nextChar() != 'a'.code || nextChar() != 'l'.code || nextChar() != 's'.code || nextChar() != 'e'.code) { | ||
|  | 						pos = save; | ||
|  | 						invalidChar(); | ||
|  | 					} | ||
|  | 					return false; | ||
|  | 				case 'n'.code: | ||
|  | 					var save = pos; | ||
|  | 					if (nextChar() != 'u'.code || nextChar() != 'l'.code || nextChar() != 'l'.code) { | ||
|  | 						pos = save; | ||
|  | 						invalidChar(); | ||
|  | 					} | ||
|  | 					return null; | ||
|  | 				case '"'.code: | ||
|  | 					return parseString(); | ||
|  | 				case '0'.code, '1'.code, '2'.code, '3'.code, '4'.code, '5'.code, '6'.code, '7'.code, '8'.code, '9'.code, '-'.code: | ||
|  | 					return parseNumber(c); | ||
|  | 				default: | ||
|  | 					invalidChar(); | ||
|  | 			} | ||
|  | 		} | ||
|  | 	} | ||
|  | 
 | ||
|  | 	function parseString() { | ||
|  | 		var start = pos; | ||
|  | 		var buf:StringBuf = null; | ||
|  | 		#if target.unicode | ||
|  | 		var prev = -1; | ||
|  | 		inline function cancelSurrogate() { | ||
|  | 			// invalid high surrogate (not followed by low surrogate) | ||
|  | 			buf.addChar(0xFFFD); | ||
|  | 			prev = -1; | ||
|  | 		} | ||
|  | 		#end | ||
|  | 		while (true) { | ||
|  | 			var c = nextChar(); | ||
|  | 			if (c == '"'.code) | ||
|  | 				break; | ||
|  | 			if (c == '\\'.code) { | ||
|  | 				if (buf == null) { | ||
|  | 					buf = new StringBuf(); | ||
|  | 				} | ||
|  | 				buf.addSub(str, start, pos - start - 1); | ||
|  | 				c = nextChar(); | ||
|  | 				#if target.unicode | ||
|  | 				if (c != "u".code && prev != -1) | ||
|  | 					cancelSurrogate(); | ||
|  | 				#end | ||
|  | 				switch (c) { | ||
|  | 					case "r".code: | ||
|  | 						buf.addChar("\r".code); | ||
|  | 					case "n".code: | ||
|  | 						buf.addChar("\n".code); | ||
|  | 					case "t".code: | ||
|  | 						buf.addChar("\t".code); | ||
|  | 					case "b".code: | ||
|  | 						buf.addChar(8); | ||
|  | 					case "f".code: | ||
|  | 						buf.addChar(12); | ||
|  | 					case "/".code, '\\'.code, '"'.code: | ||
|  | 						buf.addChar(c); | ||
|  | 					case 'u'.code: | ||
|  | 						var uc:Int = Std.parseInt("0x" + str.substr(pos, 4)); | ||
|  | 						pos += 4; | ||
|  | 						#if !target.unicode | ||
|  | 						if (uc <= 0x7F) | ||
|  | 							buf.addChar(uc); | ||
|  | 						else if (uc <= 0x7FF) { | ||
|  | 							buf.addChar(0xC0 | (uc >> 6)); | ||
|  | 							buf.addChar(0x80 | (uc & 63)); | ||
|  | 						} else if (uc <= 0xFFFF) { | ||
|  | 							buf.addChar(0xE0 | (uc >> 12)); | ||
|  | 							buf.addChar(0x80 | ((uc >> 6) & 63)); | ||
|  | 							buf.addChar(0x80 | (uc & 63)); | ||
|  | 						} else { | ||
|  | 							buf.addChar(0xF0 | (uc >> 18)); | ||
|  | 							buf.addChar(0x80 | ((uc >> 12) & 63)); | ||
|  | 							buf.addChar(0x80 | ((uc >> 6) & 63)); | ||
|  | 							buf.addChar(0x80 | (uc & 63)); | ||
|  | 						} | ||
|  | 						#else | ||
|  | 						if (prev != -1) { | ||
|  | 							if (uc < 0xDC00 || uc > 0xDFFF) | ||
|  | 								cancelSurrogate(); | ||
|  | 							else { | ||
|  | 								buf.addChar(((prev - 0xD800) << 10) + (uc - 0xDC00) + 0x10000); | ||
|  | 								prev = -1; | ||
|  | 							} | ||
|  | 						} else if (uc >= 0xD800 && uc <= 0xDBFF) | ||
|  | 							prev = uc; | ||
|  | 						else | ||
|  | 							buf.addChar(uc); | ||
|  | 						#end | ||
|  | 					default: | ||
|  | 						throw "Invalid escape sequence \\" + String.fromCharCode(c) + " at position " + (pos - 1); | ||
|  | 				} | ||
|  | 				start = pos; | ||
|  | 			} | ||
|  | 			#if !(target.unicode) | ||
|  | 			// ensure utf8 chars are not cut | ||
|  | 			else if (c >= 0x80) { | ||
|  | 				pos++; | ||
|  | 				if (c >= 0xFC) | ||
|  | 					pos += 4; | ||
|  | 				else if (c >= 0xF8) | ||
|  | 					pos += 3; | ||
|  | 				else if (c >= 0xF0) | ||
|  | 					pos += 2; | ||
|  | 				else if (c >= 0xE0) | ||
|  | 					pos++; | ||
|  | 			} | ||
|  | 			#end | ||
|  | 		else if (StringTools.isEof(c)) | ||
|  | 			throw "Unclosed string"; | ||
|  | 		} | ||
|  | 		#if target.unicode | ||
|  | 		if (prev != -1) | ||
|  | 			cancelSurrogate(); | ||
|  | 		#end | ||
|  | 		if (buf == null) { | ||
|  | 			return str.substr(start, pos - start - 1); | ||
|  | 		} else { | ||
|  | 			buf.addSub(str, start, pos - start - 1); | ||
|  | 			return buf.toString(); | ||
|  | 		} | ||
|  | 	} | ||
|  | 
 | ||
|  | 	inline function parseNumber(c:Int):Dynamic { | ||
|  | 		var start = pos - 1; | ||
|  | 		var minus = c == '-'.code, digit = !minus, zero = c == '0'.code; | ||
|  | 		var point = false, e = false, pm = false, end = false; | ||
|  | 		while (true) { | ||
|  | 			c = nextChar(); | ||
|  | 			switch (c) { | ||
|  | 				case '0'.code: | ||
|  | 					if (zero && !point) | ||
|  | 						invalidNumber(start); | ||
|  | 					if (minus) { | ||
|  | 						minus = false; | ||
|  | 						zero = true; | ||
|  | 					} | ||
|  | 					digit = true; | ||
|  | 				case '1'.code, '2'.code, '3'.code, '4'.code, '5'.code, '6'.code, '7'.code, '8'.code, '9'.code: | ||
|  | 					if (zero && !point) | ||
|  | 						invalidNumber(start); | ||
|  | 					if (minus) | ||
|  | 						minus = false; | ||
|  | 					digit = true; | ||
|  | 					zero = false; | ||
|  | 				case '.'.code: | ||
|  | 					if (minus || point || e) | ||
|  | 						invalidNumber(start); | ||
|  | 					digit = false; | ||
|  | 					point = true; | ||
|  | 				case 'e'.code, 'E'.code: | ||
|  | 					if (minus || zero || e) | ||
|  | 						invalidNumber(start); | ||
|  | 					digit = false; | ||
|  | 					e = true; | ||
|  | 				case '+'.code, '-'.code: | ||
|  | 					if (!e || pm) | ||
|  | 						invalidNumber(start); | ||
|  | 					digit = false; | ||
|  | 					pm = true; | ||
|  | 				default: | ||
|  | 					if (!digit) | ||
|  | 						invalidNumber(start); | ||
|  | 					pos--; | ||
|  | 					end = true; | ||
|  | 			} | ||
|  | 			if (end) | ||
|  | 				break; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		var f = Std.parseFloat(str.substr(start, pos - start)); | ||
|  | 		if(point) { | ||
|  | 			return f; | ||
|  | 		} else { | ||
|  | 			var i = Std.int(f); | ||
|  | 			return if (i == f) i else f; | ||
|  | 		} | ||
|  | 	} | ||
|  | 
 | ||
|  | 	inline function nextChar() { | ||
|  | 		return StringTools.fastCodeAt(str, pos++); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	function invalidChar() { | ||
|  | 		pos--; // rewind | ||
|  | 		throw "Invalid char " + StringTools.fastCodeAt(str, pos) + " at position " + pos; | ||
|  | 	} | ||
|  | 
 | ||
|  | 	function invalidNumber(start:Int) { | ||
|  | 		throw "Invalid number at position " + start + ": " + str.substr(start, pos - start); | ||
|  | 	} | ||
|  | } |