testjsonreader.pp 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821
  1. unit testjsonreader;
  2. {$mode objfpc}
  3. interface
  4. uses
  5. Classes, SysUtils, fpcunit, testregistry,fpjson,jsonscanner,jsonreader, testjsondata;
  6. Const
  7. DefaultOpts = [joUTF8,joStrict];
  8. type
  9. { TMyJSONReader }
  10. TMyJSONReader = Class(TBaseJSONReader)
  11. Private
  12. FList : TStrings;
  13. function GetList: TStrings;
  14. procedure Push(const aType : String; const AValue: String='');
  15. protected
  16. procedure BooleanValue(const AValue: Boolean); override;
  17. procedure EndArray; override;
  18. procedure EndObject; override;
  19. procedure FloatValue(const AValue: Double); override;
  20. procedure Int64Value(const AValue: int64); override;
  21. procedure IntegerValue(const AValue: integer); override;
  22. procedure KeyValue(const AKey: TJSONStringType); override;
  23. procedure NullValue; override;
  24. procedure NumberValue(const AValue: TJSONStringType); override;
  25. procedure QWordValue(const AValue: QWord); override;
  26. procedure StartArray; override;
  27. procedure StartObject; override;
  28. procedure StringValue(const AValue: TJSONStringType); override;
  29. Public
  30. destructor Destroy; override;
  31. Property List : TStrings Read GetList;
  32. end;
  33. { TTestParser }
  34. { TTestReader }
  35. { TBaseTestReader }
  36. TBaseTestReader = class(TTestJSON)
  37. private
  38. FOptions : TJSONOptions;
  39. procedure CallNoHandlerStream;
  40. procedure DoTestFloat(F: TJSONFloat); overload;
  41. procedure DoTestFloat(F: TJSONFloat; S: String); overload;
  42. procedure DoTestString(S: String; AValue: String='');
  43. procedure DoTrailingCommaErrorArray;
  44. procedure DoTrailingCommaErrorObject;
  45. Protected
  46. procedure DoTestError(S: String; Options : TJSONOptions = DefaultOpts); virtual; abstract;
  47. Procedure TestRead(aJSON : String; AResult : Array of string); virtual; abstract;
  48. published
  49. procedure TestEmpty;
  50. procedure TestNull;
  51. procedure TestTrue;
  52. procedure TestFalse;
  53. procedure TestFloat;
  54. procedure TestFloatError;
  55. procedure TestInteger;
  56. procedure TestInt64;
  57. procedure TestString;
  58. procedure TestArray;
  59. procedure TestObject;
  60. procedure TestObjectError;
  61. procedure TestTrailingComma;
  62. procedure TestTrailingCommaErrorArray;
  63. procedure TestTrailingCommaErrorObject;
  64. procedure TestMixed;
  65. Procedure TestComment;
  66. procedure TestErrors;
  67. end;
  68. TTestReader = Class(TBaseTestReader)
  69. Private
  70. FReader: TMyJSONReader;
  71. Protected
  72. Procedure Teardown; override;
  73. Public
  74. procedure DoTestError(S: String; Options : TJSONOptions = DefaultOpts); override;
  75. Procedure TestRead(aJSON : String; AResult : Array of string); override;
  76. Property Reader : TMyJSONReader Read FReader;
  77. end;
  78. { TJSONConsumer }
  79. TJSONConsumer = Class(TInterfacedObject,IJSONConsumer)
  80. Private
  81. FList : TStrings;
  82. procedure Push(const aType : String; const AValue: String='');
  83. protected
  84. procedure BooleanValue(const AValue: Boolean);
  85. procedure EndArray;
  86. procedure EndObject;
  87. procedure FloatValue(const AValue: Double);
  88. procedure Int64Value(const AValue: int64);
  89. procedure IntegerValue(const AValue: integer);
  90. procedure KeyName(const AKey: TJSONStringType);
  91. procedure NullValue;
  92. procedure NumberValue(const AValue: TJSONStringType);
  93. procedure QWordValue(const AValue: QWord);
  94. procedure StartArray;
  95. procedure StartObject;
  96. procedure StringValue(const AValue: TJSONStringType);
  97. Public
  98. Constructor Create(AList : TStrings);
  99. Property List : TStrings Read FList;
  100. end;
  101. { TTestJSONConsumerReader }
  102. TTestJSONConsumerReader = Class(TBaseTestReader)
  103. Private
  104. FList : TStrings;
  105. FReader: TJSONConsumerReader;
  106. Protected
  107. Procedure Teardown; override;
  108. Public
  109. procedure DoTestError(S: String; Options : TJSONOptions = DefaultOpts); override;
  110. Procedure TestRead(aJSON : String; AResult : Array of string); override;
  111. Property Reader : TJSONConsumerReader Read FReader;
  112. end;
  113. { TTestJSONEventReader }
  114. TTestJSONEventReader = Class(TBaseTestReader)
  115. Private
  116. FList : TStrings;
  117. FReader: TJSONEventReader;
  118. Protected
  119. procedure Push(const aType : String; const AValue: String='');
  120. procedure BooleanValue(Sender: TObject; const AValue: Boolean);
  121. procedure EndArray(Sender: TObject);
  122. procedure EndObject(Sender: TObject);
  123. procedure FloatValue(Sender: TObject; const AValue: Double);
  124. procedure Int64Value(Sender: TObject; const AValue: int64);
  125. procedure IntegerValue(Sender: TObject; const AValue: integer);
  126. procedure KeyValue(Sender: TObject; const AKey: TJSONStringType);
  127. procedure NullValue(Sender: TObject);
  128. procedure NumberValue(Sender: TObject; const AValue: TJSONStringType);
  129. procedure QWordValue(Sender: TObject; const AValue: QWord);
  130. procedure StartArray(Sender: TObject);
  131. procedure StartObject(Sender: TObject);
  132. procedure StringValue(Sender: TObject; const AValue: TJSONStringType);
  133. Procedure HookupEvents(AReader: TJSONEventReader);
  134. Procedure Teardown; override;
  135. Public
  136. procedure DoTestError(S: String; Options : TJSONOptions = DefaultOpts); override;
  137. Procedure TestRead(aJSON : String; AResult : Array of string); override;
  138. Property Reader : TJSONEventReader Read FReader;
  139. end;
  140. implementation
  141. { TMyJSONReader }
  142. function TMyJSONReader.GetList: TStrings;
  143. begin
  144. If FList=Nil then
  145. FList:=TStringList.Create;
  146. Result:=Flist;
  147. end;
  148. procedure TMyJSONReader.Push(const aType : String; const AValue : String = '');
  149. begin
  150. if AValue<>'' then
  151. List.Add(aType+':'+AValue)
  152. else
  153. List.Add(aType);
  154. end;
  155. procedure TMyJSONReader.BooleanValue(const AValue: Boolean);
  156. begin
  157. Push('boolean',BoolToStr(AValue));
  158. end;
  159. procedure TMyJSONReader.EndArray;
  160. begin
  161. Push('ea');
  162. end;
  163. procedure TMyJSONReader.EndObject;
  164. begin
  165. Push('eo');
  166. end;
  167. procedure TMyJSONReader.FloatValue(const AValue: Double);
  168. begin
  169. List.Add('float:'+formatFloat('##.##',AVAlue));
  170. end;
  171. procedure TMyJSONReader.Int64Value(const AValue: int64);
  172. begin
  173. Push('int64',IntToStr(aValue));
  174. end;
  175. procedure TMyJSONReader.IntegerValue(const AValue: integer);
  176. begin
  177. Push('integer',IntToStr(aValue));
  178. end;
  179. procedure TMyJSONReader.KeyValue(const AKey: TJSONStringType);
  180. begin
  181. Push('key',akey);
  182. end;
  183. procedure TMyJSONReader.NullValue;
  184. begin
  185. Push('null');
  186. end;
  187. procedure TMyJSONReader.NumberValue(const AValue: TJSONStringType);
  188. begin
  189. Push('number',aValue);
  190. end;
  191. procedure TMyJSONReader.QWordValue(const AValue: QWord);
  192. begin
  193. Push('qword',IntToStr(AValue));
  194. end;
  195. procedure TMyJSONReader.StartArray;
  196. begin
  197. Push('sa');
  198. end;
  199. procedure TMyJSONReader.StartObject;
  200. begin
  201. Push('so');
  202. end;
  203. procedure TMyJSONReader.StringValue(const AValue: TJSONStringType);
  204. begin
  205. List.Add('string:'+AValue)
  206. end;
  207. destructor TMyJSONReader.Destroy;
  208. begin
  209. FreeAndNil(Flist);
  210. inherited Destroy;
  211. end;
  212. procedure TBaseTestReader.TestEmpty;
  213. begin
  214. TestRead('',[]);
  215. end;
  216. procedure TBaseTestReader.TestInteger;
  217. begin
  218. TestRead('1',['number:1','integer:1']);
  219. end;
  220. procedure TBaseTestReader.TestInt64;
  221. begin
  222. TestRead('123456789012345',['number:123456789012345','int64:123456789012345']);
  223. end;
  224. procedure TBaseTestReader.TestNull;
  225. begin
  226. TestRead('null',['null']);
  227. end;
  228. procedure TBaseTestReader.TestTrue;
  229. begin
  230. TestRead('true',['boolean:'+BoolToStr(true)]);
  231. end;
  232. procedure TBaseTestReader.TestFalse;
  233. begin
  234. TestRead('false',['boolean:'+BoolToStr(false)]);
  235. end;
  236. procedure TBaseTestReader.TestFloat;
  237. begin
  238. DoTestFloat(1.2);
  239. DoTestFloat(-1.2);
  240. DoTestFloat(0);
  241. DoTestFloat(1.2e1);
  242. DoTestFloat(-1.2e1);
  243. DoTestFloat(0);
  244. DoTestFloat(1.2,'1.2');
  245. DoTestFloat(-1.2,'-1.2');
  246. DoTestFloat(0,'0.0');
  247. end;
  248. procedure TBaseTestReader.TestFloatError;
  249. begin
  250. DoTestError('.12',[joStrict]);
  251. DoTestError('.12E',[]);
  252. DoTestError('0.12E+',[]);
  253. DoTestError('.12E+-1',[]);
  254. end;
  255. procedure TBaseTestReader.TestString;
  256. begin
  257. DoTestString('A string');
  258. DoTestString('');
  259. DoTestString('\"','"');
  260. end;
  261. procedure TBaseTestReader.TestArray;
  262. Var
  263. S1,S2,S3 : String;
  264. begin
  265. TestRead('[]',['sa','ea']);
  266. TestRead('[null]',['sa','null','ea']);
  267. TestRead('[true]',['sa','boolean:'+BoolToStr(true),'ea']);
  268. TestRead('[false]',['sa','boolean:'+BoolToStr(false),'ea']);
  269. TestRead('[1]',['sa','number:1','integer:1','ea']);
  270. TestRead('[1, 2]',['sa','number:1','integer:1','number:2','integer:2','ea']);
  271. TestRead('[1, 2, 3]',['sa','number:1','integer:1','number:2','integer:2','number:3','integer:3','ea']);
  272. TestRead('[1234567890123456]',['sa','number:1234567890123456','int64:1234567890123456','ea']);
  273. TestRead('[1234567890123456, 2234567890123456]',
  274. ['sa','number:1234567890123456','int64:1234567890123456','number:2234567890123456','int64:2234567890123456','ea']);
  275. TestRead('[1234567890123456, 2234567890123456, 3234567890123456]',
  276. ['sa','number:1234567890123456','int64:1234567890123456','number:2234567890123456','int64:2234567890123456',
  277. 'number:3234567890123456','int64:3234567890123456','ea']);
  278. Str(12/10,S1);
  279. Delete(S1,1,1);
  280. Str(34/10,S2);
  281. Delete(S2,1,1);
  282. Str(34/10,S3);
  283. Delete(S3,1,1);
  284. TestRead('['+S1+']',['sa','number:'+s1,'float:'+formatfloat('##.##',12/10),'ea']);
  285. {
  286. TestRead('['+S1+', '+S2+']',2,true);
  287. TestRead('['+S1+', '+S2+', '+S3+']',3,true);
  288. TestRead('["A string"]',1);
  289. TestRead('["A string", "Another string"]',2);
  290. TestRead('["A string", "Another string", "Yet another string"]',3);
  291. TestRead('[null, false]',2);
  292. TestRead('[true, false]',2);
  293. TestRead('[null, 1]',2);
  294. TestRead('[1, "A string"]',2);
  295. TestRead('[1, []]',2);
  296. TestRead('[1, [1, 2]]',2);}
  297. end;
  298. procedure TBaseTestReader.TestTrailingComma;
  299. begin
  300. FOptions:=[joIgnoreTrailingComma];
  301. TestRead('[1, 2, ]',['sa','number:1','integer:1','number:2','integer:2','ea']);
  302. TestRead('{ "a" : 1, }',['so','key:a', 'number:1','integer:1','eo']);
  303. end;
  304. procedure TBaseTestReader.TestTrailingCommaErrorArray;
  305. begin
  306. AssertException('Need joIgnoreTrailingComma in options to allow trailing comma',EJSONParser,@DoTrailingCommaErrorArray) ;
  307. end;
  308. procedure TBaseTestReader.TestTrailingCommaErrorObject;
  309. begin
  310. AssertException('Need joIgnoreTrailingComma in options to allow trailing comma',EJSONParser,@DoTrailingCommaErrorObject);
  311. end;
  312. procedure TBaseTestReader.DoTrailingCommaErrorArray;
  313. begin
  314. TestRead('[1, 2, ]',['sa','number:1','integer:1','number:2','integer:2','ea']);
  315. end;
  316. procedure TBaseTestReader.DoTrailingCommaErrorObject;
  317. begin
  318. TestRead('{ "a" : 1, }',['so','key:a', 'number:1','integer:1','eo']);
  319. end;
  320. procedure TBaseTestReader.TestMixed;
  321. begin
  322. TestRead('[1, {}]',['sa','number:1','integer:1','so','eo','ea']);
  323. TestRead('[1, { "a" : 1 }]',['sa','number:1','integer:1','so','key:a','number:1','integer:1','eo','ea']);
  324. TestRead('[1, { "a" : 1 }, 1]',['sa','number:1','integer:1','so','key:a','number:1','integer:1','eo','number:1','integer:1','ea']);
  325. TestRead('{ "a" : [1, 2] }',['so','key:a','sa','number:1','integer:1','number:2','integer:2','ea','eo']);
  326. TestRead('{ "a" : [1, 2], "B" : { "c" : "d" } }',
  327. ['so','key:a','sa','number:1','integer:1','number:2','integer:2','ea','key:B','so','key:c','string:d','eo','eo']);
  328. end;
  329. procedure TBaseTestReader.TestComment;
  330. begin
  331. FOptions:=[joComments];
  332. TestRead('/* */ [1, {}]',['sa','number:1','integer:1','so','eo','ea']);
  333. TestRead('//'+sLineBreak+' [1, {}]',['sa','number:1','integer:1','so','eo','ea']);
  334. TestRead('/* '+sLineBreak+' */ [1, {}]',['sa','number:1','integer:1','so','eo','ea']);
  335. TestRead('/*'+sLineBreak+'*/ [1, {}]',['sa','number:1','integer:1','so','eo','ea']);
  336. TestRead('/*'+sLineBreak+'*'+sLineBreak+'*/ [1, {}]',['sa','number:1','integer:1','so','eo','ea']);
  337. TestRead('/**'+sLineBreak+'**'+sLineBreak+'**/ [1, {}]',['sa','number:1','integer:1','so','eo','ea']);
  338. TestRead('/* */ [1, {}]',['sa','number:1','integer:1','so','eo','ea']);
  339. TestRead('[1, {}]//',['sa','number:1','integer:1','so','eo','ea']);
  340. TestRead('[1, {}]/* '+sLineBreak+' */',['sa','number:1','integer:1','so','eo','ea']);
  341. TestRead('[1, {}]/* '+sLineBreak+' */ ',['sa','number:1','integer:1','so','eo','ea']);
  342. TestRead('[1, {}]/* '+sLineBreak+'*'+sLineBreak+'*/ ',['sa','number:1','integer:1','so','eo','ea']);
  343. TestRead('[1, {}]/**'+sLineBreak+'**'+sLineBreak+'**/ ',['sa','number:1','integer:1','so','eo','ea']);
  344. end;
  345. procedure TBaseTestReader.TestObject;
  346. begin
  347. TestRead('{}',['so','eo']);
  348. TestRead('{ "a" : 1 }',['so','key:a','number:1','integer:1','eo']);
  349. TestRead('{ "a" : 1, "B" : "String" }',['so','key:a','number:1','integer:1','key:B','string:String','eo']);
  350. TestRead('{ "a" : 1, "B" : {} }',['so','key:a','number:1','integer:1','key:B','so','eo','eo']);
  351. TestRead('{ "a" : 1, "B" : { "c" : "d" } }',['so','key:a','number:1','integer:1','key:B','so','key:c','string:d','eo','eo']);
  352. end;
  353. procedure TBaseTestReader.TestObjectError;
  354. begin
  355. DoTestError('{ "name" : value }',[joUTF8]);
  356. end;
  357. procedure TBaseTestReader.TestErrors;
  358. begin
  359. DoTestError('a');
  360. DoTestError('"b');
  361. DoTestError('1Tru');
  362. DoTestError('b"');
  363. DoTestError('{"a" : }');
  364. DoTestError('{"a" : ""');
  365. DoTestError('{"a : ""');
  366. DoTestError('[1,]');
  367. DoTestError('[,]');
  368. DoTestError('[,,]');
  369. DoTestError('[1,,]');
  370. end;
  371. procedure TBaseTestReader.CallNoHandlerStream;
  372. Var
  373. S : TStringStream;
  374. begin
  375. S:=TstringStream.Create('1');
  376. try
  377. GetJSON(S,True).Free;
  378. finally
  379. S.Free;
  380. end;
  381. end;
  382. procedure TBaseTestReader.DoTestString(S: String; AValue : String = '');
  383. begin
  384. if AValue='' then
  385. AValue:=S;
  386. TestRead('"'+S+'"',['string:'+AValue]);
  387. end;
  388. procedure TBaseTestReader.DoTestFloat(F : TJSONFloat);
  389. Var
  390. S : String;
  391. begin
  392. Str(F,S);
  393. DoTestFloat(F,S);
  394. end;
  395. procedure TBaseTestReader.DoTestFloat(F : TJSONFloat; S : String);
  396. begin
  397. TestRead(S,['number:'+trim(S),'float:'+formatfloat('##.##',F)]);
  398. end;
  399. procedure TTestReader.Teardown;
  400. begin
  401. FreeAndNil(FReader);
  402. inherited Teardown;
  403. end;
  404. procedure TTestReader.TestRead(aJSON: String; AResult: array of string);
  405. Var
  406. I : Integer;
  407. begin
  408. FreeAndNil(FReader);
  409. FReader:=TMyJSONReader.Create(aJSON,Foptions);
  410. TMyJSONReader(FReader).DoExecute;
  411. AssertEquals(aJSON+': Number of events',Length(AResult),FReader.List.Count);
  412. For I:=0 to Length(AResult)-1 do
  413. AssertEquals(aJSON+': Event number '+IntToStr(I),AResult[i],FReader.List[I]);
  414. end;
  415. procedure TTestReader.DoTestError(S : String; Options : TJSONOptions = DefaultOpts);
  416. Var
  417. P:TMyJSONReader;
  418. ParseOK : Boolean;
  419. begin
  420. ParseOK:=False;
  421. P:=TMyJSONReader.Create(S,FOptions);
  422. P.OPtions:=Options;
  423. Try
  424. Try
  425. P.DoExecute;
  426. ParseOk:=True;
  427. Finally
  428. FreeAndNil(P);
  429. end;
  430. except
  431. ParseOk:=False;
  432. end;
  433. If ParseOK then
  434. Fail('Parse of JSON string "'+S+'" should fail, but succeeded');
  435. end;
  436. { TJSONConsumer }
  437. procedure TJSONConsumer.Push(const aType : String; const AValue : String = '');
  438. begin
  439. if AValue<>'' then
  440. List.Add(aType+':'+AValue)
  441. else
  442. List.Add(aType);
  443. end;
  444. procedure TJSONConsumer.BooleanValue(const AValue: Boolean);
  445. begin
  446. Push('boolean',BoolToStr(AValue));
  447. end;
  448. procedure TJSONConsumer.EndArray;
  449. begin
  450. Push('ea');
  451. end;
  452. procedure TJSONConsumer.EndObject;
  453. begin
  454. Push('eo');
  455. end;
  456. procedure TJSONConsumer.FloatValue(const AValue: Double);
  457. begin
  458. List.Add('float:'+formatFloat('##.##',AVAlue));
  459. end;
  460. procedure TJSONConsumer.Int64Value(const AValue: int64);
  461. begin
  462. Push('int64',IntToStr(aValue));
  463. end;
  464. procedure TJSONConsumer.IntegerValue(const AValue: integer);
  465. begin
  466. Push('integer',IntToStr(aValue));
  467. end;
  468. procedure TJSONConsumer.KeyName(const AKey: TJSONStringType);
  469. begin
  470. Push('key',akey);
  471. end;
  472. procedure TJSONConsumer.NullValue;
  473. begin
  474. Push('null');
  475. end;
  476. procedure TJSONConsumer.NumberValue(const AValue: TJSONStringType);
  477. begin
  478. Push('number',aValue);
  479. end;
  480. procedure TJSONConsumer.QWordValue(const AValue: QWord);
  481. begin
  482. Push('qword',IntToStr(AValue));
  483. end;
  484. procedure TJSONConsumer.StartArray;
  485. begin
  486. Push('sa');
  487. end;
  488. procedure TJSONConsumer.StartObject;
  489. begin
  490. Push('so');
  491. end;
  492. procedure TJSONConsumer.StringValue(const AValue: TJSONStringType);
  493. begin
  494. List.Add('string:'+AValue)
  495. end;
  496. constructor TJSONConsumer.Create(AList: TStrings);
  497. begin
  498. FList:=AList;
  499. end;
  500. procedure TTestJSONConsumerReader.TestRead(aJSON: String; AResult: array of string);
  501. Var
  502. I : Integer;
  503. begin
  504. FreeAndNil(FReader);
  505. FreeAndNil(Flist);
  506. FList:=TStringList.Create;
  507. FReader:=TJSONConsumerReader.Create(aJSON,Foptions);
  508. FReader.Consumer:=TJSONConsumer.Create(FList);
  509. TJSONConsumerReader(FReader).Execute;
  510. AssertEquals(aJSON+': Number of events',Length(AResult),FList.Count);
  511. For I:=0 to Length(AResult)-1 do
  512. AssertEquals(aJSON+': Event number '+IntToStr(I),AResult[i],FList[I]);
  513. end;
  514. procedure TTestJSONConsumerReader.Teardown;
  515. begin
  516. FreeAndNil(FReader);
  517. FreeAndNil(FList);
  518. inherited Teardown;
  519. end;
  520. procedure TTestJSONConsumerReader.DoTestError(S : String; Options : TJSONOptions = DefaultOpts);
  521. Var
  522. P:TJSONConsumerReader;
  523. ParseOK : Boolean;
  524. begin
  525. ParseOK:=False;
  526. FreeAndNil(FReader);
  527. FreeAndNil(Flist);
  528. FList:=TStringList.Create;
  529. P:=TJSONConsumerReader.Create(S,Options);
  530. P.Consumer:=TJSONConsumer.Create(FList);
  531. P.OPtions:=Options;
  532. Try
  533. Try
  534. P.Execute;
  535. ParseOk:=True;
  536. Finally
  537. FreeAndNil(P);
  538. end;
  539. except
  540. ParseOk:=False;
  541. end;
  542. If ParseOK then
  543. Fail('Parse of JSON string "'+S+'" should fail, but succeeded');
  544. end;
  545. { TTestJSONEventReader }
  546. procedure TTestJSONEventReader.Teardown;
  547. begin
  548. FreeAndNil(Freader);
  549. FreeAndNil(Flist);
  550. inherited Teardown;
  551. end;
  552. procedure TTestJSONEventReader.DoTestError(S: String; Options: TJSONOptions);
  553. Var
  554. P:TJSONEventReader;
  555. ParseOK : Boolean;
  556. begin
  557. ParseOK:=False;
  558. FreeAndNil(FReader);
  559. FreeAndNil(Flist);
  560. FList:=TStringList.Create;
  561. P:=TJSONEventReader.Create(S,Options);
  562. HookupEvents(P);
  563. P.OPtions:=Options;
  564. Try
  565. Try
  566. P.Execute;
  567. ParseOk:=True;
  568. Finally
  569. FreeAndNil(P);
  570. end;
  571. except
  572. ParseOk:=False;
  573. end;
  574. If ParseOK then
  575. Fail('Parse of JSON string "'+S+'" should fail, but succeeded');
  576. end;
  577. procedure TTestJSONEventReader.TestRead(aJSON: String; AResult: array of string);
  578. Var
  579. I : Integer;
  580. begin
  581. FreeAndNil(FReader);
  582. FreeAndNil(Flist);
  583. FList:=TStringList.Create;
  584. FReader:=TJSONEventReader.Create(aJSON,Foptions);
  585. HookupEvents(FReader);
  586. FReader.Execute;
  587. AssertEquals(aJSON+': Number of events',Length(AResult),FList.Count);
  588. For I:=0 to Length(AResult)-1 do
  589. AssertEquals(aJSON+': Event number '+IntToStr(I),AResult[i],FList[I]);
  590. end;
  591. procedure TTestJSONEventReader.Push(const aType: String; const AValue: String);
  592. begin
  593. if AValue<>'' then
  594. FList.Add(aType+':'+AValue)
  595. else
  596. FList.Add(aType);
  597. end;
  598. procedure TTestJSONEventReader.BooleanValue(Sender: TObject; const AValue: Boolean);
  599. begin
  600. Push('boolean',BoolToStr(AValue));
  601. end;
  602. procedure TTestJSONEventReader.EndArray(Sender: TObject);
  603. begin
  604. Push('ea');
  605. end;
  606. procedure TTestJSONEventReader.EndObject(Sender: TObject);
  607. begin
  608. Push('eo');
  609. end;
  610. procedure TTestJSONEventReader.FloatValue(Sender: TObject; const AValue: Double);
  611. begin
  612. FList.Add('float:'+formatFloat('##.##',AVAlue));
  613. end;
  614. procedure TTestJSONEventReader.Int64Value(Sender: TObject; const AValue: int64);
  615. begin
  616. Push('int64',IntToStr(aValue));
  617. end;
  618. procedure TTestJSONEventReader.IntegerValue(Sender: TObject; const AValue: integer);
  619. begin
  620. Push('integer',IntToStr(aValue));
  621. end;
  622. procedure TTestJSONEventReader.KeyValue(Sender: TObject; const AKey: TJSONStringType);
  623. begin
  624. Push('key',akey);
  625. end;
  626. procedure TTestJSONEventReader.NullValue(Sender: TObject);
  627. begin
  628. Push('null');
  629. end;
  630. procedure TTestJSONEventReader.NumberValue(Sender: TObject; const AValue: TJSONStringType);
  631. begin
  632. Push('number',aValue);
  633. end;
  634. procedure TTestJSONEventReader.QWordValue(Sender: TObject; const AValue: QWord);
  635. begin
  636. Push('qword',IntToStr(AValue));
  637. end;
  638. procedure TTestJSONEventReader.StartArray(Sender: TObject);
  639. begin
  640. Push('sa');
  641. end;
  642. procedure TTestJSONEventReader.StartObject(Sender: TObject);
  643. begin
  644. Push('so');
  645. end;
  646. procedure TTestJSONEventReader.StringValue(Sender: TObject; const AValue: TJSONStringType);
  647. begin
  648. FList.Add('string:'+AValue)
  649. end;
  650. procedure TTestJSONEventReader.HookupEvents(AReader: TJSONEventReader);
  651. begin
  652. With Areader do
  653. begin
  654. OnNullValue:=@NullValue;
  655. OnBooleanValue:=@BooleanValue;
  656. OnNumberValue:=@NumberValue;
  657. OnFloatValue:=@FloatValue;
  658. OnIntegerValue:=@IntegerValue;
  659. OnInt64Value:=@Int64Value;
  660. OnQWordValue:=@QWordValue;
  661. OnStringValue:=@StringValue;
  662. OnKeyName:=@KeyValue;
  663. OnStartObject:=@StartObject;
  664. OnEndObject:=@EndObject;
  665. OnStartArray:=@StartArray;
  666. OnEndArray:=@EndArray;
  667. end;
  668. end;
  669. initialization
  670. RegisterTests([TTestReader,TTestJSONConsumerReader,TTestJSONEventReader]);
  671. end.