tcbaseparser.pas 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496
  1. unit tcbaseparser;
  2. {$mode objfpc}{$H+}
  3. interface
  4. uses
  5. Classes, SysUtils, fpcunit, pastree, pscanner, pparser, testregistry;
  6. Type
  7. { TTestEngine }
  8. TTestEngine = Class(TPasTreeContainer)
  9. Private
  10. FList : TFPList;
  11. public
  12. Destructor Destroy; override;
  13. function CreateElement(AClass: TPTreeElement; const AName: String;
  14. AParent: TPasElement; AVisibility: TPasMemberVisibility;
  15. const ASourceFilename: String; ASourceLinenumber: Integer): TPasElement;
  16. override;
  17. function FindElement(const AName: String): TPasElement; override;
  18. end;
  19. TTestPasParser = Class(TPasParser);
  20. { TTestParser }
  21. TTestParser= class(TTestCase)
  22. Private
  23. FDeclarations: TPasDeclarations;
  24. FDefinition: TPasElement;
  25. FEngine : TTestEngine;
  26. FModule: TPasModule;
  27. FParseResult: TPasElement;
  28. FScanner : TPascalScanner;
  29. FResolver : TStreamResolver;
  30. FParser : TTestPasParser;
  31. FSource: TStrings;
  32. FFileName : string;
  33. FIsUnit : Boolean;
  34. FImplementation : Boolean;
  35. FEndSource: Boolean;
  36. FUseImplementation: Boolean;
  37. function GetPL: TPasLibrary;
  38. function GetPP: TPasProgram;
  39. protected
  40. procedure SetUp; override;
  41. procedure TearDown; override;
  42. Procedure StartUnit(AUnitName : String);
  43. Procedure StartProgram(AFileName : String; AIn : String = ''; AOut : String = '');
  44. Procedure StartLibrary(AFileName : String);
  45. Procedure UsesClause(Units : Array of string);
  46. Procedure StartImplementation;
  47. Procedure EndSource;
  48. Procedure Add(Const ALine : String);
  49. Procedure StartParsing;
  50. Procedure ParseDeclarations;
  51. Procedure ParseModule;
  52. Procedure CheckHint(AHint : TPasMemberHint);
  53. Function AssertExpression(Const Msg: String; AExpr : TPasExpr; aKind : TPasExprKind; AClass : TClass) : TPasExpr;
  54. Function AssertExpression(Const Msg: String; AExpr : TPasExpr; aKind : TPasExprKind; AValue : String) : TPrimitiveExpr;
  55. Procedure AssertExportSymbol(Const Msg: String; AIndex : Integer; AName,AExportName : String; AExportIndex : Integer = -1);
  56. Procedure AssertEquals(Const Msg : String; AExpected, AActual: TPasExprKind); overload;
  57. Procedure AssertEquals(Const Msg : String; AExpected, AActual: TLoopType); overload;
  58. Procedure AssertEquals(Const Msg : String; AExpected, AActual: TPasObjKind); overload;
  59. Procedure AssertEquals(Const Msg : String; AExpected, AActual: TexprOpcode); overload;
  60. Procedure AssertEquals(Const Msg : String; AExpected, AActual: TPasMemberHint); overload;
  61. Procedure AssertEquals(Const Msg : String; AExpected, AActual: TCallingConvention); overload;
  62. Procedure AssertEquals(Const Msg : String; AExpected, AActual: TArgumentAccess); overload;
  63. Procedure AssertEquals(Const Msg : String; AExpected, AActual: TVariableModifier); overload;
  64. Procedure AssertEquals(Const Msg : String; AExpected, AActual: TVariableModifiers); overload;
  65. Procedure AssertEquals(Const Msg : String; AExpected, AActual: TPasMemberVisibility); overload;
  66. Procedure AssertEquals(Const Msg : String; AExpected, AActual: TProcedureModifier); overload;
  67. Procedure AssertEquals(Const Msg : String; AExpected, AActual: TProcedureModifiers); overload;
  68. Procedure AssertEquals(Const Msg : String; AExpected, AActual: TAssignKind); overload;
  69. Procedure AssertEquals(Const Msg : String; AExpected, AActual: TProcedureMessageType); overload;
  70. Procedure HaveHint(AHint : TPasMemberHint; AHints : TPasMemberHints);
  71. Property Resolver : TStreamResolver Read FResolver;
  72. Property Scanner : TPascalScanner Read FScanner;
  73. Property Parser : TTestPasParser read FParser ;
  74. Property Source : TStrings Read FSource;
  75. Property Module : TPasModule Read FModule;
  76. Property PasProgram : TPasProgram Read GetPP;
  77. Property PasLibrary : TPasLibrary Read GetPL;
  78. Property Declarations : TPasDeclarations read FDeclarations Write FDeclarations;
  79. Property Definition : TPasElement Read FDefinition Write FDefinition;
  80. // If set, Will be freed in teardown
  81. Property ParseResult : TPasElement Read FParseResult Write FParseResult;
  82. Property UseImplementation : Boolean Read FUseImplementation Write FUseImplementation;
  83. end;
  84. implementation
  85. uses typinfo;
  86. { TTestEngine }
  87. destructor TTestEngine.Destroy;
  88. begin
  89. FreeAndNil(FList);
  90. inherited Destroy;
  91. end;
  92. function TTestEngine.CreateElement(AClass: TPTreeElement; const AName: String;
  93. AParent: TPasElement; AVisibility: TPasMemberVisibility;
  94. const ASourceFilename: String; ASourceLinenumber: Integer): TPasElement;
  95. begin
  96. Result := AClass.Create(AName, AParent);
  97. Result.Visibility := AVisibility;
  98. Result.SourceFilename := ASourceFilename;
  99. Result.SourceLinenumber := ASourceLinenumber;
  100. If not Assigned(FList) then
  101. FList:=TFPList.Create;
  102. FList.Add(Result);
  103. end;
  104. function TTestEngine.FindElement(const AName: String): TPasElement;
  105. Var
  106. I : Integer;
  107. begin
  108. Result:=Nil;
  109. if Assigned(FList) then
  110. begin
  111. I:=FList.Count-1;
  112. While (Result=Nil) and (I>=0) do
  113. begin
  114. if CompareText(TPasElement(FList[I]).Name,AName)=0 then
  115. Result:=TPasElement(Flist[i]);
  116. Dec(i);
  117. end;
  118. end;
  119. end;
  120. function TTestParser.GetPP: TPasProgram;
  121. begin
  122. Result:=Module as TPasProgram;
  123. end;
  124. function TTestParser.GetPL: TPasLibrary;
  125. begin
  126. Result:=Module as TPasLibrary;
  127. end;
  128. procedure TTestParser.SetUp;
  129. begin
  130. FResolver:=TStreamResolver.Create;
  131. FResolver.OwnsStreams:=True;
  132. FScanner:=TPascalScanner.Create(FResolver);
  133. FEngine:=TTestEngine.Create;
  134. FParser:=TTestPasParser.Create(FScanner,FResolver,FEngine);
  135. FSource:=TStringList.Create;
  136. FModule:=Nil;
  137. FDeclarations:=Nil;
  138. FEndSource:=False;
  139. FImplementation:=False;
  140. FIsUnit:=False;
  141. end;
  142. procedure TTestParser.TearDown;
  143. begin
  144. if Not Assigned(FModule) then
  145. FreeAndNil(FDeclarations)
  146. else
  147. FDeclarations:=Nil;
  148. FImplementation:=False;
  149. FEndSource:=False;
  150. FIsUnit:=False;
  151. FreeAndNil(FModule);
  152. FreeAndNil(FSource);
  153. FreeAndNil(FParseResult);
  154. FreeAndNil(FParser);
  155. FreeAndNil(FEngine);
  156. FreeAndNil(FScanner);
  157. FreeAndNil(FResolver);
  158. end;
  159. procedure TTestParser.StartUnit(AUnitName: String);
  160. begin
  161. FIsUnit:=True;
  162. If (AUnitName='') then
  163. AUnitName:='afile';
  164. Add('unit '+aUnitName+';');
  165. Add('');
  166. Add('interface');
  167. Add('');
  168. FFileName:=AUnitName+'.pp';
  169. end;
  170. procedure TTestParser.StartProgram(AFileName : String; AIn : String = ''; AOut : String = '');
  171. begin
  172. FIsUnit:=False;
  173. If (AFileName='') then
  174. AFileName:='proga';
  175. FFileName:=AFileName+'.pp';
  176. If (AIn<>'') then
  177. begin
  178. AFileName:=AFileName+'('+AIn;
  179. if (AOut<>'') then
  180. AFileName:=AFIleName+','+AOut;
  181. AFileName:=AFileName+')';
  182. end;
  183. Add('program '+AFileName+';');
  184. FImplementation:=True;
  185. end;
  186. procedure TTestParser.StartLibrary(AFileName: String);
  187. begin
  188. FIsUnit:=False;
  189. If (AFileName='') then
  190. AFileName:='liba';
  191. FFileName:=AFileName+'.pp';
  192. Add('library '+AFileName+';');
  193. FImplementation:=True;
  194. end;
  195. procedure TTestParser.UsesClause(Units: array of string);
  196. Var
  197. S : String;
  198. I : integer;
  199. begin
  200. S:='';
  201. For I:=Low(units) to High(units) do
  202. begin
  203. If (S<>'') then
  204. S:=S+', ';
  205. S:=S+Units[i];
  206. end;
  207. Add('uses '+S+';');
  208. Add('');
  209. end;
  210. procedure TTestParser.StartImplementation;
  211. begin
  212. if Not FImplementation then
  213. begin
  214. if UseImplementation then
  215. begin
  216. FSource.Insert(0,'');
  217. FSource.Insert(0,'Implementation');
  218. FSource.Insert(0,'');
  219. end
  220. else
  221. begin
  222. Add('');
  223. Add('Implementation');
  224. Add('');
  225. end;
  226. FImplementation:=True;
  227. end;
  228. end;
  229. procedure TTestParser.EndSource;
  230. begin
  231. if Not FEndSource then
  232. begin
  233. Add('end.');
  234. FEndSource:=True;
  235. end;
  236. end;
  237. procedure TTestParser.Add(const ALine: String);
  238. begin
  239. FSource.Add(ALine);
  240. end;
  241. procedure TTestParser.StartParsing;
  242. begin
  243. If FIsUnit then
  244. StartImplementation;
  245. EndSource;
  246. If (FFileName='') then
  247. FFileName:='afile.pp';
  248. FResolver.AddStream(FFileName,TStringStream.Create(FSource.text));
  249. FScanner.OpenFile(FFileName);
  250. Writeln('// Test : ',Self.TestName);
  251. Writeln(FSource.Text);
  252. end;
  253. procedure TTestParser.ParseDeclarations;
  254. begin
  255. if UseImplementation then
  256. StartImplementation;
  257. FSource.Insert(0,'');
  258. FSource.Insert(0,'interface');
  259. FSource.Insert(0,'');
  260. FSource.Insert(0,'unit afile;');
  261. if Not UseImplementation then
  262. StartImplementation;
  263. EndSource;
  264. ParseModule;
  265. if UseImplementation then
  266. FDeclarations:=Module.ImplementationSection
  267. else
  268. FDeclarations:=Module.InterfaceSection;
  269. end;
  270. procedure TTestParser.ParseModule;
  271. begin
  272. StartParsing;
  273. FParser.ParseMain(FModule);
  274. AssertNotNull('Module resulted in Module',FModule);
  275. AssertEquals('modulename',ChangeFileExt(FFileName,''),Module.Name);
  276. end;
  277. procedure TTestParser.CheckHint(AHint: TPasMemberHint);
  278. begin
  279. HaveHint(AHint,Definition.Hints);
  280. end;
  281. function TTestParser.AssertExpression(const Msg: String; AExpr: TPasExpr;
  282. aKind: TPasExprKind; AClass: TClass): TPasExpr;
  283. begin
  284. AssertEquals(Msg+': Correct expression kind',aKind,AExpr.Kind);
  285. AssertEquals(Msg+': Correct expression class',AClass,AExpr.ClassType);
  286. Result:=AExpr;
  287. end;
  288. function TTestParser.AssertExpression(const Msg: String; AExpr: TPasExpr;
  289. aKind: TPasExprKind; AValue: String): TPrimitiveExpr;
  290. begin
  291. Result:=AssertExpression(Msg,AExpr,aKind,TPrimitiveExpr) as TPrimitiveExpr;
  292. AssertEquals(Msg+': Primitive expression value',AValue,TPrimitiveExpr(AExpr).Value);
  293. end;
  294. procedure TTestParser.AssertExportSymbol(const Msg: String; AIndex: Integer;
  295. AName, AExportName: String; AExportIndex: Integer);
  296. Var
  297. E: TPasExportSymbol;
  298. begin
  299. AssertNotNull(Msg+'Have export symbols list',PasLibrary.LibrarySection.ExportSymbols);
  300. if AIndex>=PasLibrary.LibrarySection.ExportSymbols.Count then
  301. Fail(Format(Msg+'%d not a valid export list symbol',[AIndex]));
  302. AssertNotNull(Msg+'Have export symbol',PasLibrary.LibrarySection.ExportSymbols[Aindex]);
  303. AssertEquals(Msg+'Correct export symbol class',TPasExportSymbol,TObject(PasLibrary.LibrarySection.ExportSymbols[Aindex]).ClassType);
  304. E:=TPasExportSymbol(PasLibrary.LibrarySection.ExportSymbols[Aindex]);
  305. AssertEquals(Msg+'Correct export symbol name',AName,E.Name);
  306. if (AExportName='') then
  307. AssertNull(Msg+'No export name',E.ExportName)
  308. else
  309. begin
  310. AssertNotNull(Msg+'Export name symbol',E.ExportName);
  311. AssertEquals(Msg+'TPrimitiveExpr',TPrimitiveExpr,E.ExportName.CLassType);
  312. AssertEquals(Msg+'Correct export symbol export name ',''''+AExportName+'''',TPrimitiveExpr(E.ExportName).Value);
  313. end;
  314. If AExportIndex=-1 then
  315. AssertNull(Msg+'No export name',E.ExportIndex)
  316. else
  317. begin
  318. AssertNotNull(Msg+'Export name symbol',E.ExportIndex);
  319. AssertEquals(Msg+'TPrimitiveExpr',TPrimitiveExpr,E.ExportIndex.CLassType);
  320. AssertEquals(Msg+'Correct export symbol export index',IntToStr(AExportindex),TPrimitiveExpr(E.ExportIndex).Value);
  321. end;
  322. end;
  323. procedure TTestParser.AssertEquals(const Msg: String; AExpected,
  324. AActual: TPasExprKind);
  325. begin
  326. AssertEquals(Msg,GetEnumName(TypeInfo(TPasExprKind),Ord(AExpected)),
  327. GetEnumName(TypeInfo(TPasExprKind),Ord(AActual)));
  328. end;
  329. procedure TTestParser.AssertEquals(const Msg: String; AExpected,
  330. AActual: TLoopType);
  331. begin
  332. AssertEquals(Msg,GetEnumName(TypeInfo(TLoopType),Ord(AExpected)),
  333. GetEnumName(TypeInfo(TLoopType),Ord(AActual)));
  334. end;
  335. procedure TTestParser.AssertEquals(const Msg: String; AExpected,
  336. AActual: TPasObjKind);
  337. begin
  338. AssertEquals(Msg,GetEnumName(TypeInfo(TexprOpcode),Ord(AExpected)),
  339. GetEnumName(TypeInfo(TexprOpcode),Ord(AActual)));
  340. end;
  341. procedure TTestParser.AssertEquals(const Msg: String; AExpected,
  342. AActual: TexprOpcode);
  343. begin
  344. AssertEquals(Msg,GetEnumName(TypeInfo(TexprOpcode),Ord(AExpected)),
  345. GetEnumName(TypeInfo(TexprOpcode),Ord(AActual)));
  346. end;
  347. procedure TTestParser.AssertEquals(const Msg: String; AExpected,
  348. AActual: TPasMemberHint);
  349. begin
  350. AssertEquals(Msg,GetEnumName(TypeInfo(TPasMemberHint),Ord(AExpected)),
  351. GetEnumName(TypeInfo(TPasMemberHint),Ord(AActual)));
  352. end;
  353. procedure TTestParser.AssertEquals(const Msg: String; AExpected,
  354. AActual: TCallingConvention);
  355. begin
  356. AssertEquals(Msg,GetEnumName(TypeInfo(TCallingConvention),Ord(AExpected)),
  357. GetEnumName(TypeInfo(TCallingConvention),Ord(AActual)));
  358. end;
  359. procedure TTestParser.AssertEquals(const Msg: String; AExpected,
  360. AActual: TArgumentAccess);
  361. begin
  362. AssertEquals(Msg,GetEnumName(TypeInfo(TArgumentAccess),Ord(AExpected)),
  363. GetEnumName(TypeInfo(TArgumentAccess),Ord(AActual)));
  364. end;
  365. procedure TTestParser.AssertEquals(const Msg: String; AExpected,
  366. AActual: TVariableModifier);
  367. begin
  368. AssertEquals(Msg,GetEnumName(TypeInfo(TVariableModifier),Ord(AExpected)),
  369. GetEnumName(TypeInfo(TVariableModifier),Ord(AActual)));
  370. end;
  371. procedure TTestParser.AssertEquals(const Msg: String; AExpected,
  372. AActual: TVariableModifiers);
  373. Function sn (S : TVariableModifiers) : string;
  374. Var
  375. M : TVariableModifier;
  376. begin
  377. Result:='';
  378. For M:=Low(TVariableModifier) to High(TVariableModifier) do
  379. if M in S then
  380. begin
  381. if (Result<>'') then
  382. Result:=Result+',';
  383. end;
  384. Result:='['+Result+']';
  385. end;
  386. begin
  387. AssertEquals(Msg,Sn(AExpected),Sn(AActual));
  388. end;
  389. procedure TTestParser.AssertEquals(const Msg: String; AExpected,
  390. AActual: TPasMemberVisibility);
  391. begin
  392. AssertEquals(Msg,GetEnumName(TypeInfo(TPasMemberVisibility),Ord(AExpected)),
  393. GetEnumName(TypeInfo(TPasMemberVisibility),Ord(AActual)));
  394. end;
  395. procedure TTestParser.AssertEquals(const Msg: String; AExpected,
  396. AActual: TProcedureModifier);
  397. begin
  398. AssertEquals(Msg,GetEnumName(TypeInfo(TProcedureModifier),Ord(AExpected)),
  399. GetEnumName(TypeInfo(TProcedureModifier),Ord(AActual)));
  400. end;
  401. procedure TTestParser.AssertEquals(const Msg: String; AExpected,
  402. AActual: TProcedureModifiers);
  403. Function Sn (S : TProcedureModifiers) : String;
  404. Var
  405. m : TProcedureModifier;
  406. begin
  407. Result:='';
  408. For M:=Low(TProcedureModifier) to High(TProcedureModifier) do
  409. If (m in S) then
  410. begin
  411. If (Result<>'') then
  412. Result:=Result+',';
  413. Result:=Result+GetEnumName(TypeInfo(TProcedureModifier),Ord(m))
  414. end;
  415. end;
  416. begin
  417. AssertEquals(Msg,Sn(AExpected),SN(AActual));
  418. end;
  419. procedure TTestParser.AssertEquals(const Msg: String; AExpected,
  420. AActual: TAssignKind);
  421. begin
  422. AssertEquals(Msg,GetEnumName(TypeInfo(TAssignKind),Ord(AExpected)),
  423. GetEnumName(TypeInfo(TAssignKind),Ord(AActual)));
  424. end;
  425. procedure TTestParser.AssertEquals(const Msg: String; AExpected,
  426. AActual: TProcedureMessageType);
  427. begin
  428. AssertEquals(Msg,GetEnumName(TypeInfo(TProcedureMessageType),Ord(AExpected)),
  429. GetEnumName(TypeInfo(TProcedureMessageType),Ord(AActual)));
  430. end;
  431. procedure TTestParser.HaveHint(AHint: TPasMemberHint; AHints: TPasMemberHints);
  432. begin
  433. If not (AHint in AHints) then
  434. Fail(GetEnumName(TypeInfo(TPasMemberHint),Ord(AHint))+'hint expected.');
  435. end;
  436. end.