2018-01-26 11:07:02 +00:00
|
|
|
///
|
|
|
|
/// Copyright 2018 Oliver Giles
|
|
|
|
///
|
|
|
|
/// This file is part of Laminar
|
|
|
|
///
|
|
|
|
/// Laminar is free software: you can redistribute it and/or modify
|
|
|
|
/// it under the terms of the GNU General Public License as published by
|
|
|
|
/// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
/// (at your option) any later version.
|
|
|
|
///
|
|
|
|
/// Laminar is distributed in the hope that it will be useful,
|
|
|
|
/// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
/// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
/// GNU General Public License for more details.
|
|
|
|
///
|
|
|
|
/// You should have received a copy of the GNU General Public License
|
|
|
|
/// along with Laminar. If not, see <http://www.gnu.org/licenses/>
|
|
|
|
///
|
|
|
|
#include <gtest/gtest.h>
|
|
|
|
#include "run.h"
|
|
|
|
#include "log.h"
|
|
|
|
#include "node.h"
|
|
|
|
#include "conf.h"
|
2018-09-28 07:36:10 +00:00
|
|
|
#include "tempdir.h"
|
2018-01-26 11:07:02 +00:00
|
|
|
|
2018-09-28 07:36:10 +00:00
|
|
|
class RunTest : public testing::Test {
|
2018-01-26 11:07:02 +00:00
|
|
|
protected:
|
2018-09-28 07:36:10 +00:00
|
|
|
RunTest() :
|
|
|
|
testing::Test(),
|
|
|
|
node(std::make_shared<Node>()),
|
|
|
|
tmp(),
|
|
|
|
run("foo", ParamMap{}, tmp.path.clone())
|
|
|
|
{
|
2018-01-26 11:07:02 +00:00
|
|
|
}
|
2018-09-28 07:36:10 +00:00
|
|
|
|
|
|
|
~RunTest() noexcept {}
|
|
|
|
|
2018-02-24 16:53:11 +00:00
|
|
|
void wait() {
|
|
|
|
int state = -1;
|
2018-08-03 11:36:24 +00:00
|
|
|
waitpid(run.current_pid.orDefault(0), &state, 0);
|
2018-02-24 16:53:11 +00:00
|
|
|
run.reaped(state);
|
|
|
|
}
|
2018-09-28 07:36:10 +00:00
|
|
|
|
2018-01-26 11:07:02 +00:00
|
|
|
void runAll() {
|
2018-02-24 16:53:11 +00:00
|
|
|
while(!run.step())
|
|
|
|
wait();
|
2018-01-26 11:07:02 +00:00
|
|
|
}
|
2018-09-28 07:36:10 +00:00
|
|
|
|
2018-01-26 11:07:02 +00:00
|
|
|
std::string readAllOutput() {
|
|
|
|
std::string res;
|
|
|
|
char tmp[64];
|
2018-07-20 11:15:59 +00:00
|
|
|
for(ssize_t n = read(run.output_fd, tmp, 64); n > 0; n = read(run.output_fd, tmp, 64))
|
2018-01-26 11:07:02 +00:00
|
|
|
res += std::string(tmp, n);
|
|
|
|
// strip the first "[laminar] executing.. line
|
|
|
|
return strchr(res.c_str(), '\n') + 1;
|
|
|
|
}
|
|
|
|
StringMap parseFromString(std::string content) {
|
|
|
|
char tmp[16] = "/tmp/lt.XXXXXX";
|
|
|
|
int fd = mkstemp(tmp);
|
|
|
|
write(fd, content.data(), content.size());
|
|
|
|
close(fd);
|
|
|
|
StringMap map = parseConfFile(tmp);
|
|
|
|
unlink(tmp);
|
|
|
|
return map;
|
|
|
|
}
|
|
|
|
|
2018-09-28 07:36:10 +00:00
|
|
|
std::shared_ptr<Node> node;
|
|
|
|
TempDir tmp;
|
2018-01-26 11:07:02 +00:00
|
|
|
class Run run;
|
2018-09-28 07:36:10 +00:00
|
|
|
|
|
|
|
void setRunLink(const char* path) {
|
|
|
|
tmp.fs->symlink(kj::Path{"cfg","jobs",run.name+".run"}, path, kj::WriteMode::CREATE|kj::WriteMode::CREATE_PARENT|kj::WriteMode::EXECUTABLE);
|
|
|
|
}
|
2018-01-26 11:07:02 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
TEST_F(RunTest, WorkingDirectory) {
|
2018-09-28 07:36:10 +00:00
|
|
|
setRunLink("/bin/pwd");
|
|
|
|
run.configure(1, node, *tmp.fs);
|
2018-01-26 11:07:02 +00:00
|
|
|
runAll();
|
2018-09-28 07:36:10 +00:00
|
|
|
std::string cwd{tmp.path.append(kj::Path{"run","foo","1"}).toString(true).cStr()};
|
|
|
|
EXPECT_EQ(cwd + "\n", readAllOutput());
|
2018-01-26 11:07:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(RunTest, SuccessStatus) {
|
2018-09-28 07:36:10 +00:00
|
|
|
setRunLink("/bin/true");
|
|
|
|
run.configure(1, node, *tmp.fs);
|
2018-01-26 11:07:02 +00:00
|
|
|
runAll();
|
|
|
|
EXPECT_EQ(RunState::SUCCESS, run.result);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(RunTest, FailedStatus) {
|
2018-09-28 07:36:10 +00:00
|
|
|
setRunLink("/bin/false");
|
|
|
|
run.configure(1, node, *tmp.fs);
|
2018-01-26 11:07:02 +00:00
|
|
|
runAll();
|
|
|
|
EXPECT_EQ(RunState::FAILED, run.result);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(RunTest, Environment) {
|
2018-09-28 07:36:10 +00:00
|
|
|
setRunLink("/usr/bin/env");
|
|
|
|
run.configure(1234, node, *tmp.fs);
|
2018-01-26 11:07:02 +00:00
|
|
|
runAll();
|
2018-09-28 07:36:10 +00:00
|
|
|
|
|
|
|
std::string ws{tmp.path.append(kj::Path{"run","foo","workspace"}).toString(true).cStr()};
|
|
|
|
std::string archive{tmp.path.append(kj::Path{"archive","foo","1234"}).toString(true).cStr()};
|
|
|
|
|
2018-01-26 11:07:02 +00:00
|
|
|
StringMap map = parseFromString(readAllOutput());
|
|
|
|
EXPECT_EQ("1234", map["RUN"]);
|
|
|
|
EXPECT_EQ("foo", map["JOB"]);
|
|
|
|
EXPECT_EQ("success", map["RESULT"]);
|
|
|
|
EXPECT_EQ("unknown", map["LAST_RESULT"]);
|
2018-09-28 07:36:10 +00:00
|
|
|
EXPECT_EQ(ws, map["WORKSPACE"]);
|
|
|
|
EXPECT_EQ(archive, map["ARCHIVE"]);
|
2018-01-26 11:07:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(RunTest, ParamsToEnv) {
|
2018-09-28 07:36:10 +00:00
|
|
|
setRunLink("/usr/bin/env");
|
2018-01-26 11:07:02 +00:00
|
|
|
run.params["foo"] = "bar";
|
2018-09-28 07:36:10 +00:00
|
|
|
run.configure(1, node, *tmp.fs);
|
2018-01-26 11:07:02 +00:00
|
|
|
runAll();
|
|
|
|
StringMap map = parseFromString(readAllOutput());
|
|
|
|
EXPECT_EQ("bar", map["foo"]);
|
|
|
|
}
|
2018-02-24 16:53:11 +00:00
|
|
|
|
|
|
|
TEST_F(RunTest, Abort) {
|
2018-09-28 07:36:10 +00:00
|
|
|
setRunLink("/usr/bin/yes");
|
|
|
|
run.configure(1, node, *tmp.fs);
|
2018-02-24 16:53:11 +00:00
|
|
|
run.step();
|
|
|
|
usleep(200); // TODO fix
|
2018-09-09 09:15:23 +00:00
|
|
|
run.abort(false);
|
2018-02-24 16:53:11 +00:00
|
|
|
wait();
|
|
|
|
EXPECT_EQ(RunState::ABORTED, run.result);
|
|
|
|
}
|
|
|
|
|