| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678 | unit tcwebidl2wasmjob;{$mode ObjFPC}{$H+}interfaceuses  Classes, SysUtils, fpcunit, testregistry, webidltowasmjob, pascodegen;type  { TCustomTestWebIDL2WasmJob }  TCustomTestWebIDL2WasmJob = Class(TTestCase)  private    FHeaderSrc: String;    FWebIDLToPas: TWebIDLToPasWasmJob;    procedure OnLog(Sender: TObject; LogType: TCodegenLogType; const Msg: String      );  protected    procedure Setup; override;    procedure TearDown; override;  public    procedure TestWebIDL(const WebIDLSrc, ExpectedPascalSrc: array of string); virtual;    procedure CheckDiff(Msg, Expected, Actual: string); virtual;    property WebIDLToPas: TWebIDLToPasWasmJob read FWebIDLToPas;    property HeaderSrc: String read FHeaderSrc write FHeaderSrc;  end;  { TTestWebIDL2WasmJob }  TTestWebIDL2WasmJob = Class(TCustomTestWebIDL2WasmJob)  published    procedure TestWJ_Empty;    // typedefs    procedure TestWJ_Typedef_Boolean;    procedure TestWJ_Typedef_Sequence;    // attributes    procedure TestWJ_IntfAttribute_Boolean;    // todo procedure TestWJ_IntfAttribute_Any;    // functions    procedure TestWJ_IntfFunction_Void;    procedure TestWJ_IntfFunction_SetEventHandler;    procedure TestWJ_IntfFunction_Promise;    procedure TestWJ_IntfFunction_ArgAny;  end;function LinesToStr(Args: array of const): string;function CheckSrcDiff(Expected, Actual: string; out Msg: string): boolean;implementationfunction LinesToStr(Args: array of const): string;var  s: String;  i: Integer;begin  s:='';  for i:=Low(Args) to High(Args) do    case Args[i].VType of      vtChar:         s += Args[i].VChar+LineEnding;      vtString:       s += Args[i].VString^+LineEnding;      vtPChar:        s += Args[i].VPChar+LineEnding;      vtWideChar:     s += AnsiString(Args[i].VWideChar)+LineEnding;      vtPWideChar:    s += AnsiString(Args[i].VPWideChar)+LineEnding;      vtAnsiString:   s += AnsiString(Args[i].VAnsiString)+LineEnding;      vtWidestring:   s += AnsiString(WideString(Args[i].VWideString))+LineEnding;      vtUnicodeString:s += AnsiString(UnicodeString(Args[i].VUnicodeString))+LineEnding;    end;  Result:=s;end;function CheckSrcDiff(Expected, Actual: string; out Msg: string): boolean;// search diff, ignore changes in spacesconst  SpaceChars = [#9,#10,#13,' '];var  ExpectedP, ActualP: PChar;  function FindLineEnd(p: PChar): PChar;  begin    Result:=p;    while not (Result^ in [#0,#10,#13]) do inc(Result);  end;  function FindLineStart(p, MinP: PChar): PChar;  begin    while (p>MinP) and not (p[-1] in [#10,#13]) do dec(p);    Result:=p;  end;  procedure SkipLineEnd(var p: PChar);  begin    if p^ in [#10,#13] then    begin      if (p[1] in [#10,#13]) and (p^<>p[1]) then        inc(p,2)      else        inc(p);    end;  end;  function HasSpecialChar(s: string): boolean;  var    i: Integer;  begin    for i:=1 to length(s) do      if s[i] in [#0..#31,#127..#255] then        exit(true);    Result:=false;  end;  function HashSpecialChars(s: string): string;  var    i: Integer;  begin    Result:='';    for i:=1 to length(s) do      if s[i] in [#0..#31,#127..#255] then        Result:=Result+'#'+hexstr(ord(s[i]),2)      else        Result:=Result+s[i];  end;  procedure DiffFound;  var    ActLineStartP, ActLineEndP, p, StartPos: PChar;    ExpLine, ActLine: String;    i, LineNo, DiffLineNo: Integer;  begin    writeln('Diff found "',Msg,'". Lines:');    // write correct lines    p:=PChar(Expected);    LineNo:=0;    DiffLineNo:=0;    repeat      StartPos:=p;      while not (p^ in [#0,#10,#13]) do inc(p);      ExpLine:=copy(Expected,StartPos-PChar(Expected)+1,p-StartPos);      SkipLineEnd(p);      inc(LineNo);      if (p<=ExpectedP) and (p^<>#0) then      begin        writeln('= ',ExpLine);      end else begin        // diff line        if DiffLineNo=0 then DiffLineNo:=LineNo;        // write actual line        ActLineStartP:=FindLineStart(ActualP,PChar(Actual));        ActLineEndP:=FindLineEnd(ActualP);        ActLine:=copy(Actual,ActLineStartP-PChar(Actual)+1,ActLineEndP-ActLineStartP);        writeln('- ',ActLine);        if HasSpecialChar(ActLine) then          writeln('- ',HashSpecialChars(ActLine));        // write expected line        writeln('+ ',ExpLine);        if HasSpecialChar(ExpLine) then          writeln('- ',HashSpecialChars(ExpLine));        // write empty line with pointer ^        for i:=1 to 2+ExpectedP-StartPos do write(' ');        writeln('^');        Msg:='expected "'+ExpLine+'", but got "'+ActLine+'".';        CheckSrcDiff:=false;        // write up to ten following actual lines to get some context        for i:=1 to 10 do begin          ActLineStartP:=ActLineEndP;          SkipLineEnd(ActLineStartP);          if ActLineStartP^=#0 then break;          ActLineEndP:=FindLineEnd(ActLineStartP);          ActLine:=copy(Actual,ActLineStartP-PChar(Actual)+1,ActLineEndP-ActLineStartP);          writeln('~ ',ActLine);        end;        exit;      end;    until p^=#0;    // internal error:    writeln('DiffFound Actual:-----------------------');    writeln(Actual);    writeln('DiffFound Expected:---------------------');    writeln(Expected);    writeln('DiffFound ------------------------------');    Msg:='diff found, but lines are the same, internal error';    CheckSrcDiff:=false;  end;var  IsSpaceNeeded: Boolean;  LastChar, Quote: Char;begin  Result:=true;  Msg:='';  if Expected='' then Expected:=' ';  if Actual='' then Actual:=' ';  ExpectedP:=PChar(Expected);  ActualP:=PChar(Actual);  repeat    //writeln('TTestModule.CheckDiff Exp="',ExpectedP^,'" Act="',ActualP^,'"');    case ExpectedP^ of    #0:      begin      // check that rest of Actual has only spaces      while ActualP^ in SpaceChars do inc(ActualP);      if ActualP^<>#0 then        begin        DiffFound;        exit;        end;      exit(true);      end;    ' ',#9,#10,#13:      begin      // skip space in Expected      IsSpaceNeeded:=false;      if ExpectedP>PChar(Expected) then        LastChar:=ExpectedP[-1]      else        LastChar:=#0;      while ExpectedP^ in SpaceChars do inc(ExpectedP);      if (LastChar in ['a'..'z','A'..'Z','0'..'9','_','$'])          and (ExpectedP^ in ['a'..'z','A'..'Z','0'..'9','_','$']) then        IsSpaceNeeded:=true;      if IsSpaceNeeded and (not (ActualP^ in SpaceChars)) then        begin        DiffFound;        exit;        end;      while ActualP^ in SpaceChars do inc(ActualP);      end;    '''','"':      begin      while ActualP^ in SpaceChars do inc(ActualP);      if ExpectedP^<>ActualP^ then        begin        DiffFound;        exit;        end;      Quote:=ExpectedP^;      repeat        inc(ExpectedP);        inc(ActualP);        if ExpectedP^<>ActualP^ then          begin          DiffFound;          exit;          end;        if (ExpectedP^ in [#0,#10,#13]) then          break        else if (ExpectedP^=Quote) then          begin          inc(ExpectedP);          inc(ActualP);          break;          end;      until false;      end;    else      while ActualP^ in SpaceChars do inc(ActualP);      if ExpectedP^<>ActualP^ then        begin        DiffFound;        exit;        end;      inc(ExpectedP);      inc(ActualP);    end;  until false;end;{ TCustomTestWebIDL2WasmJob }procedure TCustomTestWebIDL2WasmJob.OnLog(Sender: TObject;  LogType: TCodegenLogType; const Msg: String);begin  if LogType=cltInfo then ;  if Sender=nil then ;  writeln('TCustomTestWebIDL2WasmJob.OnLog ',Msg);end;procedure TCustomTestWebIDL2WasmJob.Setup;begin  inherited Setup;  FWebIDLToPas:=TWebIDLToPasWasmJob.Create(nil);  WebIDLToPas.OnLog:=@OnLog;  WebIDLToPas.InputFileName:='test1.webidl';  WebIDLToPas.InputStream:=TMemoryStream.Create;  WebIDLToPas.OutputFileName:='test1.pas';  WebIDLToPas.OutputStream:=TMemoryStream.Create;  HeaderSrc:=LinesToStr([    'Unit test1;',    '',    '{$MODE ObjFPC}',    '{$H+}',    'interface',    '',    'uses SysUtils, JOB_JS;',    '']);end;procedure TCustomTestWebIDL2WasmJob.TearDown;begin  WebIDLToPas.InputStream.Free;  WebIDLToPas.InputStream:=nil;  WebIDLToPas.OutputStream.Free;  WebIDLToPas.OutputStream:=nil;  FreeAndNil(FWebIDLToPas);  inherited TearDown;end;procedure TCustomTestWebIDL2WasmJob.TestWebIDL(const WebIDLSrc,  ExpectedPascalSrc: array of string);var  i: Integer;  Line, ExpectedSrc, InputSrc, OutputSrc: String;  InputMS: TMemoryStream;begin  {$IFDEF VerboseWebidl2WasmJob}  writeln('TCustomTestWebIDL2WasmJob.TestWebIDL WebIDL:----------------------');  {$ENDIF}  InputMS:=WebIDLToPas.InputStream as TMemoryStream;  for i:=0 to high(WebIDLSrc) do    begin    Line:=WebIDLSrc[i]+sLineBreak;    InputMS.Write(Line[1],length(Line));    {$IFDEF VerboseWebidl2WasmJob}    write(Line);    {$ENDIF}    end;  InputMS.Position:=0;  {$IFDEF VerboseWebidl2WasmJob}  writeln('TCustomTestWebIDL2WasmJob.TestWebIDL ExpectedPascal: BEGIN--------');  {$ENDIF}  ExpectedSrc:=HeaderSrc;  for i:=0 to high(ExpectedPascalSrc) do    ExpectedSrc:=ExpectedSrc+ExpectedPascalSrc[i]+sLineBreak;  {$IFDEF VerboseWebidl2WasmJob}  writeln(ExpectedSrc);  writeln('TCustomTestWebIDL2WasmJob.TestWebIDL ExpectedPascal END-----------');  {$ENDIF}  WebIDLToPas.Execute;  SetLength(InputSrc{%H-},InputMS.Size);  if length(InputSrc)>0 then    Move(InputMS.Memory^,InputSrc[1],length(InputSrc));  OutputSrc:=WebIDLToPas.Source.Text;  {$IFDEF VerboseWebidl2WasmJob}  writeln('TCustomTestWebIDL2WasmJob.TestWebIDL ActualPascal: BEGIN----------');  writeln(OutputSrc);  writeln('TCustomTestWebIDL2WasmJob.TestWebIDL ActualPascal: END------------');  {$ENDIF}  CheckDiff('TCustomTestWebIDL2WasmJob.TestWebIDL',ExpectedSrc,OutputSrc);end;procedure TCustomTestWebIDL2WasmJob.CheckDiff(Msg, Expected, Actual: string);// search diff, ignore changes in spacesvar  s: string;begin  if CheckSrcDiff(Expected,Actual,s) then exit;  Fail(Msg+': '+s);end;{ TTestWebIDL2WasmJob }procedure TTestWebIDL2WasmJob.TestWJ_Empty;begin  TestWebIDL([  ''],  ['Type',  '  // Forward class definitions',  'implementation',  'end.',  '']);end;procedure TTestWebIDL2WasmJob.TestWJ_Typedef_Boolean;begin  TestWebIDL([  'typedef boolean PerformanceEntry;',  ''],  ['Type',  '  // Forward class definitions',  '  TPerformanceEntry = Boolean;',  'implementation',  'end.',  '']);end;procedure TTestWebIDL2WasmJob.TestWJ_Typedef_Sequence;begin  TestWebIDL([  'typedef boolean PerformanceEntry;',  'typedef sequence <PerformanceEntry> PerformanceEntryList;',  ''],  ['Type',  '  // Forward class definitions',  '  TPerformanceEntry = Boolean;',  '  TPerformanceEntryList = IJSArray; // array of TPerformanceEntry',  'implementation',  'end.',  '']);end;procedure TTestWebIDL2WasmJob.TestWJ_IntfAttribute_Boolean;begin  TestWebIDL([  'interface Attr {',  '  attribute boolean aBoolean;',  '};',  ''],  ['Type',  '  // Forward class definitions',  '  IJSAttr = interface;',  '  TJSAttr = class;',  '  { --------------------------------------------------------------------',  '    TJSAttr',  '    --------------------------------------------------------------------}',  '',  '  IJSAttr = interface(IJSObject)',  '    [''{AA94F48A-7955-3EBA-B086-85B24440AF2A}'']',  '    function _GetaBoolean: Boolean;',  '    procedure _SetaBoolean(const aValue: Boolean);',  '    property aBoolean: Boolean read _GetaBoolean write _SetaBoolean;',  '  end;',  '',  '  TJSAttr = class(TJSObject,IJSAttr)',  '  Private',  '    function _GetaBoolean: Boolean;',  '    procedure _SetaBoolean(const aValue: Boolean);',  '  Public',  '    class function Cast(Intf: IJSObject): IJSAttr;',  '    property aBoolean: Boolean read _GetaBoolean write _SetaBoolean;',  '  end;',  '',  'implementation',  '',  'function TJSAttr._GetaBoolean: Boolean;',  'begin',  '  Result:=ReadJSPropertyBoolean(''aBoolean'');',  'end;',  '',  'procedure TJSAttr._SetaBoolean(const aValue: Boolean);',  'begin',  '  WriteJSPropertyBoolean(''aBoolean'',aValue);',  'end;',  '',  'class function TJSAttr.Cast(Intf: IJSObject): IJSAttr;',  'begin',  '  Result:=TJSAttr.JOBCast(Intf);',  'end;',  '',  'end.',  '']);end;procedure TTestWebIDL2WasmJob.TestWJ_IntfFunction_Void;begin  TestWebIDL([  'interface Attr {',  '  void append(Attr node);',  '};',  ''],  ['Type',  '  // Forward class definitions',  '  IJSAttr = interface;',  '  TJSAttr = class;',  '  { --------------------------------------------------------------------',  '    TJSAttr',  '    --------------------------------------------------------------------}',  '',  '  IJSAttr = interface(IJSObject)',  '    [''{AA94F48A-84D7-3FAA-A2A6-208CA4B2AF2A}'']',  '    procedure append(aNode: IJSAttr);',  '  end;',  '',  '  TJSAttr = class(TJSObject,IJSAttr)',  '  Private',  '  Public',  '    procedure append(aNode: IJSAttr);',  '    class function Cast(Intf: IJSObject): IJSAttr;',  '  end;',  '',  'implementation',  '',  'procedure TJSAttr.append(aNode: IJSAttr);',  'begin',  '  InvokeJSNoResult(''append'',[aNode]);',  'end;',  '',  'class function TJSAttr.Cast(Intf: IJSObject): IJSAttr;',  'begin',  '  Result:=TJSAttr.JOBCast(Intf);',  'end;',  '',  'end.',  '']);end;procedure TTestWebIDL2WasmJob.TestWJ_IntfFunction_SetEventHandler;begin  TestWebIDL([  '[LegacyTreatNonObjectAsNull]',  'callback EventHandlerNonNull = any (long event);',  'typedef EventHandlerNonNull? EventHandler;',  '',  'interface Attr {',  '  void setEventHandler([TreatNonCallableAsNull] EventHandler handler);',  '};',  ''],  ['Type',  '  // Forward class definitions',  '  IJSAttr = interface;',  '  TJSAttr = class;',  '  TEventHandlerNonNull = function (event: Integer): Variant of object;',  '  TEventHandler = TEventHandlerNonNull;',  '',  '  { --------------------------------------------------------------------',  '    TJSAttr',  '    --------------------------------------------------------------------}',  '',  '  IJSAttr = interface(IJSObject)',  '    [''{AA94F48A-121D-33BC-96FE-420246F2AF2A}'']',  '    procedure setEventHandler(const aHandler: TEventHandler);',  '  end;',  '',  '  TJSAttr = class(TJSObject,IJSAttr)',  '  Private',  '  Public',  '    procedure setEventHandler(const aHandler: TEventHandler);',  '    class function Cast(Intf: IJSObject): IJSAttr;',  '  end;',  '',  'implementation',  '',  'function JOBCallTEventHandlerNonNull(const aMethod: TMethod; var H: TJOBCallbackHelper): PByte;',  'var',  '  event: Integer;',  'begin',  '  event:=H.GetLongInt;',  '  Result:=H.AllocVariant(TEventHandlerNonNull(aMethod)(event));',  'end;',  '',  'procedure TJSAttr.setEventHandler(const aHandler: TEventHandler);',  'var',  '  m: TJOB_Method;',  'begin',  '  m:=TJOB_Method.Create(TMethod(aHandler),@JOBCallTEventHandlerNonNull);',  '  try',  '    InvokeJSNoResult(''setEventHandler'',[m]);',  '  finally',  '    m.free;',  '  end;',  'end;',  '',  'class function TJSAttr.Cast(Intf: IJSObject): IJSAttr;',  'begin',  '  Result:=TJSAttr.JOBCast(Intf);',  'end;',  '',  'end.',  '']);end;procedure TTestWebIDL2WasmJob.TestWJ_IntfFunction_Promise;begin  //  Promise<void> exitFullscreen();  TestWebIDL([  'interface Attr {',  '  Promise<void> exitFullscreen();',  '  Promise<any> addCertException(boolean isTemporary);',  '  Promise<Attr> fly();',  '};',  ''],  ['Type',  '  // Forward class definitions',  '  IJSAttr = interface;',  '  TJSAttr = class;',  '  { --------------------------------------------------------------------',  '    TJSAttr',  '    --------------------------------------------------------------------}',  '',  '  IJSAttr = interface(IJSObject)',  '    [''{74BB0007-0E0F-3C5D-B270-B1C656002861}'']',  '    function exitFullscreen: IJSPromise; // Promise<void>',  '    function addCertException(aIsTemporary: Boolean): IJSPromise; // Promise<any>',  '    function fly: IJSPromise; // Promise<Attr>',  '  end;',  '',  '  TJSAttr = class(TJSObject,IJSAttr)',  '  Private',  '  Public',  '    function exitFullscreen: IJSPromise; // Promise<void>',  '    function addCertException(aIsTemporary: Boolean): IJSPromise; // Promise<any>',  '    function fly: IJSPromise; // Promise<Attr>',  '    class function Cast(Intf: IJSObject): IJSAttr;',  '  end;',  '',  'implementation',  '',  'function TJSAttr.exitFullscreen: IJSPromise; // Promise<void>',  'begin',  '  Result:=InvokeJSObjectResult(''exitFullscreen'',[],TJSPromise) as IJSPromise;',  'end;',  '',  'function TJSAttr.addCertException(aIsTemporary: Boolean): IJSPromise; // Promise<any>',  'begin',  '  Result:=InvokeJSObjectResult(''addCertException'',[aIsTemporary],TJSPromise) as IJSPromise;',  'end;',  '',  'function TJSAttr.fly: IJSPromise; // Promise<Attr>',  'begin',  '  Result:=InvokeJSObjectResult(''fly'',[],TJSPromise) as IJSPromise;',  'end;',  '',  'class function TJSAttr.Cast(Intf: IJSObject): IJSAttr;',  'begin',  '  Result:=TJSAttr.JOBCast(Intf);',  'end;',  '',  'end.',  '']);end;procedure TTestWebIDL2WasmJob.TestWJ_IntfFunction_ArgAny;begin  TestWebIDL([  'interface Attr {',  '  void append(any node);',  '};',  ''],  ['Type',  '  // Forward class definitions',  '  IJSAttr = interface;',  '  TJSAttr = class;',  '  { --------------------------------------------------------------------',  '    TJSAttr',  '    --------------------------------------------------------------------}',  '',  '  IJSAttr = interface(IJSObject)',  '    [''{AA94F48A-84D7-3FAA-A2A6-208CA4B2AF2A}'']',  '    procedure append(const aNode: Variant);',  '  end;',  '',  '  TJSAttr = class(TJSObject,IJSAttr)',  '  Private',  '  Public',  '    procedure append(const aNode: Variant);',  '    class function Cast(Intf: IJSObject): IJSAttr;',  '  end;',  '',  'implementation',  '',  'procedure TJSAttr.append(const aNode: Variant);',  'begin',  '  InvokeJSNoResult(''append'',[aNode]);',  'end;',  '',  'class function TJSAttr.Cast(Intf: IJSObject): IJSAttr;',  'begin',  '  Result:=TJSAttr.JOBCast(Intf);',  'end;',  '',  'end.',  '']);end;initialization  RegisterTests([TTestWebIDL2Wasmjob]);end.
 |