2019-02-21 15:07:20 +00:00
|
|
|
// Copyright 2019 Sorint.lab
|
|
|
|
//
|
|
|
|
// 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 common
|
|
|
|
|
|
|
|
import (
|
|
|
|
"io"
|
|
|
|
"io/ioutil"
|
|
|
|
"net/url"
|
|
|
|
"os"
|
|
|
|
"path"
|
|
|
|
|
2019-07-01 09:40:20 +00:00
|
|
|
"agola.io/agola/internal/etcd"
|
|
|
|
"agola.io/agola/internal/objectstorage"
|
|
|
|
"agola.io/agola/internal/objectstorage/posix"
|
|
|
|
"agola.io/agola/internal/objectstorage/s3"
|
|
|
|
"agola.io/agola/internal/services/config"
|
2019-02-21 15:07:20 +00:00
|
|
|
"go.uber.org/zap"
|
2019-05-23 09:23:14 +00:00
|
|
|
errors "golang.org/x/xerrors"
|
2019-02-21 15:07:20 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
StorePrefix = "agola"
|
|
|
|
)
|
|
|
|
|
|
|
|
// WriteFileAtomicFunc atomically writes a file, it achieves this by creating a
|
|
|
|
// temporary file and then moving it. writeFunc is the func that will write
|
|
|
|
// data to the file.
|
|
|
|
// This function is taken from
|
|
|
|
// https://github.com/youtube/vitess/blob/master/go/ioutil2/ioutil.go
|
|
|
|
// Copyright 2012, Google Inc. BSD-license, see licenses/LICENSE-BSD-3-Clause
|
|
|
|
func WriteFileAtomicFunc(filename string, perm os.FileMode, writeFunc func(f io.Writer) error) error {
|
|
|
|
dir, name := path.Split(filename)
|
|
|
|
f, err := ioutil.TempFile(dir, name)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
err = writeFunc(f)
|
|
|
|
if err == nil {
|
|
|
|
err = f.Sync()
|
|
|
|
}
|
|
|
|
if closeErr := f.Close(); err == nil {
|
|
|
|
err = closeErr
|
|
|
|
}
|
|
|
|
if permErr := os.Chmod(f.Name(), perm); err == nil {
|
|
|
|
err = permErr
|
|
|
|
}
|
|
|
|
if err == nil {
|
|
|
|
err = os.Rename(f.Name(), filename)
|
|
|
|
}
|
|
|
|
// Any err should result in full cleanup.
|
|
|
|
if err != nil {
|
|
|
|
os.Remove(f.Name())
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// WriteFileAtomic atomically writes a file
|
|
|
|
func WriteFileAtomic(filename string, data []byte, perm os.FileMode) error {
|
|
|
|
return WriteFileAtomicFunc(filename, perm,
|
|
|
|
func(f io.Writer) error {
|
|
|
|
_, err := f.Write(data)
|
|
|
|
return err
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-04-27 13:16:48 +00:00
|
|
|
func NewObjectStorage(c *config.ObjectStorage) (*objectstorage.ObjStorage, error) {
|
2019-02-21 15:07:20 +00:00
|
|
|
var (
|
|
|
|
err error
|
2019-04-27 13:16:48 +00:00
|
|
|
ost objectstorage.Storage
|
2019-02-21 15:07:20 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
switch c.Type {
|
2019-04-27 13:16:48 +00:00
|
|
|
case config.ObjectStorageTypePosix:
|
2019-05-21 13:17:53 +00:00
|
|
|
ost, err = posix.New(c.Path)
|
2019-02-21 15:07:20 +00:00
|
|
|
if err != nil {
|
2019-05-23 09:23:14 +00:00
|
|
|
return nil, errors.Errorf("failed to create posix object storage: %w", err)
|
2019-02-21 15:07:20 +00:00
|
|
|
}
|
2019-04-27 13:16:48 +00:00
|
|
|
case config.ObjectStorageTypeS3:
|
2019-02-21 15:07:20 +00:00
|
|
|
// minio golang client doesn't accept an url as an endpoint
|
|
|
|
endpoint := c.Endpoint
|
|
|
|
secure := !c.DisableTLS
|
|
|
|
if u, err := url.Parse(c.Endpoint); err == nil {
|
|
|
|
endpoint = u.Host
|
|
|
|
switch u.Scheme {
|
|
|
|
case "https":
|
|
|
|
secure = true
|
|
|
|
case "http":
|
|
|
|
secure = false
|
|
|
|
default:
|
|
|
|
return nil, errors.Errorf("wrong s3 endpoint scheme %q (must be http or https)", u.Scheme)
|
|
|
|
}
|
|
|
|
}
|
2019-05-21 13:17:53 +00:00
|
|
|
ost, err = s3.New(c.Bucket, c.Location, endpoint, c.AccessKey, c.SecretAccessKey, secure)
|
2019-02-21 15:07:20 +00:00
|
|
|
if err != nil {
|
2019-05-23 09:23:14 +00:00
|
|
|
return nil, errors.Errorf("failed to create s3 object storage: %w", err)
|
2019-02-21 15:07:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-27 13:16:48 +00:00
|
|
|
return objectstorage.NewObjStorage(ost, "/"), nil
|
2019-02-21 15:07:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func NewEtcd(c *config.Etcd, logger *zap.Logger, prefix string) (*etcd.Store, error) {
|
|
|
|
e, err := etcd.New(etcd.Config{
|
|
|
|
Logger: logger,
|
|
|
|
Endpoints: c.Endpoints,
|
|
|
|
Prefix: prefix,
|
|
|
|
CertFile: c.TLSCertFile,
|
|
|
|
KeyFile: c.TLSKeyFile,
|
|
|
|
CAFile: c.TLSCAFile,
|
|
|
|
SkipTLSVerify: c.TLSSkipVerify,
|
|
|
|
})
|
|
|
|
if err != nil {
|
2019-05-23 09:23:14 +00:00
|
|
|
return nil, errors.Errorf("failed to create etcd store: %w", err)
|
2019-02-21 15:07:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return e, nil
|
|
|
|
}
|