99 lines
		
	
	
		
			2.4 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			99 lines
		
	
	
		
			2.4 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| /*
 | |
|    Copyright The containerd Authors.
 | |
| 
 | |
|    Licensed under the Apache License, Version 2.0 (the "License");
 | |
|    you may not use this file except in compliance with the License.
 | |
|    You may obtain a copy of the License at
 | |
| 
 | |
|        http://www.apache.org/licenses/LICENSE-2.0
 | |
| 
 | |
|    Unless required by applicable law or agreed to in writing, software
 | |
|    distributed under the License is distributed on an "AS IS" BASIS,
 | |
|    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | |
|    See the License for the specific language governing permissions and
 | |
|    limitations under the License.
 | |
| */
 | |
| 
 | |
| /*
 | |
|    This file is copied and customized based on
 | |
|    https://github.com/moby/moby/blob/master/pkg/idtools/idtools.go
 | |
| */
 | |
| 
 | |
| package userns
 | |
| 
 | |
| import (
 | |
| 	"errors"
 | |
| 	"fmt"
 | |
| 
 | |
| 	"github.com/opencontainers/runtime-spec/specs-go"
 | |
| )
 | |
| 
 | |
| const invalidID = 1<<32 - 1
 | |
| 
 | |
| var invalidUser = User{Uid: invalidID, Gid: invalidID}
 | |
| 
 | |
| // User is a Uid and Gid pair of a user
 | |
| //
 | |
| //nolint:revive
 | |
| type User struct {
 | |
| 	Uid uint32
 | |
| 	Gid uint32
 | |
| }
 | |
| 
 | |
| // IDMap contains the mappings of Uids and Gids.
 | |
| //
 | |
| //nolint:revive
 | |
| type IDMap struct {
 | |
| 	UidMap []specs.LinuxIDMapping `json:"UidMap"`
 | |
| 	GidMap []specs.LinuxIDMapping `json:"GidMap"`
 | |
| }
 | |
| 
 | |
| // ToHost returns the host user ID pair for the container ID pair.
 | |
| func (i IDMap) ToHost(pair User) (User, error) {
 | |
| 	var (
 | |
| 		target User
 | |
| 		err    error
 | |
| 	)
 | |
| 	target.Uid, err = toHost(pair.Uid, i.UidMap)
 | |
| 	if err != nil {
 | |
| 		return invalidUser, err
 | |
| 	}
 | |
| 	target.Gid, err = toHost(pair.Gid, i.GidMap)
 | |
| 	if err != nil {
 | |
| 		return invalidUser, err
 | |
| 	}
 | |
| 	return target, nil
 | |
| }
 | |
| 
 | |
| // toHost takes an id mapping and a remapped ID, and translates the
 | |
| // ID to the mapped host ID. If no map is provided, then the translation
 | |
| // assumes a 1-to-1 mapping and returns the passed in id #
 | |
| func toHost(contID uint32, idMap []specs.LinuxIDMapping) (uint32, error) {
 | |
| 	if idMap == nil {
 | |
| 		return contID, nil
 | |
| 	}
 | |
| 	for _, m := range idMap {
 | |
| 		high, err := safeSum(m.ContainerID, m.Size)
 | |
| 		if err != nil {
 | |
| 			break
 | |
| 		}
 | |
| 		if contID >= m.ContainerID && contID < high {
 | |
| 			hostID, err := safeSum(m.HostID, contID-m.ContainerID)
 | |
| 			if err != nil || hostID == invalidID {
 | |
| 				break
 | |
| 			}
 | |
| 			return hostID, nil
 | |
| 		}
 | |
| 	}
 | |
| 	return invalidID, fmt.Errorf("container ID %d cannot be mapped to a host ID", contID)
 | |
| }
 | |
| 
 | |
| // safeSum returns the sum of x and y. or an error if the result overflows
 | |
| func safeSum(x, y uint32) (uint32, error) {
 | |
| 	z := x + y
 | |
| 	if z < x || z < y {
 | |
| 		return invalidID, errors.New("ID overflow")
 | |
| 	}
 | |
| 	return z, nil
 | |
| }
 | 
