1 module dlangide.builders.builder;
2 
3 import dlangui.core.logger;
4 import dlangide.workspace.project;
5 import dlangide.workspace.workspace;
6 import dlangide.ui.outputpanel;
7 import dlangide.builders.extprocess;
8 import dlangui.widgets.appframe;
9 import std.algorithm;
10 import std.array;
11 import core.thread;
12 import std..string;
13 import std.conv;
14 
15 alias BuildResultListener = void delegate(int);
16 
17 dstring replaceVars(dstring source, dstring[dstring] valueMap) {
18     if (!valueMap.length)
19         return source; // no var values
20     if (source.indexOf("{") < 0)
21         return source; // no vars
22     dchar[] res;
23     res.assumeSafeAppend;
24     res.reserve(source.length + 10);
25     int varNameStart = -1;
26     for(int i = 0; i < source.length; i++) {
27         dchar ch = source[i];
28         if (ch == '{') {
29             varNameStart = i + 1;
30         } else if (ch == '}') {
31             if (varNameStart >= 0) {
32                 dstring varName = source[varNameStart .. i];
33                 if (varName.length) {
34                     if (auto value = varName in valueMap) {
35                         res ~= *value;
36                     }
37                 }
38             } else {
39                 res ~= ch;
40             }
41             varNameStart = -1;
42         } else if (varNameStart < 0) {
43             res ~= ch;
44         } // else it's inside variable name
45     }
46     return cast(dstring)res;
47 }
48 
49 class Builder : BackgroundOperationWatcher {
50     protected Project _project;
51     protected ExternalProcess _extprocess;
52     protected OutputPanel _log;
53     protected ProtectedTextStorage _box;
54     protected ProjectConfiguration _projectConfig;
55     protected BuildConfiguration _buildConfig;
56     protected BuildOperation _buildOp;
57     protected string _executable;
58     protected string _additionalParams;
59     protected BuildResultListener _listener;
60     protected int _exitCode = int.min;
61     protected string _toolchain;
62     protected string _arch;
63     protected dstring _description;
64 
65     @property Project project() { return _project; }
66     @property void project(Project p) { _project = p; }
67 
68     this(AppFrame frame, Project project, OutputPanel log, ProjectConfiguration projectConfig, BuildConfiguration buildConfig, 
69              BuildOperation buildOp, 
70              string dubExecutable,
71              string dubAdditionalParams,
72              string toolchain = null,
73              string arch = null,
74              BuildResultListener listener = null) {
75         super(frame);
76         _listener = listener;
77         _projectConfig = projectConfig;
78         _buildConfig = buildConfig;
79         _buildOp = buildOp;
80         _executable = dubExecutable.empty ? "dub" : dubExecutable;
81         _additionalParams = dubAdditionalParams;
82         _project = project;
83         _log = log;
84         _toolchain = toolchain;
85         _arch = arch;
86         _extprocess = new ExternalProcess();
87         _box = new ProtectedTextStorage();
88         string opDescriptionId;
89         switch(buildOp) with(BuildOperation) {
90             case Build:
91                 opDescriptionId = "BUILD_OP_DESCRIPTION_BUILD";
92                 break;
93             case Clean:
94                 opDescriptionId = "BUILD_OP_DESCRIPTION_CLEAN";
95                 break;
96             case Rebuild:
97                 opDescriptionId = "BUILD_OP_DESCRIPTION_REBUILD";
98                 break;
99             case Run:
100                 opDescriptionId = "BUILD_OP_DESCRIPTION_RUN";
101                 break;
102             case Upgrade:
103                 opDescriptionId = "BUILD_OP_DESCRIPTION_UPGRADE";
104                 break;
105             default:
106                 break;
107         }
108         if (opDescriptionId) {
109             import dlangui.core.i18n;
110             _description = UIString.fromId(opDescriptionId).value;
111             dstring[dstring] values;
112             values["projectName"d] = project.name;
113             _description = replaceVars(_description, values);
114         }
115     }
116 
117     /// log lines
118     void pollText() {
119         dstring text = _box.readText();
120         if (text.length) {
121             _log.appendText(null, text);
122         }
123     }
124 
125     /// returns description of background operation to show in status line
126     override @property dstring description() { return _description; }
127     /// returns icon of background operation to show in status line
128     override @property string icon() { return "folder"; }
129     /// update background operation status
130     override void update() {
131         scope(exit)pollText();
132         ExternalProcessState state = _extprocess.state;
133         if (state == ExternalProcessState.None) {
134             import dlangui.core.files;
135             string exepath = findExecutablePath(_executable);
136             if (!exepath) {
137                 _finished = true;
138                 destroy(_extprocess);
139                 _extprocess = null;
140                 return;
141             }
142 
143             _log.clear();
144             char[] program = exepath.dup;
145             char[][] params;
146             char[] dir = _project.dir.dup;
147 
148             {
149                 // dub
150                 if (_buildOp == BuildOperation.Build || _buildOp == BuildOperation.Rebuild) {
151                     params ~= "build".dup;
152                     if (_buildOp == BuildOperation.Rebuild) {
153                         params ~= "--force".dup;
154                     }
155                     if (!_arch.empty)
156                         params ~= ("--arch=" ~ _arch).dup;
157                     if (!_toolchain.empty)
158                         params ~= ("--compiler=" ~ _toolchain).dup;
159                     params ~= "--build-mode=allAtOnce".dup;
160                 } else if (_buildOp == BuildOperation.Clean) {
161                     params ~= "clean".dup;
162                 } else if (_buildOp == BuildOperation.Run) {
163                     if (_projectConfig.type == ProjectConfiguration.Type.Library) {
164                         params ~= "test".dup;
165                     } else {
166                         params ~= "run".dup;
167                     }
168                 } else if (_buildOp == BuildOperation.Upgrade) {
169                     params ~= "upgrade".dup;
170                     params ~= "--force-remove".dup;
171                     import std.path;
172                     import std.file;
173                     string projectFile = project.filename;
174                     string selectionsFile = projectFile.stripExtension ~ ".selections.json";
175                     if (selectionsFile.exists && selectionsFile.isFile) {
176                         Log.i("Removing file ", selectionsFile);
177                         remove(selectionsFile);
178                     }
179                 }
180 
181                 if (_buildOp != BuildOperation.Clean && _buildOp != BuildOperation.Upgrade) {
182                     switch (_buildConfig) {
183                         default:
184                         case BuildConfiguration.Debug:
185                             params ~= "--build=debug".dup;
186                             break;
187                         case BuildConfiguration.Release:
188                             params ~= "--build=release".dup;
189                             break;
190                         case BuildConfiguration.Unittest:
191                             params ~= "--build=unittest".dup;
192                             break;
193                     }
194                     if (!_additionalParams.empty)
195                         params ~= _additionalParams.dup;
196                 }
197 
198                 if(_projectConfig.name != ProjectConfiguration.DEFAULT_NAME) {
199                     params ~= "--config=".dup ~ _projectConfig.name;
200                 }
201             }
202             
203             auto text = "Running (in " ~ dir ~ "): " ~ program ~ " " ~ params.join(' ') ~ "\n";
204             _box.writeText(to!dstring(text));
205             state = _extprocess.run(program, params, dir, _box, null);
206             if (state != ExternalProcessState.Running) {
207                 _box.writeText("Failed to run builder tool\n"d);
208                 _finished = true;
209                 destroy(_extprocess);
210                 _extprocess = null;
211                 return;
212             }
213         }
214         state = _extprocess.poll();
215         if (state == ExternalProcessState.Stopped) {
216             _exitCode = _extprocess.result;
217             _box.writeText("Builder finished with result "d ~ to!dstring(_extprocess.result) ~ "\n"d);
218             _finished = true;
219             return;
220         }
221         if (_cancelRequested) {
222             _extprocess.kill();
223             _extprocess.wait();
224             _finished = true;
225             return;
226         }
227         super.update();
228     }
229     override void removing() {
230         super.removing();
231         if (_exitCode != int.min && _listener)
232             _listener(_exitCode);
233     }
234 }
235