Skip to content
This repository has been archived by the owner on May 12, 2021. It is now read-only.

Commit

Permalink
shim: use exec id instead of pid
Browse files Browse the repository at this point in the history
As discssed in kata-containers/agent#72, we'll
replace guest pid with runtime generated exec id to uniquely identify
a process.

Fixes: #18

Signed-off-by: Peng Tao <bergwolf@gmail.com>
  • Loading branch information
bergwolf committed Jan 11, 2018
1 parent 2e214cb commit e791731
Show file tree
Hide file tree
Showing 19 changed files with 1,041 additions and 662 deletions.
5 changes: 3 additions & 2 deletions Gopkg.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

2 changes: 1 addition & 1 deletion Gopkg.toml
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,7 @@
go-tests = true

[[constraint]]
revision = "9b50dbac719a0def765e3f4ad11eaa44d77b17e5"
revision = "8f22514ae5790f3b6953cf93692e663fa29469e3"
name = "github.com/kata-containers/agent"

[[constraint]]
Expand Down
18 changes: 10 additions & 8 deletions agent_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -106,22 +106,23 @@ func newTestSpec() *pb.Spec {
}
}

func (t *testAgent) addContainer(containerId string) (uint32, error) {
func (t *testAgent) addContainer(containerId, execId string) error {
_, err := t.client.CreateContainer(t.ctx, &pb.CreateContainerRequest{
ContainerId: containerId,
ExecId: execId,
StringUser: &pb.StringUser{Uid: "root", Gid: "root"},
OCI: newTestSpec(),
})
if err != nil {
return 0, fmt.Errorf("failed to create new container: %s", err)
return fmt.Errorf("failed to create new container: %s", err)
}

resp, err := t.client.StartContainer(t.ctx, &pb.StartContainerRequest{ContainerId: containerId})
_, err = t.client.StartContainer(t.ctx, &pb.StartContainerRequest{ContainerId: containerId})
if err != nil {
return 0, fmt.Errorf("failed to create new container: %s", err)
return fmt.Errorf("failed to create new container: %s", err)
}

return resp.PID, nil
return nil
}

func TestNewShimAgent(t *testing.T) {
Expand Down Expand Up @@ -152,9 +153,10 @@ func TestAddContainer(t *testing.T) {
agent := testSetup(t)
defer testTearDown(agent)

id := "foobar"
_, err := agent.addContainer(id)
contId := "testContainer"
execId := "testExec"
err := agent.addContainer(contId, execId)
assert.Nil(t, err, "%s", err)
_, err = agent.addContainer(id)
err = agent.addContainer(contId, execId)
assert.NotNil(t, err, "unexpected success when adding duplicated container")
}
12 changes: 6 additions & 6 deletions main.go
Original file line number Diff line number Diff line change
Expand Up @@ -50,7 +50,7 @@ func main() {
logLevel string
agentAddr string
container string
pid uint
execId string
proxyExitCode bool
showVersion bool
)
Expand All @@ -60,7 +60,7 @@ func main() {
flag.StringVar(&agentAddr, "agent", "", "agent gRPC socket endpoint")

flag.StringVar(&container, "container", "", "container id for the shim")
flag.UintVar(&pid, "pid", 0, "process id for the shim")
flag.StringVar(&execId, "exec-id", "", "process id for the shim")
flag.BoolVar(&proxyExitCode, "proxy-exit-code", true, "proxy exit code of the process")

flag.Parse()
Expand All @@ -70,8 +70,8 @@ func main() {
os.Exit(0)
}

if agentAddr == "" || container == "" || pid == 0 {
shimLog.WithField("agentAddr", agentAddr).WithField("container", container).WithField("pid", pid).Error("container ID, process ID and agent socket endpoint must be set")
if agentAddr == "" || container == "" || execId == "" {
shimLog.WithField("agentAddr", agentAddr).WithField("container", container).WithField("exec-id", execId).Error("container ID, exec ID and agent socket endpoint must be set")
os.Exit(exitFailure)
}

Expand All @@ -81,7 +81,7 @@ func main() {
os.Exit(exitFailure)
}

shim, err := newShim(agentAddr, container, uint32(pid))
shim, err := newShim(agentAddr, container, execId)
if err != nil {
shimLog.WithError(err).Error("failed to create new shim")
os.Exit(exitFailure)
Expand All @@ -108,7 +108,7 @@ func main() {
// wait until exit
exitcode, err := shim.wait()
if err != nil {
shimLog.WithError(err).WithField("pid", pid).Error("failed waiting for process")
shimLog.WithError(err).WithField("exec-id", execId).Error("failed waiting for process")
os.Exit(exitFailure)
} else if proxyExitCode {
shimLog.WithField("exitcode", exitcode).Info("using shim to proxy exit code")
Expand Down
24 changes: 12 additions & 12 deletions pipe.go
Original file line number Diff line number Diff line change
Expand Up @@ -19,13 +19,13 @@ type inPipe struct {
ctx context.Context
agent *shimAgent
containerId string
pid uint32
execId string
}

func (p *inPipe) Write(data []byte) (n int, err error) {
resp, err := p.agent.WriteStdin(p.ctx, &pb.WriteStreamRequest{
ContainerId: p.containerId,
PID: p.pid,
ExecId: p.execId,
Data: data})
if err != nil {
return 0, err
Expand All @@ -37,17 +37,17 @@ func (p *inPipe) Write(data []byte) (n int, err error) {
func (p *inPipe) Close() error {
_, err := p.agent.CloseStdin(p.ctx, &pb.CloseStdinRequest{
ContainerId: p.containerId,
PID: p.pid})
ExecId: p.execId})

return err
}

type readFn func(context.Context, *pb.ReadStreamRequest, ...grpc.CallOption) (*pb.ReadStreamResponse, error)

func pipeRead(ctx context.Context, containerId string, pid uint32, data []byte, read readFn) (n int, err error) {
func pipeRead(ctx context.Context, containerId, execId string, data []byte, read readFn) (n int, err error) {
resp, err := read(ctx, &pb.ReadStreamRequest{
ContainerId: containerId,
PID: pid,
ExecId: execId,
Len: uint32(len(data))})
if err == nil {
copy(data, resp.Data)
Expand All @@ -65,25 +65,25 @@ type outPipe struct {
ctx context.Context
agent *shimAgent
containerId string
pid uint32
execId string
}

func (p *outPipe) Read(data []byte) (n int, err error) {
return pipeRead(p.ctx, p.containerId, p.pid, data, p.agent.ReadStdout)
return pipeRead(p.ctx, p.containerId, p.execId, data, p.agent.ReadStdout)
}

type errPipe struct {
ctx context.Context
agent *shimAgent
containerId string
pid uint32
execId string
}

func (p *errPipe) Read(data []byte) (n int, err error) {
return pipeRead(p.ctx, p.containerId, p.pid, data, p.agent.ReadStderr)
return pipeRead(p.ctx, p.containerId, p.execId, data, p.agent.ReadStderr)
}

func shimStdioPipe(ctx context.Context, agent *shimAgent, containerId string, pid uint32) (io.WriteCloser, io.Reader, io.Reader) {
return &inPipe{ctx: ctx, agent: agent, containerId: containerId, pid: pid},
&outPipe{ctx: ctx, agent: agent, containerId: containerId, pid: pid}, &errPipe{ctx: ctx, agent: agent, containerId: containerId, pid: pid}
func shimStdioPipe(ctx context.Context, agent *shimAgent, containerId, execId string) (io.WriteCloser, io.Reader, io.Reader) {
return &inPipe{ctx: ctx, agent: agent, containerId: containerId, execId: execId},
&outPipe{ctx: ctx, agent: agent, containerId: containerId, execId: execId}, &errPipe{ctx: ctx, agent: agent, containerId: containerId, execId: execId}
}
7 changes: 4 additions & 3 deletions pipe_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -16,10 +16,11 @@ func TestPipe(t *testing.T) {
defer testTearDown(agent)

containerId := "testContainer"
pid, err := agent.addContainer(containerId)
execId := "testExec"
err := agent.addContainer(containerId, execId)
assert.Nil(t, err, "failed to add new container: %s", err)

inPipe, outPipe, errPipe := shimStdioPipe(agent.ctx, agent.client, containerId, pid)
inPipe, outPipe, errPipe := shimStdioPipe(agent.ctx, agent.client, containerId, execId)

buf := []byte("foobar")
size, err := inPipe.Write(buf[:])
Expand All @@ -35,7 +36,7 @@ func TestPipe(t *testing.T) {
assert.Equal(t, size, 0, "unmatched write stdin pipe len %d:%d", 0, size)

// wrong process
inPipe, outPipe, errPipe = shimStdioPipe(agent.ctx, agent.client, containerId, pid+100)
inPipe, outPipe, errPipe = shimStdioPipe(agent.ctx, agent.client, containerId, execId+"foobar")
_, err = inPipe.Write(buf[:])
assert.NotNil(t, err, "Unexpected success writing stdin pipe")

Expand Down
20 changes: 10 additions & 10 deletions shim.go
Original file line number Diff line number Diff line change
Expand Up @@ -32,33 +32,33 @@ var sigIgnored = map[syscall.Signal]bool{

type shim struct {
containerId string
pid uint32
execId string

ctx context.Context
agent *shimAgent
}

func newShim(addr, containerId string, pid uint32) (*shim, error) {
func newShim(addr, containerId, execId string) (*shim, error) {
if agent, err := newShimAgent(addr); err != nil {
return nil, err
} else {
return &shim{containerId: containerId,
pid: pid,
ctx: context.Background(),
agent: agent}, nil
execId: execId,
ctx: context.Background(),
agent: agent}, nil
}
}

func (s *shim) proxyStdio(wg *sync.WaitGroup) {
// don't wait the copying of the stdin, because `io.Copy(inPipe, os.Stdin)`
// can't terminate when no input. todo: find a better way.
wg.Add(2)
inPipe, outPipe, errPipe := shimStdioPipe(s.ctx, s.agent, s.containerId, s.pid)
inPipe, outPipe, errPipe := shimStdioPipe(s.ctx, s.agent, s.containerId, s.execId)
go func() {
_, err1 := io.Copy(inPipe, os.Stdin)
_, err2 := s.agent.CloseStdin(s.ctx, &pb.CloseStdinRequest{
ContainerId: s.containerId,
PID: s.pid})
ExecId: s.execId})
if err1 != nil {
shimLog.WithError(err1).Warn("copy stdin failed")
}
Expand Down Expand Up @@ -101,7 +101,7 @@ func (s *shim) forwardAllSignals() chan os.Signal {
// forward this signal to container
_, err := s.agent.SignalProcess(s.ctx, &pb.SignalProcessRequest{
ContainerId: s.containerId,
PID: s.pid,
ExecId: s.execId,
Signal: uint32(sysSig)})
if err != nil {
shimLog.WithError(err).WithField("signal", sig.String()).Error("forward signal failed")
Expand All @@ -120,7 +120,7 @@ func (s *shim) resizeTty(fromTty *os.File) error {

_, err = s.agent.TtyWinResize(s.ctx, &pb.TtyWinResizeRequest{
ContainerId: s.containerId,
PID: s.pid,
ExecId: s.execId,
Row: uint32(ws.Height),
Column: uint32(ws.Width)})
if err != nil {
Expand All @@ -144,7 +144,7 @@ func (s *shim) monitorTtySize(tty *os.File) {
func (s *shim) wait() (int32, error) {
resp, err := s.agent.WaitProcess(s.ctx, &pb.WaitProcessRequest{
ContainerId: s.containerId,
PID: s.pid})
ExecId: s.execId})
if err != nil {
return 0, err
}
Expand Down
16 changes: 9 additions & 7 deletions shim_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -18,27 +18,29 @@ func TestNewShim(t *testing.T) {
agent := testSetup(t)
defer testTearDown(agent)

id := "foobar"
pid, err := agent.addContainer(id)
contId := "foobarContainer"
execId := "testExec"
err := agent.addContainer(contId, execId)
assert.Nil(t, err, "%s", err)

shim, err := newShim(mockSockAddr, id, pid)
shim, err := newShim(mockSockAddr, contId, execId)
assert.Nil(t, err, "%s", err)
defer shim.agent.Close()

_, err = newShim(badMockAddr, id, pid)
_, err = newShim(badMockAddr, contId, execId)
assert.NotNil(t, err, "New shim with wrong socket address should fail")
}

func TestShimOps(t *testing.T) {
agent := testSetup(t)
defer testTearDown(agent)

id := "foobar"
pid, err := agent.addContainer(id)
contId := "foobarContainer"
execId := "testExec"
err := agent.addContainer(contId, execId)
assert.Nil(t, err, "%s", err)

shim, err := newShim(mockSockAddr, id, pid)
shim, err := newShim(mockSockAddr, contId, execId)
assert.Nil(t, err, "%s", err)
defer shim.agent.Close()

Expand Down
3 changes: 3 additions & 0 deletions vendor/github.com/kata-containers/agent/.travis.yml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

6 changes: 3 additions & 3 deletions vendor/github.com/kata-containers/agent/Gopkg.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

Loading

0 comments on commit e791731

Please sign in to comment.