mirror of https://github.com/1Panel-dev/1Panel
fix: 解决容器创建批量端口失败的问题 (#830)
parent
c424e22924
commit
379b171f0a
|
@ -166,35 +166,27 @@ func (u *ContainerService) Inspect(req dto.InspectReq) (string, error) {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (u *ContainerService) ContainerCreate(req dto.ContainerCreate) error {
|
func (u *ContainerService) ContainerCreate(req dto.ContainerCreate) error {
|
||||||
if len(req.ExposedPorts) != 0 {
|
portMap, err := checkPortStats(req.ExposedPorts)
|
||||||
for _, port := range req.ExposedPorts {
|
if err != nil {
|
||||||
if strings.Contains(port.HostPort, "-") {
|
return err
|
||||||
portStart, _ := strconv.Atoi(strings.Split(port.HostPort, "-")[0])
|
|
||||||
portEnd, _ := strconv.Atoi(strings.Split(port.HostPort, "-")[1])
|
|
||||||
for i := portStart; i <= portEnd; i++ {
|
|
||||||
if common.ScanPort(i) {
|
|
||||||
return buserr.WithDetail(constant.ErrPortInUsed, i, nil)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
portItem, _ := strconv.Atoi(port.HostPort)
|
|
||||||
if common.ScanPort(portItem) {
|
|
||||||
return buserr.WithDetail(constant.ErrPortInUsed, portItem, nil)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
client, err := docker.NewDockerClient()
|
client, err := docker.NewDockerClient()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
exposeds := make(nat.PortSet)
|
||||||
|
for port := range portMap {
|
||||||
|
exposeds[port] = struct{}{}
|
||||||
|
}
|
||||||
config := &container.Config{
|
config := &container.Config{
|
||||||
Image: req.Image,
|
Image: req.Image,
|
||||||
Cmd: req.Cmd,
|
Cmd: req.Cmd,
|
||||||
Env: req.Env,
|
Env: req.Env,
|
||||||
Labels: stringsToMap(req.Labels),
|
Labels: stringsToMap(req.Labels),
|
||||||
Tty: true,
|
Tty: true,
|
||||||
OpenStdin: true,
|
OpenStdin: true,
|
||||||
|
ExposedPorts: exposeds,
|
||||||
}
|
}
|
||||||
hostConf := &container.HostConfig{
|
hostConf := &container.HostConfig{
|
||||||
AutoRemove: req.AutoRemove,
|
AutoRemove: req.AutoRemove,
|
||||||
|
@ -211,11 +203,7 @@ func (u *ContainerService) ContainerCreate(req dto.ContainerCreate) error {
|
||||||
hostConf.Memory = req.Memory
|
hostConf.Memory = req.Memory
|
||||||
}
|
}
|
||||||
if len(req.ExposedPorts) != 0 {
|
if len(req.ExposedPorts) != 0 {
|
||||||
hostConf.PortBindings = make(nat.PortMap)
|
hostConf.PortBindings = portMap
|
||||||
for _, port := range req.ExposedPorts {
|
|
||||||
bindItem := nat.PortBinding{HostPort: port.HostPort, HostIP: port.HostIP}
|
|
||||||
hostConf.PortBindings[nat.Port(fmt.Sprintf("%s/%s", port.ContainerPort, port.Protocol))] = []nat.PortBinding{bindItem}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
if len(req.Volumes) != 0 {
|
if len(req.Volumes) != 0 {
|
||||||
config.Volumes = make(map[string]struct{})
|
config.Volumes = make(map[string]struct{})
|
||||||
|
@ -423,3 +411,44 @@ func loadCpuAndMem(client *client.Client, container string) (float64, float64) {
|
||||||
MemPercent := calculateMemPercentUnix(stats.MemoryStats)
|
MemPercent := calculateMemPercentUnix(stats.MemoryStats)
|
||||||
return CPUPercent, MemPercent
|
return CPUPercent, MemPercent
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func checkPortStats(ports []dto.PortHelper) (nat.PortMap, error) {
|
||||||
|
portMap := make(nat.PortMap)
|
||||||
|
if len(ports) == 0 {
|
||||||
|
return portMap, nil
|
||||||
|
}
|
||||||
|
for _, port := range ports {
|
||||||
|
if strings.Contains(port.HostPort, "-") {
|
||||||
|
if !strings.Contains(port.ContainerPort, "-") {
|
||||||
|
return portMap, buserr.New(constant.ErrPortRules)
|
||||||
|
}
|
||||||
|
hostStart, _ := strconv.Atoi(strings.Split(port.HostPort, "-")[0])
|
||||||
|
hostEnd, _ := strconv.Atoi(strings.Split(port.HostPort, "-")[1])
|
||||||
|
containerStart, _ := strconv.Atoi(strings.Split(port.ContainerPort, "-")[0])
|
||||||
|
containerEnd, _ := strconv.Atoi(strings.Split(port.ContainerPort, "-")[1])
|
||||||
|
if (hostEnd-hostStart) <= 0 || (containerEnd-containerStart) <= 0 {
|
||||||
|
return portMap, buserr.New(constant.ErrPortRules)
|
||||||
|
}
|
||||||
|
if (containerEnd - containerStart) != (hostEnd - hostStart) {
|
||||||
|
return portMap, buserr.New(constant.ErrPortRules)
|
||||||
|
}
|
||||||
|
for i := 0; i <= hostEnd-hostStart; i++ {
|
||||||
|
bindItem := nat.PortBinding{HostPort: strconv.Itoa(hostStart + i), HostIP: port.HostIP}
|
||||||
|
portMap[nat.Port(fmt.Sprintf("%d/%s", containerStart+i, port.Protocol))] = []nat.PortBinding{bindItem}
|
||||||
|
}
|
||||||
|
for i := hostStart; i <= hostEnd; i++ {
|
||||||
|
if common.ScanPort(i) {
|
||||||
|
return portMap, buserr.WithDetail(constant.ErrPortInUsed, i, nil)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
portItem, _ := strconv.Atoi(port.HostPort)
|
||||||
|
if common.ScanPort(portItem) {
|
||||||
|
return portMap, buserr.WithDetail(constant.ErrPortInUsed, portItem, nil)
|
||||||
|
}
|
||||||
|
bindItem := nat.PortBinding{HostPort: strconv.Itoa(portItem), HostIP: port.HostIP}
|
||||||
|
portMap[nat.Port(fmt.Sprintf("%s/%s", port.ContainerPort, port.Protocol))] = []nat.PortBinding{bindItem}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return portMap, nil
|
||||||
|
}
|
||||||
|
|
|
@ -106,6 +106,7 @@ var (
|
||||||
var (
|
var (
|
||||||
ErrInUsed = "ErrInUsed"
|
ErrInUsed = "ErrInUsed"
|
||||||
ErrObjectInUsed = "ErrObjectInUsed"
|
ErrObjectInUsed = "ErrObjectInUsed"
|
||||||
|
ErrPortRules = "ErrPortRules"
|
||||||
)
|
)
|
||||||
|
|
||||||
// runtime
|
// runtime
|
||||||
|
|
|
@ -63,6 +63,7 @@ ErrTypeOfRedis: "The recovery file type does not match the current persistence m
|
||||||
#container
|
#container
|
||||||
ErrInUsed: "{{ .detail }} is in use and cannot be deleted"
|
ErrInUsed: "{{ .detail }} is in use and cannot be deleted"
|
||||||
ErrObjectInUsed: "This object is in use and cannot be deleted"
|
ErrObjectInUsed: "This object is in use and cannot be deleted"
|
||||||
|
ErrPortRules: "Invalid port specification"
|
||||||
|
|
||||||
#runtime
|
#runtime
|
||||||
ErrDirNotFound: "The build folder does not exist! Please check file integrity!"
|
ErrDirNotFound: "The build folder does not exist! Please check file integrity!"
|
||||||
|
|
|
@ -63,6 +63,7 @@ ErrTypeOfRedis: "恢复文件类型与当前持久化方式不符,请修改后
|
||||||
#container
|
#container
|
||||||
ErrInUsed: "{{ .detail }} 正被使用,无法删除"
|
ErrInUsed: "{{ .detail }} 正被使用,无法删除"
|
||||||
ErrObjectInUsed: "该对象正被使用,无法删除"
|
ErrObjectInUsed: "该对象正被使用,无法删除"
|
||||||
|
ErrPortRules: "无效的端口规格"
|
||||||
|
|
||||||
#runtime
|
#runtime
|
||||||
ErrDirNotFound: "build 文件夹不存在!请检查文件完整性!"
|
ErrDirNotFound: "build 文件夹不存在!请检查文件完整性!"
|
||||||
|
|
|
@ -364,7 +364,7 @@ const onSubmit = async (formEl: FormInstance | undefined) => {
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const checkPortValid = async () => {
|
const checkPortValid = () => {
|
||||||
if (form.exposedPorts.length === 0) {
|
if (form.exposedPorts.length === 0) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue