2020-05-07 04:58:04 +02:00
|
|
|
package convert
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"net/url"
|
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
|
|
|
|
2020-05-19 15:50:57 +02:00
|
|
|
"github.com/pkg/errors"
|
|
|
|
|
2020-05-07 04:58:04 +02:00
|
|
|
"github.com/compose-spec/compose-go/types"
|
2020-05-19 15:50:57 +02:00
|
|
|
|
|
|
|
"github.com/docker/api/errdefs"
|
2020-05-07 04:58:04 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
// 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
|
|
|
|
}
|
|
|
|
projectVolumes[vi.name] = types.VolumeConfig{
|
|
|
|
Name: vi.name,
|
|
|
|
Driver: azureFileDriverName,
|
|
|
|
DriverOpts: map[string]string{
|
|
|
|
volumeDriveroptsAccountNameKey: vi.username,
|
|
|
|
volumeDriveroptsAccountKeyKey: vi.key,
|
|
|
|
volumeDriveroptsShareNameKey: vi.share,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
sv := types.ServiceVolumeConfig{
|
|
|
|
Type: azureFileDriverName,
|
|
|
|
Source: vi.name,
|
|
|
|
Target: vi.target,
|
|
|
|
}
|
|
|
|
serviceConfigVolumes = append(serviceConfigVolumes, sv)
|
|
|
|
}
|
|
|
|
|
|
|
|
return projectVolumes, serviceConfigVolumes, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type volumeInput struct {
|
|
|
|
name string
|
|
|
|
username string
|
|
|
|
key string
|
|
|
|
share string
|
|
|
|
target string
|
|
|
|
}
|
|
|
|
|
2020-05-19 15:26:59 +02:00
|
|
|
func escapeKeySlashes(rawURL string) (string, error) {
|
2020-05-07 04:58:04 +02:00
|
|
|
urlSplit := strings.Split(rawURL, "@")
|
|
|
|
if len(urlSplit) < 1 {
|
2020-05-19 19:45:03 +02:00
|
|
|
return "", errors.Wrap(errdefs.ErrParsingFailed, "invalid url format "+rawURL)
|
2020-05-07 04:58:04 +02:00
|
|
|
}
|
|
|
|
userPasswd := strings.ReplaceAll(urlSplit[0], "/", "_")
|
2020-05-19 19:45:03 +02:00
|
|
|
|
|
|
|
atIndex := strings.Index(rawURL, "@")
|
|
|
|
if atIndex < 0 {
|
|
|
|
return "", errors.Wrap(errdefs.ErrParsingFailed, "no share specified in "+rawURL)
|
|
|
|
}
|
|
|
|
|
|
|
|
scaped := userPasswd + rawURL[atIndex:]
|
2020-05-07 04:58:04 +02:00
|
|
|
|
|
|
|
return scaped, nil
|
|
|
|
}
|
|
|
|
|
2020-05-19 15:26:59 +02:00
|
|
|
func unescapeKey(key string) string {
|
|
|
|
return strings.ReplaceAll(key, "_", "/")
|
2020-05-07 04:58:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Removes the second ':' that separates the source from target
|
|
|
|
func volumeURL(pathURL string) (*url.URL, error) {
|
2020-05-19 15:26:59 +02:00
|
|
|
scapedURL, err := escapeKeySlashes(pathURL)
|
2020-05-07 04:58:04 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
pathURL = "//" + scapedURL
|
|
|
|
|
|
|
|
count := strings.Count(pathURL, ":")
|
|
|
|
if count > 2 {
|
2020-05-19 15:50:57 +02:00
|
|
|
return nil, errors.Wrap(errdefs.ErrParsingFailed, fmt.Sprintf("unable to parse volume mount %q", pathURL))
|
2020-05-07 04:58:04 +02:00
|
|
|
}
|
|
|
|
if count == 2 {
|
|
|
|
tokens := strings.Split(pathURL, ":")
|
|
|
|
pathURL = fmt.Sprintf("%s:%s%s", tokens[0], tokens[1], tokens[2])
|
|
|
|
}
|
|
|
|
return url.Parse(pathURL)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (v *volumeInput) parse(name string, s string) error {
|
|
|
|
volumeURL, err := volumeURL(s)
|
|
|
|
if err != nil {
|
2020-05-19 15:50:57 +02:00
|
|
|
return errors.Wrap(errdefs.ErrParsingFailed, fmt.Sprintf("volume specification %q could not be parsed %q", s, err))
|
2020-05-07 04:58:04 +02:00
|
|
|
}
|
|
|
|
v.username = volumeURL.User.Username()
|
|
|
|
if v.username == "" {
|
2020-05-19 15:50:57 +02:00
|
|
|
return errors.Wrap(errdefs.ErrParsingFailed, fmt.Sprintf("volume specification %q does not include a storage username", v))
|
2020-05-07 04:58:04 +02:00
|
|
|
}
|
2020-05-19 15:26:59 +02:00
|
|
|
key, ok := volumeURL.User.Password()
|
|
|
|
if !ok || key == "" {
|
2020-05-19 15:50:57 +02:00
|
|
|
return errors.Wrap(errdefs.ErrParsingFailed, fmt.Sprintf("volume specification %q does not include a storage key", v))
|
2020-05-07 04:58:04 +02:00
|
|
|
}
|
2020-05-19 15:26:59 +02:00
|
|
|
v.key = unescapeKey(key)
|
2020-05-07 04:58:04 +02:00
|
|
|
v.share = volumeURL.Host
|
|
|
|
if v.share == "" {
|
2020-05-19 15:50:57 +02:00
|
|
|
return errors.Wrap(errdefs.ErrParsingFailed, fmt.Sprintf("volume specification %q does not include a storage file share", v))
|
2020-05-07 04:58:04 +02:00
|
|
|
}
|
|
|
|
v.name = name
|
|
|
|
v.target = volumeURL.Path
|
|
|
|
if v.target == "" {
|
|
|
|
v.target = filepath.Join("/run/volumes/", v.share)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|