diff --git a/src/main/nim/strawbosspkg/configuration.nim b/src/main/nim/strawbosspkg/configuration.nim index 1ab100d..edff17f 100644 --- a/src/main/nim/strawbosspkg/configuration.nim +++ b/src/main/nim/strawbosspkg/configuration.nim @@ -13,17 +13,16 @@ type complete, failed, queued, rejected, running, setup, stepComplete BuildStatus* = object - runId*, details*: string + runId*, details*, version*: string state*: BuildState Step* = object - name*, stepCmd*, workingDir*: string + containerImage, name*, stepCmd*, workingDir*: string artifacts*, cmdInput*, depends*, expectedEnv*: seq[string] dontSkip*: bool ProjectConfig* = object - name*: string - versionCmd*: string + containerImage*, name*, versionCmd*: string steps*: Table[string, Step] ProjectDef* = object @@ -170,14 +169,15 @@ proc loadProjectConfig*(cfgFile: string): ProjectConfig = var steps = initTable[string, Step]() for sName, pJson in jsonCfg.getOrFail("steps", "project configuration").getFields: steps[sName] = Step( - name: sName, - workingDir: pJson.getIfExists("workingDir").getStr("."), - stepCmd: pJson.getIfExists("stepCmd").getStr("NOT GIVEN"), - depends: pJson.getIfExists("depends").getElems.mapIt(it.getStr), - artifacts: pJson.getIfExists("artifacts").getElems.mapIt(it.getStr), - cmdInput: pJson.getIfExists("cmdInput").getElems.mapIt(it.getStr), - expectedEnv: pJson.getIfExists("expectedEnv").getElems.mapIt(it.getStr), - dontSkip: pJson.getIfExists("dontSkip").getBVal(false)) + name: sName, + workingDir: pJson.getIfExists("workingDir").getStr("."), + stepCmd: pJson.getIfExists("stepCmd").getStr("NOT GIVEN"), + depends: pJson.getIfExists("depends").getElems.mapIt(it.getStr), + artifacts: pJson.getIfExists("artifacts").getElems.mapIt(it.getStr), + cmdInput: pJson.getIfExists("cmdInput").getElems.mapIt(it.getStr), + expectedEnv: pJson.getIfExists("expectedEnv").getElems.mapIt(it.getStr), + containerImage: pJson.getIfExists("containerImage").getStr(""), + dontSkip: pJson.getIfExists("dontSkip").getBVal(false)) # cmdInput and stepCmd are related, so we have a conditional defaulting. # Four possibilities: @@ -194,6 +194,7 @@ proc loadProjectConfig*(cfgFile: string): ProjectConfig = result = ProjectConfig( name: jsonCfg.getOrFail("name", "project configuration").getStr, + containerImage: jsonCfg.getIfExists("containerImage").getStr(""), versionCmd: jsonCfg.getIfExists("versionCmd").getStr("git describe --tags --always"), steps: steps) @@ -259,6 +260,9 @@ proc `%`*(s: Step): JsonNode = "expectedEnv": s.expectedEnv, "dontSkip": s.dontSkip } + if not s.containerImage.isNullOrEmpty: + result["containerImage"] = %s.containerImage + proc `%`*(p: ProjectConfig): JsonNode = result = %* { "name": p.name, @@ -268,6 +272,9 @@ proc `%`*(p: ProjectConfig): JsonNode = for name, step in p.steps: result["steps"][name] = %step + if not p.containerImage.isNilOrEmpty: + result["containerImage"] = %p.containerImage + proc `%`*(req: RunRequest): JsonNode = result = %* { "runId": $(req.runId), diff --git a/src/main/nim/strawbosspkg/core.nim b/src/main/nim/strawbosspkg/core.nim index e124ab3..5e7eada 100644 --- a/src/main/nim/strawbosspkg/core.nim +++ b/src/main/nim/strawbosspkg/core.nim @@ -1,5 +1,5 @@ import cliutils, logging, json, os, ospaths, osproc, sequtils, streams, - strtabs, strutils, tables, times, uuids + strtabs, strutils, tables, tempfile, times, uuids import ./configuration import nre except toSeq @@ -49,7 +49,51 @@ proc newCopy(w: Workspace): Workspace = step: w.step, version: w.version) -# Logging wrappers around +const WKSP_ROOT = "/strawboss/wksp" +const ARTIFACTS_ROOT = "/strawboss/artifacts" + +proc execWithOutput(w: Workspace, cmd, workingDir: string, + args: openarray[string], env: StringTableRef, + options: set[ProcessOption] = {poUsePath}, + msgCB: HandleProcMsgCB = nil): + tuple[output: TaintedString, error: TaintedString, exitCode: int] + {.tags: [ExecIOEffect, ReadIOEffect, RootEffect] .} = + + # Look for a container image to use + let containerImage = + if not isNilOrEmpty(w.step.containerImage): w.step.containerImage + else: w.project.containerImage + + if containerImage.isNilOrEmpty: + return exec(cmd, workingDir, args, env, options, msgCB + + var fullEnv = newStringTable(modeCaseSensitive) + for k,v in env: fullEnv[k] = v + + var fullArgs = @["run", "-w", WKSP_ROOT, "-v", wksp.dir & ":" & WKSP_ROOT ] + + if w.step.name.isNilOrEmpty: + for depStep in step.depends: + fullArgs.add("-v", ARTIFACTS_ROOT / depStep) + fullEnv[depStep & "_DIR"] = ARTIFACTS_DIR / depStep) + + let envFile = mkstemp()[0] + writeFile(envFile, toSeq(fullEnv.pairs()).mapIt(it[0] & "=" & it[1]).join("\n")) + + fullArgs.add("--env-file", envFile) + fullArgs.add(containerImage) + fullArgs.add(cmd) + + echo "Executing docker command: \n\t" & "docker " & $(fullArgs & args) + return execWithOutput("docker", wksp.dir, fullArgs & args, fullEnv, options, msgCB) + +proc exec(w: Workspace, cmd, workingDir: string, args: openarray[string], + env: StringTableRef, options: set[ProcessingOption] = {poUsePath}, + msgCB: HandleProcMsgCG = nil): int + {.tags: [ExecIOEffect, ReadIOEffect, RootEffect] .} = + + return execWithOutput(w, cmd, workingDir, args, env, options, msgCB)[2] + # Utility methods for Workspace activities proc sendStatusMsg(oh: HandleProcMsgCB, status: BuildStatus): void = if not oh.isNil: @@ -89,7 +133,7 @@ proc publishStatus(wksp: Workspace, state: BuildState, details: string): void = $wksp.runRequest.runId & ".status.json", $wksp.status) # If we have our step we can save status to the step status - if not wksp.step.name.isNilOrEmpty(): + if not wksp.step.name.isNilOrEmpty: let stepStatusDir = wksp.buildDataDir / "status" / wksp.step.name if not existsDir(stepStatusDir): createDir(stepStatusDir) writeFile(stepStatusDir / wksp.version & ".json", $wksp.status) diff --git a/src/test/json/dummy-project.config.json b/src/test/json/dummy-project.config.json index d0cf115..1219d38 100644 --- a/src/test/json/dummy-project.config.json +++ b/src/test/json/dummy-project.config.json @@ -1,8 +1,10 @@ { "name": "dummy-project", "versionCmd": "git describe --all --always", + "containerImage": "ubuntu", "steps": { "build": { + "containerImage": "alpine", "depends": ["test"], "workingDir": "dir1", "stepCmd": "cust-build", diff --git a/src/test/nim/unit/tconfiguration.nim b/src/test/nim/unit/tconfiguration.nim index 7a8255d..4cc1710 100644 --- a/src/test/nim/unit/tconfiguration.nim +++ b/src/test/nim/unit/tconfiguration.nim @@ -99,6 +99,7 @@ suite "load and save configuration objects": check: pc.name == "dummy-project" pc.versionCmd == "git describe --all --always" + pc.containerImage == "ubuntu" pc.steps.len == 2 # Explicitly set properties @@ -106,6 +107,7 @@ suite "load and save configuration objects": pc.steps["build"].dontSkip == true pc.steps["build"].stepCmd == "cust-build" pc.steps["build"].workingDir == "dir1" + pc.steps["containerImage"] == "alpine" sameContents(pc.steps["build"].artifacts, @["bin1", "doc1"]) sameContents(pc.steps["build"].depends, @["test"]) sameContents(pc.steps["build"].expectedEnv, @["VAR1"]) @@ -115,7 +117,8 @@ suite "load and save configuration objects": pc.steps["test"].name == "test" pc.steps["test"].dontSkip == false pc.steps["test"].stepCmd == "true" - pc.steps["test"].workingDir == "." + pc.steps["test"].workingDir == ".: + pc.steps["test"].containerImage.isNilOrEmpty sameContents(pc.steps["test"].artifacts, @[]) sameContents(pc.steps["test"].depends, @[]) sameContents(pc.steps["test"].expectedEnv, @[])