{ /*************************************************************************** idecmdline.pas -------------------- A unit to manage command lines issue used inside the ide ***************************************************************************/ *************************************************************************** * * * This source is free software; you can redistribute it and/or modify * * it under the terms of the GNU General Public License as published by * * the Free Software Foundation; either version 2 of the License, or * * (at your option) any later version. * * * * This code is distributed in the hope that it will be useful, but * * WITHOUT ANY WARRANTY; without even the implied warranty of * * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * * General Public License for more details. * * * * A copy of the GNU General Public License is available on the World * * Wide Web at . You can also * * obtain it by writing to the Free Software Foundation, * * Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. * * * *************************************************************************** Author: Ido Kanner This unit manages the commandline utils that are used across Lazarus. It was created for avoding duplicates and easier access for commandline utils that are required by the IDE. } unit IDECmdLine; {$mode objfpc}{$H+} interface uses Classes, SysUtils, FileUtil, LazFileUtils, LazUTF8, LazUTF8Classes, LazLogger, LazConf, LCLProc; const ShowSetupDialogOptLong='--setup'; PrimaryConfPathOptLong='--primary-config-path='; PrimaryConfPathOptShort='--pcp='; SecondaryConfPathOptLong='--secondary-config-path='; SecondaryConfPathOptShort='--scp='; NoSplashScreenOptLong='--no-splash-screen'; NoSplashScreenOptShort='--nsc'; StartedByStartLazarusOpt='--started-by-startlazarus'; ForceNewInstanceOpt='--force-new-instance'; SkipLastProjectOpt='--skip-last-project'; DebugLogOpt='--debug-log='; DebugLogOptEnable='--debug-enable='; LanguageOpt='--language='; LazarusDirOpt ='--lazarusdir='; procedure ParseCommandLine(aCmdLineParams: TStrings; out IDEPid : Integer; out ShowSplashScreen: boolean); function GetCommandLineParameters(aCmdLineParams: TStrings; isStartLazarus: Boolean = False) : string; function ExtractPrimaryConfigPath(aCmdLineParams: TStrings): string; function IsHelpRequested : Boolean; function IsVersionRequested : boolean; function GetLanguageSpecified : string; function ParamIsOption(ParamIndex : integer; const Option : string) : boolean; function ParamIsOptionPlusValue(ParamIndex : integer; const Option : string; out AValue : string) : boolean; procedure ParseNoGuiCmdLineParams; function ExtractCmdLineFilenames : TStrings; // options from CFG file function GetCfgFileContent: TStrings; function GetParamsAndCfgFile: TStrings; function ParamsAndCfgCount: Integer; function ParamsAndCfgStr(Idx: Integer): String; procedure ResetParamsAndCfg; var ConsoleVerbosity: integer = 0; // 0=normal, -1=quiet, 1=verbose, 2=very verbose implementation var CfgFileName: String = ''; CfgFileDone: Boolean = False; CfgFileContent: TStrings = nil; ParamsAndCfgFileContent: TStrings = nil; function GetCfgFileContent: TStrings; begin Result := CfgFileContent; if CfgFileDone then exit; CfgFileDone := True; CfgFileName := AppendPathDelim(ProgramDirectory) + 'lazarus.cfg'; if FileExistsUTF8(CfgFileName) then begin DebugLn(['using config file ', CfgFileName]); CfgFileContent := TStringListUTF8.Create; CfgFileContent.LoadFromFile(CfgFileName); end; Result := CfgFileContent; end; function GetParamsAndCfgFile: TStrings; var Cfg: TStrings; i: Integer; s: String; Warn: String; begin Result := ParamsAndCfgFileContent; if Result <> nil then exit; ParamsAndCfgFileContent := TStringList.Create; ParamsAndCfgFileContent.Add(ParamStrUTF8(0)); Cfg := GetCfgFileContent; if Cfg <> nil then begin Warn := ''; // insert Cfg at start. For duplicates the latest occurence takes precedence for i := 0 to Cfg.Count - 1 do begin s := Cfg[i]; if (s <> '') and (s[1] = '-') then begin s := Trim(s); {$ifdef windows} //cfg file is made by Windows installer and probably is Windows default codepage if FindInvalidUTF8Character(PChar(s), Length(s), True) > 0 then s := WinCPToUtf8(s); {$endif windows} ParamsAndCfgFileContent.Add(s) end else if (Trim(s) <> '') and (s[1] <> '#') then Warn := Warn + IntToStr(i)+': ' + s + LineEnding; end; if Warn<>'' then begin debugln('WARNING: invalid lines in lazarus.cfg:'); debugln(Warn); end; end; for i := 1 to Paramcount do ParamsAndCfgFileContent.Add(ParamStrUTF8(i)); Result := ParamsAndCfgFileContent; end; function ParamsAndCfgCount: Integer; begin Result := GetParamsAndCfgFile.Count; end; function ParamsAndCfgStr(Idx: Integer): String; begin if (Idx < 0) or (Idx >= GetParamsAndCfgFile.Count) then Result := '' else Result := GetParamsAndCfgFile[Idx]; end; procedure ResetParamsAndCfg; begin FreeAndNil(ParamsAndCfgFileContent); end; procedure ParseCommandLine(aCmdLineParams: TStrings; out IDEPid: Integer; out ShowSplashScreen: boolean); const LazarusPidOpt = '--lazarus-pid='; LazarusDebugOpt = '--debug'; var i : Integer; Param : string; HasDebugLog: Boolean; begin IDEPid := 0; HasDebugLog := False; for i := 1 to ParamsAndCfgCount do begin Param := ParamsAndCfgStr(i); if SysUtils.CompareText(LeftStr(Param, length(DebugLogOpt)), DebugLogOpt) = 0 then HasDebugLog := HasDebugLog or (length(Param) > length(DebugLogOpt)); if (Param=LazarusDebugOpt) and (not HasDebugLog) then begin aCmdLineParams.Add('--debug-log=' + AppendPathDelim(UTF8ToSys(GetPrimaryConfigPath)) + 'debug.log'); end; if LeftStr(Param,length(LazarusPidOpt))=LazarusPidOpt then begin try IDEPid := StrToInt(RightStr(Param,Length(Param)-Length(LazarusPidOpt))); except DebugLn('Failed to parse %s',[Param]); IDEPid := 0; end; end else if ParamIsOption(i, NoSplashScreenOptLong) or ParamIsOption(i, NoSplashScreenOptShort) then begin ShowSplashScreen := false; end else begin // Do not add file to the parameter list if not (Copy(Param,1,1) = '-') and (FileExistsUTF8(ExpandFileNameUTF8(Param))) then begin DebugLn('%s is a file', [Param]); continue; end; // pass these parameters to Lazarus DebugLn('Adding "%s" as a parameter', [Param]); aCmdLineParams.Add(Param); end; end; end; function GetCommandLineParameters(aCmdLineParams : TStrings; isStartLazarus : Boolean = False) : String; var i: Integer; s: String; begin if isStartLazarus then Result := ' --no-splash-screen --started-by-startlazarus' else Result := ''; for i := 0 to aCmdLineParams.Count - 1 do begin s := aCmdLineParams[i]; // make sure that command line parameters are still // double quoted, if they contain spaces if pos(' ', s) > 0 then s := '"' + s + '"'; Result := Result + ' ' + s; end; end; function ExtractPrimaryConfigPath(aCmdLineParams: TStrings): string; procedure GetParam(Param, Prefix: string; var Value: string); begin if LeftStr(Param,length(Prefix))=Prefix then Value:=copy(Param,length(Prefix)+1,length(Param)); end; var i: Integer; begin Result:=''; for i:=0 to aCmdLineParams.Count-1 do begin GetParam(aCmdLineParams[i],PrimaryConfPathOptLong,Result); GetParam(aCmdLineParams[i],PrimaryConfPathOptShort,Result); end; end; function IsHelpRequested : Boolean; var i: integer; begin Result := false; i:=1; while (i <= ParamsAndCfgCount) and (Result = false) do begin Result := ParamIsOption(i, '--help') or ParamIsOption(i, '-help') or ParamIsOption(i, '-?') or ParamIsOption(i, '-h'); inc(i); end; end; function IsVersionRequested: boolean; begin Result := (ParamsAndCfgCount=1) and (ParamIsOption(1, '--version') or ParamIsOption(1, '-v')); end; function GetLanguageSpecified : string; var i: integer; AValue: string; begin // return language specified in command line (empty string if no language specified) Result := ''; AValue := ''; i := 1; while i <= ParamsAndCfgCount do begin if ParamIsOptionPlusValue(i, LanguageOpt, AValue) = true then begin Result := AValue; exit; end; inc(i); end; end; function ParamIsOption(ParamIndex : integer; const Option : string) : boolean; begin Result:=SysUtils.CompareText(ParamsAndCfgStr(ParamIndex),Option) = 0; end; function ParamIsOptionPlusValue(ParamIndex : integer; const Option : string; out AValue : string) : boolean; var p : String; begin p := ParamsAndCfgStr(ParamIndex); Result := SysUtils.CompareText(LeftStr(p, length(Option)), Option) = 0; if Result then AValue := copy(p, length(Option) + 1, length(p)) else AValue := ''; end; procedure ParseNoGuiCmdLineParams; var i : integer; AValue : String; begin for i:= 1 to ParamsAndCfgCount do begin //DebugLn(['ParseNoGuiCmdLineParams ',i,' "',ParamsAndCfgStr(i),'"']); if ParamIsOptionPlusValue(i, PrimaryConfPathOptLong, AValue) then SetPrimaryConfigPath(AValue) else if ParamIsOptionPlusValue(i, PrimaryConfPathOptShort, AValue) then SetPrimaryConfigPath(AValue) else if ParamIsOptionPlusValue(i, SecondaryConfPathOptLong, AValue) then SetSecondaryConfigPath(AValue) else if ParamIsOptionPlusValue(i, SecondaryConfPathOptShort, AValue) then SetSecondaryConfigPath(AValue); end; end; function ExtractCmdLineFilenames : TStrings; var i : LongInt; Filename : String; begin Result := nil; for i := 1 to ParamsAndCfgCount do begin Filename := ParamsAndCfgStr(i); if (Filename = '') or (Filename[1] = '-') then continue; if Result = nil then Result := TStringList.Create; Result.Add(Filename); end; end; procedure InitLogger; var i : integer; AValue : String; begin for i:= 1 to ParamsAndCfgCount do begin if ParamIsOptionPlusValue(i, DebugLogOpt, AValue) then LazLogger.DebugLogger.LogName := AValue; end; end; initialization InitLogger; finalization FreeAndNil(CfgFileContent); FreeAndNil(ParamsAndCfgFileContent); end.