mirror of https://github.com/k3s-io/k3s
447 lines
14 KiB
Go
447 lines
14 KiB
Go
package hcn
|
|
|
|
import (
|
|
"encoding/json"
|
|
"os"
|
|
"syscall"
|
|
|
|
"github.com/Microsoft/go-winio/pkg/guid"
|
|
icni "github.com/Microsoft/hcsshim/internal/cni"
|
|
"github.com/Microsoft/hcsshim/internal/interop"
|
|
"github.com/Microsoft/hcsshim/internal/regstate"
|
|
"github.com/Microsoft/hcsshim/internal/runhcs"
|
|
"github.com/sirupsen/logrus"
|
|
)
|
|
|
|
// NamespaceResourceEndpoint represents an Endpoint attached to a Namespace.
|
|
type NamespaceResourceEndpoint struct {
|
|
Id string `json:"ID,"`
|
|
}
|
|
|
|
// NamespaceResourceContainer represents a Container attached to a Namespace.
|
|
type NamespaceResourceContainer struct {
|
|
Id string `json:"ID,"`
|
|
}
|
|
|
|
// NamespaceResourceType determines whether the Namespace resource is a Container or Endpoint.
|
|
type NamespaceResourceType string
|
|
|
|
var (
|
|
// NamespaceResourceTypeContainer are contianers associated with a Namespace.
|
|
NamespaceResourceTypeContainer NamespaceResourceType = "Container"
|
|
// NamespaceResourceTypeEndpoint are endpoints associated with a Namespace.
|
|
NamespaceResourceTypeEndpoint NamespaceResourceType = "Endpoint"
|
|
)
|
|
|
|
// NamespaceResource is associated with a namespace
|
|
type NamespaceResource struct {
|
|
Type NamespaceResourceType `json:","` // Container, Endpoint
|
|
Data json.RawMessage `json:","`
|
|
}
|
|
|
|
// NamespaceType determines whether the Namespace is for a Host or Guest
|
|
type NamespaceType string
|
|
|
|
var (
|
|
// NamespaceTypeHost are host namespaces.
|
|
NamespaceTypeHost NamespaceType = "Host"
|
|
// NamespaceTypeHostDefault are host namespaces in the default compartment.
|
|
NamespaceTypeHostDefault NamespaceType = "HostDefault"
|
|
// NamespaceTypeGuest are guest namespaces.
|
|
NamespaceTypeGuest NamespaceType = "Guest"
|
|
// NamespaceTypeGuestDefault are guest namespaces in the default compartment.
|
|
NamespaceTypeGuestDefault NamespaceType = "GuestDefault"
|
|
)
|
|
|
|
// HostComputeNamespace represents a namespace (AKA compartment) in
|
|
type HostComputeNamespace struct {
|
|
Id string `json:"ID,omitempty"`
|
|
NamespaceId uint32 `json:",omitempty"`
|
|
Type NamespaceType `json:",omitempty"` // Host, HostDefault, Guest, GuestDefault
|
|
Resources []NamespaceResource `json:",omitempty"`
|
|
SchemaVersion SchemaVersion `json:",omitempty"`
|
|
}
|
|
|
|
// ModifyNamespaceSettingRequest is the structure used to send request to modify a namespace.
|
|
// Used to Add/Remove an endpoints and containers to/from a namespace.
|
|
type ModifyNamespaceSettingRequest struct {
|
|
ResourceType NamespaceResourceType `json:",omitempty"` // Container, Endpoint
|
|
RequestType RequestType `json:",omitempty"` // Add, Remove, Update, Refresh
|
|
Settings json.RawMessage `json:",omitempty"`
|
|
}
|
|
|
|
func getNamespace(namespaceGuid guid.GUID, query string) (*HostComputeNamespace, error) {
|
|
// Open namespace.
|
|
var (
|
|
namespaceHandle hcnNamespace
|
|
resultBuffer *uint16
|
|
propertiesBuffer *uint16
|
|
)
|
|
hr := hcnOpenNamespace(&namespaceGuid, &namespaceHandle, &resultBuffer)
|
|
if err := checkForErrors("hcnOpenNamespace", hr, resultBuffer); err != nil {
|
|
return nil, err
|
|
}
|
|
// Query namespace.
|
|
hr = hcnQueryNamespaceProperties(namespaceHandle, query, &propertiesBuffer, &resultBuffer)
|
|
if err := checkForErrors("hcnQueryNamespaceProperties", hr, resultBuffer); err != nil {
|
|
return nil, err
|
|
}
|
|
properties := interop.ConvertAndFreeCoTaskMemString(propertiesBuffer)
|
|
// Close namespace.
|
|
hr = hcnCloseNamespace(namespaceHandle)
|
|
if err := checkForErrors("hcnCloseNamespace", hr, nil); err != nil {
|
|
return nil, err
|
|
}
|
|
// Convert output to HostComputeNamespace
|
|
var outputNamespace HostComputeNamespace
|
|
if err := json.Unmarshal([]byte(properties), &outputNamespace); err != nil {
|
|
return nil, err
|
|
}
|
|
return &outputNamespace, nil
|
|
}
|
|
|
|
func enumerateNamespaces(query string) ([]HostComputeNamespace, error) {
|
|
// Enumerate all Namespace Guids
|
|
var (
|
|
resultBuffer *uint16
|
|
namespaceBuffer *uint16
|
|
)
|
|
hr := hcnEnumerateNamespaces(query, &namespaceBuffer, &resultBuffer)
|
|
if err := checkForErrors("hcnEnumerateNamespaces", hr, resultBuffer); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
namespaces := interop.ConvertAndFreeCoTaskMemString(namespaceBuffer)
|
|
var namespaceIds []guid.GUID
|
|
if err := json.Unmarshal([]byte(namespaces), &namespaceIds); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
var outputNamespaces []HostComputeNamespace
|
|
for _, namespaceGuid := range namespaceIds {
|
|
namespace, err := getNamespace(namespaceGuid, query)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
outputNamespaces = append(outputNamespaces, *namespace)
|
|
}
|
|
return outputNamespaces, nil
|
|
}
|
|
|
|
func createNamespace(settings string) (*HostComputeNamespace, error) {
|
|
// Create new namespace.
|
|
var (
|
|
namespaceHandle hcnNamespace
|
|
resultBuffer *uint16
|
|
propertiesBuffer *uint16
|
|
)
|
|
namespaceGuid := guid.GUID{}
|
|
hr := hcnCreateNamespace(&namespaceGuid, settings, &namespaceHandle, &resultBuffer)
|
|
if err := checkForErrors("hcnCreateNamespace", hr, resultBuffer); err != nil {
|
|
return nil, err
|
|
}
|
|
// Query namespace.
|
|
hcnQuery := defaultQuery()
|
|
query, err := json.Marshal(hcnQuery)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
hr = hcnQueryNamespaceProperties(namespaceHandle, string(query), &propertiesBuffer, &resultBuffer)
|
|
if err := checkForErrors("hcnQueryNamespaceProperties", hr, resultBuffer); err != nil {
|
|
return nil, err
|
|
}
|
|
properties := interop.ConvertAndFreeCoTaskMemString(propertiesBuffer)
|
|
// Close namespace.
|
|
hr = hcnCloseNamespace(namespaceHandle)
|
|
if err := checkForErrors("hcnCloseNamespace", hr, nil); err != nil {
|
|
return nil, err
|
|
}
|
|
// Convert output to HostComputeNamespace
|
|
var outputNamespace HostComputeNamespace
|
|
if err := json.Unmarshal([]byte(properties), &outputNamespace); err != nil {
|
|
return nil, err
|
|
}
|
|
return &outputNamespace, nil
|
|
}
|
|
|
|
func modifyNamespace(namespaceId string, settings string) (*HostComputeNamespace, error) {
|
|
namespaceGuid, err := guid.FromString(namespaceId)
|
|
if err != nil {
|
|
return nil, errInvalidNamespaceID
|
|
}
|
|
// Open namespace.
|
|
var (
|
|
namespaceHandle hcnNamespace
|
|
resultBuffer *uint16
|
|
propertiesBuffer *uint16
|
|
)
|
|
hr := hcnOpenNamespace(&namespaceGuid, &namespaceHandle, &resultBuffer)
|
|
if err := checkForErrors("hcnOpenNamespace", hr, resultBuffer); err != nil {
|
|
return nil, err
|
|
}
|
|
// Modify namespace.
|
|
hr = hcnModifyNamespace(namespaceHandle, settings, &resultBuffer)
|
|
if err := checkForErrors("hcnModifyNamespace", hr, resultBuffer); err != nil {
|
|
return nil, err
|
|
}
|
|
// Query namespace.
|
|
hcnQuery := defaultQuery()
|
|
query, err := json.Marshal(hcnQuery)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
hr = hcnQueryNamespaceProperties(namespaceHandle, string(query), &propertiesBuffer, &resultBuffer)
|
|
if err := checkForErrors("hcnQueryNamespaceProperties", hr, resultBuffer); err != nil {
|
|
return nil, err
|
|
}
|
|
properties := interop.ConvertAndFreeCoTaskMemString(propertiesBuffer)
|
|
// Close namespace.
|
|
hr = hcnCloseNamespace(namespaceHandle)
|
|
if err := checkForErrors("hcnCloseNamespace", hr, nil); err != nil {
|
|
return nil, err
|
|
}
|
|
// Convert output to Namespace
|
|
var outputNamespace HostComputeNamespace
|
|
if err := json.Unmarshal([]byte(properties), &outputNamespace); err != nil {
|
|
return nil, err
|
|
}
|
|
return &outputNamespace, nil
|
|
}
|
|
|
|
func deleteNamespace(namespaceId string) error {
|
|
namespaceGuid, err := guid.FromString(namespaceId)
|
|
if err != nil {
|
|
return errInvalidNamespaceID
|
|
}
|
|
var resultBuffer *uint16
|
|
hr := hcnDeleteNamespace(&namespaceGuid, &resultBuffer)
|
|
if err := checkForErrors("hcnDeleteNamespace", hr, resultBuffer); err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// ListNamespaces makes a call to list all available namespaces.
|
|
func ListNamespaces() ([]HostComputeNamespace, error) {
|
|
hcnQuery := defaultQuery()
|
|
namespaces, err := ListNamespacesQuery(hcnQuery)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return namespaces, nil
|
|
}
|
|
|
|
// ListNamespacesQuery makes a call to query the list of available namespaces.
|
|
func ListNamespacesQuery(query HostComputeQuery) ([]HostComputeNamespace, error) {
|
|
queryJson, err := json.Marshal(query)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
namespaces, err := enumerateNamespaces(string(queryJson))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return namespaces, nil
|
|
}
|
|
|
|
// GetNamespaceByID returns the Namespace specified by Id.
|
|
func GetNamespaceByID(namespaceId string) (*HostComputeNamespace, error) {
|
|
hcnQuery := defaultQuery()
|
|
mapA := map[string]string{"ID": namespaceId}
|
|
filter, err := json.Marshal(mapA)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
hcnQuery.Filter = string(filter)
|
|
|
|
namespaces, err := ListNamespacesQuery(hcnQuery)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if len(namespaces) == 0 {
|
|
return nil, NamespaceNotFoundError{NamespaceID: namespaceId}
|
|
}
|
|
|
|
return &namespaces[0], err
|
|
}
|
|
|
|
// GetNamespaceEndpointIds returns the endpoints of the Namespace specified by Id.
|
|
func GetNamespaceEndpointIds(namespaceId string) ([]string, error) {
|
|
namespace, err := GetNamespaceByID(namespaceId)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
var endpointsIds []string
|
|
for _, resource := range namespace.Resources {
|
|
if resource.Type == "Endpoint" {
|
|
var endpointResource NamespaceResourceEndpoint
|
|
if err := json.Unmarshal([]byte(resource.Data), &endpointResource); err != nil {
|
|
return nil, err
|
|
}
|
|
endpointsIds = append(endpointsIds, endpointResource.Id)
|
|
}
|
|
}
|
|
return endpointsIds, nil
|
|
}
|
|
|
|
// GetNamespaceContainerIds returns the containers of the Namespace specified by Id.
|
|
func GetNamespaceContainerIds(namespaceId string) ([]string, error) {
|
|
namespace, err := GetNamespaceByID(namespaceId)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
var containerIds []string
|
|
for _, resource := range namespace.Resources {
|
|
if resource.Type == "Container" {
|
|
var contaienrResource NamespaceResourceContainer
|
|
if err := json.Unmarshal([]byte(resource.Data), &contaienrResource); err != nil {
|
|
return nil, err
|
|
}
|
|
containerIds = append(containerIds, contaienrResource.Id)
|
|
}
|
|
}
|
|
return containerIds, nil
|
|
}
|
|
|
|
// NewNamespace creates a new Namespace object
|
|
func NewNamespace(nsType NamespaceType) *HostComputeNamespace {
|
|
return &HostComputeNamespace{
|
|
Type: nsType,
|
|
SchemaVersion: V2SchemaVersion(),
|
|
}
|
|
}
|
|
|
|
// Create Namespace.
|
|
func (namespace *HostComputeNamespace) Create() (*HostComputeNamespace, error) {
|
|
logrus.Debugf("hcn::HostComputeNamespace::Create id=%s", namespace.Id)
|
|
|
|
jsonString, err := json.Marshal(namespace)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
logrus.Debugf("hcn::HostComputeNamespace::Create JSON: %s", jsonString)
|
|
namespace, hcnErr := createNamespace(string(jsonString))
|
|
if hcnErr != nil {
|
|
return nil, hcnErr
|
|
}
|
|
return namespace, nil
|
|
}
|
|
|
|
// Delete Namespace.
|
|
func (namespace *HostComputeNamespace) Delete() error {
|
|
logrus.Debugf("hcn::HostComputeNamespace::Delete id=%s", namespace.Id)
|
|
|
|
if err := deleteNamespace(namespace.Id); err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Sync Namespace endpoints with the appropriate sandbox container holding the
|
|
// network namespace open. If no sandbox container is found for this namespace
|
|
// this method is determined to be a success and will not return an error in
|
|
// this case. If the sandbox container is found and a sync is initiated any
|
|
// failures will be returned via this method.
|
|
//
|
|
// This call initiates a sync between endpoints and the matching UtilityVM
|
|
// hosting those endpoints. It is safe to call for any `NamespaceType` but
|
|
// `NamespaceTypeGuest` is the only case when a sync will actually occur. For
|
|
// `NamespaceTypeHost` the process container will be automatically synchronized
|
|
// when the the endpoint is added via `AddNamespaceEndpoint`.
|
|
//
|
|
// Note: This method sync's both additions and removals of endpoints from a
|
|
// `NamespaceTypeGuest` namespace.
|
|
func (namespace *HostComputeNamespace) Sync() error {
|
|
logrus.WithField("id", namespace.Id).Debugf("hcs::HostComputeNamespace::Sync")
|
|
|
|
// We only attempt a sync for namespace guest.
|
|
if namespace.Type != NamespaceTypeGuest {
|
|
return nil
|
|
}
|
|
|
|
// Look in the registry for the key to map from namespace id to pod-id
|
|
cfg, err := icni.LoadPersistedNamespaceConfig(namespace.Id)
|
|
if err != nil {
|
|
if regstate.IsNotFoundError(err) {
|
|
return nil
|
|
}
|
|
return err
|
|
}
|
|
req := runhcs.VMRequest{
|
|
ID: cfg.ContainerID,
|
|
Op: runhcs.OpSyncNamespace,
|
|
}
|
|
shimPath := runhcs.VMPipePath(cfg.HostUniqueID)
|
|
if err := runhcs.IssueVMRequest(shimPath, &req); err != nil {
|
|
// The shim is likey gone. Simply ignore the sync as if it didn't exist.
|
|
if perr, ok := err.(*os.PathError); ok && perr.Err == syscall.ERROR_FILE_NOT_FOUND {
|
|
// Remove the reg key there is no point to try again
|
|
_ = cfg.Remove()
|
|
return nil
|
|
}
|
|
f := map[string]interface{}{
|
|
"id": namespace.Id,
|
|
"container-id": cfg.ContainerID,
|
|
}
|
|
logrus.WithFields(f).
|
|
WithError(err).
|
|
Debugf("hcs::HostComputeNamespace::Sync failed to connect to shim pipe: '%s'", shimPath)
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// ModifyNamespaceSettings updates the Endpoints/Containers of a Namespace.
|
|
func ModifyNamespaceSettings(namespaceId string, request *ModifyNamespaceSettingRequest) error {
|
|
logrus.Debugf("hcn::HostComputeNamespace::ModifyNamespaceSettings id=%s", namespaceId)
|
|
|
|
namespaceSettings, err := json.Marshal(request)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
_, err = modifyNamespace(namespaceId, string(namespaceSettings))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// AddNamespaceEndpoint adds an endpoint to a Namespace.
|
|
func AddNamespaceEndpoint(namespaceId string, endpointId string) error {
|
|
logrus.Debugf("hcn::HostComputeEndpoint::AddNamespaceEndpoint id=%s", endpointId)
|
|
|
|
mapA := map[string]string{"EndpointId": endpointId}
|
|
settingsJson, err := json.Marshal(mapA)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
requestMessage := &ModifyNamespaceSettingRequest{
|
|
ResourceType: NamespaceResourceTypeEndpoint,
|
|
RequestType: RequestTypeAdd,
|
|
Settings: settingsJson,
|
|
}
|
|
|
|
return ModifyNamespaceSettings(namespaceId, requestMessage)
|
|
}
|
|
|
|
// RemoveNamespaceEndpoint removes an endpoint from a Namespace.
|
|
func RemoveNamespaceEndpoint(namespaceId string, endpointId string) error {
|
|
logrus.Debugf("hcn::HostComputeNamespace::RemoveNamespaceEndpoint id=%s", endpointId)
|
|
|
|
mapA := map[string]string{"EndpointId": endpointId}
|
|
settingsJson, err := json.Marshal(mapA)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
requestMessage := &ModifyNamespaceSettingRequest{
|
|
ResourceType: NamespaceResourceTypeEndpoint,
|
|
RequestType: RequestTypeRemove,
|
|
Settings: settingsJson,
|
|
}
|
|
|
|
return ModifyNamespaceSettings(namespaceId, requestMessage)
|
|
}
|