sq/libsq/driver/registry.go
Neil O'Toole c7bba4dfe4
go1.21: changes to support slog as part of stdlib (#299)
* go1.21: changes to support slog as part of stdlib

* Removed accidentially checked-in line of code

* Fixed minor linting issues; reenable typecheck

* go1.21: switched to stdlib slices pkg
2023-08-12 12:54:14 -06:00

117 lines
2.7 KiB
Go

package driver
import (
"log/slog"
"sync"
"github.com/neilotoole/sq/libsq/core/lg/lga"
"github.com/neilotoole/sq/libsq/core/errz"
"github.com/neilotoole/sq/libsq/source"
)
// NewRegistry returns a new Registry instance that provides
// access to driver implementations. Note that Registry
// implements Provider.
func NewRegistry(log *slog.Logger) *Registry {
return &Registry{
log: log,
providers: map[source.DriverType]Provider{},
}
}
// Registry provides access to driver implementations.
type Registry struct {
log *slog.Logger
mu sync.Mutex
providers map[source.DriverType]Provider
types []source.DriverType
}
// AddProvider registers the provider for the specified driver type.
// This method has no effect if there's already a provider for typ.
func (r *Registry) AddProvider(typ source.DriverType, p Provider) {
r.mu.Lock()
defer r.mu.Unlock()
if existingType, ok := r.providers[typ]; ok {
r.log.Warn("failed to add driver provider (%T) for driver type %s: provider (%T) already registered", p, typ,
existingType)
return
}
r.types = append(r.types, typ)
r.providers[typ] = p
}
// ProviderFor returns the provider for typ, or nil if no
// registered provider.
func (r *Registry) ProviderFor(typ source.DriverType) Provider {
r.mu.Lock()
defer r.mu.Unlock()
p := r.providers[typ]
return p
}
// DriverFor implements Provider.
func (r *Registry) DriverFor(typ source.DriverType) (Driver, error) {
r.mu.Lock()
defer r.mu.Unlock()
p, ok := r.providers[typ]
if !ok {
return nil, errz.Errorf("no registered driver for {%s}", typ)
}
return p.DriverFor(typ)
}
// SQLDriverFor for is a convenience method for getting a SQLDriver.
func (r *Registry) SQLDriverFor(typ source.DriverType) (SQLDriver, error) {
drvr, err := r.DriverFor(typ)
if err != nil {
return nil, err
}
sqlDrvr, ok := drvr.(SQLDriver)
if !ok {
return nil, errz.Errorf("driver %T is not of type %T", drvr, sqlDrvr)
}
return sqlDrvr, nil
}
// DriversMetadata returns metadata for each registered driver type.
func (r *Registry) DriversMetadata() []Metadata {
var md []Metadata
for _, typ := range r.types {
drv, err := r.DriverFor(typ)
if err != nil {
// Should never happen
r.log.Error("Error getting driver", lga.Type, typ, lga.Err, err)
continue
}
md = append(md, drv.DriverMetadata())
}
return md
}
// Drivers returns the registered drivers.
func (r *Registry) Drivers() []Driver {
var drvrs []Driver
for _, typ := range r.types {
drvr, err := r.DriverFor(typ)
if err != nil {
// Should never happen
r.log.Error("Error getting driver", lga.Type, typ, lga.Err, err)
continue
}
drvrs = append(drvrs, drvr)
}
return drvrs
}