add better generate
Signed-off-by: Jess Frazelle <acidburn@microsoft.com>
This commit is contained in:
		
							parent
							
								
									3fc6abf56b
								
							
						
					
					
						commit
						cdd93563f5
					
				
					 5655 changed files with 1187011 additions and 392 deletions
				
			
		
							
								
								
									
										133
									
								
								vendor/github.com/docker/docker-ce/components/engine/restartmanager/restartmanager.go
									
										
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										133
									
								
								vendor/github.com/docker/docker-ce/components/engine/restartmanager/restartmanager.go
									
										
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							|  | @ -0,0 +1,133 @@ | |||
| package restartmanager // import "github.com/docker/docker/restartmanager" | ||||
| 
 | ||||
| import ( | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"sync" | ||||
| 	"time" | ||||
| 
 | ||||
| 	"github.com/docker/docker/api/types/container" | ||||
| ) | ||||
| 
 | ||||
| const ( | ||||
| 	backoffMultiplier = 2 | ||||
| 	defaultTimeout    = 100 * time.Millisecond | ||||
| 	maxRestartTimeout = 1 * time.Minute | ||||
| ) | ||||
| 
 | ||||
| // ErrRestartCanceled is returned when the restart manager has been | ||||
| // canceled and will no longer restart the container. | ||||
| var ErrRestartCanceled = errors.New("restart canceled") | ||||
| 
 | ||||
| // RestartManager defines object that controls container restarting rules. | ||||
| type RestartManager interface { | ||||
| 	Cancel() error | ||||
| 	ShouldRestart(exitCode uint32, hasBeenManuallyStopped bool, executionDuration time.Duration) (bool, chan error, error) | ||||
| } | ||||
| 
 | ||||
| type restartManager struct { | ||||
| 	sync.Mutex | ||||
| 	sync.Once | ||||
| 	policy       container.RestartPolicy | ||||
| 	restartCount int | ||||
| 	timeout      time.Duration | ||||
| 	active       bool | ||||
| 	cancel       chan struct{} | ||||
| 	canceled     bool | ||||
| } | ||||
| 
 | ||||
| // New returns a new restartManager based on a policy. | ||||
| func New(policy container.RestartPolicy, restartCount int) RestartManager { | ||||
| 	return &restartManager{policy: policy, restartCount: restartCount, cancel: make(chan struct{})} | ||||
| } | ||||
| 
 | ||||
| func (rm *restartManager) SetPolicy(policy container.RestartPolicy) { | ||||
| 	rm.Lock() | ||||
| 	rm.policy = policy | ||||
| 	rm.Unlock() | ||||
| } | ||||
| 
 | ||||
