| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617 | unit httpcompiler;{$mode objfpc}{$H+}interfaceuses  sysutils, classes, fpjson, contnrs, syncobjs, fpmimetypes, custhttpapp,  fpwebfile, httproute, httpdefs, dirwatch, Pas2JSFSCompiler, Pas2JSCompilerCfg;Const  nErrTooManyThreads = -1;Type  TDirWatcher = Class;  THTTPCompilerApplication = Class;  { TCompileItem }  TCompileItem = Class(TCollectionItem)  private    FBaseDir: string;    FConfigFile: String;    FFileName: string;    FOutput : TStrings;    FOptions : TStrings;    FSuccess: Boolean;    FThread: TThread;    function GetOptions: TStrings;    function GetOutput: TStrings;  Public    Destructor Destroy; override;    Property BaseDir : string Read FBaseDir Write FBaseDir;    Property FileName : string Read FFileName Write FFileName;    Property ConfigFile: String Read FConfigFile Write FConfigFile;    Property Options : TStrings Read GetOptions;    Property Output : TStrings Read GetOutput;    Property Thread : TThread Read FThread;    Property Success : Boolean  Read FSuccess;  end;  { TCompiles }  TCompiles = Class(TCollection)  private    function GetC(AIndex : Integer): TCompileItem;  Public     Property Compiles[AIndex : Integer] : TCompileItem Read GetC; default;  end;  { TCompileThread }  TCompileThread = class(TThread)  private    FApp : THTTPCompilerApplication;    FItem: TCompileItem;    procedure DoCompilerLog(Sender: TObject; const Msg: String);    procedure SetItem(AValue: TCompileItem);  Public    Constructor create(App : THTTPCompilerApplication; aItem : TCompileItem);    Procedure Execute; override;    Property Item : TCompileItem read FItem write SetItem;  end;  { TDirWatcher }  TDirWatcher = Class(TComponent)  Private    FApp : THTTPCompilerApplication;    FDW : TDirWatch;    procedure DoChange(Sender: TObject; aEntry: TDirectoryEntry; AEvents: TFileEvents);  Public    Constructor Create(App : THTTPCompilerApplication; ADir : String);overload;    Destructor Destroy; override;  end;  { THTTPCompilerApplication }  THTTPCompilerApplication = Class(TCustomHTTPApplication)  private    FBaseDir: String;    FConfigFile: String;    FProjectFile: String;    FStatusLock : TCriticalSection;    FQuiet: Boolean;    FWatch: Boolean;    FDW : TDirWatcher;    FStatusList : TFPObjectList;    FCompiles : TCompiles;    FServeOnly  : Boolean;    procedure AddToStatus(O: TJSONObject);    function HandleCompileOptions(aDir: String): Boolean;    function ProcessOptions: Boolean;    Procedure ReportBuilding(AItem : TCompileItem);    Procedure ReportBuilt(AItem : TCompileItem);    Procedure AddToStatus(AEntry : TDirectoryEntry; AEvents : TFileEvents);    procedure DoStatusRequest(ARequest: TRequest; AResponse: TResponse);    procedure DoRecompile(ARequest: TRequest; AResponse: TResponse);    function ScheduleCompile(const aProjectFile: String; Options : TStrings = Nil): Integer;    procedure StartWatch(ADir: String);  protected    procedure Usage(Msg: String); virtual;    function GetDefaultMimeTypesFile: string; virtual;    procedure LoadDefaultMimeTypes; virtual;  public    Constructor Create(AOWner : TComponent); override;    Destructor Destroy; override;    procedure DoLog(EventType: TEventType; const Msg: String); override;    Procedure DoRun; override;    property Quiet : Boolean read FQuiet Write FQuiet;    Property Watch : Boolean Read FWatch Write FWatch;    Property ProjectFile : String Read FProjectFile Write FProjectFile;    Property ConfigFile : String Read FConfigFile Write FConfigFile;    Property BaseDir : String Read FBaseDir;    Property ServeOnly : Boolean Read FServeOnly;  end;Implementation{ TCompileThread }procedure TCompileThread.SetItem(AValue: TCompileItem);begin  if FItem=AValue then Exit;  FItem:=AValue;end;procedure TCompileThread.DoCompilerLog(Sender: TObject; const Msg: String);begin  If Assigned(Item) then    Item.Output.Add(Msg);end;constructor TCompileThread.create(App: THTTPCompilerApplication; aItem: TCompileItem);begin  FItem:=aItem;  FApp:=App;  FreeOnTerminate:=True;  inherited create(False);end;procedure TCompileThread.Execute;Var  C : TPas2JSFSCompiler;  L : TStrings;begin  L:=Nil;  C:=TPas2JSFSCompiler.Create;  Try    C.ConfigSupport:=TPas2JSFileConfigSupport.Create(C);    FApp.ReportBuilding(Item);    L:=TStringList.Create;    L.Assign(Item.Options);    if (Item.ConfigFile<>'') then      L.Add('@'+Item.ConfigFile);    L.Add(Item.FileName);    C.Log.OnLog:=@DoCompilerLog;    try      C.Run(ParamStr(0),Item.BaseDir,L,True);      Item.FSuccess:=True;    except      On E : Exception do        Item.Output.Add(Format('Error %s compiling %s: %s',[E.ClassName,Item.FileName,E.Message]));    end;    FApp.ReportBuilt(Item);  Finally    C.Free;    L.Free;  end;  Item.FThread:=Nil;end;{ TCompiles }function TCompiles.GetC(AIndex : Integer): TCompileItem;begin  Result:=Items[Aindex] as TCompileItem;end;{ TCompileItem }function TCompileItem.GetOutput: TStrings;begin  If (FOutput=Nil) then    FOutput:=TStringList.Create;  Result:=FOutput;end;function TCompileItem.GetOptions: TStrings;begin  If (FOptions=Nil) then    FOptions:=TStringList.Create;  Result:=FOptions;end;destructor TCompileItem.Destroy;begin  FreeAndNil(FOutput);  FreeAndNil(FOptions);  inherited Destroy;end;{ TDirWatcher }procedure TDirWatcher.DoChange(Sender: TObject; aEntry: TDirectoryEntry; AEvents: TFileEvents);begin  if Assigned(FApp) then    FApp.AddToStatus(AEntry,AEvents);end;constructor TDirWatcher.Create(App: THTTPCompilerApplication; ADir: String);begin  Inherited create(APP);  FApp:=App;  FDW:=TDirwatch.Create(Self);  FDW.AddWatch(ADir,allEvents);  FDW.OnChange:=@DoChange;  TThread.ExecuteInThread(@FDW.StartWatch);end;destructor TDirWatcher.Destroy;begin  FApp:=Nil;  FDW.Terminate;  FreeAndNil(FDW);  inherited Destroy;end;{ THTTPCompilerApplication }procedure THTTPCompilerApplication.DoLog(EventType: TEventType; const Msg: String);begin  {AllowWriteln}  if Quiet then    exit;  if IsConsole then    Writeln(FormatDateTime('yyyy-mm-dd hh:nn:ss.zzz',Now),' [',EventType,'] ',Msg)  else    inherited DoLog(EventType, Msg);  {AllowWriteln-}end;procedure THTTPCompilerApplication.Usage(Msg : String);begin  {AllowWriteln}  if (Msg<>'') then    Writeln('Error: ',Msg);  Writeln('Usage ',ExtractFileName(ParamStr(0)),' [options] ');  Writeln('Where options is one or more of : ');  Writeln('-d --directory=dir  Base directory from which to serve files.');  Writeln('                    Default is current working directory: ',GetCurrentDir);  Writeln('-h --help           This help text');  Writeln('-i --indexpage=name Directory index page to use (default: index.html)');  Writeln('-n --noindexpage    Do not allow index page.');  Writeln('-p --port=NNNN      TCP/IP port to listen on (default is 3000)');  Writeln('-q --quiet          Do not write diagnostic messages');  Writeln('-w --watch          Watch directory for changes');  Writeln('-c --compile[=proj] Recompile project if pascal files change. Default project is app.lpr');  Writeln('-m --mimetypes=file filename of mimetypes. Default is ',GetDefaultMimeTypesFile);  Writeln('-s --simpleserver   Only serve files, do not enable compilation.');  Halt(Ord(Msg<>''));  {AllowWriteln-}end;function THTTPCompilerApplication.GetDefaultMimeTypesFile: string;begin  {$ifdef unix}  Result:='/etc/mime.types';  {$ifdef darwin}  if not FileExists(Result) then    Result:='/private/etc/apache2/mime.types';  {$endif}  {$else}  Result:=ExtractFilePath(System.ParamStr(0))+'mime.types';  {$endif}end;procedure THTTPCompilerApplication.LoadDefaultMimeTypes;begin  MimeTypes.LoadKnownTypes;  // To be sure  MimeTypes.AddType('application/xhtml+xml','xhtml;xht');  MimeTypes.AddType('text/html','html;htm');  MimeTypes.AddType('text/plain','txt');  MimeTypes.AddType('application/javascript','js');  MimeTypes.AddType('text/plain','map');  MimeTypes.AddType('application/json','json');  MimeTypes.AddType('image/png','png');  MimeTypes.AddType('image/jpeg','jpeg;jpg');  MimeTypes.AddType('image/gif','gif');  MimeTypes.AddType('image/jp2','jp2');  MimeTypes.AddType('image/tiff','tiff;tif');  MimeTypes.AddType('application/pdf','pdf');  MimeTypes.AddType('text/css','css');end;constructor THTTPCompilerApplication.Create(AOWner: TComponent);begin  inherited Create(AOWner);  FStatusLock:=TCriticalSection.Create;  FStatusList:=TFPObjectList.Create(False);  FCompiles:=TCompiles.Create(TCompileItem);end;destructor THTTPCompilerApplication.Destroy;begin  FStatusLock.Enter;  try    FreeAndNil(FCompiles);    FreeAndNil(FStatusList);  finally    FStatusLock.Leave;  end;  FreeAndNil(FStatusLock);  inherited Destroy;end;procedure THTTPCompilerApplication.StartWatch(ADir : String);begin  FDW:=TDirWatcher.Create(Self,ADir);end;procedure THTTPCompilerApplication.ReportBuilding(AItem: TCompileItem);Var  O : TJSONObject;begin  O:=TJSONObject.Create(['action','building','compileID',AItem.ID,'project',AItem.FileName,'config',AItem.ConfigFile]);  AddToStatus(O);end;procedure THTTPCompilerApplication.ReportBuilt(AItem: TCompileItem);Var  O : TJSONObject;  A : TJSONArray;  I : Integer;begin  A:=TJSONArray.Create;  For I:=0 to AItem.Output.Count-1 do    A.Add(AItem.Output[i]);  O:=TJSONObject.Create(['action','built','compileID',AItem.ID,'project',AItem.FileName,'config',AItem.ConfigFile,'output',A,'success',AItem.Success]);  AddToStatus(O);end;procedure THTTPCompilerApplication.AddToStatus(O : TJSONObject);begin  FStatusLock.Enter;  try    {$ifdef VerboseHTTPCompiler}    Writeln('Adding to status ',Assigned(O),' : ',O.ClassName);    {$endif}    FStatusList.Add(O);  finally    FStatusLock.Leave;  end;end;procedure THTTPCompilerApplication.AddToStatus(AEntry: TDirectoryEntry; AEvents: TFileEvents);Var    O : TJSONObject;  FN : String;begin  Log(etDebug,'File change detected: %s (%s)',[AEntry.name,FileEventsToStr(AEvents)]);  O:=TJSONObject.Create(['action','file','name',AEntry.name,'events',FileEventsToStr(AEvents)]);  if Pos(ExtractFileExt(AEntry.Name),'.lpr.pas.pp.inc.dpr')>0 then    FN:=AEntry.Name;  if (FN<>'') then    O.Add('recompile',true);  AddToStatus(O);  if (FN<>'') then    begin    Log(etDebug,'File change forces recompile: %s',[AEntry.name]);    ScheduleCompile('',Nil);    end;end;procedure THTTPCompilerApplication.DoStatusRequest(ARequest : TRequest; AResponse : TResponse);Var  R,O : TJSONObject;  A : TJSONArray;  I : integer;begin  Log(etDebug,'Status request from: %s',[ARequest.RemoteAddress]);  R:=Nil;  try    FStatusLock.Enter;    try      if (FStatusList.Count=0) then        R:=TJSONObject.Create(['ping',True])      else        begin        {$ifdef VerboseHTTPCompiler}        Writeln(FStatusList[0].ClassName);        {$endif}        O:=FStatusList[0] as TJSONObject;        FStatusList.Delete(0);        if O.Get('action','')<>'file' then          R:=O        else          begin          // If first event is file event, then add and delete all file events in list.          A:=TJSONArray.Create([O]);          O.Delete('action');          R:=TJSONObject.Create(['action','sync','files',A]);          For I:=FStatusList.Count-1 downto 0 do            begin            O:=FStatusList[I] as TJSONObject;            if (O.Get('action','')='file') then              begin              A.Add(O);              O.Delete('action');              FStatusList.Delete(I);              end;            end;          end        end;    finally      FStatusLock.Leave;    end;    AResponse.ContentType:='application/json';    AResponse.Content:=R.AsJSON;    AResponse.SendResponse;  finally    R.Free;  end;end;function THTTPCompilerApplication.ScheduleCompile(const aProjectFile: String;  Options: TStrings): Integer;Var  CI : TCompileItem;  I,TC : Integer;begin  TC:=0;  For I:=0 to FCompiles.Count-1 do    if Assigned(FCompiles[I].THread) then      Inc(TC);  if TC>10 then    begin    Log(etError,'Refusing compile of file "%s" using config file "%s"',[AProjectFile, ConfigFile]);    Exit(nErrTooManyThreads);    end;  CI:=FCompiles.Add as TCompileItem;  Log(etInfo,'Scheduling compile ID %d of file "%s" using config file "%s"',[CI.ID,AProjectFile, ConfigFile]);  CI.BaseDir:=BaseDir;  CI.FileName:=AProjectFile;  CI.ConfigFile:=ConfigFile;  if Assigned(Options) then    CI.Options.Assign(Options);  TCompileThread.Create(Self,CI);  Result:=CI.ID;end;procedure THTTPCompilerApplication.DoRecompile(ARequest: TRequest; AResponse: TResponse);Var  ID : Integer;  PF,CL : String;  Options: TStrings;begin  PF:=ARequest.ContentFields.Values['ProjectFile'];  CL:=ARequest.ContentFields.Values['CompileOptions'];  if PF='' then    PF:=ProjectFile;  If (PF='') then    begin    AResponse.Code:=404;    AResponse.CodeText:='No project file';    AResponse.ContentType:='application/json';    AResponse.Content:='{ "success" : false, "message": "no project file set or provided" }';    end  else    begin    Options:=Nil;    try      if CL<>'' then        begin        Options:=TStringList.Create;        Options.Text:=Cl;        end;      ID:=ScheduleCompile(PF,Options);    finally      FreeAndNil(Options);    end;    if ID=nErrTooManyThreads then      begin      AResponse.Code:=403;      AResponse.CodeText:='Too many compiles';      AResponse.ContentType:='application/json';      AResponse.Content:='{ "success" : false, "message": "Too many compiles running" }';      end    else      begin      AResponse.Code:=200;      AResponse.ContentType:='application/json';      AResponse.Content:=Format('{ "success" : true, "file": "%s", "commandLine" : "%s", "compileID": %d }',[StringToJSONString(PF),StringToJSONString(CL),ID]);      end    end;  AResponse.SendResponse;end;function THTTPCompilerApplication.HandleCompileOptions(aDir: String): Boolean;begin  Result:=False;  Watch:=HasOption('w','watch');  if Hasoption('P','project') then    begin    ProjectFile:=GetOptionValue('P','project');    if ProjectFile='' then      ProjectFile:=IncludeTrailingPathDelimiter(aDir)+'app.lpr';    If Not FileExists(ProjectFile) then      begin      Terminate;      Log(etError,'Project file "%s" does not exist, aborting.',[ProjectFile]);      Exit;      end;    ConfigFile:=GetOptionValue('c','config');    if (ConfigFile='') then      ConfigFile:=ChangeFileExt(Projectfile,'.cfg');    if not FileExists(ConfigFile) then      ConfigFile:='';    end;  if Watch then    begin    if (ProjectFile='') then      Log(etWarning,'No project file specified, disabling watch.')   ;    StartWatch(aDir);    end;  Result:=True;end;function THTTPCompilerApplication.ProcessOptions: Boolean;Var  S,IndexPage,D : String;begin  Result:=False;  S:=Checkoptions('shqd:ni:p:wP::cm:',['help','quiet','noindexpage','directory:','port:','indexpage:','watch','project::','config:','simpleserver','mimetypes:']);  if (S<>'') or HasOption('h','help') then    usage(S);  FServeOnly:=HasOption('s','serve-only');  Quiet:=HasOption('q','quiet');  Port:=StrToIntDef(GetOptionValue('p','port'),3000);  D:=GetOptionValue('d','directory');  if D='' then    D:=GetCurrentDir;  if HasOption('m','mimetypes') then    MimeTypesFile:=GetOptionValue('m','mimetypes');  if MimeTypesFile='' then    begin    MimeTypesFile:=GetDefaultMimeTypesFile;    if not FileExists(MimeTypesFile) then      begin      MimeTypesFile:='';      LoadDefaultMimeTypes;      end;    end  else if not FileExists(MimeTypesFile) then    Log(etWarning,'mimetypes file not found: '+MimeTypesFile);  FBaseDir:=D;  if not ServeOnly then    if not HandleCompileOptions(D) then      exit(False);  TSimpleFileModule.BaseDir:=IncludeTrailingPathDelimiter(D);  TSimpleFileModule.OnLog:=@Log;  Log(etInfo,'Listening on port %d, serving files from directory: %s',[Port,D]);  if ServeOnly then    Log(etInfo,'Compile requests will be ignored.');  If not HasOption('n','noindexpage') then    begin    IndexPage:=GetOptionValue('i','indexpage');    if (IndexPage='') then      IndexPage:='index.html';    Log(etInfo,'Using index page %s',[IndexPage]);    TSimpleFileModule.IndexPageName:=IndexPage;    end;  Result:=True;end;procedure THTTPCompilerApplication.DoRun;begin  If not ProcessOptions then    begin    Terminate;    exit;    end;  if not ServeOnly then    begin    httprouter.RegisterRoute('$sys/compile',rmPost,@DoRecompile);    httprouter.RegisterRoute('$sys/status',rmGet,@DoStatusRequest);    end;  TSimpleFileModule.RegisterDefaultRoute;  inherited;end;end.
 |