Refactor shim terminal and io handling
This also finishes the service implementation of the shim behind GRPC Signed-off-by: Michael Crosby <crosbymichael@gmail.com>
This commit is contained in:
		
							parent
							
								
									6e9e0a895a
								
							
						
					
					
						commit
						d5d2e586cd
					
				
					 12 changed files with 547 additions and 630 deletions
				
			
		|  | @ -51,9 +51,14 @@ var _ = math.Inf | ||||||
| const _ = proto.GoGoProtoPackageIsVersion2 // please upgrade the proto package | const _ = proto.GoGoProtoPackageIsVersion2 // please upgrade the proto package | ||||||
| 
 | 
 | ||||||
| type CreateRequest struct { | type CreateRequest struct { | ||||||
| 	ID      string `protobuf:"bytes,1,opt,name=id,proto3" json:"id,omitempty"` | 	ID       string `protobuf:"bytes,1,opt,name=id,proto3" json:"id,omitempty"` | ||||||
| 	Bundle  string `protobuf:"bytes,2,opt,name=bundle,proto3" json:"bundle,omitempty"` | 	Bundle   string `protobuf:"bytes,2,opt,name=bundle,proto3" json:"bundle,omitempty"` | ||||||
| 	Runtime string `protobuf:"bytes,3,opt,name=runtime,proto3" json:"runtime,omitempty"` | 	Runtime  string `protobuf:"bytes,3,opt,name=runtime,proto3" json:"runtime,omitempty"` | ||||||
|  | 	NoPivot  bool   `protobuf:"varint,4,opt,name=no_pivot,json=noPivot,proto3" json:"no_pivot,omitempty"` | ||||||
|  | 	Terminal bool   `protobuf:"varint,5,opt,name=terminal,proto3" json:"terminal,omitempty"` | ||||||
|  | 	Stdin    string `protobuf:"bytes,6,opt,name=stdin,proto3" json:"stdin,omitempty"` | ||||||
|  | 	Stdout   string `protobuf:"bytes,7,opt,name=stdout,proto3" json:"stdout,omitempty"` | ||||||
|  | 	Stderr   string `protobuf:"bytes,8,opt,name=stderr,proto3" json:"stderr,omitempty"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (m *CreateRequest) Reset()                    { *m = CreateRequest{} } | func (m *CreateRequest) Reset()                    { *m = CreateRequest{} } | ||||||
|  | @ -76,6 +81,7 @@ func (*StartRequest) ProtoMessage()               {} | ||||||
| func (*StartRequest) Descriptor() ([]byte, []int) { return fileDescriptorShim, []int{2} } | func (*StartRequest) Descriptor() ([]byte, []int) { return fileDescriptorShim, []int{2} } | ||||||
| 
 | 
 | ||||||
| type DeleteRequest struct { | type DeleteRequest struct { | ||||||
|  | 	Pid uint32 `protobuf:"varint,1,opt,name=pid,proto3" json:"pid,omitempty"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (m *DeleteRequest) Reset()                    { *m = DeleteRequest{} } | func (m *DeleteRequest) Reset()                    { *m = DeleteRequest{} } | ||||||
|  | @ -106,7 +112,7 @@ func (*ExecResponse) ProtoMessage()               {} | ||||||
| func (*ExecResponse) Descriptor() ([]byte, []int) { return fileDescriptorShim, []int{6} } | func (*ExecResponse) Descriptor() ([]byte, []int) { return fileDescriptorShim, []int{6} } | ||||||
| 
 | 
 | ||||||
| type PtyRequest struct { | type PtyRequest struct { | ||||||
| 	ID     string `protobuf:"bytes,1,opt,name=id,proto3" json:"id,omitempty"` | 	Pid    uint32 `protobuf:"varint,1,opt,name=pid,proto3" json:"pid,omitempty"` | ||||||
| 	Width  uint32 `protobuf:"varint,2,opt,name=width,proto3" json:"width,omitempty"` | 	Width  uint32 `protobuf:"varint,2,opt,name=width,proto3" json:"width,omitempty"` | ||||||
| 	Height uint32 `protobuf:"varint,3,opt,name=height,proto3" json:"height,omitempty"` | 	Height uint32 `protobuf:"varint,3,opt,name=height,proto3" json:"height,omitempty"` | ||||||
| } | } | ||||||
|  | @ -129,11 +135,16 @@ func (this *CreateRequest) GoString() string { | ||||||
| 	if this == nil { | 	if this == nil { | ||||||
| 		return "nil" | 		return "nil" | ||||||
| 	} | 	} | ||||||
| 	s := make([]string, 0, 7) | 	s := make([]string, 0, 12) | ||||||
| 	s = append(s, "&shim.CreateRequest{") | 	s = append(s, "&shim.CreateRequest{") | ||||||
| 	s = append(s, "ID: "+fmt.Sprintf("%#v", this.ID)+",\n") | 	s = append(s, "ID: "+fmt.Sprintf("%#v", this.ID)+",\n") | ||||||
| 	s = append(s, "Bundle: "+fmt.Sprintf("%#v", this.Bundle)+",\n") | 	s = append(s, "Bundle: "+fmt.Sprintf("%#v", this.Bundle)+",\n") | ||||||
| 	s = append(s, "Runtime: "+fmt.Sprintf("%#v", this.Runtime)+",\n") | 	s = append(s, "Runtime: "+fmt.Sprintf("%#v", this.Runtime)+",\n") | ||||||
|  | 	s = append(s, "NoPivot: "+fmt.Sprintf("%#v", this.NoPivot)+",\n") | ||||||
|  | 	s = append(s, "Terminal: "+fmt.Sprintf("%#v", this.Terminal)+",\n") | ||||||
|  | 	s = append(s, "Stdin: "+fmt.Sprintf("%#v", this.Stdin)+",\n") | ||||||
|  | 	s = append(s, "Stdout: "+fmt.Sprintf("%#v", this.Stdout)+",\n") | ||||||
|  | 	s = append(s, "Stderr: "+fmt.Sprintf("%#v", this.Stderr)+",\n") | ||||||
| 	s = append(s, "}") | 	s = append(s, "}") | ||||||
| 	return strings.Join(s, "") | 	return strings.Join(s, "") | ||||||
| } | } | ||||||
|  | @ -160,8 +171,9 @@ func (this *DeleteRequest) GoString() string { | ||||||
| 	if this == nil { | 	if this == nil { | ||||||
| 		return "nil" | 		return "nil" | ||||||
| 	} | 	} | ||||||
| 	s := make([]string, 0, 4) | 	s := make([]string, 0, 5) | ||||||
| 	s = append(s, "&shim.DeleteRequest{") | 	s = append(s, "&shim.DeleteRequest{") | ||||||
|  | 	s = append(s, "Pid: "+fmt.Sprintf("%#v", this.Pid)+",\n") | ||||||
| 	s = append(s, "}") | 	s = append(s, "}") | ||||||
| 	return strings.Join(s, "") | 	return strings.Join(s, "") | ||||||
| } | } | ||||||
|  | @ -200,7 +212,7 @@ func (this *PtyRequest) GoString() string { | ||||||
| 	} | 	} | ||||||
| 	s := make([]string, 0, 7) | 	s := make([]string, 0, 7) | ||||||
| 	s = append(s, "&shim.PtyRequest{") | 	s = append(s, "&shim.PtyRequest{") | ||||||
| 	s = append(s, "ID: "+fmt.Sprintf("%#v", this.ID)+",\n") | 	s = append(s, "Pid: "+fmt.Sprintf("%#v", this.Pid)+",\n") | ||||||
| 	s = append(s, "Width: "+fmt.Sprintf("%#v", this.Width)+",\n") | 	s = append(s, "Width: "+fmt.Sprintf("%#v", this.Width)+",\n") | ||||||
| 	s = append(s, "Height: "+fmt.Sprintf("%#v", this.Height)+",\n") | 	s = append(s, "Height: "+fmt.Sprintf("%#v", this.Height)+",\n") | ||||||
| 	s = append(s, "}") | 	s = append(s, "}") | ||||||
|  | @ -470,6 +482,44 @@ func (m *CreateRequest) MarshalTo(dAtA []byte) (int, error) { | ||||||
| 		i = encodeVarintShim(dAtA, i, uint64(len(m.Runtime))) | 		i = encodeVarintShim(dAtA, i, uint64(len(m.Runtime))) | ||||||
| 		i += copy(dAtA[i:], m.Runtime) | 		i += copy(dAtA[i:], m.Runtime) | ||||||
| 	} | 	} | ||||||
|  | 	if m.NoPivot { | ||||||
|  | 		dAtA[i] = 0x20 | ||||||
|  | 		i++ | ||||||
|  | 		if m.NoPivot { | ||||||
|  | 			dAtA[i] = 1 | ||||||
|  | 		} else { | ||||||
|  | 			dAtA[i] = 0 | ||||||
|  | 		} | ||||||
|  | 		i++ | ||||||
|  | 	} | ||||||
|  | 	if m.Terminal { | ||||||
|  | 		dAtA[i] = 0x28 | ||||||
|  | 		i++ | ||||||
|  | 		if m.Terminal { | ||||||
|  | 			dAtA[i] = 1 | ||||||
|  | 		} else { | ||||||
|  | 			dAtA[i] = 0 | ||||||
|  | 		} | ||||||
|  | 		i++ | ||||||
|  | 	} | ||||||
|  | 	if len(m.Stdin) > 0 { | ||||||
|  | 		dAtA[i] = 0x32 | ||||||
|  | 		i++ | ||||||
|  | 		i = encodeVarintShim(dAtA, i, uint64(len(m.Stdin))) | ||||||
|  | 		i += copy(dAtA[i:], m.Stdin) | ||||||
|  | 	} | ||||||
|  | 	if len(m.Stdout) > 0 { | ||||||
|  | 		dAtA[i] = 0x3a | ||||||
|  | 		i++ | ||||||
|  | 		i = encodeVarintShim(dAtA, i, uint64(len(m.Stdout))) | ||||||
|  | 		i += copy(dAtA[i:], m.Stdout) | ||||||
|  | 	} | ||||||
|  | 	if len(m.Stderr) > 0 { | ||||||
|  | 		dAtA[i] = 0x42 | ||||||
|  | 		i++ | ||||||
|  | 		i = encodeVarintShim(dAtA, i, uint64(len(m.Stderr))) | ||||||
|  | 		i += copy(dAtA[i:], m.Stderr) | ||||||
|  | 	} | ||||||
| 	return i, nil | 	return i, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -529,6 +579,11 @@ func (m *DeleteRequest) MarshalTo(dAtA []byte) (int, error) { | ||||||
| 	_ = i | 	_ = i | ||||||
| 	var l int | 	var l int | ||||||
| 	_ = l | 	_ = l | ||||||
|  | 	if m.Pid != 0 { | ||||||
|  | 		dAtA[i] = 0x8 | ||||||
|  | 		i++ | ||||||
|  | 		i = encodeVarintShim(dAtA, i, uint64(m.Pid)) | ||||||
|  | 	} | ||||||
| 	return i, nil | 	return i, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -611,11 +666,10 @@ func (m *PtyRequest) MarshalTo(dAtA []byte) (int, error) { | ||||||
| 	_ = i | 	_ = i | ||||||
| 	var l int | 	var l int | ||||||
| 	_ = l | 	_ = l | ||||||
| 	if len(m.ID) > 0 { | 	if m.Pid != 0 { | ||||||
| 		dAtA[i] = 0xa | 		dAtA[i] = 0x8 | ||||||
| 		i++ | 		i++ | ||||||
| 		i = encodeVarintShim(dAtA, i, uint64(len(m.ID))) | 		i = encodeVarintShim(dAtA, i, uint64(m.Pid)) | ||||||
| 		i += copy(dAtA[i:], m.ID) |  | ||||||
| 	} | 	} | ||||||
| 	if m.Width != 0 { | 	if m.Width != 0 { | ||||||
| 		dAtA[i] = 0x10 | 		dAtA[i] = 0x10 | ||||||
|  | @ -672,6 +726,24 @@ func (m *CreateRequest) Size() (n int) { | ||||||
| 	if l > 0 { | 	if l > 0 { | ||||||
| 		n += 1 + l + sovShim(uint64(l)) | 		n += 1 + l + sovShim(uint64(l)) | ||||||
| 	} | 	} | ||||||
|  | 	if m.NoPivot { | ||||||
|  | 		n += 2 | ||||||
|  | 	} | ||||||
|  | 	if m.Terminal { | ||||||
|  | 		n += 2 | ||||||
|  | 	} | ||||||
|  | 	l = len(m.Stdin) | ||||||
|  | 	if l > 0 { | ||||||
|  | 		n += 1 + l + sovShim(uint64(l)) | ||||||
|  | 	} | ||||||
|  | 	l = len(m.Stdout) | ||||||
|  | 	if l > 0 { | ||||||
|  | 		n += 1 + l + sovShim(uint64(l)) | ||||||
|  | 	} | ||||||
|  | 	l = len(m.Stderr) | ||||||
|  | 	if l > 0 { | ||||||
|  | 		n += 1 + l + sovShim(uint64(l)) | ||||||
|  | 	} | ||||||
| 	return n | 	return n | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -693,6 +765,9 @@ func (m *StartRequest) Size() (n int) { | ||||||
| func (m *DeleteRequest) Size() (n int) { | func (m *DeleteRequest) Size() (n int) { | ||||||
| 	var l int | 	var l int | ||||||
| 	_ = l | 	_ = l | ||||||
|  | 	if m.Pid != 0 { | ||||||
|  | 		n += 1 + sovShim(uint64(m.Pid)) | ||||||
|  | 	} | ||||||
| 	return n | 	return n | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -723,9 +798,8 @@ func (m *ExecResponse) Size() (n int) { | ||||||
| func (m *PtyRequest) Size() (n int) { | func (m *PtyRequest) Size() (n int) { | ||||||
| 	var l int | 	var l int | ||||||
| 	_ = l | 	_ = l | ||||||
| 	l = len(m.ID) | 	if m.Pid != 0 { | ||||||
| 	if l > 0 { | 		n += 1 + sovShim(uint64(m.Pid)) | ||||||
| 		n += 1 + l + sovShim(uint64(l)) |  | ||||||
| 	} | 	} | ||||||
| 	if m.Width != 0 { | 	if m.Width != 0 { | ||||||
| 		n += 1 + sovShim(uint64(m.Width)) | 		n += 1 + sovShim(uint64(m.Width)) | ||||||
|  | @ -757,6 +831,11 @@ func (this *CreateRequest) String() string { | ||||||
| 		`ID:` + fmt.Sprintf("%v", this.ID) + `,`, | 		`ID:` + fmt.Sprintf("%v", this.ID) + `,`, | ||||||
| 		`Bundle:` + fmt.Sprintf("%v", this.Bundle) + `,`, | 		`Bundle:` + fmt.Sprintf("%v", this.Bundle) + `,`, | ||||||
| 		`Runtime:` + fmt.Sprintf("%v", this.Runtime) + `,`, | 		`Runtime:` + fmt.Sprintf("%v", this.Runtime) + `,`, | ||||||
|  | 		`NoPivot:` + fmt.Sprintf("%v", this.NoPivot) + `,`, | ||||||
|  | 		`Terminal:` + fmt.Sprintf("%v", this.Terminal) + `,`, | ||||||
|  | 		`Stdin:` + fmt.Sprintf("%v", this.Stdin) + `,`, | ||||||
|  | 		`Stdout:` + fmt.Sprintf("%v", this.Stdout) + `,`, | ||||||
|  | 		`Stderr:` + fmt.Sprintf("%v", this.Stderr) + `,`, | ||||||
| 		`}`, | 		`}`, | ||||||
| 	}, "") | 	}, "") | ||||||
| 	return s | 	return s | ||||||
|  | @ -785,6 +864,7 @@ func (this *DeleteRequest) String() string { | ||||||
| 		return "nil" | 		return "nil" | ||||||
| 	} | 	} | ||||||
| 	s := strings.Join([]string{`&DeleteRequest{`, | 	s := strings.Join([]string{`&DeleteRequest{`, | ||||||
|  | 		`Pid:` + fmt.Sprintf("%v", this.Pid) + `,`, | ||||||
| 		`}`, | 		`}`, | ||||||
| 	}, "") | 	}, "") | ||||||
| 	return s | 	return s | ||||||
|  | @ -823,7 +903,7 @@ func (this *PtyRequest) String() string { | ||||||
| 		return "nil" | 		return "nil" | ||||||
| 	} | 	} | ||||||
| 	s := strings.Join([]string{`&PtyRequest{`, | 	s := strings.Join([]string{`&PtyRequest{`, | ||||||
| 		`ID:` + fmt.Sprintf("%v", this.ID) + `,`, | 		`Pid:` + fmt.Sprintf("%v", this.Pid) + `,`, | ||||||
| 		`Width:` + fmt.Sprintf("%v", this.Width) + `,`, | 		`Width:` + fmt.Sprintf("%v", this.Width) + `,`, | ||||||
| 		`Height:` + fmt.Sprintf("%v", this.Height) + `,`, | 		`Height:` + fmt.Sprintf("%v", this.Height) + `,`, | ||||||
| 		`}`, | 		`}`, | ||||||
|  | @ -954,6 +1034,133 @@ func (m *CreateRequest) Unmarshal(dAtA []byte) error { | ||||||
| 			} | 			} | ||||||
| 			m.Runtime = string(dAtA[iNdEx:postIndex]) | 			m.Runtime = string(dAtA[iNdEx:postIndex]) | ||||||
| 			iNdEx = postIndex | 			iNdEx = postIndex | ||||||
|  | 		case 4: | ||||||
|  | 			if wireType != 0 { | ||||||
|  | 				return fmt.Errorf("proto: wrong wireType = %d for field NoPivot", wireType) | ||||||
|  | 			} | ||||||
|  | 			var v int | ||||||
|  | 			for shift := uint(0); ; shift += 7 { | ||||||
|  | 				if shift >= 64 { | ||||||
|  | 					return ErrIntOverflowShim | ||||||
|  | 				} | ||||||
|  | 				if iNdEx >= l { | ||||||
|  | 					return io.ErrUnexpectedEOF | ||||||
|  | 				} | ||||||
|  | 				b := dAtA[iNdEx] | ||||||
|  | 				iNdEx++ | ||||||
|  | 				v |= (int(b) & 0x7F) << shift | ||||||
|  | 				if b < 0x80 { | ||||||
|  | 					break | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 			m.NoPivot = bool(v != 0) | ||||||
|  | 		case 5: | ||||||
|  | 			if wireType != 0 { | ||||||
|  | 				return fmt.Errorf("proto: wrong wireType = %d for field Terminal", wireType) | ||||||
|  | 			} | ||||||
|  | 			var v int | ||||||
|  | 			for shift := uint(0); ; shift += 7 { | ||||||
|  | 				if shift >= 64 { | ||||||
|  | 					return ErrIntOverflowShim | ||||||
|  | 				} | ||||||
|  | 				if iNdEx >= l { | ||||||
|  | 					return io.ErrUnexpectedEOF | ||||||
|  | 				} | ||||||
|  | 				b := dAtA[iNdEx] | ||||||
|  | 				iNdEx++ | ||||||
|  | 				v |= (int(b) & 0x7F) << shift | ||||||
|  | 				if b < 0x80 { | ||||||
|  | 					break | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 			m.Terminal = bool(v != 0) | ||||||
|  | 		case 6: | ||||||
|  | 			if wireType != 2 { | ||||||
|  | 				return fmt.Errorf("proto: wrong wireType = %d for field Stdin", wireType) | ||||||
|  | 			} | ||||||
|  | 			var stringLen uint64 | ||||||
|  | 			for shift := uint(0); ; shift += 7 { | ||||||
|  | 				if shift >= 64 { | ||||||
|  | 					return ErrIntOverflowShim | ||||||
|  | 				} | ||||||
|  | 				if iNdEx >= l { | ||||||
|  | 					return io.ErrUnexpectedEOF | ||||||
|  | 				} | ||||||
|  | 				b := dAtA[iNdEx] | ||||||
|  | 				iNdEx++ | ||||||
|  | 				stringLen |= (uint64(b) & 0x7F) << shift | ||||||
|  | 				if b < 0x80 { | ||||||
|  | 					break | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 			intStringLen := int(stringLen) | ||||||
|  | 			if intStringLen < 0 { | ||||||
|  | 				return ErrInvalidLengthShim | ||||||
|  | 			} | ||||||
|  | 			postIndex := iNdEx + intStringLen | ||||||
|  | 			if postIndex > l { | ||||||
|  | 				return io.ErrUnexpectedEOF | ||||||
|  | 			} | ||||||
|  | 			m.Stdin = string(dAtA[iNdEx:postIndex]) | ||||||
|  | 			iNdEx = postIndex | ||||||
|  | 		case 7: | ||||||
|  | 			if wireType != 2 { | ||||||
|  | 				return fmt.Errorf("proto: wrong wireType = %d for field Stdout", wireType) | ||||||
|  | 			} | ||||||
|  | 			var stringLen uint64 | ||||||
|  | 			for shift := uint(0); ; shift += 7 { | ||||||
|  | 				if shift >= 64 { | ||||||
|  | 					return ErrIntOverflowShim | ||||||
|  | 				} | ||||||
|  | 				if iNdEx >= l { | ||||||
|  | 					return io.ErrUnexpectedEOF | ||||||
|  | 				} | ||||||
|  | 				b := dAtA[iNdEx] | ||||||
|  | 				iNdEx++ | ||||||
|  | 				stringLen |= (uint64(b) & 0x7F) << shift | ||||||
|  | 				if b < 0x80 { | ||||||
|  | 					break | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 			intStringLen := int(stringLen) | ||||||
|  | 			if intStringLen < 0 { | ||||||
|  | 				return ErrInvalidLengthShim | ||||||
|  | 			} | ||||||
|  | 			postIndex := iNdEx + intStringLen | ||||||
|  | 			if postIndex > l { | ||||||
|  | 				return io.ErrUnexpectedEOF | ||||||
|  | 			} | ||||||
|  | 			m.Stdout = string(dAtA[iNdEx:postIndex]) | ||||||
|  | 			iNdEx = postIndex | ||||||
|  | 		case 8: | ||||||
|  | 			if wireType != 2 { | ||||||
|  | 				return fmt.Errorf("proto: wrong wireType = %d for field Stderr", wireType) | ||||||
|  | 			} | ||||||
|  | 			var stringLen uint64 | ||||||
|  | 			for shift := uint(0); ; shift += 7 { | ||||||
|  | 				if shift >= 64 { | ||||||
|  | 					return ErrIntOverflowShim | ||||||
|  | 				} | ||||||
|  | 				if iNdEx >= l { | ||||||
|  | 					return io.ErrUnexpectedEOF | ||||||
|  | 				} | ||||||
|  | 				b := dAtA[iNdEx] | ||||||
|  | 				iNdEx++ | ||||||
|  | 				stringLen |= (uint64(b) & 0x7F) << shift | ||||||
|  | 				if b < 0x80 { | ||||||
|  | 					break | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 			intStringLen := int(stringLen) | ||||||
|  | 			if intStringLen < 0 { | ||||||
|  | 				return ErrInvalidLengthShim | ||||||
|  | 			} | ||||||
|  | 			postIndex := iNdEx + intStringLen | ||||||
|  | 			if postIndex > l { | ||||||
|  | 				return io.ErrUnexpectedEOF | ||||||
|  | 			} | ||||||
|  | 			m.Stderr = string(dAtA[iNdEx:postIndex]) | ||||||
|  | 			iNdEx = postIndex | ||||||
| 		default: | 		default: | ||||||
| 			iNdEx = preIndex | 			iNdEx = preIndex | ||||||
| 			skippy, err := skipShim(dAtA[iNdEx:]) | 			skippy, err := skipShim(dAtA[iNdEx:]) | ||||||
|  | @ -1123,6 +1330,25 @@ func (m *DeleteRequest) Unmarshal(dAtA []byte) error { | ||||||
| 			return fmt.Errorf("proto: DeleteRequest: illegal tag %d (wire type %d)", fieldNum, wire) | 			return fmt.Errorf("proto: DeleteRequest: illegal tag %d (wire type %d)", fieldNum, wire) | ||||||
| 		} | 		} | ||||||
| 		switch fieldNum { | 		switch fieldNum { | ||||||
|  | 		case 1: | ||||||
|  | 			if wireType != 0 { | ||||||
|  | 				return fmt.Errorf("proto: wrong wireType = %d for field Pid", wireType) | ||||||
|  | 			} | ||||||
|  | 			m.Pid = 0 | ||||||
|  | 			for shift := uint(0); ; shift += 7 { | ||||||
|  | 				if shift >= 64 { | ||||||
|  | 					return ErrIntOverflowShim | ||||||
|  | 				} | ||||||
|  | 				if iNdEx >= l { | ||||||
|  | 					return io.ErrUnexpectedEOF | ||||||
|  | 				} | ||||||
|  | 				b := dAtA[iNdEx] | ||||||
|  | 				iNdEx++ | ||||||
|  | 				m.Pid |= (uint32(b) & 0x7F) << shift | ||||||
|  | 				if b < 0x80 { | ||||||
|  | 					break | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
| 		default: | 		default: | ||||||
| 			iNdEx = preIndex | 			iNdEx = preIndex | ||||||
| 			skippy, err := skipShim(dAtA[iNdEx:]) | 			skippy, err := skipShim(dAtA[iNdEx:]) | ||||||
|  | @ -1362,10 +1588,10 @@ func (m *PtyRequest) Unmarshal(dAtA []byte) error { | ||||||
| 		} | 		} | ||||||
| 		switch fieldNum { | 		switch fieldNum { | ||||||
| 		case 1: | 		case 1: | ||||||
| 			if wireType != 2 { | 			if wireType != 0 { | ||||||
| 				return fmt.Errorf("proto: wrong wireType = %d for field ID", wireType) | 				return fmt.Errorf("proto: wrong wireType = %d for field Pid", wireType) | ||||||
| 			} | 			} | ||||||
| 			var stringLen uint64 | 			m.Pid = 0 | ||||||
| 			for shift := uint(0); ; shift += 7 { | 			for shift := uint(0); ; shift += 7 { | ||||||
| 				if shift >= 64 { | 				if shift >= 64 { | ||||||
| 					return ErrIntOverflowShim | 					return ErrIntOverflowShim | ||||||
|  | @ -1375,21 +1601,11 @@ func (m *PtyRequest) Unmarshal(dAtA []byte) error { | ||||||
| 				} | 				} | ||||||
| 				b := dAtA[iNdEx] | 				b := dAtA[iNdEx] | ||||||
| 				iNdEx++ | 				iNdEx++ | ||||||
| 				stringLen |= (uint64(b) & 0x7F) << shift | 				m.Pid |= (uint32(b) & 0x7F) << shift | ||||||
| 				if b < 0x80 { | 				if b < 0x80 { | ||||||
| 					break | 					break | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| 			intStringLen := int(stringLen) |  | ||||||
| 			if intStringLen < 0 { |  | ||||||
| 				return ErrInvalidLengthShim |  | ||||||
| 			} |  | ||||||
| 			postIndex := iNdEx + intStringLen |  | ||||||
| 			if postIndex > l { |  | ||||||
| 				return io.ErrUnexpectedEOF |  | ||||||
| 			} |  | ||||||
| 			m.ID = string(dAtA[iNdEx:postIndex]) |  | ||||||
| 			iNdEx = postIndex |  | ||||||
| 		case 2: | 		case 2: | ||||||
| 			if wireType != 0 { | 			if wireType != 0 { | ||||||
| 				return fmt.Errorf("proto: wrong wireType = %d for field Width", wireType) | 				return fmt.Errorf("proto: wrong wireType = %d for field Width", wireType) | ||||||
|  | @ -1557,32 +1773,36 @@ var ( | ||||||
| func init() { proto.RegisterFile("shim.proto", fileDescriptorShim) } | func init() { proto.RegisterFile("shim.proto", fileDescriptorShim) } | ||||||
| 
 | 
 | ||||||
| var fileDescriptorShim = []byte{ | var fileDescriptorShim = []byte{ | ||||||
| 	// 419 bytes of a gzipped FileDescriptorProto | 	// 487 bytes of a gzipped FileDescriptorProto | ||||||
| 	0x1f, 0x8b, 0x08, 0x00, 0x00, 0x09, 0x6e, 0x88, 0x02, 0xff, 0x84, 0x92, 0xbf, 0x8e, 0xd3, 0x40, | 	0x1f, 0x8b, 0x08, 0x00, 0x00, 0x09, 0x6e, 0x88, 0x02, 0xff, 0x74, 0x52, 0x4d, 0x6f, 0xda, 0x40, | ||||||
| 	0x10, 0xc6, 0xcf, 0x0e, 0x67, 0xc4, 0xe4, 0x1c, 0xd0, 0xea, 0x14, 0x19, 0xe7, 0xf0, 0x9d, 0x5c, | 	0x10, 0x8d, 0x21, 0x7c, 0x74, 0x88, 0x51, 0xb5, 0x8a, 0x90, 0x63, 0x52, 0x87, 0xfa, 0x94, 0x93, | ||||||
| 	0x51, 0x39, 0x3a, 0x68, 0x28, 0x90, 0x90, 0x42, 0x52, 0xd0, 0x45, 0x76, 0x45, 0x85, 0x9c, 0x78, | 	0x51, 0xda, 0x4b, 0x0f, 0x95, 0x2a, 0xa5, 0xe1, 0x50, 0xa9, 0x07, 0x64, 0x7e, 0x40, 0x04, 0x78, | ||||||
| 	0xb0, 0x57, 0x8a, 0xbd, 0xc6, 0x5e, 0x87, 0xa4, 0xe3, 0xf1, 0x52, 0x52, 0x50, 0x50, 0x21, 0xe2, | 	0x0a, 0x2b, 0x81, 0xd7, 0x5d, 0x8f, 0x69, 0xb8, 0xf5, 0xe7, 0xe5, 0xd8, 0x43, 0x55, 0xf5, 0x54, | ||||||
| 	0x27, 0xe0, 0x11, 0xd0, 0xae, 0xd7, 0x90, 0x38, 0x44, 0x74, 0x3b, 0x7f, 0xf4, 0xed, 0x37, 0xbf, | 	0x15, 0xff, 0x82, 0xfe, 0x84, 0x6a, 0x77, 0xed, 0x24, 0x40, 0xb8, 0xed, 0x9b, 0xf7, 0x76, 0xf7, | ||||||
| 	0x19, 0x80, 0x32, 0xa1, 0xa9, 0x97, 0x17, 0x8c, 0x33, 0x62, 0x2e, 0x59, 0xc6, 0x43, 0x9a, 0x61, | 	0xcd, 0x9b, 0x01, 0x48, 0xe7, 0x7c, 0x19, 0x24, 0x52, 0x90, 0x60, 0xf6, 0x54, 0xc4, 0x34, 0xe6, | ||||||
| 	0x11, 0x79, 0xeb, 0x7b, 0x7b, 0x14, 0x33, 0x16, 0xaf, 0x70, 0x2c, 0x8b, 0x8b, 0xea, 0xe3, 0x18, | 	0x31, 0xca, 0x28, 0x58, 0x5d, 0xb9, 0xdd, 0x99, 0x10, 0xb3, 0x05, 0xf6, 0x35, 0x39, 0xc9, 0xbe, | ||||||
| 	0xd3, 0x9c, 0x6f, 0x9b, 0x5e, 0xfb, 0x3a, 0x66, 0x31, 0x93, 0xcf, 0xb1, 0x78, 0x35, 0x59, 0xf7, | 	0xf4, 0x71, 0x99, 0xd0, 0xda, 0x68, 0xdd, 0xd3, 0x99, 0x98, 0x09, 0x7d, 0xec, 0xab, 0x93, 0xa9, | ||||||
| 	0x3d, 0x98, 0x6f, 0x0b, 0x0c, 0x39, 0xfa, 0xf8, 0xa9, 0xc2, 0x92, 0x93, 0x21, 0xe8, 0x34, 0xb2, | 	0xfa, 0xbf, 0x2c, 0xb0, 0x3f, 0x4a, 0x1c, 0x13, 0x86, 0xf8, 0x35, 0xc3, 0x94, 0x58, 0x07, 0x2a, | ||||||
| 	0xb4, 0x3b, 0xed, 0xf9, 0xa3, 0x89, 0x51, 0xff, 0xb8, 0xd5, 0xdf, 0x4d, 0x7d, 0x9d, 0x46, 0x64, | 	0x3c, 0x72, 0xac, 0x9e, 0x75, 0xf9, 0xe2, 0xba, 0x9e, 0xff, 0xb9, 0xa8, 0x7c, 0xba, 0x09, 0x2b, | ||||||
| 	0x08, 0xc6, 0xa2, 0xca, 0xa2, 0x15, 0x5a, 0xba, 0xa8, 0xf9, 0x2a, 0x22, 0x16, 0x3c, 0x2c, 0xaa, | 	0x3c, 0x62, 0x1d, 0xa8, 0x4f, 0xb2, 0x38, 0x5a, 0xa0, 0x53, 0x51, 0x5c, 0x58, 0x20, 0xe6, 0x40, | ||||||
| 	0x8c, 0xd3, 0x14, 0xad, 0x9e, 0x2c, 0xb4, 0xa1, 0xeb, 0xc2, 0xa0, 0x95, 0x2e, 0x73, 0x96, 0x95, | 	0x43, 0x66, 0x31, 0xf1, 0x25, 0x3a, 0x55, 0x4d, 0x94, 0x90, 0x9d, 0x41, 0x33, 0x16, 0xb7, 0x09, | ||||||
| 	0x48, 0x9e, 0x40, 0x2f, 0x57, 0xe2, 0xa6, 0x2f, 0x9e, 0xee, 0x00, 0xae, 0x02, 0x1e, 0x16, 0x5c, | 	0x5f, 0x09, 0x72, 0x8e, 0x7b, 0xd6, 0x65, 0x33, 0x6c, 0xc4, 0x62, 0xa8, 0x20, 0x73, 0xa1, 0x49, | ||||||
| 	0xfd, 0xee, 0x3e, 0x06, 0x73, 0x8a, 0x2b, 0xfc, 0x63, 0xc7, 0xbd, 0x87, 0x41, 0x9b, 0x50, 0x22, | 	0x28, 0x97, 0x3c, 0x1e, 0x2f, 0x9c, 0x9a, 0xa6, 0x1e, 0x30, 0x3b, 0x85, 0x5a, 0x4a, 0x11, 0x8f, | ||||||
| 	0xb7, 0xd0, 0xc7, 0x0d, 0xe5, 0x1f, 0x4a, 0x1e, 0xf2, 0xaa, 0x54, 0x62, 0x20, 0x52, 0x81, 0xcc, | 	0x9d, 0xba, 0x7e, 0xce, 0x00, 0xf5, 0x7d, 0x4a, 0x91, 0xc8, 0xc8, 0x69, 0x98, 0xef, 0x0d, 0x2a, | ||||||
| 	0xb8, 0x26, 0xf4, 0x67, 0x1b, 0x5c, 0xb6, 0x0a, 0x77, 0x70, 0xd5, 0x84, 0x67, 0x4d, 0xf8, 0x00, | 	0xea, 0x28, 0xa5, 0xd3, 0x7c, 0xa8, 0xa3, 0x94, 0xbe, 0x0f, 0xed, 0xb2, 0xaf, 0x34, 0x11, 0x71, | ||||||
| 	0x73, 0xbe, 0xfd, 0x1f, 0x80, 0x6b, 0xb8, 0xfc, 0x4c, 0x23, 0x9e, 0xc8, 0xf9, 0x4d, 0xbf, 0x09, | 	0x8a, 0xec, 0x25, 0x54, 0x93, 0xa2, 0x33, 0x3b, 0x54, 0x47, 0xbf, 0x0d, 0x27, 0x23, 0x1a, 0x4b, | ||||||
| 	0x04, 0x96, 0x04, 0x69, 0x9c, 0x70, 0x39, 0xbd, 0xe9, 0xab, 0xe8, 0xc5, 0x37, 0x1d, 0xfa, 0x41, | 	0x2a, 0x5a, 0xf7, 0x5f, 0x83, 0x7d, 0x83, 0x0b, 0x7c, 0xcc, 0x62, 0xff, 0xca, 0x15, 0xb4, 0x4b, | ||||||
| 	0x42, 0xd3, 0x00, 0x8b, 0x35, 0x5d, 0x22, 0x99, 0x81, 0xd1, 0xc0, 0x20, 0x37, 0xde, 0xd1, 0xd2, | 	0x49, 0xf1, 0xec, 0x05, 0xb4, 0xf0, 0x8e, 0xd3, 0x6d, 0x4a, 0x63, 0xca, 0xd2, 0x42, 0x0b, 0xaa, | ||||||
| 	0xbc, 0x23, 0xfc, 0xf6, 0xb3, 0x33, 0x55, 0x65, 0xfe, 0x35, 0x5c, 0x4a, 0x5e, 0x64, 0xd4, 0xe9, | 	0x34, 0xd2, 0x15, 0xdf, 0x86, 0xd6, 0xe0, 0x0e, 0xa7, 0xe5, 0x27, 0x3d, 0x38, 0x31, 0xf0, 0xa0, | ||||||
| 	0x3b, 0xa4, 0x68, 0x0f, 0xbd, 0xe6, 0x10, 0xbc, 0xf6, 0x10, 0xbc, 0x99, 0x38, 0x04, 0x61, 0xa2, | 	0xad, 0xcf, 0x00, 0x43, 0x5a, 0x1f, 0xf4, 0xa0, 0x02, 0xfa, 0xc6, 0x23, 0x9a, 0xeb, 0x41, 0xd8, | ||||||
| 	0x81, 0x79, 0x62, 0xe2, 0x08, 0xfa, 0x89, 0x89, 0xce, 0x06, 0xde, 0xc0, 0x03, 0x41, 0x94, 0xd8, | 	0xa1, 0x01, 0x2a, 0x88, 0x39, 0xf2, 0xd9, 0x9c, 0xf4, 0x18, 0xec, 0xb0, 0x40, 0x6f, 0x7e, 0x56, | ||||||
| 	0x9d, 0xb6, 0x03, 0xea, 0xf6, 0xe8, 0x9f, 0x35, 0x25, 0xf0, 0x0a, 0x7a, 0x73, 0xbe, 0x25, 0x4f, | 	0xa0, 0x35, 0x9a, 0xf3, 0xe5, 0x08, 0xe5, 0x8a, 0x4f, 0x91, 0x0d, 0xa0, 0x6e, 0x82, 0x61, 0xe7, | ||||||
| 	0x3b, 0x3d, 0x7f, 0x97, 0x70, 0x6e, 0x82, 0xc9, 0xcd, 0x6e, 0xef, 0x5c, 0x7c, 0xdf, 0x3b, 0x17, | 	0xc1, 0xd6, 0xfa, 0x04, 0x5b, 0x7b, 0xe0, 0xbe, 0x3a, 0xc0, 0x16, 0xb6, 0xdf, 0x43, 0x4d, 0x67, | ||||||
| 	0xbf, 0xf6, 0x8e, 0xf6, 0xa5, 0x76, 0xb4, 0x5d, 0xed, 0x68, 0x5f, 0x6b, 0x47, 0xfb, 0x59, 0x3b, | 	0xc7, 0xba, 0x3b, 0xba, 0xa7, 0x89, 0xba, 0x9d, 0xc0, 0xac, 0x64, 0x50, 0xae, 0x64, 0x30, 0x50, | ||||||
| 	0xda, 0xc2, 0x90, 0xdd, 0x2f, 0x7f, 0x07, 0x00, 0x00, 0xff, 0xff, 0x98, 0x18, 0x12, 0xde, 0x23, | 	0x2b, 0xa9, 0x4c, 0x98, 0x18, 0xf7, 0x4c, 0x6c, 0x0d, 0x60, 0xcf, 0xc4, 0x4e, 0xf6, 0x1f, 0xe0, | ||||||
| 	0x03, 0x00, 0x00, | 	0x58, 0x65, 0xc9, 0xdc, 0x1d, 0xd9, 0x93, 0xbc, 0xdd, 0xee, 0xb3, 0x5c, 0xf1, 0xc0, 0x3b, 0xa8, | ||||||
|  | 	0x0e, 0x69, 0xcd, 0xce, 0x76, 0x34, 0x8f, 0xf1, 0x1f, 0xea, 0xe0, 0xfa, 0xfc, 0x7e, 0xe3, 0x1d, | ||||||
|  | 	0xfd, 0xde, 0x78, 0x47, 0xff, 0x36, 0x9e, 0xf5, 0x3d, 0xf7, 0xac, 0xfb, 0xdc, 0xb3, 0x7e, 0xe4, | ||||||
|  | 	0x9e, 0xf5, 0x37, 0xf7, 0xac, 0x49, 0x5d, 0xab, 0xdf, 0xfe, 0x0f, 0x00, 0x00, 0xff, 0xff, 0xff, | ||||||
|  | 	0xb1, 0x72, 0x31, 0xad, 0x03, 0x00, 0x00, | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -17,6 +17,11 @@ message CreateRequest { | ||||||
| 	string id = 1 [(gogoproto.customname) = "ID"]; | 	string id = 1 [(gogoproto.customname) = "ID"]; | ||||||
|     string bundle = 2; |     string bundle = 2; | ||||||
|     string runtime = 3; |     string runtime = 3; | ||||||
|  |     bool no_pivot = 4; | ||||||
|  |     bool terminal = 5; | ||||||
|  |     string stdin = 6; | ||||||
|  |     string stdout = 7; | ||||||
|  |     string stderr = 8; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| message CreateResponse { | message CreateResponse { | ||||||
|  | @ -27,7 +32,7 @@ message StartRequest { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| message DeleteRequest { | message DeleteRequest { | ||||||
| 
 |     uint32 pid = 1; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| message DeleteResponse { | message DeleteResponse { | ||||||
|  | @ -43,7 +48,7 @@ message ExecResponse { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| message PtyRequest { | message PtyRequest { | ||||||
| 	string id = 1 [(gogoproto.customname) = "ID"]; |     uint32 pid = 1; | ||||||
| 	uint32 width = 2; | 	uint32 width = 2; | ||||||
| 	uint32 height = 3; | 	uint32 height = 3; | ||||||
| } | } | ||||||
|  |  | ||||||
							
								
								
									
										38
									
								
								cmd/containerd-shim/checkpoint.go
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								cmd/containerd-shim/checkpoint.go
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,38 @@ | ||||||
|  | package main | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"encoding/json" | ||||||
|  | 	"os" | ||||||
|  | 	"path/filepath" | ||||||
|  | 	"time" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | type checkpoint struct { | ||||||
|  | 	// Timestamp is the time that checkpoint happened | ||||||
|  | 	Created time.Time `json:"created"` | ||||||
|  | 	// Name is the name of the checkpoint | ||||||
|  | 	Name string `json:"name"` | ||||||
|  | 	// TCP checkpoints open tcp connections | ||||||
|  | 	TCP bool `json:"tcp"` | ||||||
|  | 	// UnixSockets persists unix sockets in the checkpoint | ||||||
|  | 	UnixSockets bool `json:"unixSockets"` | ||||||
|  | 	// Shell persists tty sessions in the checkpoint | ||||||
|  | 	Shell bool `json:"shell"` | ||||||
|  | 	// Exit exits the container after the checkpoint is finished | ||||||
|  | 	Exit bool `json:"exit"` | ||||||
|  | 	// EmptyNS tells CRIU not to restore a particular namespace | ||||||
|  | 	EmptyNS []string `json:"emptyNS,omitempty"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func loadCheckpoint(checkpointPath string) (*checkpoint, error) { | ||||||
|  | 	f, err := os.Open(filepath.Join(checkpointPath, "config.json")) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	defer f.Close() | ||||||
|  | 	var cpt checkpoint | ||||||
|  | 	if err := json.NewDecoder(f).Decode(&cpt); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	return &cpt, nil | ||||||
|  | } | ||||||
|  | @ -1,82 +0,0 @@ | ||||||
| // +build !solaris |  | ||||||
| 
 |  | ||||||
| package main |  | ||||||
| 
 |  | ||||||
| import ( |  | ||||||
| 	"fmt" |  | ||||||
| 	"os" |  | ||||||
| 	"syscall" |  | ||||||
| 	"unsafe" |  | ||||||
| ) |  | ||||||
| 
 |  | ||||||
| // NewConsole returns an initialized console that can be used within a container by copying bytes |  | ||||||
| // from the master side to the slave that is attached as the tty for the container's init process. |  | ||||||
| func newConsole(uid, gid int) (*os.File, string, error) { |  | ||||||
| 	master, err := os.OpenFile("/dev/ptmx", syscall.O_RDWR|syscall.O_NOCTTY|syscall.O_CLOEXEC, 0) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, "", err |  | ||||||
| 	} |  | ||||||
| 	if err = saneTerminal(master); err != nil { |  | ||||||
| 		return nil, "", err |  | ||||||
| 	} |  | ||||||
| 	console, err := ptsname(master) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, "", err |  | ||||||
| 	} |  | ||||||
| 	if err := unlockpt(master); err != nil { |  | ||||||
| 		return nil, "", err |  | ||||||
| 	} |  | ||||||
| 	if err := os.Chmod(console, 0600); err != nil { |  | ||||||
| 		return nil, "", err |  | ||||||
| 	} |  | ||||||
| 	if err := os.Chown(console, uid, gid); err != nil { |  | ||||||
| 		return nil, "", err |  | ||||||
| 	} |  | ||||||
| 	return master, console, nil |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // saneTerminal sets the necessary tty_ioctl(4)s to ensure that a pty pair |  | ||||||
| // created by us acts normally. In particular, a not-very-well-known default of |  | ||||||
| // Linux unix98 ptys is that they have +onlcr by default. While this isn't a |  | ||||||
| // problem for terminal emulators, because we relay data from the terminal we |  | ||||||
| // also relay that funky line discipline. |  | ||||||
| func saneTerminal(terminal *os.File) error { |  | ||||||
| 	// Go doesn't have a wrapper for any of the termios ioctls. |  | ||||||
| 	var termios syscall.Termios |  | ||||||
| 
 |  | ||||||
| 	if err := ioctl(terminal.Fd(), syscall.TCGETS, uintptr(unsafe.Pointer(&termios))); err != nil { |  | ||||||
| 		return fmt.Errorf("ioctl(tty, tcgets): %s", err.Error()) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	// Set -onlcr so we don't have to deal with \r. |  | ||||||
| 	termios.Oflag &^= syscall.ONLCR |  | ||||||
| 
 |  | ||||||
| 	if err := ioctl(terminal.Fd(), syscall.TCSETS, uintptr(unsafe.Pointer(&termios))); err != nil { |  | ||||||
| 		return fmt.Errorf("ioctl(tty, tcsets): %s", err.Error()) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func ioctl(fd uintptr, flag, data uintptr) error { |  | ||||||
| 	if _, _, err := syscall.Syscall(syscall.SYS_IOCTL, fd, flag, data); err != 0 { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // unlockpt unlocks the slave pseudoterminal device corresponding to the master pseudoterminal referred to by f. |  | ||||||
| // unlockpt should be called before opening the slave side of a pty. |  | ||||||
| func unlockpt(f *os.File) error { |  | ||||||
| 	var u int32 |  | ||||||
| 	return ioctl(f.Fd(), syscall.TIOCSPTLCK, uintptr(unsafe.Pointer(&u))) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // ptsname retrieves the name of the first available pts for the given master. |  | ||||||
| func ptsname(f *os.File) (string, error) { |  | ||||||
| 	var n int32 |  | ||||||
| 	if err := ioctl(f.Fd(), syscall.TIOCGPTN, uintptr(unsafe.Pointer(&n))); err != nil { |  | ||||||
| 		return "", err |  | ||||||
| 	} |  | ||||||
| 	return fmt.Sprintf("/dev/pts/%d", n), nil |  | ||||||
| } |  | ||||||
|  | @ -1,14 +0,0 @@ | ||||||
| // +build solaris |  | ||||||
| 
 |  | ||||||
| package main |  | ||||||
| 
 |  | ||||||
| import ( |  | ||||||
| 	"errors" |  | ||||||
| 	"os" |  | ||||||
| ) |  | ||||||
| 
 |  | ||||||
| // NewConsole returns an initialized console that can be used within a container by copying bytes |  | ||||||
| // from the master side to the slave that is attached as the tty for the container's init process. |  | ||||||
| func newConsole(uid, gid int) (*os.File, string, error) { |  | ||||||
| 	return nil, "", errors.New("newConsole not implemented on Solaris") |  | ||||||
| } |  | ||||||
							
								
								
									
										5
									
								
								cmd/containerd-shim/exec.go
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								cmd/containerd-shim/exec.go
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,5 @@ | ||||||
|  | package main | ||||||
|  | 
 | ||||||
|  | func newExecProcess(id, bundle, runtimeName string) (process, error) { | ||||||
|  | 	return nil, nil | ||||||
|  | } | ||||||
							
								
								
									
										122
									
								
								cmd/containerd-shim/init.go
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										122
									
								
								cmd/containerd-shim/init.go
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,122 @@ | ||||||
|  | package main | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"context" | ||||||
|  | 	"os" | ||||||
|  | 	"path/filepath" | ||||||
|  | 	"sync" | ||||||
|  | 	"syscall" | ||||||
|  | 
 | ||||||
|  | 	runc "github.com/crosbymichael/go-runc" | ||||||
|  | 	"github.com/docker/containerd/api/shim" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | type initProcess struct { | ||||||
|  | 	sync.WaitGroup | ||||||
|  | 
 | ||||||
|  | 	id      string | ||||||
|  | 	bundle  string | ||||||
|  | 	console *runc.Console | ||||||
|  | 	io      runc.IO | ||||||
|  | 	runc    *runc.Runc | ||||||
|  | 	status  int | ||||||
|  | 	pid     int | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func newInitProcess(context context.Context, r *shim.CreateRequest) (process, error) { | ||||||
|  | 	cwd, err := os.Getwd() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	runtime := &runc.Runc{ | ||||||
|  | 		Command:      r.Runtime, | ||||||
|  | 		Log:          filepath.Join(cwd, "log.json"), | ||||||
|  | 		LogFormat:    runc.JSON, | ||||||
|  | 		PdeathSignal: syscall.SIGKILL, | ||||||
|  | 	} | ||||||
|  | 	p := &initProcess{ | ||||||
|  | 		id:     r.ID, | ||||||
|  | 		bundle: r.Bundle, | ||||||
|  | 		runc:   runtime, | ||||||
|  | 	} | ||||||
|  | 	var ( | ||||||
|  | 		socket *runc.ConsoleSocket | ||||||
|  | 		io     runc.IO | ||||||
|  | 	) | ||||||
|  | 	if r.Terminal { | ||||||
|  | 		if socket, err = runc.NewConsoleSocket(filepath.Join(cwd, "pty.sock")); err != nil { | ||||||
|  | 			return nil, err | ||||||
|  | 		} | ||||||
|  | 	} else { | ||||||
|  | 		// TODO: get uid/gid | ||||||
|  | 		if io, err = runc.NewPipeIO(0, 0); err != nil { | ||||||
|  | 			return nil, err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	opts := &runc.CreateOpts{ | ||||||
|  | 		PidFile:       filepath.Join(cwd, "pid"), | ||||||
|  | 		ConsoleSocket: socket, | ||||||
|  | 		IO:            io, | ||||||
|  | 		NoPivot:       r.NoPivot, | ||||||
|  | 	} | ||||||
|  | 	if err := p.runc.Create(context, r.ID, r.Bundle, opts); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	if socket != nil { | ||||||
|  | 		console, err := socket.ReceiveMaster() | ||||||
|  | 		if err != nil { | ||||||
|  | 			return nil, err | ||||||
|  | 		} | ||||||
|  | 		p.console = console | ||||||
|  | 		if err := copyConsole(context, console, r.Stdin, r.Stdout, r.Stderr, &p.WaitGroup); err != nil { | ||||||
|  | 			return nil, err | ||||||
|  | 		} | ||||||
|  | 	} else { | ||||||
|  | 		if err := copyPipes(context, io, r.Stdin, r.Stdout, r.Stderr, &p.WaitGroup); err != nil { | ||||||
|  | 			return nil, err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	pid, err := runc.ReadPidFile(opts.PidFile) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	p.pid = pid | ||||||
|  | 	return p, nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (p *initProcess) Pid() int { | ||||||
|  | 	return p.pid | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (p *initProcess) Status() int { | ||||||
|  | 	return p.status | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (p *initProcess) Start(context context.Context) error { | ||||||
|  | 	return p.runc.Start(context, p.id) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (p *initProcess) Exited(status int) { | ||||||
|  | 	p.status = status | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (p *initProcess) Delete(context context.Context) error { | ||||||
|  | 	p.killAll(context) | ||||||
|  | 	p.Wait() | ||||||
|  | 	err := p.runc.Delete(context, p.id) | ||||||
|  | 	p.io.Close() | ||||||
|  | 	return err | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (p *initProcess) Resize(ws runc.WinSize) error { | ||||||
|  | 	if p.console == nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	return p.console.Resize(ws) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (p *initProcess) killAll(context context.Context) error { | ||||||
|  | 	return p.runc.Kill(context, p.id, int(syscall.SIGKILL), &runc.KillOpts{ | ||||||
|  | 		All: true, | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  | @ -52,7 +52,9 @@ func main() { | ||||||
| 		} | 		} | ||||||
| 		var ( | 		var ( | ||||||
| 			server = grpc.NewServer() | 			server = grpc.NewServer() | ||||||
| 			sv     = &service{} | 			sv     = &service{ | ||||||
|  | 				processes: make(map[int]process), | ||||||
|  | 			} | ||||||
| 		) | 		) | ||||||
| 		shim.RegisterShimServiceServer(server, sv) | 		shim.RegisterShimServiceServer(server, sv) | ||||||
| 		l, err := utils.CreateUnixSocket("shim.sock") | 		l, err := utils.CreateUnixSocket("shim.sock") | ||||||
|  |  | ||||||
|  | @ -1,301 +1,25 @@ | ||||||
| package main | package main | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"encoding/json" | 	"context" | ||||||
| 	"errors" | 	"errors" | ||||||
| 	"fmt" | 
 | ||||||
| 	"io" | 	runc "github.com/crosbymichael/go-runc" | ||||||
| 	"io/ioutil" |  | ||||||
| 	"os" |  | ||||||
| 	"os/exec" |  | ||||||
| 	"path/filepath" |  | ||||||
| 	"runtime" |  | ||||||
| 	"strconv" |  | ||||||
| 	"sync" |  | ||||||
| 	"syscall" |  | ||||||
| 	"time" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| var errRuntime = errors.New("shim: runtime execution error") | var errRuntime = errors.New("shim: runtime execution error") | ||||||
| 
 | 
 | ||||||
| type checkpoint struct { | type process interface { | ||||||
| 	// Timestamp is the time that checkpoint happened | 	// Pid returns the pid for the process | ||||||
| 	Created time.Time `json:"created"` | 	Pid() int | ||||||
| 	// Name is the name of the checkpoint | 	// Start starts the user's defined process inside | ||||||
| 	Name string `json:"name"` | 	Start(context.Context) error | ||||||
| 	// TCP checkpoints open tcp connections | 	// Delete deletes the process and closes all open pipes | ||||||
| 	TCP bool `json:"tcp"` | 	Delete(context.Context) error | ||||||
| 	// UnixSockets persists unix sockets in the checkpoint | 	// Resize resizes the process console | ||||||
| 	UnixSockets bool `json:"unixSockets"` | 	Resize(ws runc.WinSize) error | ||||||
| 	// Shell persists tty sessions in the checkpoint | 	// Exited sets the exit status for the process | ||||||
| 	Shell bool `json:"shell"` | 	Exited(status int) | ||||||
| 	// Exit exits the container after the checkpoint is finished | 	// Status returns the exit status | ||||||
| 	Exit bool `json:"exit"` | 	Status() int | ||||||
| 	// EmptyNS tells CRIU not to restore a particular namespace |  | ||||||
| 	EmptyNS []string `json:"emptyNS,omitempty"` |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /* |  | ||||||
| type processState struct { |  | ||||||
| 	Terminal    bool     `json:"terminal"` |  | ||||||
| 	Exec        bool     `json:"exec"` |  | ||||||
| 	Stdin       string   `json:"containerdStdin"` |  | ||||||
| 	Stdout      string   `json:"containerdStdout"` |  | ||||||
| 	Stderr      string   `json:"containerdStderr"` |  | ||||||
| 	RuntimeArgs []string `json:"runtimeArgs"` |  | ||||||
| 
 |  | ||||||
| 	NoPivotRoot    bool   `json:"noPivotRoot"` |  | ||||||
| 	CheckpointPath string `json:"checkpoint"` |  | ||||||
| 	RootUID        int    `json:"rootUID"` |  | ||||||
| 	RootGID        int    `json:"rootGID"` |  | ||||||
| } |  | ||||||
| */ |  | ||||||
| 
 |  | ||||||
| type process struct { |  | ||||||
| 	sync.WaitGroup |  | ||||||
| 	id             string |  | ||||||
| 	bundle         string |  | ||||||
| 	stdio          *stdio |  | ||||||
| 	exec           bool |  | ||||||
| 	containerPid   int |  | ||||||
| 	checkpoint     *checkpoint |  | ||||||
| 	checkpointPath string |  | ||||||
| 	shimIO         *IO |  | ||||||
| 	stdinCloser    io.Closer |  | ||||||
| 	console        *os.File |  | ||||||
| 	consolePath    string |  | ||||||
| 	runtime        string |  | ||||||
| 	exitStatus     int |  | ||||||
| 	Stdin          string `json:"containerdStdin"` |  | ||||||
| 	Stdout         string `json:"containerdStdout"` |  | ||||||
| 	Stderr         string `json:"containerdStderr"` |  | ||||||
| 	Terminal       bool   `json:"terminal"` |  | ||||||
| 	RootUID        int    `json:"rootUID"` |  | ||||||
| 	RootGID        int    `json:"rootGID"` |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func newProcess(id, bundle, runtimeName string) (*process, error) { |  | ||||||
| 	p := &process{ |  | ||||||
| 		id:      id, |  | ||||||
| 		bundle:  bundle, |  | ||||||
| 		runtime: runtimeName, |  | ||||||
| 	} |  | ||||||
| 	if err := p.openIO(); err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
| 	return p, nil |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func loadCheckpoint(checkpointPath string) (*checkpoint, error) { |  | ||||||
| 	f, err := os.Open(filepath.Join(checkpointPath, "config.json")) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
| 	defer f.Close() |  | ||||||
| 	var cpt checkpoint |  | ||||||
| 	if err := json.NewDecoder(f).Decode(&cpt); err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
| 	return &cpt, nil |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (p *process) create(isExec bool) error { |  | ||||||
| 	cwd, err := os.Getwd() |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	logPath := filepath.Join(cwd, "log.json") |  | ||||||
| 	args := []string{ |  | ||||||
| 		"--log", logPath, |  | ||||||
| 		"--log-format", "json", |  | ||||||
| 	} |  | ||||||
| 	if isExec { |  | ||||||
| 		args = append(args, "exec", |  | ||||||
| 			"-d", |  | ||||||
| 			"--process", filepath.Join(cwd, "process.json"), |  | ||||||
| 			"--console-socket", p.consolePath, |  | ||||||
| 		) |  | ||||||
| 	} else if p.checkpoint != nil { |  | ||||||
| 		args = append(args, "restore", |  | ||||||
| 			"-d", |  | ||||||
| 			"--image-path", p.checkpointPath, |  | ||||||
| 			"--work-path", filepath.Join(p.checkpointPath, "criu.work", "restore-"+time.Now().Format(time.RFC3339)), |  | ||||||
| 		) |  | ||||||
| 		add := func(flags ...string) { |  | ||||||
| 			args = append(args, flags...) |  | ||||||
| 		} |  | ||||||
| 		if p.checkpoint.Shell { |  | ||||||
| 			add("--shell-job") |  | ||||||
| 		} |  | ||||||
| 		if p.checkpoint.TCP { |  | ||||||
| 			add("--tcp-established") |  | ||||||
| 		} |  | ||||||
| 		if p.checkpoint.UnixSockets { |  | ||||||
| 			add("--ext-unix-sk") |  | ||||||
| 		} |  | ||||||
| 		/* |  | ||||||
| 			if p.state.NoPivotRoot { |  | ||||||
| 				add("--no-pivot") |  | ||||||
| 			} |  | ||||||
| 		*/ |  | ||||||
| 		for _, ns := range p.checkpoint.EmptyNS { |  | ||||||
| 			add("--empty-ns", ns) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 	} else { |  | ||||||
| 		args = append(args, "create", |  | ||||||
| 			"--bundle", p.bundle, |  | ||||||
| 			"--console-socket", p.consolePath, |  | ||||||
| 		) |  | ||||||
| 		/* |  | ||||||
| 			if p.state.NoPivotRoot { |  | ||||||
| 				args = append(args, "--no-pivot") |  | ||||||
| 			} |  | ||||||
| 		*/ |  | ||||||
| 	} |  | ||||||
| 	args = append(args, |  | ||||||
| 		"--pid-file", filepath.Join(cwd, "pid"), |  | ||||||
| 		p.id, |  | ||||||
| 	) |  | ||||||
| 	cmd := exec.Command(p.runtime, args...) |  | ||||||
| 	cmd.Dir = p.bundle |  | ||||||
| 	if p.stdio != nil { |  | ||||||
| 		cmd.Stdin = p.stdio.stdin |  | ||||||
| 		cmd.Stdout = p.stdio.stdout |  | ||||||
| 		cmd.Stderr = p.stdio.stderr |  | ||||||
| 	} |  | ||||||
| 	// Call out to setPDeathSig to set SysProcAttr as elements are platform specific |  | ||||||
| 	cmd.SysProcAttr = setPDeathSig() |  | ||||||
| 
 |  | ||||||
| 	if err := cmd.Start(); err != nil { |  | ||||||
| 		if exErr, ok := err.(*exec.Error); ok { |  | ||||||
| 			if exErr.Err == exec.ErrNotFound || exErr.Err == os.ErrNotExist { |  | ||||||
| 				return fmt.Errorf("%s not installed on system", p.runtime) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	if runtime.GOOS != "solaris" { |  | ||||||
| 		// Since current logic dictates that we need a pid at the end of p.create |  | ||||||
| 		// we need to call runtime start as well on Solaris hence we need the |  | ||||||
| 		// pipes to stay open. |  | ||||||
| 		if p.stdio != nil { |  | ||||||
| 			p.stdio.stdout.Close() |  | ||||||
| 			p.stdio.stderr.Close() |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	if err := cmd.Wait(); err != nil { |  | ||||||
| 		if _, ok := err.(*exec.ExitError); ok { |  | ||||||
| 			return errRuntime |  | ||||||
| 		} |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	data, err := ioutil.ReadFile("pid") |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	pid, err := strconv.Atoi(string(data)) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	p.containerPid = pid |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (p *process) pid() int { |  | ||||||
| 	return p.containerPid |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (p *process) delete() error { |  | ||||||
| 	cmd := exec.Command(p.runtime, "delete", p.id) |  | ||||||
| 	cmd.SysProcAttr = setPDeathSig() |  | ||||||
| 	out, err := cmd.CombinedOutput() |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("%s: %v", out, err) |  | ||||||
| 	} |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // IO holds all 3 standard io Reader/Writer (stdin,stdout,stderr) |  | ||||||
| type IO struct { |  | ||||||
| 	Stdin  io.WriteCloser |  | ||||||
| 	Stdout io.ReadCloser |  | ||||||
| 	Stderr io.ReadCloser |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (p *process) initializeIO(rootuid int) (i *IO, err error) { |  | ||||||
| 	var fds []uintptr |  | ||||||
| 	i = &IO{} |  | ||||||
| 	// cleanup in case of an error |  | ||||||
| 	defer func() { |  | ||||||
| 		if err != nil { |  | ||||||
| 			for _, fd := range fds { |  | ||||||
| 				syscall.Close(int(fd)) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	}() |  | ||||||
| 	// STDIN |  | ||||||
| 	r, w, err := os.Pipe() |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
| 	fds = append(fds, r.Fd(), w.Fd()) |  | ||||||
| 	p.stdio.stdin, i.Stdin = r, w |  | ||||||
| 	// STDOUT |  | ||||||
| 	if r, w, err = os.Pipe(); err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
| 	fds = append(fds, r.Fd(), w.Fd()) |  | ||||||
| 	p.stdio.stdout, i.Stdout = w, r |  | ||||||
| 	// STDERR |  | ||||||
| 	if r, w, err = os.Pipe(); err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
| 	fds = append(fds, r.Fd(), w.Fd()) |  | ||||||
| 	p.stdio.stderr, i.Stderr = w, r |  | ||||||
| 	// change ownership of the pipes in case we are in a user namespace |  | ||||||
| 	for _, fd := range fds { |  | ||||||
| 		if err := syscall.Fchown(int(fd), rootuid, rootuid); err != nil { |  | ||||||
| 			return nil, err |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	return i, nil |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (p *process) Close() error { |  | ||||||
| 	if p.stdio == nil { |  | ||||||
| 		return nil |  | ||||||
| 	} |  | ||||||
| 	return p.stdio.Close() |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (p *process) start() error { |  | ||||||
| 	cmd := exec.Command(p.runtime, "start", p.id) |  | ||||||
| 	cmd.SysProcAttr = setPDeathSig() |  | ||||||
| 	out, err := cmd.CombinedOutput() |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("%s: %v", out, err) |  | ||||||
| 	} |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (p *process) setExited(status int) { |  | ||||||
| 	p.exitStatus = status |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| type stdio struct { |  | ||||||
| 	stdin  *os.File |  | ||||||
| 	stdout *os.File |  | ||||||
| 	stderr *os.File |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (s *stdio) Close() error { |  | ||||||
| 	err := s.stdin.Close() |  | ||||||
| 	if oerr := s.stdout.Close(); err == nil { |  | ||||||
| 		err = oerr |  | ||||||
| 	} |  | ||||||
| 	if oerr := s.stderr.Close(); err == nil { |  | ||||||
| 		err = oerr |  | ||||||
| 	} |  | ||||||
| 	return err |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -1,95 +1,57 @@ | ||||||
| // +build !solaris |  | ||||||
| 
 |  | ||||||
| package main | package main | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
|  | 	"context" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"io" | 	"io" | ||||||
| 	"os/exec" | 	"sync" | ||||||
| 	"syscall" | 	"syscall" | ||||||
| 	"time" |  | ||||||
| 
 | 
 | ||||||
|  | 	runc "github.com/crosbymichael/go-runc" | ||||||
| 	"github.com/tonistiigi/fifo" | 	"github.com/tonistiigi/fifo" | ||||||
| 	"golang.org/x/net/context" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| // setPDeathSig sets the parent death signal to SIGKILL so that if the | func copyConsole(ctx context.Context, console *runc.Console, stdin, stdout, stderr string, wg *sync.WaitGroup) error { | ||||||
| // shim dies the container process also dies. | 	in, err := fifo.OpenFifo(ctx, stdin, syscall.O_RDONLY, 0) | ||||||
| func setPDeathSig() *syscall.SysProcAttr { | 	if err != nil { | ||||||
| 	return &syscall.SysProcAttr{ | 		return err | ||||||
| 		Pdeathsig: syscall.SIGKILL, |  | ||||||
| 	} | 	} | ||||||
|  | 	go io.Copy(console, in) | ||||||
|  | 	outw, err := fifo.OpenFifo(ctx, stdout, syscall.O_WRONLY, 0) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	outr, err := fifo.OpenFifo(ctx, stdout, syscall.O_RDONLY, 0) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	wg.Add(1) | ||||||
|  | 	go func() { | ||||||
|  | 		io.Copy(outw, console) | ||||||
|  | 		console.Close() | ||||||
|  | 		outr.Close() | ||||||
|  | 		outw.Close() | ||||||
|  | 		wg.Done() | ||||||
|  | 	}() | ||||||
|  | 	return nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // openIO opens the pre-created fifo's for use with the container | func copyPipes(ctx context.Context, rio runc.IO, stdin, stdout, stderr string, wg *sync.WaitGroup) error { | ||||||
| // in RDWR so that they remain open if the other side stops listening |  | ||||||
| func (p *process) openIO() error { |  | ||||||
| 	return nil |  | ||||||
| 	p.stdio = &stdio{} |  | ||||||
| 	var ( |  | ||||||
| 		uid = p.RootUID |  | ||||||
| 		gid = p.RootGID |  | ||||||
| 	) |  | ||||||
| 
 |  | ||||||
| 	ctx, _ := context.WithTimeout(context.Background(), 15*time.Second) |  | ||||||
| 
 |  | ||||||
| 	stdinCloser, err := fifo.OpenFifo(ctx, p.Stdin, syscall.O_WRONLY|syscall.O_NONBLOCK, 0) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	p.stdinCloser = stdinCloser |  | ||||||
| 
 |  | ||||||
| 	if p.Terminal { |  | ||||||
| 		master, console, err := newConsole(uid, gid) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 		p.console = master |  | ||||||
| 		p.consolePath = console |  | ||||||
| 		stdin, err := fifo.OpenFifo(ctx, p.Stdin, syscall.O_RDONLY, 0) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 		go io.Copy(master, stdin) |  | ||||||
| 		stdoutw, err := fifo.OpenFifo(ctx, p.Stdout, syscall.O_WRONLY, 0) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 		stdoutr, err := fifo.OpenFifo(ctx, p.Stdout, syscall.O_RDONLY, 0) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 		p.Add(1) |  | ||||||
| 		go func() { |  | ||||||
| 			io.Copy(stdoutw, master) |  | ||||||
| 			master.Close() |  | ||||||
| 			stdoutr.Close() |  | ||||||
| 			stdoutw.Close() |  | ||||||
| 			p.Done() |  | ||||||
| 		}() |  | ||||||
| 		return nil |  | ||||||
| 	} |  | ||||||
| 	i, err := p.initializeIO(uid) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	p.shimIO = i |  | ||||||
| 	// non-tty |  | ||||||
| 	for name, dest := range map[string]func(wc io.WriteCloser, rc io.Closer){ | 	for name, dest := range map[string]func(wc io.WriteCloser, rc io.Closer){ | ||||||
| 		p.Stdout: func(wc io.WriteCloser, rc io.Closer) { | 		stdout: func(wc io.WriteCloser, rc io.Closer) { | ||||||
| 			p.Add(1) | 			wg.Add(1) | ||||||
| 			go func() { | 			go func() { | ||||||
| 				io.Copy(wc, i.Stdout) | 				io.Copy(wc, rio.Stdout()) | ||||||
| 				p.Done() | 				wg.Done() | ||||||
| 				wc.Close() | 				wc.Close() | ||||||
| 				rc.Close() | 				rc.Close() | ||||||
| 			}() | 			}() | ||||||
| 		}, | 		}, | ||||||
| 		p.Stderr: func(wc io.WriteCloser, rc io.Closer) { | 		stderr: func(wc io.WriteCloser, rc io.Closer) { | ||||||
| 			p.Add(1) | 			wg.Add(1) | ||||||
| 			go func() { | 			go func() { | ||||||
| 				io.Copy(wc, i.Stderr) | 				io.Copy(wc, rio.Stderr()) | ||||||
| 				p.Done() | 				wg.Done() | ||||||
| 				wc.Close() | 				wc.Close() | ||||||
| 				rc.Close() | 				rc.Close() | ||||||
| 			}() | 			}() | ||||||
|  | @ -106,25 +68,14 @@ func (p *process) openIO() error { | ||||||
| 		dest(fw, fr) | 		dest(fw, fr) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	f, err := fifo.OpenFifo(ctx, p.Stdin, syscall.O_RDONLY, 0) | 	f, err := fifo.OpenFifo(ctx, stdin, syscall.O_RDONLY, 0) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return fmt.Errorf("containerd-shim: opening %s failed: %s", p.Stdin, err) | 		return fmt.Errorf("containerd-shim: opening %s failed: %s", stdin, err) | ||||||
| 	} | 	} | ||||||
| 	go func() { | 	go func() { | ||||||
| 		io.Copy(i.Stdin, f) | 		io.Copy(rio.Stdin(), f) | ||||||
| 		i.Stdin.Close() | 		rio.Stdin().Close() | ||||||
| 		f.Close() | 		f.Close() | ||||||
| 	}() | 	}() | ||||||
| 
 |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (p *process) killAll() error { |  | ||||||
| 	cmd := exec.Command(p.runtime, "kill", "--all", p.id, "SIGKILL") |  | ||||||
| 	cmd.SysProcAttr = setPDeathSig() |  | ||||||
| 	out, err := cmd.CombinedOutput() |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("%s: %v", out, err) |  | ||||||
| 	} |  | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -1,70 +0,0 @@ | ||||||
| // +build solaris |  | ||||||
| 
 |  | ||||||
| package main |  | ||||||
| 
 |  | ||||||
| import ( |  | ||||||
| 	"io" |  | ||||||
| 	"os" |  | ||||||
| 	"syscall" |  | ||||||
| ) |  | ||||||
| 
 |  | ||||||
| // setPDeathSig is a no-op on Solaris as Pdeathsig is not defined. |  | ||||||
| func setPDeathSig() *syscall.SysProcAttr { |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // TODO: Update to using fifo's package in openIO. Need to |  | ||||||
| // 1. Merge and vendor changes in the package to use sys/unix. |  | ||||||
| // 2. Figure out why context.Background is timing out. |  | ||||||
| // openIO opens the pre-created fifo's for use with the container |  | ||||||
| // in RDWR so that they remain open if the other side stops listening |  | ||||||
| func (p *process) openIO() error { |  | ||||||
| 	p.stdio = &stdio{} |  | ||||||
| 	var ( |  | ||||||
| 		uid = p.state.RootUID |  | ||||||
| 	) |  | ||||||
| 	i, err := p.initializeIO(uid) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	p.shimIO = i |  | ||||||
| 	// Both tty and non-tty mode are handled by the runtime using |  | ||||||
| 	// the following pipes |  | ||||||
| 	for name, dest := range map[string]func(f *os.File){ |  | ||||||
| 		p.state.Stdout: func(f *os.File) { |  | ||||||
| 			p.Add(1) |  | ||||||
| 			go func() { |  | ||||||
| 				io.Copy(f, i.Stdout) |  | ||||||
| 				p.Done() |  | ||||||
| 			}() |  | ||||||
| 		}, |  | ||||||
| 		p.state.Stderr: func(f *os.File) { |  | ||||||
| 			p.Add(1) |  | ||||||
| 			go func() { |  | ||||||
| 				io.Copy(f, i.Stderr) |  | ||||||
| 				p.Done() |  | ||||||
| 			}() |  | ||||||
| 		}, |  | ||||||
| 	} { |  | ||||||
| 		f, err := os.OpenFile(name, syscall.O_RDWR, 0) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 		dest(f) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	f, err := os.OpenFile(p.state.Stdin, syscall.O_RDONLY, 0) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	go func() { |  | ||||||
| 		io.Copy(i.Stdin, f) |  | ||||||
| 		i.Stdin.Close() |  | ||||||
| 	}() |  | ||||||
| 
 |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (p *process) killAll() error { |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  | @ -1,9 +1,12 @@ | ||||||
| package main | package main | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"sync" | ||||||
|  | 
 | ||||||
|  | 	runc "github.com/crosbymichael/go-runc" | ||||||
| 	"github.com/docker/containerd/api/shim" | 	"github.com/docker/containerd/api/shim" | ||||||
| 	"github.com/docker/containerd/utils" | 	"github.com/docker/containerd/utils" | ||||||
| 	"github.com/docker/docker/pkg/term" |  | ||||||
| 	google_protobuf "github.com/golang/protobuf/ptypes/empty" | 	google_protobuf "github.com/golang/protobuf/ptypes/empty" | ||||||
| 	"golang.org/x/net/context" | 	"golang.org/x/net/context" | ||||||
| ) | ) | ||||||
|  | @ -11,43 +14,51 @@ import ( | ||||||
| var emptyResponse = &google_protobuf.Empty{} | var emptyResponse = &google_protobuf.Empty{} | ||||||
| 
 | 
 | ||||||
| type service struct { | type service struct { | ||||||
| 	init *process | 	initPid   int | ||||||
|  | 	mu        sync.Mutex | ||||||
|  | 	processes map[int]process | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *service) Create(ctx context.Context, r *shim.CreateRequest) (*shim.CreateResponse, error) { | func (s *service) Create(ctx context.Context, r *shim.CreateRequest) (*shim.CreateResponse, error) { | ||||||
| 	process, err := newProcess(r.ID, r.Bundle, r.Runtime) | 	process, err := newInitProcess(ctx, r) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	s.init = process | 	s.mu.Lock() | ||||||
| 	if err := process.create(false); err != nil { | 	pid := process.Pid() | ||||||
| 		return nil, err | 	s.initPid, s.processes[pid] = pid, process | ||||||
| 	} | 	s.mu.Unlock() | ||||||
| 	return &shim.CreateResponse{ | 	return &shim.CreateResponse{ | ||||||
| 		Pid: uint32(process.pid()), | 		Pid: uint32(pid), | ||||||
| 	}, nil | 	}, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *service) Start(ctx context.Context, r *shim.StartRequest) (*google_protobuf.Empty, error) { | func (s *service) Start(ctx context.Context, r *shim.StartRequest) (*google_protobuf.Empty, error) { | ||||||
| 	if err := s.init.start(); err != nil { | 	s.mu.Lock() | ||||||
|  | 	p := s.processes[s.initPid] | ||||||
|  | 	s.mu.Unlock() | ||||||
|  | 	if err := p.Start(ctx); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	return emptyResponse, nil | 	return emptyResponse, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *service) Delete(ctx context.Context, r *shim.DeleteRequest) (*shim.DeleteResponse, error) { | func (s *service) Delete(ctx context.Context, r *shim.DeleteRequest) (*shim.DeleteResponse, error) { | ||||||
| 	// TODO: error when container has not stopped | 	s.mu.Lock() | ||||||
| 	err := s.init.killAll() | 	p, ok := s.processes[int(r.Pid)] | ||||||
| 	s.init.Wait() | 	s.mu.Unlock() | ||||||
| 	if derr := s.init.delete(); err == nil { | 	if !ok { | ||||||
| 		err = derr | 		return nil, fmt.Errorf("process does not exist %d", r.Pid) | ||||||
| 	} | 	} | ||||||
| 	if cerr := s.init.Close(); err == nil { | 	if err := p.Delete(ctx); err != nil { | ||||||
| 		err = cerr | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 	s.mu.Lock() | ||||||
|  | 	delete(s.processes, int(r.Pid)) | ||||||
|  | 	s.mu.Unlock() | ||||||
| 	return &shim.DeleteResponse{ | 	return &shim.DeleteResponse{ | ||||||
| 		ExitStatus: uint32(s.init.exitStatus), | 		ExitStatus: uint32(p.Status()), | ||||||
| 	}, err | 	}, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *service) Exec(ctx context.Context, r *shim.ExecRequest) (*shim.ExecResponse, error) { | func (s *service) Exec(ctx context.Context, r *shim.ExecRequest) (*shim.ExecResponse, error) { | ||||||
|  | @ -55,22 +66,27 @@ func (s *service) Exec(ctx context.Context, r *shim.ExecRequest) (*shim.ExecResp | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *service) Pty(ctx context.Context, r *shim.PtyRequest) (*google_protobuf.Empty, error) { | func (s *service) Pty(ctx context.Context, r *shim.PtyRequest) (*google_protobuf.Empty, error) { | ||||||
| 	if s.init.console == nil { | 	ws := runc.WinSize{ | ||||||
| 		return emptyResponse, nil |  | ||||||
| 	} |  | ||||||
| 	ws := term.Winsize{ |  | ||||||
| 		Width:  uint16(r.Width), | 		Width:  uint16(r.Width), | ||||||
| 		Height: uint16(r.Height), | 		Height: uint16(r.Height), | ||||||
| 	} | 	} | ||||||
| 	if err := term.SetWinsize(s.init.console.Fd(), &ws); err != nil { | 	s.mu.Lock() | ||||||
|  | 	p, ok := s.processes[int(r.Pid)] | ||||||
|  | 	s.mu.Unlock() | ||||||
|  | 	if !ok { | ||||||
|  | 		return nil, fmt.Errorf("process does not exist %d", r.Pid) | ||||||
|  | 	} | ||||||
|  | 	if err := p.Resize(ws); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	return emptyResponse, nil | 	return emptyResponse, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *service) processExited(e utils.Exit) error { | func (s *service) processExited(e utils.Exit) error { | ||||||
| 	if s.init.pid() == e.Pid { | 	s.mu.Lock() | ||||||
| 		s.init.setExited(e.Status) | 	if p, ok := s.processes[e.Pid]; ok { | ||||||
|  | 		p.Exited(e.Status) | ||||||
| 	} | 	} | ||||||
|  | 	s.mu.Unlock() | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue