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