И под внутренним пакетом у меня есть еще 6 пакетов, которым нужен этот логгер.и каждый из них зависит от этого.(В пакете журналов) есть ли хорошее начало, чтобы справиться с этим?
Хороший общий принцип - уважать выбор приложения (регистрировать или нет)вместо установки политики.
Пакеты Let Go в вашем каталоге internal
будут пакетами поддержки , которые
- будут возвращать только
error
в случае проблем - не будет регистрироваться (консоль или иным образом)
- не будет
panic
Пусть ваше приложение(пакеты в вашей директории cmd
) решают, какое поведение следует предпринять в случае ошибки (запись / постепенное завершение работы / восстановление до 100% целостности)
Это упростит разработку благодаря ведению журналатолько на определенном слое.Примечание: не забудьте дать приложению достаточный контекст для определения действия
internal / process / process.go
package process
import (
"errors"
)
var (
ErrNotFound = errors.New("Not Found")
ErrConnFail = errors.New("Connection Failed")
)
// function Process is a dummy function that returns error for certain arguments received
func Process(i int) error {
switch i {
case 6:
return ErrNotFound
case 7:
return ErrConnFail
default:
return nil
}
}
cmd / servi / main.go
package main
import (
"log"
p "../../internal/process"
)
func main() {
// sample: generic logging on any failure
err := p.Process(6)
if err != nil {
log.Println("FAIL", err)
}
// sample: this application decides how to handle error based on context
err = p.Process(7)
if err != nil {
switch err {
case p.ErrNotFound:
log.Println("DOESN'T EXIST. TRY ANOTHER")
case p.ErrConnFail:
log.Println("UNABLE TO CONNECT; WILL RETRY LATER")
}
}
}
на случай, если у меня будет больше вещей, которые мне нужно передать (например, логгер), какой здесь будет подход / шаблон
Зависимостьинъекция всегда хороший выбор.Рассматривайте другие только тогда, когда простейшей реализации недостаточно.
Код ниже 'связывает' пакеты template
и logger
вместе, используя внедрение зависимостей и первоклассную передачу функций.
internal / logs / logger.go
package logger
import (
"github.com/sirupsen/logrus"
"github.com/x-cray/logrus-prefixed-formatter"
"os"
)
var Logger *logrus.Logger
func NewLogger() *logrus.Logger {
var level logrus.Level
level = LogLevel("info")
logger := &logrus.Logger{
Out: os.Stdout,
Level: level,
Formatter: &prefixed.TextFormatter{
DisableColors: true,
TimestampFormat: "2009-06-03 11:04:075",
},
}
Logger = logger
return Logger
}
func LogLevel(lvl string) logrus.Level {
switch lvl {
case "info":
return logrus.InfoLevel
case "error":
return logrus.ErrorLevel
default:
panic("Not supported")
}
}
внутренняя / template / template.go
package template
import (
"fmt"
"github.com/sirupsen/logrus"
)
type Template struct {
Name string
logger *logrus.Logger
}
// factory function New accepts a logging function and some data
func New(logger *logrus.Logger, data string) *Template {
return &Template{data, logger}
}
// dummy function DoSomething should do something and log using the given logger
func (t *Template) DoSomething() {
t.logger.Info(fmt.Sprintf("%s, %s", t.Name, "did something"))
}
cmd / servi2 / main.go
package main
import (
"../../internal/logs"
"../../internal/template"
)
func main() {
// wire our template and logger together
loggingFunc := logger.NewLogger()
t := template.New(loggingFunc, "Penguin Template")
// use the stuff we've set up
t.DoSomething()
}
Надеюсь, это поможет.Cheers,