File: fpmakecreatefile.pas

package info (click to toggle)
fpc 3.2.2%2Bdfsg-48
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 341,456 kB
  • sloc: pascal: 3,820,194; xml: 194,356; ansic: 9,637; asm: 8,482; java: 5,346; sh: 4,813; yacc: 3,956; makefile: 2,705; lex: 2,661; javascript: 2,454; sql: 929; php: 474; cpp: 145; perl: 136; sed: 132; csh: 34; tcl: 7
file content (247 lines) | stat: -rw-r--r-- 7,939 bytes parent folder | download | duplicates (10)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
unit fpmakecreatefile;

{$mode objfpc}{$H+}

interface

uses
  Classes, SysUtils,
  fpmakeParseJSon,
  fpTemplate,
  fpmkunit;

procedure CreateFile(AOutputFileName: string; ATemplate: TStringList; APackages: TPackages; ASkipBackup, ACreateDir: boolean);
function TemplateParser: TTemplateParser;

implementation

type

  { TfpmakeTemplateParser }

  TfpmakeTemplateParser = class(TTemplateParser)
  public
    constructor Create;
    Procedure OnGetParamProc(Sender : TObject; Const TagString : String; TagParams:TStringList; Out ReplaceText : String);
  end;

var
  GTemplateParser: TTemplateParser;

resourcestring
  SErrDelBackupFailed = 'Error: Delete of old backup file "%s" failed.';
  SErrCreateDirFailed = 'Error: Could not create the directory for file "%s".';
  SErrNoSuchDirectory = 'Error: Directory of file "%s" does not exists. User -p to force creation.';
  SErrBackupFailed    = 'Error: Backup of file "%s" to "%s" failed.';
  SBackupCreated      = 'Saved old "%s" to "%s"';


