mirror of
https://github.com/MichaelMure/git-bug.git
synced 2024-12-16 19:34:41 +03:00
343 lines
6.6 KiB
Go
343 lines
6.6 KiB
Go
// Package core contains the target-agnostic code to define and run a bridge
|
|
package core
|
|
|
|
import (
|
|
"fmt"
|
|
"reflect"
|
|
"regexp"
|
|
"strings"
|
|
|
|
"github.com/MichaelMure/git-bug/cache"
|
|
"github.com/MichaelMure/git-bug/repository"
|
|
"github.com/pkg/errors"
|
|
)
|
|
|
|
var ErrImportNotSupported = errors.New("import is not supported")
|
|
var ErrExportNotSupported = errors.New("export is not supported")
|
|
|
|
const bridgeConfigKeyPrefix = "git-bug.bridge"
|
|
|
|
var bridgeImpl map[string]reflect.Type
|
|
|
|
// Bridge is a wrapper around a BridgeImpl that will bind low-level
|
|
// implementation with utility code to provide high-level functions.
|
|
type Bridge struct {
|
|
Name string
|
|
repo *cache.RepoCache
|
|
impl BridgeImpl
|
|
importer Importer
|
|
exporter Exporter
|
|
conf Configuration
|
|
initDone bool
|
|
}
|
|
|
|
// 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
|
|
}
|
|
|
|
// Instantiate a new Bridge for a repo, from the given target and name
|
|
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,
|
|
impl: impl,
|
|
}
|
|
|
|
return bridge, nil
|
|
}
|
|
|
|
// Instantiate a new bridge for a repo, from the combined target and name contained
|
|
// in the full name
|
|
func NewBridgeFromFullName(repo *cache.RepoCache, fullName string) (*Bridge, error) {
|
|
target, name, err := splitFullName(fullName)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return NewBridge(repo, target, name)
|
|
}
|
|
|
|
// Attempt to retrieve a default bridge for the given repo. If zero or multiple
|
|
// bridge exist, it fails.
|
|
func DefaultBridge(repo *cache.RepoCache) (*Bridge, error) {
|
|
bridges, err := ConfiguredBridges(repo)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if len(bridges) == 0 {
|
|
return nil, fmt.Errorf("no configured bridge")
|
|
}
|
|
|
|
if len(bridges) > 1 {
|
|
return nil, fmt.Errorf("multiple bridge are configured, you need to select one explicitely")
|
|
}
|
|
|
|
target, name, err := splitFullName(bridges[0])
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return NewBridge(repo, target, name)
|
|
}
|
|
|
|
func splitFullName(fullName string) (string, string, error) {
|
|
split := strings.Split(fullName, ".")
|
|
|
|
if len(split) != 2 {
|
|
return "", "", fmt.Errorf("bad bridge fullname: %s", fullName)
|
|
}
|
|
|
|
return split[0], split[1], nil
|
|
}
|
|
|
|
// ConfiguredBridges return the list of bridge that are configured for the given
|
|
// repo
|
|
func ConfiguredBridges(repo repository.RepoCommon) ([]string, error) {
|
|
configs, err := repo.ReadConfigs(bridgeConfigKeyPrefix + ".")
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "can't read configured bridges")
|
|
}
|
|
|
|
re, err := regexp.Compile(bridgeConfigKeyPrefix + `.([^.]+\.[^.]+)`)
|
|
if err != nil {
|
|
panic(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
|
|
}
|
|
|
|
// Remove a configured bridge
|
|
func RemoveBridge(repo repository.RepoCommon, fullName string) error {
|
|
re, err := regexp.Compile(`^[^.]+\.[^.]+$`)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
if !re.MatchString(fullName) {
|
|
return fmt.Errorf("bad bridge fullname: %s", fullName)
|
|
}
|
|
|
|
keyPrefix := fmt.Sprintf("git-bug.bridge.%s", fullName)
|
|
return repo.RmConfigs(keyPrefix)
|
|
}
|
|
|
|
// Configure run the target specific configuration process
|
|
func (b *Bridge) Configure() error {
|
|
conf, err := b.impl.Configure(b.repo)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
b.conf = conf
|
|
|
|
return b.storeConfig(conf)
|
|
}
|
|
|
|
func (b *Bridge) storeConfig(conf Configuration) error {
|
|
for key, val := range conf {
|
|
storeKey := fmt.Sprintf("git-bug.bridge.%s.%s.%s", b.impl.Target(), b.Name, key)
|
|
|
|
err := b.repo.StoreConfig(storeKey, val)
|
|
if err != nil {
|
|
return errors.Wrap(err, "error while storing bridge configuration")
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (b *Bridge) ensureConfig() error {
|
|
if b.conf == nil {
|
|
conf, err := b.loadConfig()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
b.conf = conf
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (b *Bridge) loadConfig() (Configuration, error) {
|
|
keyPrefix := fmt.Sprintf("git-bug.bridge.%s.%s.", b.impl.Target(), b.Name)
|
|
|
|
pairs, err := b.repo.ReadConfigs(keyPrefix)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "error while reading bridge configuration")
|
|
}
|
|
|
|
result := make(Configuration, len(pairs))
|
|
for key, value := range pairs {
|
|
key := strings.TrimPrefix(key, keyPrefix)
|
|
result[key] = value
|
|
}
|
|
|
|
err = b.impl.ValidateConfig(result)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "invalid configuration")
|
|
}
|
|
|
|
return result, nil
|
|
}
|
|
|
|
func (b *Bridge) getImporter() Importer {
|
|
if b.importer == nil {
|
|
b.importer = b.impl.NewImporter()
|
|
}
|
|
|
|
return b.importer
|
|
}
|
|
|
|
func (b *Bridge) getExporter() Exporter {
|
|
if b.exporter == nil {
|
|
b.exporter = b.impl.NewExporter()
|
|
}
|
|
|
|
return b.exporter
|
|
}
|
|
|
|
func (b *Bridge) ensureInit() error {
|
|
if b.initDone {
|
|
return nil
|
|
}
|
|
|
|
importer := b.getImporter()
|
|
if importer != nil {
|
|
err := importer.Init(b.conf)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
exporter := b.getExporter()
|
|
if exporter != nil {
|
|
err := exporter.Init(b.conf)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
b.initDone = true
|
|
|
|
return nil
|
|
}
|
|
|
|
func (b *Bridge) ImportAll() error {
|
|
importer := b.getImporter()
|
|
if importer == nil {
|
|
return ErrImportNotSupported
|
|
}
|
|
|
|
err := b.ensureConfig()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = b.ensureInit()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return importer.ImportAll(b.repo)
|
|
}
|
|
|
|
func (b *Bridge) Import(id string) error {
|
|
importer := b.getImporter()
|
|
if importer == nil {
|
|
return ErrImportNotSupported
|
|
}
|
|
|
|
err := b.ensureConfig()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = b.ensureInit()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return importer.Import(b.repo, id)
|
|
}
|
|
|
|
func (b *Bridge) ExportAll() error {
|
|
exporter := b.getExporter()
|
|
if exporter == nil {
|
|
return ErrExportNotSupported
|
|
}
|
|
|
|
err := b.ensureConfig()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = b.ensureInit()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return exporter.ExportAll(b.repo)
|
|
}
|
|
|
|
func (b *Bridge) Export(id string) error {
|
|
exporter := b.getExporter()
|
|
if exporter == nil {
|
|
return ErrExportNotSupported
|
|
}
|
|
|
|
err := b.ensureConfig()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = b.ensureInit()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return exporter.Export(b.repo, id)
|
|
}
|