/src/com/google/maps/extras/arcgislink/json/JSONDecoder.as

http://gmaps-utility-library-flash.googlecode.com/ · ActionScript · 329 lines · 171 code · 38 blank · 120 comment · 45 complexity · d7aeebbaf93b530404585b70443f9962 MD5 · raw file

  1. /*
  2. Copyright (c) 2008, Adobe Systems Incorporated
  3. All rights reserved.
  4. Redistribution and use in source and binary forms, with or without
  5. modification, are permitted provided that the following conditions are
  6. met:
  7. * Redistributions of source code must retain the above copyright notice,
  8. this list of conditions and the following disclaimer.
  9. * Redistributions in binary form must reproduce the above copyright
  10. notice, this list of conditions and the following disclaimer in the
  11. documentation and/or other materials provided with the distribution.
  12. * Neither the name of Adobe Systems Incorporated nor the names of its
  13. contributors may be used to endorse or promote products derived from
  14. this software without specific prior written permission.
  15. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
  16. IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
  17. THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
  18. PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
  19. CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
  20. EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
  21. PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
  22. PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
  23. LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
  24. NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
  25. SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  26. */
  27. //http://code.google.com/p/as3corelib/
  28. //package com.adobe.serialization.json
  29. package com.google.maps.extras.arcgislink.json
  30. {
  31. public class JSONDecoder
  32. {
  33. /**
  34. * Flag indicating if the parser should be strict about the format
  35. * of the JSON string it is attempting to decode.
  36. */
  37. private var strict:Boolean;
  38. /** The value that will get parsed from the JSON string */
  39. private var value:*;
  40. /** The tokenizer designated to read the JSON string */
  41. private var tokenizer:JSONTokenizer;
  42. /** The current token from the tokenizer */
  43. private var token:JSONToken;
  44. /**
  45. * Constructs a new JSONDecoder to parse a JSON string
  46. * into a native object.
  47. *
  48. * @param s The JSON string to be converted
  49. * into a native object
  50. * @param strict Flag indicating if the JSON string needs to
  51. * strictly match the JSON standard or not.
  52. * @langversion ActionScript 3.0
  53. * @playerversion Flash 9.0
  54. * @tiptext
  55. */
  56. public function JSONDecoder( s:String, strict:Boolean )
  57. {
  58. this.strict = strict;
  59. tokenizer = new JSONTokenizer( s, strict );
  60. nextToken();
  61. value = parseValue();
  62. // Make sure the input stream is empty
  63. if ( strict && nextToken() != null )
  64. {
  65. tokenizer.parseError( "Unexpected characters left in input stream" );
  66. }
  67. }
  68. /**
  69. * Gets the internal object that was created by parsing
  70. * the JSON string passed to the constructor.
  71. *
  72. * @return The internal object representation of the JSON
  73. * string that was passed to the constructor
  74. * @langversion ActionScript 3.0
  75. * @playerversion Flash 9.0
  76. * @tiptext
  77. */
  78. public function getValue():*
  79. {
  80. return value;
  81. }
  82. /**
  83. * Returns the next token from the tokenzier reading
  84. * the JSON string
  85. */
  86. private function nextToken():JSONToken
  87. {
  88. return token = tokenizer.getNextToken();
  89. }
  90. /**
  91. * Attempt to parse an array.
  92. */
  93. private function parseArray():Array
  94. {
  95. // create an array internally that we're going to attempt
  96. // to parse from the tokenizer
  97. var a:Array = new Array();
  98. // grab the next token from the tokenizer to move
  99. // past the opening [
  100. nextToken();
  101. // check to see if we have an empty array
  102. if ( token.type == JSONTokenType.RIGHT_BRACKET )
  103. {
  104. // we're done reading the array, so return it
  105. return a;
  106. }
  107. // in non-strict mode an empty array is also a comma
  108. // followed by a right bracket
  109. else if ( !strict && token.type == JSONTokenType.COMMA )
  110. {
  111. // move past the comma
  112. nextToken();
  113. // check to see if we're reached the end of the array
  114. if ( token.type == JSONTokenType.RIGHT_BRACKET )
  115. {
  116. return a;
  117. }
  118. else
  119. {
  120. tokenizer.parseError( "Leading commas are not supported. Expecting ']' but found " + token.value );
  121. }
  122. }
  123. // deal with elements of the array, and use an "infinite"
  124. // loop because we could have any amount of elements
  125. while ( true )
  126. {
  127. // read in the value and add it to the array
  128. a.push( parseValue() );
  129. // after the value there should be a ] or a ,
  130. nextToken();
  131. if ( token.type == JSONTokenType.RIGHT_BRACKET )
  132. {
  133. // we're done reading the array, so return it
  134. return a;
  135. }
  136. else if ( token.type == JSONTokenType.COMMA )
  137. {
  138. // move past the comma and read another value
  139. nextToken();
  140. // Allow arrays to have a comma after the last element
  141. // if the decoder is not in strict mode
  142. if ( !strict )
  143. {
  144. // Reached ",]" as the end of the array, so return it
  145. if ( token.type == JSONTokenType.RIGHT_BRACKET )
  146. {
  147. return a;
  148. }
  149. }
  150. }
  151. else
  152. {
  153. tokenizer.parseError( "Expecting ] or , but found " + token.value );
  154. }
  155. }
  156. return null;
  157. }
  158. /**
  159. * Attempt to parse an object.
  160. */
  161. private function parseObject():Object
  162. {
  163. // create the object internally that we're going to
  164. // attempt to parse from the tokenizer
  165. var o:Object = new OrderedObject();//NL: changed new Object();
  166. // store the string part of an object member so
  167. // that we can assign it a value in the object
  168. var key:String
  169. // grab the next token from the tokenizer
  170. nextToken();
  171. // check to see if we have an empty object
  172. if ( token.type == JSONTokenType.RIGHT_BRACE )
  173. {
  174. // we're done reading the object, so return it
  175. return o;
  176. }
  177. // in non-strict mode an empty object is also a comma
  178. // followed by a right bracket
  179. else if ( !strict && token.type == JSONTokenType.COMMA )
  180. {
  181. // move past the comma
  182. nextToken();
  183. // check to see if we're reached the end of the object
  184. if ( token.type == JSONTokenType.RIGHT_BRACE )
  185. {
  186. return o;
  187. }
  188. else
  189. {
  190. tokenizer.parseError( "Leading commas are not supported. Expecting '}' but found " + token.value );
  191. }
  192. }
  193. // deal with members of the object, and use an "infinite"
  194. // loop because we could have any amount of members
  195. while ( true )
  196. {
  197. if ( token.type == JSONTokenType.STRING )
  198. {
  199. // the string value we read is the key for the object
  200. key = String( token.value );
  201. // move past the string to see what's next
  202. nextToken();
  203. // after the string there should be a :
  204. if ( token.type == JSONTokenType.COLON )
  205. {
  206. // move past the : and read/assign a value for the key
  207. nextToken();
  208. o[key] = parseValue();
  209. // move past the value to see what's next
  210. nextToken();
  211. // after the value there's either a } or a ,
  212. if ( token.type == JSONTokenType.RIGHT_BRACE )
  213. {
  214. // we're done reading the object, so return it
  215. return o;
  216. }
  217. else if ( token.type == JSONTokenType.COMMA )
  218. {
  219. // skip past the comma and read another member
  220. nextToken();
  221. // Allow objects to have a comma after the last member
  222. // if the decoder is not in strict mode
  223. if ( !strict )
  224. {
  225. // Reached ",}" as the end of the object, so return it
  226. if ( token.type == JSONTokenType.RIGHT_BRACE )
  227. {
  228. return o;
  229. }
  230. }
  231. }
  232. else
  233. {
  234. tokenizer.parseError( "Expecting } or , but found " + token.value );
  235. }
  236. }
  237. else
  238. {
  239. tokenizer.parseError( "Expecting : but found " + token.value );
  240. }
  241. }
  242. else
  243. {
  244. tokenizer.parseError( "Expecting string but found " + token.value );
  245. }
  246. }
  247. return null;
  248. }
  249. /**
  250. * Attempt to parse a value
  251. */
  252. private function parseValue():Object
  253. {
  254. // Catch errors when the input stream ends abruptly
  255. if ( token == null )
  256. {
  257. tokenizer.parseError( "Unexpected end of input" );
  258. }
  259. switch ( token.type )
  260. {
  261. case JSONTokenType.LEFT_BRACE:
  262. return parseObject();
  263. case JSONTokenType.LEFT_BRACKET:
  264. return parseArray();
  265. case JSONTokenType.STRING:
  266. case JSONTokenType.NUMBER:
  267. case JSONTokenType.TRUE:
  268. case JSONTokenType.FALSE:
  269. case JSONTokenType.NULL:
  270. return token.value;
  271. case JSONTokenType.NAN:
  272. if ( !strict )
  273. {
  274. return token.value;
  275. }
  276. else
  277. {
  278. tokenizer.parseError( "Unexpected " + token.value );
  279. }
  280. default:
  281. tokenizer.parseError( "Unexpected " + token.value );
  282. }
  283. return null;
  284. }
  285. }
  286. }