/* Copyright 2020 Docker, Inc. 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. */ package convert import ( "fmt" "strconv" "strings" "github.com/pkg/errors" "github.com/compose-spec/compose-go/types" "github.com/docker/compose-cli/errdefs" ) // GetRunVolumes return volume configurations for a project and a single service // this is meant to be used as a compose project of a single service func GetRunVolumes(volumes []string) (map[string]types.VolumeConfig, []types.ServiceVolumeConfig, error) { var serviceConfigVolumes []types.ServiceVolumeConfig projectVolumes := make(map[string]types.VolumeConfig, len(volumes)) for i, v := range volumes { var vi volumeInput err := vi.parse(fmt.Sprintf("volume-%d", i), v) if err != nil { return nil, nil, err } readOnly := strconv.FormatBool(vi.readonly) projectVolumes[vi.name] = types.VolumeConfig{ Name: vi.name, Driver: azureFileDriverName, DriverOpts: map[string]string{ volumeDriveroptsAccountNameKey: vi.storageAccount, volumeDriveroptsShareNameKey: vi.fileshare, volumeReadOnly: readOnly, }, } sv := types.ServiceVolumeConfig{ Type: azureFileDriverName, Source: vi.name, Target: vi.target, ReadOnly: vi.readonly, } serviceConfigVolumes = append(serviceConfigVolumes, sv) } return projectVolumes, serviceConfigVolumes, nil } type volumeInput struct { name string storageAccount string fileshare string target string readonly bool } // parse takes a candidate string and creates a volumeInput // Candidates take the form of [:][:] // Source is of the form `/` // If only the source is specified then the target is set to `/run/volumes/` // Target is an absolute path in the container of the form `/path/to/mount` // Permissions can only be set if the target is set // If set, permissions must be `rw` or `ro` func (v *volumeInput) parse(name string, candidate string) error { v.name = name tokens := strings.Split(candidate, ":") sourceTokens := strings.Split(tokens[0], "/") if len(sourceTokens) != 2 || sourceTokens[0] == "" { return errors.Wrapf(errdefs.ErrParsingFailed, "volume specification %q does not include a storage account before '/'", candidate) } if sourceTokens[1] == "" { return errors.Wrapf(errdefs.ErrParsingFailed, "volume specification %q does not include a storage file fileshare after '/'", candidate) } v.storageAccount = sourceTokens[0] v.fileshare = sourceTokens[1] switch len(tokens) { case 1: // source only v.target = "/run/volumes/" + v.fileshare case 2: // source and target v.target = tokens[1] case 3: // source, target, and permissions v.target = tokens[1] permissions := strings.ToLower(tokens[2]) if permissions != "ro" && permissions != "rw" { return errors.Wrapf(errdefs.ErrParsingFailed, "volume specification %q has an invalid mode %q", candidate, permissions) } v.readonly = permissions == "ro" default: return errors.Wrapf(errdefs.ErrParsingFailed, "volume specification %q has invalid format", candidate) } return nil }