1
0
mirror of https://github.com/go-kratos/kratos.git synced 2025-01-28 03:57:02 +02:00
kratos/transport/http/server.go

151 lines
3.5 KiB
Go
Raw Normal View History

2021-02-17 17:14:47 +08:00
package http
import (
"context"
2021-02-17 22:54:41 +08:00
"errors"
2021-02-17 17:14:47 +08:00
"fmt"
"net"
"net/http"
2021-05-28 15:08:25 +08:00
"strings"
2021-02-17 17:14:47 +08:00
"time"
ic "github.com/go-kratos/kratos/v2/internal/context"
2021-02-17 17:14:47 +08:00
"github.com/go-kratos/kratos/v2/internal/host"
"github.com/go-kratos/kratos/v2/log"
"github.com/go-kratos/kratos/v2/transport"
"github.com/gorilla/mux"
)
var _ transport.Server = (*Server)(nil)
2021-05-28 15:30:55 +08:00
var _ transport.Endpointer = (*Server)(nil)
2021-02-17 17:14:47 +08:00
// ServerOption is an HTTP server option.
2021-02-17 17:14:47 +08:00
type ServerOption func(*Server)
// Network with server network.
func Network(network string) ServerOption {
return func(s *Server) {
s.network = network
}
}
// Address with server address.
func Address(addr string) ServerOption {
return func(s *Server) {
s.address = addr
}
}
// Timeout with server timeout.
func Timeout(timeout time.Duration) ServerOption {
return func(s *Server) {
s.timeout = timeout
}
}
// Logger with server logger.
func Logger(logger log.Logger) ServerOption {
return func(s *Server) {
s.log = log.NewHelper(logger)
2021-02-17 17:14:47 +08:00
}
}
// Server is an HTTP server wrapper.
2021-02-17 17:14:47 +08:00
type Server struct {
*http.Server
ctx context.Context
lis net.Listener
network string
address string
timeout time.Duration
router *mux.Router
log *log.Helper
2021-02-17 17:14:47 +08:00
}
// NewServer creates an HTTP server by options.
2021-02-17 17:14:47 +08:00
func NewServer(opts ...ServerOption) *Server {
srv := &Server{
network: "tcp",
address: ":0",
timeout: 1 * time.Second,
log: log.NewHelper(log.DefaultLogger),
2021-02-17 17:14:47 +08:00
}
for _, o := range opts {
o(srv)
}
srv.router = mux.NewRouter()
srv.Server = &http.Server{Handler: srv}
return srv
}
// Handle registers a new route with a matcher for the URL path.
func (s *Server) Handle(path string, h http.Handler) {
s.router.Handle(path, h)
}
// HandlePrefix registers a new route with a matcher for the URL path prefix.
func (s *Server) HandlePrefix(prefix string, h http.Handler) {
s.router.PathPrefix(prefix).Handler(h)
}
2021-02-17 17:14:47 +08:00
// HandleFunc registers a new route with a matcher for the URL path.
func (s *Server) HandleFunc(path string, h http.HandlerFunc) {
s.router.HandleFunc(path, h)
}
// ServeHTTP should write reply headers and data to the ResponseWriter and then return.
func (s *Server) ServeHTTP(res http.ResponseWriter, req *http.Request) {
ctx, cancel := ic.Merge(req.Context(), s.ctx)
2021-02-17 17:14:47 +08:00
defer cancel()
ctx = transport.NewContext(ctx, transport.Transport{Kind: transport.KindHTTP})
2021-02-17 17:14:47 +08:00
ctx = NewServerContext(ctx, ServerInfo{Request: req, Response: res})
if s.timeout > 0 {
2021-05-30 21:43:41 +08:00
ctx, cancel = context.WithTimeout(ctx, s.timeout)
defer cancel()
}
s.router.ServeHTTP(res, req.WithContext(ctx))
2021-02-17 17:14:47 +08:00
}
// Endpoint return a real address to registry endpoint.
// examples:
// http://127.0.0.1:8000?isSecure=false
func (s *Server) Endpoint() (string, error) {
2021-05-28 15:30:55 +08:00
if s.lis == nil && strings.HasSuffix(s.address, ":0") {
2021-05-28 15:08:25 +08:00
lis, err := net.Listen(s.network, s.address)
if err != nil {
return "", err
}
s.lis = lis
}
2021-02-17 17:14:47 +08:00
addr, err := host.Extract(s.address, s.lis)
if err != nil {
return "", err
}
s.address = addr
2021-02-17 17:14:47 +08:00
return fmt.Sprintf("http://%s", addr), nil
}
// Start start the HTTP server.
func (s *Server) Start(ctx context.Context) error {
s.ctx = ctx
2021-05-28 15:08:25 +08:00
if s.lis == nil {
lis, err := net.Listen(s.network, s.address)
if err != nil {
return err
}
s.lis = lis
2021-02-17 17:14:47 +08:00
}
2021-05-28 15:08:25 +08:00
s.log.Infof("[HTTP] server listening on: %s", s.lis.Addr().String())
if err := s.Serve(s.lis); !errors.Is(err, http.ErrServerClosed) {
2021-02-17 22:54:41 +08:00
return err
}
return nil
2021-02-17 17:14:47 +08:00
}
// Stop stop the HTTP server.
func (s *Server) Stop(ctx context.Context) error {
2021-02-17 17:14:47 +08:00
s.log.Info("[HTTP] server stopping")
return s.Shutdown(context.Background())
}