function GetConditionalAdd(const Value: string; CPUs: TCPUS; OSes: TOSes; const AddName: string): string;
begin
  if (CPUs <> AllCPUs) and (OSes <> AllOSes) then
    result := result + '    '+AddName+'('''+Value+''','+ExtCPUsToString(CPUs)+','+ExtOSesToString(OSes)+');' + LineEnding
  else if (CPUs <> AllCPUs) then
    result := result + '    '+AddName+'('''+Value+''','+ExtCPUsToString(CPUs)+');' + LineEnding
  else if (OSes <> AllOSes) then
    result := result + '    '+AddName+'('''+Value+''','+ExtOSesToString(OSes)+');' + LineEnding
  else
    result := result + '    '+AddName+'('''+Value+''');' + LineEnding;
end;

function GetConditionalStringsMacro(ACondStrings: TConditionalStrings; APropName: string): string;
var
  ADependency: TConditionalString;
  i: Integer;
begin
  if ACondStrings.Count=0 then
    Exit;
  if ACondStrings.Count=1 then
    begin
    ADependency := ACondStrings[0];
    result := result + GetConditionalAdd(ADependency.Value, ADependency.CPUs, ADependency.OSes,APropName+'.Add');
    end
  else
    begin
    result := '    with '+APropName+' do' + LineEnding +
              '      begin'+LineEnding;
    for i := 0 to ACondStrings.Count-1 do
      begin
      ADependency := ACondStrings[i];
      result := result + GetConditionalAdd(ADependency.Value, ADependency.CPUs, ADependency.OSes,'  Add');
      end;
    result := result +
              '      end;' + LineEnding;

    end;
end;


function GetConditionalPackagePropertiesMacro(APackage: TPackage): string;
begin
  result := '';
  if APackage.CPUs<>AllCPUs then
    result := result + '    P.CPUs := '+ExtCPUSToString(APackage.CPUs)+';'+LineEnding;
  if APackage.OSes<>AllOSes then
    result := result + '    P.OSes := '+ExtOSesToString(APackage.OSes)+';'+LineEnding;
end;

function GetTargetsMacro(aTargets: TTargets): string;
var
  ATarget: TTarget;
  i: Integer;
  d: integer;
begin
  if aTargets.Count=0 then
    Exit;
  result := '    with P.Targets do' + LineEnding +
            '      begin'+LineEnding;
  for i := 0 to aTargets.Count-1 do
    begin
    ATarget := aTargets.Items[i] as TTarget;
    result := result + GetConditionalAdd(ATarget.Name + ATarget.Extension, ATarget.CPUs, ATarget.OSes,'  T := AddUnit');
    if atarget.ResourceStrings then
      result := result + '      T.Resourcestrings := True;'+LineEnding;
    for d := 0 to aTarget.Dependencies.Count-1 do
      begin
      if ATarget.Dependencies[d].DependencyType=depInclude then
        result := result + '      T.Dependencies.AddInclude('''+ATarget.Dependencies[d].Value+''');'+LineEnding
      else if ATarget.Dependencies[d].DependencyType=depUnit then
        result := result + '      T.Dependencies.AddUnit('''+ATarget.Dependencies[d].Value+''');'+LineEnding
      else
        result := result + '      T.Dependencies.Add('''+ATarget.Dependencies[d].Value+''');'+LineEnding;
      end;
    end;
  result := result +
            '      end;';
end;


procedure CreateFile(AOutputFileName: string; ATemplate: TStringList; APackages: TPackages; ASkipBackup, ACreateDir: boolean);

Var
  Fout : Text;
  S,BFN : String;
  I : Integer;
  PackageNr: Integer;
  APackage: TPackage;

begin
  If (AOutputFileName<>'')
     and FileExists(AOutputFileName)
     and not ASkipBackup then
    begin
    BFN:=ChangeFileExt(AOutputFileName,'.bak');
    If FileExists(BFN) and not DeleteFile(BFN) then
      begin
      Writeln(StdErr,Format(SErrDelBackupFailed,[BFN]));
      Halt(1);
      end;
    If not RenameFile(AOutputFileName,BFN) then
      begin
      Writeln(StdErr,Format(SErrBackupFailed,[AOutputFileName,BFN]));
      Halt(1);
      end
    else
      Writeln(Format(SBackupCreated,[ExtractFileName(AOutputFileName),ExtractFileName(BFN)]));
    end;
  if (AOutputFileName<>'') and (ExtractFilePath(AOutputFileName)<>'') and not DirectoryExists(ExtractFilePath(AOutputFileName)) then
    begin
    if ACreateDir then
      begin
      if not ForceDirectories(ExtractFilePath(AOutputFileName)) then
        begin
        Writeln(StdErr,Format(SErrCreateDirFailed,[AOutputFileName]));
        Halt(1);
        end;
      end
    else
      begin
      Writeln(StdErr,Format(SErrNoSuchDirectory,[AOutputFileName]));
      Halt(1);
      end;
    end;
  Assign(Fout,AOutputFileName);
  Rewrite(FOut);
  Try
    for PackageNr := 0 to APackages.Count-1 do
      begin
      APackage := APackages.Items[PackageNr] as TPackage;

      TemplateParser.Values['packagename'] := APackage.Name;
      TemplateParser.Values['directory'] := APackage.Directory;
      TemplateParser.Values['version'] := APackage.Version;
      TemplateParser.Values['author'] := APackage.Author;
      TemplateParser.Values['license'] := APackage.License;
      TemplateParser.Values['homepageurl'] := APackage.HomepageURL;
      TemplateParser.Values['downloadurl'] := APackage.DownloadURL;
      TemplateParser.Values['email'] := APackage.Email;
      TemplateParser.Values['description'] := APackage.Description;
      TemplateParser.Values['needlibc'] := BoolToStr(APackage.NeedLibC,'true','false');
      TemplateParser.Values['conditionalpackageproperties'] := GetConditionalPackagePropertiesMacro(APackage);
      TemplateParser.Values['packagedependencies'] := GetConditionalStringsMacro(APackage.Dependencies, 'P.Dependencies');
      TemplateParser.Values['packagesourcepaths'] := GetConditionalStringsMacro(APackage.SourcePath, 'P.SourcePath');
      TemplateParser.Values['targets'] := GetTargetsMacro(APackage.Targets);

      For I:=0 to ATemplate.Count-1 do
        begin
        S:=ATemplate[i];
        S := TemplateParser.ParseString(S);
        Writeln(FOut,S);
        end;

      end;
  Finally
    Close(Fout);
  end;
end;

function TemplateParser: TTemplateParser;
begin
  if not assigned(GTemplateParser) then
    begin
    GTemplateParser := TfpmakeTemplateParser.Create;
    GTemplateParser.StartDelimiter:='%';
    GTemplateParser.EndDelimiter:='%';
    GTemplateParser.ParamStartDelimiter:='(';
    GTemplateParser.ParamEndDelimiter:=')';
    GTemplateParser.Values['PWD'] := GetCurrentDir;
    GTemplateParser.Values['BUILDDATE'] := DateToStr(Date);
    GTemplateParser.Values['BUILDTIME'] := TimeToStr(Time);
    end;
  result := GTemplateParser;
end;

{ TfpmakeTemplateParser }

constructor TfpmakeTemplateParser.Create;
begin
  inherited create;
  AllowTagParams := True;
  OnReplaceTag := @OnGetParamProc;
end;

procedure TfpmakeTemplateParser.OnGetParamProc(Sender : TObject; Const TagString : String; TagParams:TStringList; Out ReplaceText : String);
var
  i: Integer;
  s: string;
begin
  if TagString = 'quotedstr' then
    begin
    i := TagParams.Count;
    ReplaceText:='';
    for i := 0 to TagParams.Count-1 do
      begin
      GetParam(TagParams[i],s);
      ReplaceText:=ReplaceText + quotedstr(s);
      end;
    end
  else
    GetParam(TagString,ReplaceText);
end;

initialization
  GTemplateParser := nil
finalization
  GTemplateParser.Free;
end.