2018-09-21 13:54:48 +03:00
|
|
|
package core
|
|
|
|
|
2018-09-21 19:23:46 +03:00
|
|
|
import (
|
|
|
|
"fmt"
|
2018-09-24 16:25:15 +03:00
|
|
|
"reflect"
|
|
|
|
"regexp"
|
2018-09-21 19:23:46 +03:00
|
|
|
"strings"
|
2018-09-21 13:54:48 +03:00
|
|
|
|
2018-09-21 19:23:46 +03:00
|
|
|
"github.com/MichaelMure/git-bug/cache"
|
|
|
|
"github.com/MichaelMure/git-bug/repository"
|
|
|
|
"github.com/pkg/errors"
|
|
|
|
)
|
|
|
|
|
|
|
|
var ErrImportNorSupported = errors.New("import is not supported")
|
|
|
|
var ErrExportNorSupported = errors.New("export is not supported")
|
|
|
|
|
2018-09-24 16:25:15 +03:00
|
|
|
var bridgeImpl map[string]reflect.Type
|
|
|
|
|
2018-09-21 19:23:46 +03:00
|
|
|
// Bridge is a wrapper around a BridgeImpl that will bind low-level
|
|
|
|
// implementation with utility code to provide high-level functions.
|
|
|
|
type Bridge struct {
|
2018-09-24 16:25:15 +03:00
|
|
|
Name string
|
|
|
|
repo *cache.RepoCache
|
2018-09-21 19:23:46 +03:00
|
|
|
impl BridgeImpl
|
|
|
|
conf Configuration
|
2018-09-21 13:54:48 +03:00
|
|
|
}
|
|
|
|
|
2018-09-24 16:25:15 +03:00
|
|
|
// Register will register a new BridgeImpl
|
|
|
|
func Register(impl BridgeImpl) {
|
|
|
|
if bridgeImpl == nil {
|
|
|
|
bridgeImpl = make(map[string]reflect.Type)
|
|
|
|
}
|
|
|
|
bridgeImpl[impl.Target()] = reflect.TypeOf(impl)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Targets return all known bridge implementation target
|
|
|
|
func Targets() []string {
|
|
|
|
var result []string
|
|
|
|
|
|
|
|
for key := range bridgeImpl {
|
|
|
|
result = append(result, key)
|
|
|
|
}
|
|
|
|
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewBridge(repo *cache.RepoCache, target string, name string) (*Bridge, error) {
|
|
|
|
implType, ok := bridgeImpl[target]
|
|
|
|
if !ok {
|
|
|
|
return nil, fmt.Errorf("unknown bridge target %v", target)
|
|
|
|
}
|
|
|
|
|
|
|
|
impl := reflect.New(implType).Elem().Interface().(BridgeImpl)
|
|
|
|
|
|
|
|
bridge := &Bridge{
|
|
|
|
Name: name,
|
|
|
|
repo: repo,
|
2018-09-21 19:23:46 +03:00
|
|
|
impl: impl,
|
|
|
|
}
|
2018-09-24 16:25:15 +03:00
|
|
|
|
|
|
|
return bridge, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func ConfiguredBridges(repo repository.RepoCommon) ([]string, error) {
|
|
|
|
configs, err := repo.ReadConfigs("git-bug.")
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "can't read configured bridges")
|
|
|
|
}
|
|
|
|
|
|
|
|
re, err := regexp.Compile(`git-bug.([^\.]+\.[^\.]+)`)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
set := make(map[string]interface{})
|
|
|
|
|
|
|
|
for key := range configs {
|
|
|
|
res := re.FindStringSubmatch(key)
|
|
|
|
|
|
|
|
if res == nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
set[res[1]] = nil
|
|
|
|
}
|
|
|
|
|
|
|
|
result := make([]string, len(set))
|
|
|
|
|
|
|
|
i := 0
|
|
|
|
for key := range set {
|
|
|
|
result[i] = key
|
|
|
|
i++
|
|
|
|
}
|
|
|
|
|
|
|
|
return result, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *Bridge) String() string {
|
|
|
|
var _type string
|
|
|
|
if b.impl.Importer() != nil && b.impl.Exporter() != nil {
|
|
|
|
_type = "import/export"
|
|
|
|
} else if b.impl.Importer() != nil {
|
|
|
|
_type = "import"
|
|
|
|
} else if b.impl.Exporter() != nil {
|
|
|
|
_type = "export"
|
|
|
|
} else {
|
|
|
|
panic("bad bridge impl, neither import nor export")
|
|
|
|
}
|
|
|
|
|
|
|
|
return fmt.Sprintf("%s.%s: %s", b.impl.Target(), b.Name, _type)
|
2018-09-21 13:54:48 +03:00
|
|
|
}
|
|
|
|
|
2018-09-24 16:25:15 +03:00
|
|
|
func (b *Bridge) Configure() error {
|
|
|
|
conf, err := b.impl.Configure(b.repo)
|
2018-09-21 19:23:46 +03:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2018-09-24 16:25:15 +03:00
|
|
|
b.conf = conf
|
|
|
|
|
|
|
|
return b.storeConfig(conf)
|
2018-09-21 13:54:48 +03:00
|
|
|
}
|
|
|
|
|
2018-09-24 16:25:15 +03:00
|
|
|
func (b *Bridge) storeConfig(conf Configuration) error {
|
2018-09-21 19:23:46 +03:00
|
|
|
for key, val := range conf {
|
2018-09-24 16:25:15 +03:00
|
|
|
storeKey := fmt.Sprintf("git-bug.%s.%s.%s", b.impl.Target(), b.Name, key)
|
2018-09-21 19:23:46 +03:00
|
|
|
|
2018-09-24 16:25:15 +03:00
|
|
|
err := b.repo.StoreConfig(storeKey, val)
|
2018-09-21 19:23:46 +03:00
|
|
|
if err != nil {
|
2018-09-24 16:25:15 +03:00
|
|
|
return errors.Wrap(err, "error while storing bridge configuration")
|
2018-09-21 19:23:46 +03:00
|
|
|
}
|
|
|
|
}
|
2018-09-21 13:54:48 +03:00
|
|
|
|
2018-09-21 19:23:46 +03:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-09-24 16:25:15 +03:00
|
|
|
func (b Bridge) getConfig() (Configuration, error) {
|
2018-09-21 19:23:46 +03:00
|
|
|
var err error
|
|
|
|
if b.conf == nil {
|
2018-09-24 16:25:15 +03:00
|
|
|
b.conf, err = b.loadConfig()
|
2018-09-21 19:23:46 +03:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return b.conf, nil
|
|
|
|
}
|
|
|
|
|
2018-09-24 16:25:15 +03:00
|
|
|
func (b Bridge) loadConfig() (Configuration, error) {
|
|
|
|
keyPrefix := fmt.Sprintf("git-bug.%s.%s.", b.impl.Target(), b.Name)
|
2018-09-21 19:23:46 +03:00
|
|
|
|
2018-09-24 16:25:15 +03:00
|
|
|
pairs, err := b.repo.ReadConfigs(keyPrefix)
|
2018-09-21 19:23:46 +03:00
|
|
|
if err != nil {
|
2018-09-24 16:25:15 +03:00
|
|
|
return nil, errors.Wrap(err, "error while reading bridge configuration")
|
2018-09-21 19:23:46 +03:00
|
|
|
}
|
|
|
|
|
2018-09-24 16:25:15 +03:00
|
|
|
result := make(Configuration, len(pairs))
|
|
|
|
for key, value := range pairs {
|
|
|
|
key := strings.TrimPrefix(key, keyPrefix)
|
|
|
|
result[key] = value
|
2018-09-21 19:23:46 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return result, nil
|
|
|
|
}
|
|
|
|
|
2018-09-24 16:25:15 +03:00
|
|
|
func (b Bridge) ImportAll() error {
|
2018-09-21 19:23:46 +03:00
|
|
|
importer := b.impl.Importer()
|
|
|
|
if importer == nil {
|
|
|
|
return ErrImportNorSupported
|
|
|
|
}
|
|
|
|
|
2018-09-24 16:25:15 +03:00
|
|
|
conf, err := b.getConfig()
|
2018-09-21 19:23:46 +03:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2018-09-24 16:25:15 +03:00
|
|
|
return b.impl.Importer().ImportAll(b.repo, conf)
|
2018-09-21 19:23:46 +03:00
|
|
|
}
|
|
|
|
|
2018-09-24 16:25:15 +03:00
|
|
|
func (b Bridge) Import(id string) error {
|
2018-09-21 19:23:46 +03:00
|
|
|
importer := b.impl.Importer()
|
|
|
|
if importer == nil {
|
|
|
|
return ErrImportNorSupported
|
|
|
|
}
|
|
|
|
|
2018-09-24 16:25:15 +03:00
|
|
|
conf, err := b.getConfig()
|
2018-09-21 19:23:46 +03:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2018-09-24 16:25:15 +03:00
|
|
|
return b.impl.Importer().Import(b.repo, conf, id)
|
2018-09-21 19:23:46 +03:00
|
|
|
}
|
|
|
|
|
2018-09-24 16:25:15 +03:00
|
|
|
func (b Bridge) ExportAll() error {
|
2018-09-21 19:23:46 +03:00
|
|
|
exporter := b.impl.Exporter()
|
|
|
|
if exporter == nil {
|
|
|
|
return ErrExportNorSupported
|
|
|
|
}
|
|
|
|
|
2018-09-24 16:25:15 +03:00
|
|
|
conf, err := b.getConfig()
|
2018-09-21 19:23:46 +03:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2018-09-24 16:25:15 +03:00
|
|
|
return b.impl.Exporter().ExportAll(b.repo, conf)
|
2018-09-21 19:23:46 +03:00
|
|
|
}
|
|
|
|
|
2018-09-24 16:25:15 +03:00
|
|
|
func (b Bridge) Export(id string) error {
|
2018-09-21 19:23:46 +03:00
|
|
|
exporter := b.impl.Exporter()
|
|
|
|
if exporter == nil {
|
|
|
|
return ErrExportNorSupported
|
|
|
|
}
|
|
|
|
|
2018-09-24 16:25:15 +03:00
|
|
|
conf, err := b.getConfig()
|
2018-09-21 19:23:46 +03:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2018-09-24 16:25:15 +03:00
|
|
|
return b.impl.Exporter().Export(b.repo, conf, id)
|
2018-09-21 19:23:46 +03:00
|
|
|
}
|