Merge pull request #445 from cyphar/conmon-split-stderr
conmon: split std{out,err} pipe for !terminal containers
This commit is contained in:
commit
a8b73e7583
6 changed files with 212 additions and 94 deletions
|
@ -2,7 +2,7 @@ src = $(wildcard *.c)
|
||||||
obj = $(src:.c=.o)
|
obj = $(src:.c=.o)
|
||||||
|
|
||||||
override LIBS += $(shell pkg-config --libs glib-2.0)
|
override LIBS += $(shell pkg-config --libs glib-2.0)
|
||||||
override CFLAGS += -Wall -Wextra $(shell pkg-config --cflags glib-2.0)
|
override CFLAGS += -std=c99 -Wall -Wextra $(shell pkg-config --cflags glib-2.0)
|
||||||
|
|
||||||
conmon: $(obj)
|
conmon: $(obj)
|
||||||
$(CC) -o $@ $^ $(CFLAGS) $(LIBS)
|
$(CC) -o $@ $^ $(CFLAGS) $(LIBS)
|
||||||
|
|
171
conmon/conmon.c
171
conmon/conmon.c
|
@ -122,21 +122,38 @@ int set_k8s_timestamp(char *buf, ssize_t buflen)
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* stdpipe_t represents one of the std pipes (or NONE). */
|
||||||
|
typedef enum {
|
||||||
|
NO_PIPE,
|
||||||
|
STDIN_PIPE, /* unused */
|
||||||
|
STDOUT_PIPE,
|
||||||
|
STDERR_PIPE,
|
||||||
|
} stdpipe_t;
|
||||||
|
|
||||||
|
const char *stdpipe_name(stdpipe_t pipe)
|
||||||
|
{
|
||||||
|
switch (pipe) {
|
||||||
|
case STDIN_PIPE:
|
||||||
|
return "stdin";
|
||||||
|
case STDOUT_PIPE:
|
||||||
|
return "stdout";
|
||||||
|
case STDERR_PIPE:
|
||||||
|
return "stderr";
|
||||||
|
default:
|
||||||
|
return "NONE";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* The CRI requires us to write logs with a (timestamp, stream, line) format
|
* The CRI requires us to write logs with a (timestamp, stream, line) format
|
||||||
* for every newline-separated line. write_k8s_log writes said format for every
|
* for every newline-separated line. write_k8s_log writes said format for every
|
||||||
* line in buf, and will partially write the final line of the log if buf is
|
* line in buf, and will partially write the final line of the log if buf is
|
||||||
* not terminated by a newline.
|
* not terminated by a newline.
|
||||||
*
|
|
||||||
* TODO: The above trick (partially writing the last line) won't work if we
|
|
||||||
* ever switch to having different streams for std{out,err}. Luckily at
|
|
||||||
* the moment that isn't an issue for us, but we need to fix this in the
|
|
||||||
* future.
|
|
||||||
*/
|
*/
|
||||||
int write_k8s_log(int fd, const char *stream_type, const char *buf, ssize_t buflen)
|
int write_k8s_log(int fd, stdpipe_t pipe, const char *buf, ssize_t buflen)
|
||||||
{
|
{
|
||||||
char tsbuf[TSBUFLEN];
|
char tsbuf[TSBUFLEN];
|
||||||
static bool trailing_line = false;
|
static stdpipe_t trailing_line = NO_PIPE;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Use the same timestamp for every line of the log in this buffer.
|
* Use the same timestamp for every line of the log in this buffer.
|
||||||
|
@ -159,10 +176,22 @@ int write_k8s_log(int fd, const char *stream_type, const char *buf, ssize_t bufl
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Write the (timestamp, stream) tuple if there isn't any trailing
|
* Write the (timestamp, stream) tuple if there isn't any trailing
|
||||||
* input from the previous line.
|
* output from the previous line (or if there is trailing output but
|
||||||
|
* the current buffer being printed is from a different pipe).
|
||||||
*/
|
*/
|
||||||
if (!trailing_line) {
|
if (trailing_line != pipe) {
|
||||||
if (dprintf(fd, "%s %s ", tsbuf, stream_type) < 0) {
|
/*
|
||||||
|
* If there was a trailing line from a different pipe, prepend a
|
||||||
|
* newline to split it properly. This technically breaks the flow
|
||||||
|
* of the previous line (adding a newline in the log where there
|
||||||
|
* wasn't one output) but without modifying the file in a
|
||||||
|
* non-append-only way there's not much we can do.
|
||||||
|
*/
|
||||||
|
char *leading = "";
|
||||||
|
if (trailing_line != NO_PIPE)
|
||||||
|
leading = "\n";
|
||||||
|
|
||||||
|
if (dprintf(fd, "%s%s %s ", leading, tsbuf, stdpipe_name(pipe)) < 0) {
|
||||||
nwarn("failed to write (timestamp, stream) to log");
|
nwarn("failed to write (timestamp, stream) to log");
|
||||||
goto next;
|
goto next;
|
||||||
}
|
}
|
||||||
|
@ -175,7 +204,7 @@ int write_k8s_log(int fd, const char *stream_type, const char *buf, ssize_t bufl
|
||||||
}
|
}
|
||||||
|
|
||||||
/* If we did not output a full line, then we are a trailing_line. */
|
/* If we did not output a full line, then we are a trailing_line. */
|
||||||
trailing_line = *line_end != '\n';
|
trailing_line = (*line_end == '\n') ? NO_PIPE : pipe;
|
||||||
|
|
||||||
next:
|
next:
|
||||||
/* Update the head of the buffer remaining to output. */
|
/* Update the head of the buffer remaining to output. */
|
||||||
|
@ -197,10 +226,13 @@ int main(int argc, char *argv[])
|
||||||
int status;
|
int status;
|
||||||
pid_t pid, create_pid;
|
pid_t pid, create_pid;
|
||||||
_cleanup_close_ int logfd = -1;
|
_cleanup_close_ int logfd = -1;
|
||||||
_cleanup_close_ int mfd = -1;
|
_cleanup_close_ int masterfd_stdout = -1;
|
||||||
|
_cleanup_close_ int masterfd_stderr = -1;
|
||||||
_cleanup_close_ int epfd = -1;
|
_cleanup_close_ int epfd = -1;
|
||||||
_cleanup_close_ int csfd = -1;
|
_cleanup_close_ int csfd = -1;
|
||||||
int runtime_mfd = -1;
|
/* Used for !terminal cases. */
|
||||||
|
int slavefd_stdout = -1;
|
||||||
|
int slavefd_stderr = -1;
|
||||||
char csname[PATH_MAX] = "/tmp/conmon-term.XXXXXXXX";
|
char csname[PATH_MAX] = "/tmp/conmon-term.XXXXXXXX";
|
||||||
char buf[BUF_SIZE];
|
char buf[BUF_SIZE];
|
||||||
int num_read;
|
int num_read;
|
||||||
|
@ -209,6 +241,7 @@ int main(int argc, char *argv[])
|
||||||
int sync_pipe_fd = -1;
|
int sync_pipe_fd = -1;
|
||||||
char *sync_pipe, *endptr;
|
char *sync_pipe, *endptr;
|
||||||
int len;
|
int len;
|
||||||
|
int num_stdio_fds = 0;
|
||||||
GError *error = NULL;
|
GError *error = NULL;
|
||||||
GOptionContext *context;
|
GOptionContext *context;
|
||||||
_cleanup_gstring_ GString *cmd = NULL;
|
_cleanup_gstring_ GString *cmd = NULL;
|
||||||
|
@ -302,7 +335,7 @@ int main(int argc, char *argv[])
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Create a "fake" master fd so that we can use the same epoll code in
|
* Create a "fake" master fd so that we can use the same epoll code in
|
||||||
* both cases. The runtime_mfd will be closed after we dup over
|
* both cases. The slavefd_*s will be closed after we dup over
|
||||||
* everything.
|
* everything.
|
||||||
*
|
*
|
||||||
* We use pipes here because open(/dev/std{out,err}) will fail if we
|
* We use pipes here because open(/dev/std{out,err}) will fail if we
|
||||||
|
@ -310,10 +343,16 @@ int main(int argc, char *argv[])
|
||||||
* pty pair in the host).
|
* pty pair in the host).
|
||||||
*/
|
*/
|
||||||
if (pipe(fds) < 0)
|
if (pipe(fds) < 0)
|
||||||
pexit("Failed to create runtime_mfd pipes");
|
pexit("Failed to create !terminal stdout pipe");
|
||||||
|
|
||||||
mfd = fds[0];
|
masterfd_stdout = fds[0];
|
||||||
runtime_mfd = fds[1];
|
slavefd_stdout = fds[1];
|
||||||
|
|
||||||
|
if (pipe(fds) < 0)
|
||||||
|
pexit("Failed to create !terminal stderr pipe");
|
||||||
|
|
||||||
|
masterfd_stderr = fds[0];
|
||||||
|
slavefd_stderr = fds[1];
|
||||||
}
|
}
|
||||||
|
|
||||||
cmd = g_string_new(runtime_path);
|
cmd = g_string_new(runtime_path);
|
||||||
|
@ -364,12 +403,12 @@ int main(int argc, char *argv[])
|
||||||
|
|
||||||
/* We only need to touch the stdio if we have terminal=false. */
|
/* We only need to touch the stdio if we have terminal=false. */
|
||||||
/* FIXME: This results in us not outputting runc error messages to ocid's log. */
|
/* FIXME: This results in us not outputting runc error messages to ocid's log. */
|
||||||
if (runtime_mfd >= 0) {
|
if (slavefd_stdout >= 0) {
|
||||||
if (dup2(runtime_mfd, STDIN_FILENO) < 0)
|
if (dup2(slavefd_stdout, STDOUT_FILENO) < 0)
|
||||||
pexit("Failed to dup over stdin");
|
|
||||||
if (dup2(runtime_mfd, STDOUT_FILENO) < 0)
|
|
||||||
pexit("Failed to dup over stdout");
|
pexit("Failed to dup over stdout");
|
||||||
if (dup2(runtime_mfd, STDERR_FILENO) < 0)
|
}
|
||||||
|
if (slavefd_stderr >= 0) {
|
||||||
|
if (dup2(slavefd_stderr, STDERR_FILENO) < 0)
|
||||||
pexit("Failed to dup over stderr");
|
pexit("Failed to dup over stderr");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -379,7 +418,8 @@ int main(int argc, char *argv[])
|
||||||
}
|
}
|
||||||
|
|
||||||
/* The runtime has that fd now. We don't need to touch it anymore. */
|
/* The runtime has that fd now. We don't need to touch it anymore. */
|
||||||
close(runtime_mfd);
|
close(slavefd_stdout);
|
||||||
|
close(slavefd_stderr);
|
||||||
|
|
||||||
/* Get the console fd. */
|
/* Get the console fd. */
|
||||||
/*
|
/*
|
||||||
|
@ -407,7 +447,10 @@ int main(int argc, char *argv[])
|
||||||
ninfo("console = {.name = '%s'; .fd = %d}", console.name, console.fd);
|
ninfo("console = {.name = '%s'; .fd = %d}", console.name, console.fd);
|
||||||
free(console.name);
|
free(console.name);
|
||||||
|
|
||||||
mfd = console.fd;
|
/* We only have a single fd for both pipes, so we just treat it as
|
||||||
|
* stdout. stderr is ignored. */
|
||||||
|
masterfd_stdout = console.fd;
|
||||||
|
masterfd_stderr = -1;
|
||||||
|
|
||||||
/* Clean up everything */
|
/* Clean up everything */
|
||||||
close(connfd);
|
close(connfd);
|
||||||
|
@ -454,71 +497,51 @@ int main(int argc, char *argv[])
|
||||||
if (epfd < 0)
|
if (epfd < 0)
|
||||||
pexit("epoll_create");
|
pexit("epoll_create");
|
||||||
ev.events = EPOLLIN;
|
ev.events = EPOLLIN;
|
||||||
/*
|
if (masterfd_stdout >= 0) {
|
||||||
ev.data.fd = STDIN_FILENO;
|
ev.data.fd = masterfd_stdout;
|
||||||
if (epoll_ctl(epfd, EPOLL_CTL_ADD, STDIN_FILENO, &ev) < 0) {
|
if (epoll_ctl(epfd, EPOLL_CTL_ADD, ev.data.fd, &ev) < 0)
|
||||||
pexit("Failed to add stdin to epoll");
|
pexit("Failed to add console masterfd_stdout to epoll");
|
||||||
|
num_stdio_fds++;
|
||||||
}
|
}
|
||||||
*/
|
if (masterfd_stderr >= 0) {
|
||||||
ev.data.fd = mfd;
|
ev.data.fd = masterfd_stderr;
|
||||||
if (epoll_ctl(epfd, EPOLL_CTL_ADD, mfd, &ev) < 0) {
|
if (epoll_ctl(epfd, EPOLL_CTL_ADD, ev.data.fd, &ev) < 0)
|
||||||
pexit("Failed to add console master fd to epoll");
|
pexit("Failed to add console masterfd_stderr to epoll");
|
||||||
|
num_stdio_fds++;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/* Log all of the container's output. */
|
||||||
* Log all of the container's output and pipe STDIN into it. Currently
|
while (num_stdio_fds > 0) {
|
||||||
* nothing using the STDIN setup (which makes its inclusion here a bit
|
|
||||||
* questionable but we need to rewrite this code soon anyway TODO).
|
|
||||||
*/
|
|
||||||
while (true) {
|
|
||||||
int ready = epoll_wait(epfd, evlist, MAX_EVENTS, -1);
|
int ready = epoll_wait(epfd, evlist, MAX_EVENTS, -1);
|
||||||
int i = 0;
|
if (ready < 0)
|
||||||
for (i = 0; i < ready; i++) {
|
continue;
|
||||||
|
|
||||||
|
for (int i = 0; i < ready; i++) {
|
||||||
if (evlist[i].events & EPOLLIN) {
|
if (evlist[i].events & EPOLLIN) {
|
||||||
if (evlist[i].data.fd == STDIN_FILENO) {
|
int masterfd = evlist[i].data.fd;
|
||||||
/*
|
stdpipe_t pipe;
|
||||||
* TODO: We need to replace STDIN_FILENO with something
|
if (masterfd == masterfd_stdout)
|
||||||
* more sophisticated so that attach actually works
|
pipe = STDOUT_PIPE;
|
||||||
* properly.
|
else if (masterfd == masterfd_stderr)
|
||||||
*/
|
pipe = STDERR_PIPE;
|
||||||
num_read = read(STDIN_FILENO, buf, BUF_SIZE);
|
else {
|
||||||
|
nwarn("unknown pipe fd");
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
num_read = read(masterfd, buf, BUF_SIZE);
|
||||||
if (num_read <= 0)
|
if (num_read <= 0)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
if (write(mfd, buf, num_read) != num_read) {
|
if (write_k8s_log(logfd, pipe, buf, num_read) < 0) {
|
||||||
nwarn("partial/failed write (masterFd)");
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
} else if (evlist[i].data.fd == mfd) {
|
|
||||||
num_read = read(mfd, buf, BUF_SIZE);
|
|
||||||
if (num_read <= 0)
|
|
||||||
goto out;
|
|
||||||
|
|
||||||
if (exec) {
|
|
||||||
/*
|
|
||||||
* If we're in ExecSync we don't output the k8s log
|
|
||||||
* format. TODO(cyphar): This code really should be
|
|
||||||
* rewritten so that we have a single conmon per
|
|
||||||
* container and the conmon is logging the main
|
|
||||||
* container process as a separate piece of logic to
|
|
||||||
* the streaming to Exec[Sync] clients.
|
|
||||||
*/
|
|
||||||
if (write(logfd, buf, num_read) < 0) {
|
|
||||||
nwarn("write failed");
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (write_k8s_log(logfd, "stdout", buf, num_read) < 0) {
|
|
||||||
nwarn("write_k8s_log failed");
|
nwarn("write_k8s_log failed");
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
||||||
} else if (evlist[i].events & (EPOLLHUP | EPOLLERR)) {
|
} else if (evlist[i].events & (EPOLLHUP | EPOLLERR)) {
|
||||||
printf("closing fd %d\n", evlist[i].data.fd);
|
printf("closing fd %d\n", evlist[i].data.fd);
|
||||||
if (close(evlist[i].data.fd) < 0)
|
if (close(evlist[i].data.fd) < 0)
|
||||||
pexit("close");
|
pexit("close");
|
||||||
goto out;
|
num_stdio_fds--;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
44
oci/oci.go
44
oci/oci.go
|
@ -236,6 +236,42 @@ func prepareExec() (pidFile, parentPipe, childPipe *os.File, err error) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func parseLog(log []byte) (stdout, stderr []byte) {
|
||||||
|
// Split the log on newlines, which is what separates entries.
|
||||||
|
lines := bytes.SplitAfter(log, []byte{'\n'})
|
||||||
|
for _, line := range lines {
|
||||||
|
// Ignore empty lines.
|
||||||
|
if len(line) == 0 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
// The format of log lines is "DATE pipe REST".
|
||||||
|
parts := bytes.SplitN(line, []byte{' '}, 3)
|
||||||
|
if len(parts) < 3 {
|
||||||
|
// Ignore the line if it's formatted incorrectly, but complain
|
||||||
|
// about it so it can be debugged.
|
||||||
|
logrus.Warnf("hit invalid log format: %q", string(line))
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
pipe := string(parts[1])
|
||||||
|
content := parts[2]
|
||||||
|
|
||||||
|
switch pipe {
|
||||||
|
case "stdout":
|
||||||
|
stdout = append(stdout, content...)
|
||||||
|
case "stderr":
|
||||||
|
stderr = append(stderr, content...)
|
||||||
|
default:
|
||||||
|
// Complain about unknown pipes.
|
||||||
|
logrus.Warnf("hit invalid log format [unknown pipe %s]: %q", pipe, string(line))
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return stdout, stderr
|
||||||
|
}
|
||||||
|
|
||||||
// ExecSync execs a command in a container and returns it's stdout, stderr and return code.
|
// ExecSync execs a command in a container and returns it's stdout, stderr and return code.
|
||||||
func (r *Runtime) ExecSync(c *Container, command []string, timeout int64) (resp *ExecSyncResponse, err error) {
|
func (r *Runtime) ExecSync(c *Container, command []string, timeout int64) (resp *ExecSyncResponse, err error) {
|
||||||
pidFile, parentPipe, childPipe, err := prepareExec()
|
pidFile, parentPipe, childPipe, err := prepareExec()
|
||||||
|
@ -386,7 +422,7 @@ func (r *Runtime) ExecSync(c *Container, command []string, timeout int64) (resp
|
||||||
// XXX: Currently runC dups the same console over both stdout and stderr,
|
// XXX: Currently runC dups the same console over both stdout and stderr,
|
||||||
// so we can't differentiate between the two.
|
// so we can't differentiate between the two.
|
||||||
|
|
||||||
outputBytes, err := ioutil.ReadFile(logPath)
|
logBytes, err := ioutil.ReadFile(logPath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, ExecSyncError{
|
return nil, ExecSyncError{
|
||||||
Stdout: stdoutBuf,
|
Stdout: stdoutBuf,
|
||||||
|
@ -396,9 +432,11 @@ func (r *Runtime) ExecSync(c *Container, command []string, timeout int64) (resp
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// We have to parse the log output into {stdout, stderr} buffers.
|
||||||
|
stdoutBytes, stderrBytes := parseLog(logBytes)
|
||||||
return &ExecSyncResponse{
|
return &ExecSyncResponse{
|
||||||
Stdout: outputBytes,
|
Stdout: stdoutBytes,
|
||||||
Stderr: nil,
|
Stderr: stderrBytes,
|
||||||
ExitCode: ec.ExitCode,
|
ExitCode: ec.ExitCode,
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,7 +2,7 @@ src = $(wildcard *.c)
|
||||||
obj = $(src:.c=.o)
|
obj = $(src:.c=.o)
|
||||||
|
|
||||||
override LIBS +=
|
override LIBS +=
|
||||||
override CFLAGS += -Os -Wall -Wextra -static
|
override CFLAGS += -std=c99 -Os -Wall -Wextra -static
|
||||||
|
|
||||||
pause: $(obj)
|
pause: $(obj)
|
||||||
$(CC) -o $@ $^ $(CFLAGS) $(LIBS)
|
$(CC) -o $@ $^ $(CFLAGS) $(LIBS)
|
||||||
|
|
|
@ -112,7 +112,7 @@ function teardown() {
|
||||||
# Create a new container.
|
# Create a new container.
|
||||||
newconfig=$(mktemp --tmpdir ocid-config.XXXXXX.json)
|
newconfig=$(mktemp --tmpdir ocid-config.XXXXXX.json)
|
||||||
cp "$TESTDATA"/container_config_logging.json "$newconfig"
|
cp "$TESTDATA"/container_config_logging.json "$newconfig"
|
||||||
sed -i 's|"%echooutput%"|"here", "is", "some", "output"|' "$newconfig"
|
sed -i 's|"%shellcommand%"|"echo here is some output \&\& echo and some from stderr >\&2"|' "$newconfig"
|
||||||
run ocic ctr create --config "$newconfig" --pod "$pod_id"
|
run ocic ctr create --config "$newconfig" --pod "$pod_id"
|
||||||
echo "$output"
|
echo "$output"
|
||||||
[ "$status" -eq 0 ]
|
[ "$status" -eq 0 ]
|
||||||
|
@ -130,10 +130,11 @@ function teardown() {
|
||||||
[ "$status" -eq 0 ]
|
[ "$status" -eq 0 ]
|
||||||
|
|
||||||
# Check that the output is what we expect.
|
# Check that the output is what we expect.
|
||||||
run cat "$DEFAULT_LOG_PATH/$pod_id/$ctr_id.log"
|
logpath="$DEFAULT_LOG_PATH/$pod_id/$ctr_id.log"
|
||||||
echo "$DEFAULT_LOG_PATH/$pod_id/$ctr_id.log ::" "$output"
|
[ -f "$logpath" ]
|
||||||
[ "$status" -eq 0 ]
|
echo "$logpath :: $(cat "$logpath")"
|
||||||
[[ "$output" == *"here is some output" ]]
|
grep -E "^[^\n]+ stdout here is some output$" "$logpath"
|
||||||
|
grep -E "^[^\n]+ stderr and some from stderr$" "$logpath"
|
||||||
|
|
||||||
run ocic pod stop --id "$pod_id"
|
run ocic pod stop --id "$pod_id"
|
||||||
echo "$output"
|
echo "$output"
|
||||||
|
@ -160,7 +161,7 @@ function teardown() {
|
||||||
# Create a new container.
|
# Create a new container.
|
||||||
newconfig=$(mktemp --tmpdir ocid-config.XXXXXX.json)
|
newconfig=$(mktemp --tmpdir ocid-config.XXXXXX.json)
|
||||||
cp "$TESTDATA"/container_config_logging.json "$newconfig"
|
cp "$TESTDATA"/container_config_logging.json "$newconfig"
|
||||||
sed -i 's|"%echooutput%"|"here", "is", "some", "output"|' "$newconfig"
|
sed -i 's|"%shellcommand%"|"echo here is some output"|' "$newconfig"
|
||||||
sed -i 's|"tty": false,|"tty": true,|' "$newconfig"
|
sed -i 's|"tty": false,|"tty": true,|' "$newconfig"
|
||||||
run ocic ctr create --config "$newconfig" --pod "$pod_id"
|
run ocic ctr create --config "$newconfig" --pod "$pod_id"
|
||||||
echo "$output"
|
echo "$output"
|
||||||
|
@ -179,10 +180,10 @@ function teardown() {
|
||||||
[ "$status" -eq 0 ]
|
[ "$status" -eq 0 ]
|
||||||
|
|
||||||
# Check that the output is what we expect.
|
# Check that the output is what we expect.
|
||||||
run cat "$DEFAULT_LOG_PATH/$pod_id/$ctr_id.log"
|
logpath="$DEFAULT_LOG_PATH/$pod_id/$ctr_id.log"
|
||||||
echo "$DEFAULT_LOG_PATH/$pod_id/$ctr_id.log ::" "$output"
|
[ -f "$logpath" ]
|
||||||
[ "$status" -eq 0 ]
|
echo "$logpath :: $(cat "$logpath")"
|
||||||
[[ "$output" == *"here is some output" ]]
|
grep -E "^[^\n]+ stdout here is some output$" "$logpath"
|
||||||
|
|
||||||
run ocic pod stop --id "$pod_id"
|
run ocic pod stop --id "$pod_id"
|
||||||
echo "$output"
|
echo "$output"
|
||||||
|
@ -450,6 +451,62 @@ function teardown() {
|
||||||
stop_ocid
|
stop_ocid
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@test "ctr execsync exit code" {
|
||||||
|
start_ocid
|
||||||
|
run ocic pod run --config "$TESTDATA"/sandbox_config.json
|
||||||
|
echo "$output"
|
||||||
|
[ "$status" -eq 0 ]
|
||||||
|
pod_id="$output"
|
||||||
|
run ocic ctr create --config "$TESTDATA"/container_redis.json --pod "$pod_id"
|
||||||
|
echo "$output"
|
||||||
|
[ "$status" -eq 0 ]
|
||||||
|
ctr_id="$output"
|
||||||
|
run ocic ctr start --id "$ctr_id"
|
||||||
|
echo "$output"
|
||||||
|
[ "$status" -eq 0 ]
|
||||||
|
run ocic ctr execsync --id "$ctr_id" false
|
||||||
|
echo "$output"
|
||||||
|
[ "$status" -eq 0 ]
|
||||||
|
[[ "$output" =~ "Exit code: 1" ]]
|
||||||
|
cleanup_ctrs
|
||||||
|
cleanup_pods
|
||||||
|
stop_ocid
|
||||||
|
}
|
||||||
|
|
||||||
|
@test "ctr execsync std{out,err}" {
|
||||||
|
start_ocid
|
||||||
|
run ocic pod run --config "$TESTDATA"/sandbox_config.json
|
||||||
|
echo "$output"
|
||||||
|
[ "$status" -eq 0 ]
|
||||||
|
pod_id="$output"
|
||||||
|
run ocic ctr create --config "$TESTDATA"/container_redis.json --pod "$pod_id"
|
||||||
|
echo "$output"
|
||||||
|
[ "$status" -eq 0 ]
|
||||||
|
ctr_id="$output"
|
||||||
|
run ocic ctr start --id "$ctr_id"
|
||||||
|
echo "$output"
|
||||||
|
[ "$status" -eq 0 ]
|
||||||
|
run ocic ctr execsync --id "$ctr_id" "echo hello0 stdout"
|
||||||
|
echo "$output"
|
||||||
|
[ "$status" -eq 0 ]
|
||||||
|
[[ "$output" == *"$(printf "Stdout:\nhello0 stdout")"* ]]
|
||||||
|
run ocic ctr execsync --id "$ctr_id" "echo hello1 stderr >&2"
|
||||||
|
echo "$output"
|
||||||
|
[ "$status" -eq 0 ]
|
||||||
|
[[ "$output" == *"$(printf "Stderr:\nhello1 stderr")"* ]]
|
||||||
|
run ocic ctr execsync --id "$ctr_id" "echo hello2 stderr >&2; echo hello3 stdout"
|
||||||
|
echo "$output"
|
||||||
|
[ "$status" -eq 0 ]
|
||||||
|
[[ "$output" == *"$(printf "Stderr:\nhello2 stderr")"* ]]
|
||||||
|
[[ "$output" == *"$(printf "Stdout:\nhello3 stdout")"* ]]
|
||||||
|
run ocic pod remove --id "$pod_id"
|
||||||
|
echo "$output"
|
||||||
|
[ "$status" -eq 0 ]
|
||||||
|
cleanup_ctrs
|
||||||
|
cleanup_pods
|
||||||
|
stop_ocid
|
||||||
|
}
|
||||||
|
|
||||||
@test "ctr stop idempotent" {
|
@test "ctr stop idempotent" {
|
||||||
start_ocid
|
start_ocid
|
||||||
run ocic pod run --config "$TESTDATA"/sandbox_config.json
|
run ocic pod run --config "$TESTDATA"/sandbox_config.json
|
||||||
|
|
4
test/testdata/container_config_logging.json
vendored
4
test/testdata/container_config_logging.json
vendored
|
@ -7,10 +7,10 @@
|
||||||
"image": "docker://busybox:latest"
|
"image": "docker://busybox:latest"
|
||||||
},
|
},
|
||||||
"command": [
|
"command": [
|
||||||
"/bin/echo"
|
"/bin/sh", "-c"
|
||||||
],
|
],
|
||||||
"args": [
|
"args": [
|
||||||
"%echooutput%"
|
"%shellcommand%"
|
||||||
],
|
],
|
||||||
"working_dir": "/",
|
"working_dir": "/",
|
||||||
"envs": [
|
"envs": [
|
||||||
|
|
Loading…
Reference in a new issue