happyDomain/services/decode.go

159 lines
4.1 KiB
Go
Raw Permalink Normal View History

2023-12-24 10:18:08 +00:00
// This file is part of the happyDomain (R) project.
// Copyright (c) 2020-2024 happyDomain
// Authors: Pierre-Olivier Mercier, et al.
2020-05-09 11:14:29 +00:00
//
2023-12-24 10:18:08 +00:00
// This program is offered under a commercial and under the AGPL license.
// For commercial licensing, contact us at <contact@happydomain.org>.
2020-05-09 11:14:29 +00:00
//
2023-12-24 10:18:08 +00:00
// For AGPL licensing:
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
2020-05-09 11:14:29 +00:00
//
2023-12-24 10:18:08 +00:00
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Affero General Public License for more details.
2020-05-09 11:14:29 +00:00
//
2023-12-24 10:18:08 +00:00
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>.
2020-05-09 11:14:29 +00:00
package svcs
import (
2020-05-10 22:14:00 +00:00
"fmt"
2020-05-09 11:14:29 +00:00
"log"
"reflect"
2020-05-09 11:14:29 +00:00
"sort"
2020-10-10 16:44:17 +00:00
"strings"
2020-05-09 11:14:29 +00:00
2023-09-07 09:37:18 +00:00
"git.happydns.org/happyDomain/model"
2020-05-09 11:14:29 +00:00
)
2020-10-10 16:44:17 +00:00
const (
Abstract = "abstract"
2023-12-06 00:52:27 +00:00
Hidden = "hidden"
2020-10-10 16:44:17 +00:00
Provider = "provider"
)
2020-05-09 11:14:29 +00:00
type ServiceCreator func() happydns.Service
type SubServiceCreator func() interface{}
2020-05-09 11:14:29 +00:00
type ServiceAnalyzer func(*Analyzer) error
2020-10-05 08:20:15 +00:00
type Svc struct {
2020-05-09 11:14:29 +00:00
Creator ServiceCreator
Analyzer ServiceAnalyzer
Infos ServiceInfos
Weight uint32
}
2020-10-05 08:20:15 +00:00
type ByWeight []*Svc
2020-05-09 11:14:29 +00:00
func (a ByWeight) Len() int { return len(a) }
func (a ByWeight) Swap(i, j int) { a[i], a[j] = a[j], a[i] }
func (a ByWeight) Less(i, j int) bool { return a[i].Weight < a[j].Weight }
var (
2020-10-05 08:20:15 +00:00
services map[string]*Svc = map[string]*Svc{}
subServices map[string]SubServiceCreator = map[string]SubServiceCreator{}
2023-09-07 09:37:18 +00:00
pathToSvcsModule string = "git.happydns.org/happyDomain/services"
2020-10-05 08:20:15 +00:00
ordered_services []*Svc
)
2020-05-09 11:14:29 +00:00
func RegisterService(creator ServiceCreator, analyzer ServiceAnalyzer, infos ServiceInfos, weight uint32, aliases ...string) {
// Invalidate ordered_services, which serve as cache
ordered_services = nil
baseType := reflect.Indirect(reflect.ValueOf(creator())).Type()
name := baseType.String()
2020-05-09 11:14:29 +00:00
log.Println("Registering new service:", name)
2020-10-12 14:34:36 +00:00
// Override given parameters by true one
2020-10-12 12:16:14 +00:00
infos.Type = name
2020-10-12 14:34:36 +00:00
if _, ok := Icons[name]; ok {
infos.Icon = "/api/service_specs/" + name + "/icon.png"
}
2020-10-12 12:16:14 +00:00
2020-10-05 08:20:15 +00:00
svc := &Svc{
2020-05-09 11:14:29 +00:00
creator,
analyzer,
infos,
weight,
}
services[name] = svc
// Register aliases
for _, alias := range aliases {
services[alias] = svc
}
// Register sub types
RegisterSubServices(baseType)
}
func RegisterSubServices(t reflect.Type) {
2020-10-10 16:44:17 +00:00
if t.Kind() == reflect.Struct && strings.HasPrefix(t.PkgPath(), pathToSvcsModule) {
if _, ok := subServices[t.String()]; !ok {
log.Println("Registering new subservice:", t.String())
subServices[t.String()] = func() interface{} {
return reflect.New(t).Interface()
}
}
for i := 0; i < t.NumField(); i += 1 {
RegisterSubServices(t.Field(i).Type)
}
} else if t.Kind() == reflect.Array || t.Kind() == reflect.Map || t.Kind() == reflect.Ptr || t.Kind() == reflect.Slice {
RegisterSubServices(t.Elem())
} else if t.PkgPath() == pathToSvcsModule {
if _, ok := subServices[t.String()]; ok {
return
}
log.Println("Registering new subservice:", t.String())
subServices[t.String()] = func() interface{} {
return reflect.New(t).Interface()
}
}
2020-05-09 11:14:29 +00:00
}
2020-10-05 08:20:15 +00:00
func OrderedServices() []*Svc {
2020-05-09 11:14:29 +00:00
if ordered_services == nil {
// Create the list
2020-05-10 22:14:00 +00:00
for _, svc := range services {
2020-05-09 11:14:29 +00:00
ordered_services = append(ordered_services, svc)
}
// Sort the list
sort.Sort(ByWeight(ordered_services))
}
return ordered_services
}
2020-05-10 22:14:00 +00:00
2020-10-05 08:20:15 +00:00
func GetServices() *map[string]*Svc {
2020-05-10 22:14:00 +00:00
return &services
}
func FindService(name string) (happydns.Service, error) {
svc, ok := services[name]
if !ok {
2020-10-10 16:44:17 +00:00
return nil, ServiceNotFoundError{name}
2020-05-10 22:14:00 +00:00
}
return svc.Creator(), nil
}
func FindSubService(name string) (interface{}, error) {
if svc, ok := services[name]; ok {
return svc.Creator(), nil
} else if ssvc, ok := subServices[name]; ok {
return ssvc(), nil
} else {
return nil, fmt.Errorf("Unable to find corresponding service `%s`.", name)
}
}