| func (rm *restartManager) ShouldRestart(exitCode uint32, hasBeenManuallyStopped bool, executionDuration time.Duration) (bool, chan error, error) { | ||||
| 	if rm.policy.IsNone() { | ||||
| 		return false, nil, nil | ||||
| 	} | ||||
| 	rm.Lock() | ||||
| 	unlockOnExit := true | ||||
| 	defer func() { | ||||
| 		if unlockOnExit { | ||||
| 			rm.Unlock() | ||||
| 		} | ||||
| 	}() | ||||
| 
 | ||||
| 	if rm.canceled { | ||||
| 		return false, nil, ErrRestartCanceled | ||||
| 	} | ||||
| 
 | ||||
| 	if rm.active { | ||||
| 		return false, nil, fmt.Errorf("invalid call on an active restart manager") | ||||
| 	} | ||||
| 	// if the container ran for more than 10s, regardless of status and policy reset the | ||||
| 	// the timeout back to the default. | ||||
| 	if executionDuration.Seconds() >= 10 { | ||||
| 		rm.timeout = 0 | ||||
| 	} | ||||
| 	switch { | ||||
| 	case rm.timeout == 0: | ||||
| 		rm.timeout = defaultTimeout | ||||
| 	case rm.timeout < maxRestartTimeout: | ||||
| 		rm.timeout *= backoffMultiplier | ||||
| 	} | ||||
| 	if rm.timeout > maxRestartTimeout { | ||||
| 		rm.timeout = maxRestartTimeout | ||||
| 	} | ||||
| 
 | ||||
| 	var restart bool | ||||
| 	switch { | ||||
| 	case rm.policy.IsAlways(): | ||||
| 		restart = true | ||||
| 	case rm.policy.IsUnlessStopped() && !hasBeenManuallyStopped: | ||||
| 		restart = true | ||||
| 	case rm.policy.IsOnFailure(): | ||||
| 		// the default value of 0 for MaximumRetryCount means that we will not enforce a maximum count | ||||
| 		if max := rm.policy.MaximumRetryCount; max == 0 || rm.restartCount < max { | ||||
| 			restart = exitCode != 0 | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	if !restart { | ||||
| 		rm.active = false | ||||
| 		return false, nil, nil | ||||
| 	} | ||||
| 
 | ||||
| 	rm.restartCount++ | ||||
| 
 | ||||
| 	unlockOnExit = false | ||||
| 	rm.active = true | ||||
| 	rm.Unlock() | ||||
| 
 | ||||
| 	ch := make(chan error) | ||||
| 	go func() { | ||||
| 		select { | ||||
| 		case <-rm.cancel: | ||||
| 			ch <- ErrRestartCanceled | ||||
| 			close(ch) | ||||
| 		case <-time.After(rm.timeout): | ||||
| 			rm.Lock() | ||||
| 			close(ch) | ||||
| 			rm.active = false | ||||
| 			rm.Unlock() | ||||
| 		} | ||||
| 	}() | ||||
| 
 | ||||
| 	return true, ch, nil | ||||
| } | ||||
| 
 | ||||
| func (rm *restartManager) Cancel() error { | ||||
| 	rm.Do(func() { | ||||
| 		rm.Lock() | ||||
| 		rm.canceled = true | ||||
| 		close(rm.cancel) | ||||
| 		rm.Unlock() | ||||
| 	}) | ||||
| 	return nil | ||||
| } | ||||
							
								
								
									
										36
									
								
								vendor/github.com/docker/docker-ce/components/engine/restartmanager/restartmanager_test.go
									
										
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								vendor/github.com/docker/docker-ce/components/engine/restartmanager/restartmanager_test.go
									
										
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							|  | @ -0,0 +1,36 @@ | |||
| package restartmanager // import "github.com/docker/docker/restartmanager" | ||||
| 
 | ||||
| import ( | ||||
| 	"testing" | ||||
| 	"time" | ||||
| 
 | ||||
| 	"github.com/docker/docker/api/types/container" | ||||
| ) | ||||
| 
 | ||||
| func TestRestartManagerTimeout(t *testing.T) { | ||||
| 	rm := New(container.RestartPolicy{Name: "always"}, 0).(*restartManager) | ||||
| 	var duration = time.Duration(1 * time.Second) | ||||
| 	should, _, err := rm.ShouldRestart(0, false, duration) | ||||
| 	if err != nil { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
| 	if !should { | ||||
| 		t.Fatal("container should be restarted") | ||||
| 	} | ||||
| 	if rm.timeout != defaultTimeout { | ||||
| 		t.Fatalf("restart manager should have a timeout of 100 ms but has %s", rm.timeout) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func TestRestartManagerTimeoutReset(t *testing.T) { | ||||
| 	rm := New(container.RestartPolicy{Name: "always"}, 0).(*restartManager) | ||||
| 	rm.timeout = 5 * time.Second | ||||
| 	var duration = time.Duration(10 * time.Second) | ||||
| 	_, _, err := rm.ShouldRestart(0, false, duration) | ||||
| 	if err != nil { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
| 	if rm.timeout != defaultTimeout { | ||||
| 		t.Fatalf("restart manager should have a timeout of 100 ms but has %s", rm.timeout) | ||||
| 	} | ||||
| } | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue