2019-06-03 20:30:43 +02:00
|
|
|
package web
|
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto/tls"
|
2019-07-08 09:01:42 +02:00
|
|
|
"fmt"
|
2019-06-03 20:30:43 +02:00
|
|
|
"net"
|
|
|
|
"net/http"
|
|
|
|
"os"
|
|
|
|
"os/signal"
|
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
|
|
|
"sync"
|
|
|
|
"time"
|
|
|
|
|
2020-01-30 00:40:43 +02:00
|
|
|
"github.com/micro/cli/v2"
|
2020-01-30 13:39:00 +02:00
|
|
|
"github.com/micro/go-micro/v2"
|
2020-03-11 19:55:39 +02:00
|
|
|
"github.com/micro/go-micro/v2/logger"
|
2020-01-30 13:39:00 +02:00
|
|
|
"github.com/micro/go-micro/v2/registry"
|
|
|
|
maddr "github.com/micro/go-micro/v2/util/addr"
|
|
|
|
mhttp "github.com/micro/go-micro/v2/util/http"
|
|
|
|
mnet "github.com/micro/go-micro/v2/util/net"
|
2020-04-21 14:00:12 +02:00
|
|
|
signalutil "github.com/micro/go-micro/v2/util/signal"
|
2020-01-30 13:39:00 +02:00
|
|
|
mls "github.com/micro/go-micro/v2/util/tls"
|
2019-06-03 20:30:43 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
type service struct {
|
|
|
|
opts Options
|
|
|
|
|
|
|
|
mux *http.ServeMux
|
|
|
|
srv *registry.Service
|
|
|
|
|
2020-04-26 16:41:36 +02:00
|
|
|
sync.RWMutex
|
2019-06-03 20:30:43 +02:00
|
|
|
running bool
|
|
|
|
static bool
|
|
|
|
exit chan chan error
|
|
|
|
}
|
|
|
|
|
|
|
|
func newService(opts ...Option) Service {
|
|
|
|
options := newOptions(opts...)
|
|
|
|
s := &service{
|
|
|
|
opts: options,
|
|
|
|
mux: http.NewServeMux(),
|
|
|
|
static: true,
|
|
|
|
}
|
|
|
|
s.srv = s.genSrv()
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *service) genSrv() *registry.Service {
|
2020-03-06 17:44:56 +02:00
|
|
|
var host string
|
|
|
|
var port string
|
|
|
|
var err error
|
|
|
|
|
2019-06-03 20:30:43 +02:00
|
|
|
// default host:port
|
2020-03-06 17:44:56 +02:00
|
|
|
if len(s.opts.Address) > 0 {
|
|
|
|
host, port, err = net.SplitHostPort(s.opts.Address)
|
|
|
|
if err != nil {
|
2020-04-26 23:03:05 +02:00
|
|
|
logger.Fatal(err)
|
2020-03-06 17:44:56 +02:00
|
|
|
}
|
|
|
|
}
|
2019-06-03 20:30:43 +02:00
|
|
|
|
|
|
|
// check the advertise address first
|
|
|
|
// if it exists then use it, otherwise
|
|
|
|
// use the address
|
|
|
|
if len(s.opts.Advertise) > 0 {
|
2020-03-06 17:44:56 +02:00
|
|
|
host, port, err = net.SplitHostPort(s.opts.Address)
|
|
|
|
if err != nil {
|
2020-04-26 23:03:05 +02:00
|
|
|
logger.Fatal(err)
|
2019-06-03 20:30:43 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
addr, err := maddr.Extract(host)
|
|
|
|
if err != nil {
|
2020-04-26 23:03:05 +02:00
|
|
|
logger.Fatal(err)
|
2020-03-06 17:44:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if strings.Count(addr, ":") > 0 {
|
|
|
|
addr = "[" + addr + "]"
|
2019-06-03 20:30:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return ®istry.Service{
|
|
|
|
Name: s.opts.Name,
|
|
|
|
Version: s.opts.Version,
|
2019-11-01 17:07:53 +02:00
|
|
|
Nodes: []*registry.Node{{
|
2019-06-03 20:30:43 +02:00
|
|
|
Id: s.opts.Id,
|
2020-03-06 17:44:56 +02:00
|
|
|
Address: fmt.Sprintf("%s:%s", addr, port),
|
2019-06-03 20:30:43 +02:00
|
|
|
Metadata: s.opts.Metadata,
|
|
|
|
}},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *service) run(exit chan bool) {
|
2020-04-26 16:41:36 +02:00
|
|
|
s.RLock()
|
2019-06-03 20:30:43 +02:00
|
|
|
if s.opts.RegisterInterval <= time.Duration(0) {
|
2020-04-26 16:41:36 +02:00
|
|
|
s.RUnlock()
|
2019-06-03 20:30:43 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
t := time.NewTicker(s.opts.RegisterInterval)
|
2020-04-26 16:41:36 +02:00
|
|
|
s.RUnlock()
|
2019-06-03 20:30:43 +02:00
|
|
|
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-t.C:
|
|
|
|
s.register()
|
|
|
|
case <-exit:
|
|
|
|
t.Stop()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *service) register() error {
|
2020-04-28 13:23:52 +02:00
|
|
|
s.RLock()
|
|
|
|
defer s.RUnlock()
|
|
|
|
|
2019-06-03 20:30:43 +02:00
|
|
|
if s.srv == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
// default to service registry
|
|
|
|
r := s.opts.Service.Client().Options().Registry
|
|
|
|
// switch to option if specified
|
|
|
|
if s.opts.Registry != nil {
|
|
|
|
r = s.opts.Registry
|
|
|
|
}
|
2019-11-02 10:39:56 +02:00
|
|
|
|
|
|
|
// service node need modify, node address maybe changed
|
|
|
|
srv := s.genSrv()
|
|
|
|
srv.Endpoints = s.srv.Endpoints
|
|
|
|
s.srv = srv
|
2019-12-05 16:15:35 +02:00
|
|
|
|
|
|
|
// use RegisterCheck func before register
|
|
|
|
if err := s.opts.RegisterCheck(s.opts.Context); err != nil {
|
2020-04-26 23:03:05 +02:00
|
|
|
if logger.V(logger.ErrorLevel, logger.DefaultLogger) {
|
|
|
|
logger.Errorf("Server %s-%s register check error: %s", s.opts.Name, s.opts.Id, err)
|
2020-03-11 19:55:39 +02:00
|
|
|
}
|
2019-12-05 16:15:35 +02:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-06-03 20:30:43 +02:00
|
|
|
return r.Register(s.srv, registry.RegisterTTL(s.opts.RegisterTTL))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *service) deregister() error {
|
2020-04-28 13:23:52 +02:00
|
|
|
s.RLock()
|
|
|
|
defer s.RUnlock()
|
|
|
|
|
2019-06-03 20:30:43 +02:00
|
|
|
if s.srv == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
// default to service registry
|
|
|
|
r := s.opts.Service.Client().Options().Registry
|
|
|
|
// switch to option if specified
|
|
|
|
if s.opts.Registry != nil {
|
|
|
|
r = s.opts.Registry
|
|
|
|
}
|
|
|
|
return r.Deregister(s.srv)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *service) start() error {
|
|
|
|
s.Lock()
|
|
|
|
defer s.Unlock()
|
|
|
|
|
|
|
|
if s.running {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-02-26 10:44:10 +02:00
|
|
|
for _, fn := range s.opts.BeforeStart {
|
|
|
|
if err := fn(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-03 20:30:43 +02:00
|
|
|
l, err := s.listen("tcp", s.opts.Address)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
s.opts.Address = l.Addr().String()
|
|
|
|
srv := s.genSrv()
|
|
|
|
srv.Endpoints = s.srv.Endpoints
|
|
|
|
s.srv = srv
|
|
|
|
|
|
|
|
var h http.Handler
|
|
|
|
|
|
|
|
if s.opts.Handler != nil {
|
|
|
|
h = s.opts.Handler
|
|
|
|
} else {
|
|
|
|
h = s.mux
|
|
|
|
var r sync.Once
|
|
|
|
|
|
|
|
// register the html dir
|
|
|
|
r.Do(func() {
|
|
|
|
// static dir
|
|
|
|
static := s.opts.StaticDir
|
|
|
|
if s.opts.StaticDir[0] != '/' {
|
|
|
|
dir, _ := os.Getwd()
|
|
|
|
static = filepath.Join(dir, static)
|
|
|
|
}
|
|
|
|
|
|
|
|
// set static if no / handler is registered
|
|
|
|
if s.static {
|
|
|
|
_, err := os.Stat(static)
|
|
|
|
if err == nil {
|
2020-04-26 23:03:05 +02:00
|
|
|
if logger.V(logger.InfoLevel, logger.DefaultLogger) {
|
|
|
|
logger.Infof("Enabling static file serving from %s", static)
|
2020-03-11 19:55:39 +02:00
|
|
|
}
|
2019-06-03 20:30:43 +02:00
|
|
|
s.mux.Handle("/", http.FileServer(http.Dir(static)))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
var httpSrv *http.Server
|
|
|
|
if s.opts.Server != nil {
|
|
|
|
httpSrv = s.opts.Server
|
|
|
|
} else {
|
|
|
|
httpSrv = &http.Server{}
|
|
|
|
}
|
|
|
|
|
|
|
|
httpSrv.Handler = h
|
|
|
|
|
|
|
|
go httpSrv.Serve(l)
|
|
|
|
|
|
|
|
for _, fn := range s.opts.AfterStart {
|
|
|
|
if err := fn(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
s.exit = make(chan chan error, 1)
|
|
|
|
s.running = true
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
ch := <-s.exit
|
|
|
|
ch <- l.Close()
|
|
|
|
}()
|
|
|
|
|
2020-04-26 23:03:05 +02:00
|
|
|
if logger.V(logger.InfoLevel, logger.DefaultLogger) {
|
|
|
|
logger.Infof("Listening on %v", l.Addr().String())
|
2020-03-11 19:55:39 +02:00
|
|
|
}
|
2019-06-03 20:30:43 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *service) stop() error {
|
|
|
|
s.Lock()
|
|
|
|
defer s.Unlock()
|
|
|
|
|
|
|
|
if !s.running {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, fn := range s.opts.BeforeStop {
|
|
|
|
if err := fn(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ch := make(chan error, 1)
|
|
|
|
s.exit <- ch
|
|
|
|
s.running = false
|
|
|
|
|
2020-04-26 23:03:05 +02:00
|
|
|
if logger.V(logger.InfoLevel, logger.DefaultLogger) {
|
|
|
|
logger.Info("Stopping")
|
2020-03-11 19:55:39 +02:00
|
|
|
}
|
2019-06-03 20:30:43 +02:00
|
|
|
|
|
|
|
for _, fn := range s.opts.AfterStop {
|
|
|
|
if err := fn(); err != nil {
|
|
|
|
if chErr := <-ch; chErr != nil {
|
|
|
|
return chErr
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return <-ch
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *service) Client() *http.Client {
|
|
|
|
rt := mhttp.NewRoundTripper(
|
2020-04-05 12:15:38 +02:00
|
|
|
mhttp.WithRegistry(s.opts.Registry),
|
2019-06-03 20:30:43 +02:00
|
|
|
)
|
|
|
|
return &http.Client{
|
|
|
|
Transport: rt,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *service) Handle(pattern string, handler http.Handler) {
|
|
|
|
var seen bool
|
2020-04-28 13:23:52 +02:00
|
|
|
s.RLock()
|
2019-06-03 20:30:43 +02:00
|
|
|
for _, ep := range s.srv.Endpoints {
|
|
|
|
if ep.Name == pattern {
|
|
|
|
seen = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
2020-04-28 13:23:52 +02:00
|
|
|
s.RUnlock()
|
2019-06-03 20:30:43 +02:00
|
|
|
|
|
|
|
// if its unseen then add an endpoint
|
|
|
|
if !seen {
|
2020-04-28 13:23:52 +02:00
|
|
|
s.Lock()
|
2019-06-03 20:30:43 +02:00
|
|
|
s.srv.Endpoints = append(s.srv.Endpoints, ®istry.Endpoint{
|
|
|
|
Name: pattern,
|
|
|
|
})
|
2020-04-28 13:23:52 +02:00
|
|
|
s.Unlock()
|
2019-06-03 20:30:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// disable static serving
|
|
|
|
if pattern == "/" {
|
|
|
|
s.Lock()
|
|
|
|
s.static = false
|
|
|
|
s.Unlock()
|
|
|
|
}
|
|
|
|
|
|
|
|
// register the handler
|
|
|
|
s.mux.Handle(pattern, handler)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *service) HandleFunc(pattern string, handler func(http.ResponseWriter, *http.Request)) {
|
2020-04-28 13:23:52 +02:00
|
|
|
|
2019-06-03 20:30:43 +02:00
|
|
|
var seen bool
|
2020-04-28 13:23:52 +02:00
|
|
|
s.RLock()
|
2019-06-03 20:30:43 +02:00
|
|
|
for _, ep := range s.srv.Endpoints {
|
|
|
|
if ep.Name == pattern {
|
|
|
|
seen = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
2020-04-28 13:23:52 +02:00
|
|
|
s.RUnlock()
|
|
|
|
|
2019-06-03 20:30:43 +02:00
|
|
|
if !seen {
|
2020-04-28 13:23:52 +02:00
|
|
|
s.Lock()
|
2019-06-03 20:30:43 +02:00
|
|
|
s.srv.Endpoints = append(s.srv.Endpoints, ®istry.Endpoint{
|
|
|
|
Name: pattern,
|
|
|
|
})
|
2020-04-28 13:23:52 +02:00
|
|
|
s.Unlock()
|
2019-06-03 20:30:43 +02:00
|
|
|
}
|
|
|
|
|
2020-04-19 21:30:38 +02:00
|
|
|
// disable static serving
|
|
|
|
if pattern == "/" {
|
|
|
|
s.Lock()
|
|
|
|
s.static = false
|
|
|
|
s.Unlock()
|
|
|
|
}
|
|
|
|
|
2019-06-03 20:30:43 +02:00
|
|
|
s.mux.HandleFunc(pattern, handler)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *service) Init(opts ...Option) error {
|
2020-04-26 16:41:36 +02:00
|
|
|
s.Lock()
|
|
|
|
|
2019-06-03 20:30:43 +02:00
|
|
|
for _, o := range opts {
|
|
|
|
o(&s.opts)
|
|
|
|
}
|
|
|
|
|
|
|
|
serviceOpts := []micro.Option{}
|
|
|
|
|
|
|
|
if len(s.opts.Flags) > 0 {
|
|
|
|
serviceOpts = append(serviceOpts, micro.Flags(s.opts.Flags...))
|
|
|
|
}
|
|
|
|
|
|
|
|
if s.opts.Registry != nil {
|
|
|
|
serviceOpts = append(serviceOpts, micro.Registry(s.opts.Registry))
|
|
|
|
}
|
|
|
|
|
2020-04-28 13:23:52 +02:00
|
|
|
s.Unlock()
|
|
|
|
|
2020-01-25 12:41:25 +02:00
|
|
|
serviceOpts = append(serviceOpts, micro.Action(func(ctx *cli.Context) error {
|
2020-04-26 16:41:36 +02:00
|
|
|
s.Lock()
|
|
|
|
defer s.Unlock()
|
|
|
|
|
2019-06-03 20:30:43 +02:00
|
|
|
if ttl := ctx.Int("register_ttl"); ttl > 0 {
|
|
|
|
s.opts.RegisterTTL = time.Duration(ttl) * time.Second
|
|
|
|
}
|
|
|
|
|
|
|
|
if interval := ctx.Int("register_interval"); interval > 0 {
|
|
|
|
s.opts.RegisterInterval = time.Duration(interval) * time.Second
|
|
|
|
}
|
|
|
|
|
|
|
|
if name := ctx.String("server_name"); len(name) > 0 {
|
|
|
|
s.opts.Name = name
|
|
|
|
}
|
|
|
|
|
|
|
|
if ver := ctx.String("server_version"); len(ver) > 0 {
|
|
|
|
s.opts.Version = ver
|
|
|
|
}
|
|
|
|
|
|
|
|
if id := ctx.String("server_id"); len(id) > 0 {
|
|
|
|
s.opts.Id = id
|
|
|
|
}
|
|
|
|
|
|
|
|
if addr := ctx.String("server_address"); len(addr) > 0 {
|
|
|
|
s.opts.Address = addr
|
|
|
|
}
|
|
|
|
|
|
|
|
if adv := ctx.String("server_advertise"); len(adv) > 0 {
|
|
|
|
s.opts.Advertise = adv
|
|
|
|
}
|
|
|
|
|
|
|
|
if s.opts.Action != nil {
|
|
|
|
s.opts.Action(ctx)
|
|
|
|
}
|
2020-01-25 12:41:25 +02:00
|
|
|
|
|
|
|
return nil
|
2019-06-03 20:30:43 +02:00
|
|
|
}))
|
|
|
|
|
2020-04-28 13:23:52 +02:00
|
|
|
s.RLock()
|
2020-04-09 10:41:50 +02:00
|
|
|
// pass in own name and version
|
|
|
|
serviceOpts = append(serviceOpts, micro.Name(s.opts.Name))
|
|
|
|
serviceOpts = append(serviceOpts, micro.Version(s.opts.Version))
|
2020-04-28 13:23:52 +02:00
|
|
|
s.RUnlock()
|
2020-04-09 10:41:50 +02:00
|
|
|
|
2019-06-03 20:30:43 +02:00
|
|
|
s.opts.Service.Init(serviceOpts...)
|
2020-04-28 13:23:52 +02:00
|
|
|
|
|
|
|
s.Lock()
|
2019-06-03 20:30:43 +02:00
|
|
|
srv := s.genSrv()
|
|
|
|
srv.Endpoints = s.srv.Endpoints
|
|
|
|
s.srv = srv
|
2020-04-28 13:23:52 +02:00
|
|
|
s.Unlock()
|
2019-06-03 20:30:43 +02:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *service) Run() error {
|
|
|
|
if err := s.start(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := s.register(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// start reg loop
|
|
|
|
ex := make(chan bool)
|
|
|
|
go s.run(ex)
|
|
|
|
|
|
|
|
ch := make(chan os.Signal, 1)
|
2020-01-10 16:25:28 +02:00
|
|
|
if s.opts.Signal {
|
2020-04-21 14:14:20 +02:00
|
|
|
signal.Notify(ch, signalutil.Shutdown()...)
|
2020-01-10 16:25:28 +02:00
|
|
|
}
|
2019-06-03 20:30:43 +02:00
|
|
|
|
|
|
|
select {
|
|
|
|
// wait on kill signal
|
|
|
|
case sig := <-ch:
|
2020-04-26 23:03:05 +02:00
|
|
|
if logger.V(logger.InfoLevel, logger.DefaultLogger) {
|
|
|
|
logger.Infof("Received signal %s", sig)
|
2020-03-11 19:55:39 +02:00
|
|
|
}
|
2019-06-03 20:30:43 +02:00
|
|
|
// wait on context cancel
|
|
|
|
case <-s.opts.Context.Done():
|
2020-04-26 23:03:05 +02:00
|
|
|
if logger.V(logger.InfoLevel, logger.DefaultLogger) {
|
|
|
|
logger.Info("Received context shutdown")
|
2020-03-11 19:55:39 +02:00
|
|
|
}
|
2019-06-03 20:30:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// exit reg loop
|
|
|
|
close(ex)
|
|
|
|
|
|
|
|
if err := s.deregister(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return s.stop()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Options returns the options for the given service
|
|
|
|
func (s *service) Options() Options {
|
|
|
|
return s.opts
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *service) listen(network, addr string) (net.Listener, error) {
|
|
|
|
var l net.Listener
|
|
|
|
var err error
|
|
|
|
|
|
|
|
// TODO: support use of listen options
|
|
|
|
if s.opts.Secure || s.opts.TLSConfig != nil {
|
|
|
|
config := s.opts.TLSConfig
|
|
|
|
|
|
|
|
fn := func(addr string) (net.Listener, error) {
|
|
|
|
if config == nil {
|
|
|
|
hosts := []string{addr}
|
|
|
|
|
|
|
|
// check if its a valid host:port
|
|
|
|
if host, _, err := net.SplitHostPort(addr); err == nil {
|
|
|
|
if len(host) == 0 {
|
|
|
|
hosts = maddr.IPs()
|
|
|
|
} else {
|
|
|
|
hosts = []string{host}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// generate a certificate
|
|
|
|
cert, err := mls.Certificate(hosts...)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
config = &tls.Config{Certificates: []tls.Certificate{cert}}
|
|
|
|
}
|
|
|
|
return tls.Listen(network, addr, config)
|
|
|
|
}
|
|
|
|
|
|
|
|
l, err = mnet.Listen(addr, fn)
|
|
|
|
} else {
|
|
|
|
fn := func(addr string) (net.Listener, error) {
|
|
|
|
return net.Listen(network, addr)
|
|
|
|
}
|
|
|
|
|
|
|
|
l, err = mnet.Listen(addr, fn)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return l, nil
|
|
|
|
}
|