Fix bug around spawning worker processes.

We were expecting to find the path to the `strawboss` binary implicitly from
the environment, which meant that configuration was also implicit, and required
more setup. Now the path to the binary is explicit in the StrawBoss runtime
configuration, and the path to the configuration file can also be explicitly given.
This commit is contained in:
Jonathan Bernard 2017-05-11 10:43:55 -05:00
parent 42f37a21e6
commit a1100f17d8
6 changed files with 47 additions and 15 deletions

View File

@ -41,6 +41,7 @@ Options
else: "strawboss.config.json" else: "strawboss.config.json"
var cfg = loadStrawBossConfig(cfgFile) var cfg = loadStrawBossConfig(cfgFile)
cfg.pathToExe = paramStr(0)
if not existsDir(cfg.artifactsRepo): if not existsDir(cfg.artifactsRepo):
echo "Artifacts repo (" & cfg.artifactsRepo & ") does not exist. Creating..." echo "Artifacts repo (" & cfg.artifactsRepo & ") does not exist. Creating..."
createDir(cfg.artifactsRepo) createDir(cfg.artifactsRepo)

View File

@ -37,7 +37,9 @@ type
StrawBossConfig* = object StrawBossConfig* = object
artifactsRepo*: string artifactsRepo*: string
authSecret*: string authSecret*: string
filePath*: string
debug*: bool debug*: bool
pathToExe*: string
projects*: seq[ProjectDef] projects*: seq[ProjectDef]
pwdCost*: int8 pwdCost*: int8
users*: seq[UserRef] users*: seq[UserRef]
@ -131,6 +133,7 @@ proc loadStrawBossConfig*(cfgFile: string): StrawBossConfig =
raiseEx "strawboss config file not found: " & cfgFile raiseEx "strawboss config file not found: " & cfgFile
result = parseStrawBossConfig(parseFile(cfgFile)) result = parseStrawBossConfig(parseFile(cfgFile))
result.filePath = cfgFile
proc loadProjectConfig*(cfgFile: string): ProjectConfig = proc loadProjectConfig*(cfgFile: string): ProjectConfig =
if not existsFile(cfgFile): if not existsFile(cfgFile):

View File

@ -77,12 +77,16 @@ proc extractSession(cfg: StrawBossConfig, request: Request): Session =
result = fromJWT(cfg, headerVal[7..^1]) result = fromJWT(cfg, headerVal[7..^1])
proc spawnWorker(req: RunRequest): Worker = proc spawnWorker(cfg: StrawBossConfig, req: RunRequest): Worker =
## Kick off a new worker process with the given run information
let dir = mkdtemp() let dir = mkdtemp()
var args = @["run", req.projectName, req.stepName, "-r", req.buildRef, "-w", dir] var args = @["run", req.projectName, req.stepName, "-r", req.buildRef,
"-w", dir, "-c", cfg.filePath]
if req.forceRebuild: args.add("-f") if req.forceRebuild: args.add("-f")
debug "Launching worker: " & cfg.pathToExe & " " & args.join(" ")
result = Worker( result = Worker(
process: startProcess("strawboss", ".", args, loadEnv(), {poUsePath}), process: startProcess(cfg.pathToExe, ".", args, loadEnv(), {poUsePath}),
workingDir: dir) workingDir: dir)
proc hashPwd*(pwd: string, cost: int8): string = proc hashPwd*(pwd: string, cost: int8): string =
@ -248,7 +252,18 @@ proc start*(cfg: StrawBossConfig): void =
projectName: @"projectName", projectName: @"projectName",
stepName: @"stepName", stepName: @"stepName",
buildRef: if @"buildRef" != "": @"buildRef" else: nil, buildRef: if @"buildRef" != "": @"buildRef" else: nil,
forceRebuild: false))) # TODO support this with optional query params forceRebuild: false) # TODO support this with optional query params
# TODO: instead of immediately spawning a worker, add the request to a
# queue to be picked up by a worker. Allows capping the number of worker
# prcesses, distributing, etc.
let worker = spawnWorker(cfg, runRequest)
workers.add(worker)
resp($(%*{
"runRequest": runRequest,
"status": { "state": "accepted", "details": "Run request has been queued." }
}))
post "/service/debug/stop": post "/service/debug/stop":
if not cfg.debug: resp(Http404, makeJsonResp(Http404), JSON) if not cfg.debug: resp(Http404, makeJsonResp(Http404), JSON)

View File

@ -87,9 +87,22 @@ suite "strawboss server":
let returnedVersions = parseJson(resp.body).getElems.mapIt(it.getStr) let returnedVersions = parseJson(resp.body).getElems.mapIt(it.getStr)
check sameContents(expectedVersions, returnedVersions) check sameContents(expectedVersions, returnedVersions)
#test "enqueue a build": test "run a successful build with artifacts":
# let http = newAuthenticatedHttpClient(apibase, "bob@builder.com", "password") let http = newAuthenticatedHttpClient(apibase, "bob@builder.com", "password")
# let resp = http.get(apiBase & "/project/" & testProjName & " let resp = http.get(apiBase & "/project/" & testProjName & "/step/build/run/0.1.0"
check resp.status.startsWith("200")
# TODO
# check that the project directory has been created in the artifacts repo
# check that the run status file has been created in the artifacts repo
# check that the run status is not failed
# wait for the build to complete
# check that the status is "complete"
# check that the artifacts we expect are present
check false
test "run a time-consuming build and check the status via the API"
check false
# Last-chance catch to kill the server in case some test err'ed and didn't # Last-chance catch to kill the server in case some test err'ed and didn't
# reach it's teardown handler # reach it's teardown handler

View File

@ -5,14 +5,14 @@ import ../../../main/nim/strawbosspkg/configuration
suite "load and save configuration objects": suite "load and save configuration objects":
# suite setup & common data # suite setup & common data
let testProjDefStr = """{ "name": "test-project-1", "repo": let testProjDefStr = """{ "name": "dummy-project", "repo":
"/non-existent/dir", "/non-existent/dir",
"cfgFilePath": "strawhat.json", "cfgFilePath": "strawhat.json",
"defaultBranch": "deploy", "defaultBranch": "deploy",
"envVars": { "VAR1": "value" } }""" "envVars": { "VAR1": "value" } }"""
let testProjDef = ProjectDef( let testProjDef = ProjectDef(
name: "test-project-1", name: "dummy-project",
repo: "/non-existent/dir", repo: "/non-existent/dir",
cfgFilePath: "strawhat.json", cfgFilePath: "strawhat.json",
defaultBranch: "deploy", defaultBranch: "deploy",
@ -23,7 +23,7 @@ suite "load and save configuration objects":
let pd = parseProjectDef(parseJson(testProjDefStr)) let pd = parseProjectDef(parseJson(testProjDefStr))
check: check:
pd.name == "test-project-1" pd.name == "dummy-project"
pd.repo == "/non-existent/dir" pd.repo == "/non-existent/dir"
pd.cfgFilePath == "strawhat.json" pd.cfgFilePath == "strawhat.json"
pd.defaultBranch == "deploy" pd.defaultBranch == "deploy"
@ -61,13 +61,13 @@ suite "load and save configuration objects":
let expectedUsers = @[UserRef(name: "bob@builder.com", hashedPwd: "testvalue"), let expectedUsers = @[UserRef(name: "bob@builder.com", hashedPwd: "testvalue"),
UserRef(name: "sam@sousa.com", hashedPwd: "testvalue")] UserRef(name: "sam@sousa.com", hashedPwd: "testvalue")]
let expectedProjects = @[ let expectedProjects = @[
ProjectDef(name: "test-project-1", ProjectDef(name: "dummy-project",
repo: "/non-existent/dir", repo: "/non-existent/dir",
defaultBranch: "deploy", defaultBranch: "deploy",
cfgFilePath: "strawhat.json", cfgFilePath: "strawhat.json",
envVars: newStringTable("VAR1", "value", modeCaseSensitive)), envVars: newStringTable("VAR1", "value", modeCaseSensitive)),
ProjectDef(name: "test-strawboss", ProjectDef(name: "test-project",
repo: "https://git.jdb-labs.com:jdb/test-strawboss.git", repo: "",
defaultBranch: "master", defaultBranch: "master",
cfgFilePath: "strawboss.json", cfgFilePath: "strawboss.json",
envVars: newStringTable(modeCaseSensitive))] envVars: newStringTable(modeCaseSensitive))]
@ -80,10 +80,10 @@ suite "load and save configuration objects":
sameContents(expectedProjects, cfg.projects) sameContents(expectedProjects, cfg.projects)
test "loadProjectConfig": test "loadProjectConfig":
let pc = loadProjectConfig("src/test/json/test-project-1.config.json") let pc = loadProjectConfig("src/test/json/dummy-project.config.json")
check: check:
pc.name == "test-project-1" pc.name == "dummy-project"
pc.versionCmd == "git describe --all --always" pc.versionCmd == "git describe --all --always"
pc.steps.len == 2 pc.steps.len == 2

Binary file not shown.