mirror of
https://github.com/valyala/fasthttp.git
synced 2026-06-15 16:07:51 +03:00
38aa88ab52
* Make the prefork mode more robust The main process will exit if one of the prefork child processes doesn't complete successfully under the current prefork mode, so it ought to make sure that all child processes run independently and the main process will only exit after all child processes are finished. * Start over those failed child processes automatically * Kill all child processes before main process exits * Remove redundant code * Add configurable threshold of starting over child processes * Return a error of RecoverThreshold * Resolved requested changes * Add logs * Resolve requested changes
277 lines
6.0 KiB
Go
277 lines
6.0 KiB
Go
package prefork
|
|
|
|
import (
|
|
"errors"
|
|
"flag"
|
|
"log"
|
|
"net"
|
|
"os"
|
|
"os/exec"
|
|
"runtime"
|
|
|
|
"github.com/valyala/fasthttp"
|
|
"github.com/valyala/fasthttp/reuseport"
|
|
)
|
|
|
|
const (
|
|
preforkChildFlag = "-prefork-child"
|
|
defaultNetwork = "tcp4"
|
|
)
|
|
|
|
var (
|
|
defaultLogger = Logger(log.New(os.Stderr, "", log.LstdFlags))
|
|
// ErrOverRecovery is returned when the times of starting over child prefork processes exceed
|
|
// the threshold.
|
|
ErrOverRecovery = errors.New("exceeding the value of RecoverThreshold")
|
|
)
|
|
|
|
// Logger is used for logging formatted messages.
|
|
type Logger interface {
|
|
// Printf must have the same semantics as log.Printf.
|
|
Printf(format string, args ...interface{})
|
|
}
|
|
|
|
// Prefork implements fasthttp server prefork
|
|
//
|
|
// Preforks master process (with all cores) between several child processes
|
|
// increases performance significantly, because Go doesn't have to share
|
|
// and manage memory between cores
|
|
//
|
|
// WARNING: using prefork prevents the use of any global state!
|
|
// Things like in-memory caches won't work.
|
|
type Prefork struct {
|
|
// The network must be "tcp", "tcp4" or "tcp6".
|
|
//
|
|
// By default is "tcp4"
|
|
Network string
|
|
|
|
// Flag to use a listener with reuseport, if not a file Listener will be used
|
|
// See: https://www.nginx.com/blog/socket-sharding-nginx-release-1-9-1/
|
|
//
|
|
// It's disabled by default
|
|
Reuseport bool
|
|
|
|
// Child prefork processes may exit with failure and will be started over until the times reach
|
|
// the value of RecoverThreshold, then it will return and terminate the server.
|
|
RecoverThreshold int
|
|
|
|
// By default standard logger from log package is used.
|
|
Logger Logger
|
|
|
|
ServeFunc func(ln net.Listener) error
|
|
ServeTLSFunc func(ln net.Listener, certFile, keyFile string) error
|
|
ServeTLSEmbedFunc func(ln net.Listener, certData, keyData []byte) error
|
|
|
|
ln net.Listener
|
|
files []*os.File
|
|
}
|
|
|
|
func init() { //nolint:gochecknoinits
|
|
// Definition flag to not break the program when the user adds their own flags
|
|
// and runs `flag.Parse()`
|
|
flag.Bool(preforkChildFlag[1:], false, "Is a child process")
|
|
}
|
|
|
|
// IsChild checks if the current thread/process is a child
|
|
func IsChild() bool {
|
|
for _, arg := range os.Args[1:] {
|
|
if arg == preforkChildFlag {
|
|
return true
|
|
}
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
// New wraps the fasthttp server to run with preforked processes
|
|
func New(s *fasthttp.Server) *Prefork {
|
|
return &Prefork{
|
|
Network: defaultNetwork,
|
|
RecoverThreshold: runtime.GOMAXPROCS(0) / 2,
|
|
Logger: s.Logger,
|
|
ServeFunc: s.Serve,
|
|
ServeTLSFunc: s.ServeTLS,
|
|
ServeTLSEmbedFunc: s.ServeTLSEmbed,
|
|
}
|
|
}
|
|
|
|
func (p *Prefork) logger() Logger {
|
|
if p.Logger != nil {
|
|
return p.Logger
|
|
}
|
|
return defaultLogger
|
|
}
|
|
|
|
func (p *Prefork) listen(addr string) (net.Listener, error) {
|
|
runtime.GOMAXPROCS(1)
|
|
|
|
if p.Network == "" {
|
|
p.Network = defaultNetwork
|
|
}
|
|
|
|
if p.Reuseport {
|
|
return reuseport.Listen(p.Network, addr)
|
|
}
|
|
|
|
return net.FileListener(os.NewFile(3, ""))
|
|
}
|
|
|
|
func (p *Prefork) setTCPListenerFiles(addr string) error {
|
|
if p.Network == "" {
|
|
p.Network = defaultNetwork
|
|
}
|
|
|
|
tcpAddr, err := net.ResolveTCPAddr(p.Network, addr)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
tcplistener, err := net.ListenTCP(p.Network, tcpAddr)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
p.ln = tcplistener
|
|
|
|
fl, err := tcplistener.File()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
p.files = []*os.File{fl}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (p *Prefork) doCommand() (*exec.Cmd, error) {
|
|
/* #nosec G204 */
|
|
cmd := exec.Command(os.Args[0], append(os.Args[1:], preforkChildFlag)...)
|
|
cmd.Stdout = os.Stdout
|
|
cmd.Stderr = os.Stderr
|
|
cmd.ExtraFiles = p.files
|
|
return cmd, cmd.Start()
|
|
}
|
|
|
|
func (p *Prefork) prefork(addr string) (err error) {
|
|
if !p.Reuseport {
|
|
if err = p.setTCPListenerFiles(addr); err != nil {
|
|
return
|
|
}
|
|
|
|
// defer for closing the net.Listener opened by setTCPListenerFiles.
|
|
defer func() {
|
|
e := p.ln.Close()
|
|
if err == nil {
|
|
err = e
|
|
}
|
|
}()
|
|
}
|
|
|
|
type procSig struct {
|
|
pid int
|
|
err error
|
|
}
|
|
|
|
goMaxProcs := runtime.GOMAXPROCS(0)
|
|
sigCh := make(chan procSig, goMaxProcs)
|
|
childProcs := make(map[int]*exec.Cmd)
|
|
|
|
defer func() {
|
|
for _, proc := range childProcs {
|
|
_ = proc.Process.Kill()
|
|
}
|
|
}()
|
|
|
|
for i := 0; i < goMaxProcs; i++ {
|
|
var cmd *exec.Cmd
|
|
if cmd, err = p.doCommand(); err != nil {
|
|
p.logger().Printf("failed to start a child prefork process, error: %v\n", err)
|
|
return
|
|
}
|
|
|
|
childProcs[cmd.Process.Pid] = cmd
|
|
go func() {
|
|
sigCh <- procSig{cmd.Process.Pid, cmd.Wait()}
|
|
}()
|
|
}
|
|
|
|
var exitedProcs int
|
|
for sig := range sigCh {
|
|
delete(childProcs, sig.pid)
|
|
|
|
p.logger().Printf("one of the child prefork processes exited with "+
|
|
"error: %v", sig.err)
|
|
|
|
if exitedProcs++; exitedProcs > p.RecoverThreshold {
|
|
p.logger().Printf("child prefork processes exit too many times, "+
|
|
"which exceeds the value of RecoverThreshold(%d), "+
|
|
"exiting the master process.\n", exitedProcs)
|
|
err = ErrOverRecovery
|
|
break
|
|
}
|
|
|
|
var cmd *exec.Cmd
|
|
if cmd, err = p.doCommand(); err != nil {
|
|
break
|
|
}
|
|
childProcs[cmd.Process.Pid] = cmd
|
|
go func() {
|
|
sigCh <- procSig{cmd.Process.Pid, cmd.Wait()}
|
|
}()
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
// ListenAndServe serves HTTP requests from the given TCP addr
|
|
func (p *Prefork) ListenAndServe(addr string) error {
|
|
if IsChild() {
|
|
ln, err := p.listen(addr)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
p.ln = ln
|
|
|
|
return p.ServeFunc(ln)
|
|
}
|
|
|
|
return p.prefork(addr)
|
|
}
|
|
|
|
// ListenAndServeTLS serves HTTPS requests from the given TCP addr
|
|
//
|
|
// certFile and keyFile are paths to TLS certificate and key files.
|
|
func (p *Prefork) ListenAndServeTLS(addr, certKey, certFile string) error {
|
|
if IsChild() {
|
|
ln, err := p.listen(addr)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
p.ln = ln
|
|
|
|
return p.ServeTLSFunc(ln, certFile, certKey)
|
|
}
|
|
|
|
return p.prefork(addr)
|
|
}
|
|
|
|
// ListenAndServeTLSEmbed serves HTTPS requests from the given TCP addr
|
|
//
|
|
// certData and keyData must contain valid TLS certificate and key data.
|
|
func (p *Prefork) ListenAndServeTLSEmbed(addr string, certData, keyData []byte) error {
|
|
if IsChild() {
|
|
ln, err := p.listen(addr)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
p.ln = ln
|
|
|
|
return p.ServeTLSEmbedFunc(ln, certData, keyData)
|
|
}
|
|
|
|
return p.prefork(addr)
|
|
}
|