2020-04-19 02:31:09 +03:00
|
|
|
package di
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
2021-02-26 02:45:45 +01:00
|
|
|
"runtime/debug"
|
2020-04-19 02:31:09 +03:00
|
|
|
"time"
|
|
|
|
|
2023-12-13 17:29:12 +01:00
|
|
|
"github.com/defval/di"
|
2020-04-20 17:12:58 +03:00
|
|
|
"github.com/getsentry/raven-go"
|
2020-04-19 02:31:09 +03:00
|
|
|
"github.com/spf13/viper"
|
|
|
|
|
2024-02-01 08:12:34 +01:00
|
|
|
. "ely.by/chrly/internal/http"
|
2020-04-19 02:31:09 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
var server = di.Options(
|
|
|
|
di.Provide(newAuthenticator, di.As(new(Authenticator))),
|
|
|
|
di.Provide(newServer),
|
|
|
|
)
|
|
|
|
|
|
|
|
func newAuthenticator(config *viper.Viper, emitter Emitter) (*JwtAuth, error) {
|
|
|
|
key := config.GetString("chrly.secret")
|
|
|
|
if key == "" {
|
|
|
|
return nil, errors.New("chrly.secret must be set in order to use authenticator")
|
|
|
|
}
|
|
|
|
|
|
|
|
return &JwtAuth{
|
|
|
|
Key: []byte(key),
|
|
|
|
Emitter: emitter,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2020-04-20 17:12:58 +03:00
|
|
|
type serverParams struct {
|
|
|
|
di.Inject
|
2020-04-19 02:31:09 +03:00
|
|
|
|
2020-04-20 17:12:58 +03:00
|
|
|
Config *viper.Viper `di:""`
|
|
|
|
Handler http.Handler `di:""`
|
|
|
|
Sentry *raven.Client `di:"" optional:"true"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func newServer(params serverParams) *http.Server {
|
|
|
|
params.Config.SetDefault("server.host", "")
|
|
|
|
params.Config.SetDefault("server.port", 80)
|
|
|
|
|
2021-02-26 02:45:45 +01:00
|
|
|
var handler http.Handler
|
2020-04-20 17:12:58 +03:00
|
|
|
if params.Sentry != nil {
|
|
|
|
// raven.Recoverer uses DefaultClient and nothing can be done about it
|
|
|
|
// To avoid code duplication, if the Sentry service is successfully initiated,
|
|
|
|
// it will also replace DefaultClient, so raven.Recoverer will work with the instance
|
|
|
|
// created in the application constructor
|
2021-02-26 02:45:45 +01:00
|
|
|
handler = raven.Recoverer(params.Handler)
|
|
|
|
} else {
|
|
|
|
// Raven's Recoverer is prints the stacktrace and sets the corresponding status itself.
|
|
|
|
// But there is no magic and if you don't define a panic handler, Mux will just reset the connection
|
|
|
|
handler = http.HandlerFunc(func(request http.ResponseWriter, response *http.Request) {
|
|
|
|
defer func() {
|
|
|
|
if recovered := recover(); recovered != nil {
|
|
|
|
debug.PrintStack() // TODO: colorize output
|
|
|
|
request.WriteHeader(http.StatusInternalServerError)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
params.Handler.ServeHTTP(request, response)
|
|
|
|
})
|
2020-04-20 17:12:58 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
address := fmt.Sprintf("%s:%d", params.Config.GetString("server.host"), params.Config.GetInt("server.port"))
|
2020-04-19 02:31:09 +03:00
|
|
|
server := &http.Server{
|
|
|
|
Addr: address,
|
|
|
|
ReadTimeout: 5 * time.Second,
|
|
|
|
WriteTimeout: 5 * time.Second,
|
|
|
|
IdleTimeout: 60 * time.Second,
|
|
|
|
MaxHeaderBytes: 1 << 16,
|
|
|
|
Handler: handler,
|
|
|
|
}
|
|
|
|
|
|
|
|
return server
|
|
|
|
}
|