2020-06-08 03:29:51 +03:00
|
|
|
package registry
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"sort"
|
|
|
|
"sync"
|
2021-08-17 13:06:32 +03:00
|
|
|
|
|
|
|
"github.com/wader/fq/pkg/decode"
|
2020-06-08 03:29:51 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
type Registry struct {
|
|
|
|
Groups map[string][]*decode.Format
|
|
|
|
resolveOnce sync.Once
|
2021-09-28 14:04:51 +03:00
|
|
|
resolved bool
|
2020-06-08 03:29:51 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
func New() *Registry {
|
|
|
|
return &Registry{
|
|
|
|
Groups: map[string][]*decode.Format{},
|
|
|
|
resolveOnce: sync.Once{},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *Registry) register(groupName string, format *decode.Format, single bool) *decode.Format { //nolint:unparam
|
2021-09-28 14:04:51 +03:00
|
|
|
if r.resolved {
|
|
|
|
// for now can't change after resolved
|
|
|
|
panic("registry already resolved")
|
|
|
|
}
|
|
|
|
|
2020-06-08 03:29:51 +03:00
|
|
|
formats, ok := r.Groups[groupName]
|
|
|
|
if ok {
|
|
|
|
if !single {
|
|
|
|
panic(fmt.Sprintf("%s: format already registered", groupName))
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
formats = []*decode.Format{}
|
|
|
|
}
|
|
|
|
|
|
|
|
r.Groups[groupName] = append(formats, format)
|
|
|
|
|
|
|
|
return format
|
|
|
|
}
|
|
|
|
|
2021-09-28 14:04:51 +03:00
|
|
|
func (r *Registry) MustRegister(format *decode.Format) *decode.Format {
|
|
|
|
r.register(format.Name, format, false)
|
|
|
|
for _, g := range format.Groups {
|
|
|
|
r.register(g, format, true)
|
|
|
|
}
|
|
|
|
r.register("all", format, true)
|
|
|
|
|
|
|
|
return format
|
|
|
|
}
|
|
|
|
|
2020-06-08 03:29:51 +03:00
|
|
|
func sortFormats(fs []*decode.Format) {
|
|
|
|
sort.Slice(fs, func(i, j int) bool {
|
|
|
|
if fs[i].ProbeOrder == fs[j].ProbeOrder {
|
|
|
|
return fs[i].Name < fs[j].Name
|
|
|
|
}
|
|
|
|
return fs[i].ProbeOrder < fs[j].ProbeOrder
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *Registry) resolve() error {
|
|
|
|
for _, fs := range r.Groups {
|
|
|
|
for _, f := range fs {
|
|
|
|
for _, d := range f.Dependencies {
|
|
|
|
var formats []*decode.Format
|
|
|
|
for _, dName := range d.Names {
|
|
|
|
df, ok := r.Groups[dName]
|
|
|
|
if !ok {
|
|
|
|
return fmt.Errorf("%s: can't find format dependency %s", f.Name, dName)
|
|
|
|
}
|
|
|
|
formats = append(formats, df...)
|
|
|
|
}
|
|
|
|
|
|
|
|
sortFormats(formats)
|
|
|
|
*d.Formats = formats
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, fs := range r.Groups {
|
|
|
|
sortFormats(fs)
|
|
|
|
}
|
|
|
|
|
2021-09-28 14:04:51 +03:00
|
|
|
r.resolved = true
|
2020-06-08 03:29:51 +03:00
|
|
|
|
2021-09-28 14:04:51 +03:00
|
|
|
return nil
|
2020-06-08 03:29:51 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
func (r *Registry) Group(name string) ([]*decode.Format, error) {
|
|
|
|
r.resolveOnce.Do(func() {
|
|
|
|
if err := r.resolve(); err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
if g, ok := r.Groups[name]; ok {
|
|
|
|
return g, nil
|
|
|
|
}
|
|
|
|
return nil, errors.New("format group not found")
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *Registry) MustGroup(name string) []*decode.Format {
|
2021-09-27 12:01:14 +03:00
|
|
|
g, err := r.Group(name)
|
|
|
|
if err == nil {
|
2020-06-08 03:29:51 +03:00
|
|
|
return g
|
|
|
|
}
|
2021-09-27 12:01:14 +03:00
|
|
|
panic(err)
|
2020-06-08 03:29:51 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
func (r *Registry) MustAll() []*decode.Format {
|
|
|
|
return r.MustGroup("all")
|
|
|
|
}
|