mirror of
https://github.com/minio/minio.git
synced 2025-01-12 07:23:23 -05:00
9a012a53ef
This PR fixes a bug that perhaps has been long introduced, with no visible workarounds. In any deployment, if an entire erasure set is deleted, there is no way the cluster recovers.
114 lines
3.8 KiB
Go
114 lines
3.8 KiB
Go
// Copyright (c) 2015-2021 MinIO, Inc.
|
|
//
|
|
// This file is part of MinIO Object Storage stack
|
|
//
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU Affero General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// This program is distributed in the hope that it will be useful
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU Affero General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
package cmd
|
|
|
|
import (
|
|
"context"
|
|
"os"
|
|
"os/exec"
|
|
"runtime"
|
|
"syscall"
|
|
|
|
xioutil "github.com/minio/minio/internal/ioutil"
|
|
)
|
|
|
|
// Type of service signals currently supported.
|
|
type serviceSignal int
|
|
|
|
const (
|
|
serviceRestart serviceSignal = iota // Restarts the server.
|
|
serviceStop // Stops the server.
|
|
serviceReloadDynamic // Reload dynamic config values.
|
|
serviceFreeze // Freeze all S3 API calls.
|
|
serviceUnFreeze // Un-Freeze previously frozen S3 API calls.
|
|
// Add new service requests here.
|
|
)
|
|
|
|
// Global service signal channel.
|
|
var globalServiceSignalCh = make(chan serviceSignal)
|
|
|
|
// GlobalContext context that is canceled when server is requested to shut down.
|
|
// cancelGlobalContext can be used to indicate server shutdown.
|
|
var GlobalContext, cancelGlobalContext = context.WithCancel(context.Background())
|
|
|
|
// restartProcess starts a new process passing it the active fd's. It
|
|
// doesn't fork, but starts a new process using the same environment and
|
|
// arguments as when it was originally started. This allows for a newly
|
|
// deployed binary to be started. It returns the pid of the newly started
|
|
// process when successful.
|
|
func restartProcess() error {
|
|
if runtime.GOOS == globalWindowsOSName {
|
|
cmd := exec.Command(os.Args[0], os.Args[1:]...)
|
|
cmd.Stdout = os.Stdout
|
|
cmd.Stderr = os.Stderr
|
|
cmd.Stdin = os.Stdin
|
|
cmd.Env = os.Environ()
|
|
err := cmd.Run()
|
|
if err == nil {
|
|
os.Exit(0)
|
|
}
|
|
return err
|
|
}
|
|
|
|
// Use the original binary location. This works with symlinks such that if
|
|
// the file it points to has been changed we will use the updated symlink.
|
|
argv0, err := exec.LookPath(os.Args[0])
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Invokes the execve system call.
|
|
// Re-uses the same pid. This preserves the pid over multiple server-respawns.
|
|
return syscall.Exec(argv0, os.Args, os.Environ())
|
|
}
|
|
|
|
// freezeServices will freeze all incoming S3 API calls.
|
|
// For each call, unfreezeServices must be called once.
|
|
func freezeServices() {
|
|
// Use atomics for globalServiceFreeze, so we can read without locking.
|
|
// We need a lock since we are need the 2 atomic values to remain in sync.
|
|
globalServiceFreezeMu.Lock()
|
|
// If multiple calls, first one creates channel.
|
|
globalServiceFreezeCnt++
|
|
if globalServiceFreezeCnt == 1 {
|
|
globalServiceFreeze.Store(make(chan struct{}))
|
|
}
|
|
globalServiceFreezeMu.Unlock()
|
|
}
|
|
|
|
// unfreezeServices will unfreeze all incoming S3 API calls.
|
|
// For each call, unfreezeServices must be called once.
|
|
func unfreezeServices() {
|
|
// We need a lock since we need the 2 atomic values to remain in sync.
|
|
globalServiceFreezeMu.Lock()
|
|
// Close when we reach 0
|
|
globalServiceFreezeCnt--
|
|
if globalServiceFreezeCnt <= 0 {
|
|
// Set to a nil channel.
|
|
var _ch chan struct{}
|
|
if val := globalServiceFreeze.Swap(_ch); val != nil {
|
|
if ch, ok := val.(chan struct{}); ok && ch != nil {
|
|
// Close previous non-nil channel.
|
|
xioutil.SafeClose(ch)
|
|
}
|
|
}
|
|
globalServiceFreezeCnt = 0 // Don't risk going negative.
|
|
}
|
|
globalServiceFreezeMu.Unlock()
|
|
}
|