2019-11-21 18:32:02 +00:00
// Copyright 2019 The Gitea Authors. All rights reserved.
2022-11-27 13:20:29 -05:00
// SPDX-License-Identifier: MIT
2019-11-21 18:32:02 +00:00
// This code is heavily inspired by the archived gofacebook/gracenet/net.go handler
2021-08-24 11:47:09 -05:00
//go:build windows
2019-11-21 18:32:02 +00:00
package graceful
import (
2019-11-30 08:40:22 -06:00
"context"
2019-11-21 18:32:02 +00:00
"os"
2022-03-25 12:47:12 +00:00
"runtime/pprof"
2019-11-21 18:32:02 +00:00
"strconv"
"sync"
"time"
"code.gitea.io/gitea/modules/log"
"code.gitea.io/gitea/modules/setting"
"golang.org/x/sys/windows/svc"
"golang.org/x/sys/windows/svc/debug"
)
2019-12-04 01:16:29 +01:00
// WindowsServiceName is the name of the Windows service
2019-11-21 18:32:02 +00:00
var WindowsServiceName = "gitea"
const (
hammerCode = 128
hammerCmd = svc . Cmd ( hammerCode )
acceptHammerCode = svc . Accepted ( hammerCode )
)
2019-12-15 09:51:28 +00:00
// Manager manages the graceful shutdown process
type Manager struct {
2019-11-30 08:40:22 -06:00
ctx context . Context
2019-11-21 18:32:02 +00:00
isChild bool
lock * sync . RWMutex
state state
2021-05-15 15:22:26 +01:00
shutdownCtx context . Context
hammerCtx context . Context
terminateCtx context . Context
2022-03-25 12:47:12 +00:00
managerCtx context . Context
2021-05-15 15:22:26 +01:00
shutdownCtxCancel context . CancelFunc
hammerCtxCancel context . CancelFunc
terminateCtxCancel context . CancelFunc
2022-03-25 12:47:12 +00:00
managerCtxCancel context . CancelFunc
2019-11-21 18:32:02 +00:00
runningServerWaitGroup sync . WaitGroup
createServerWaitGroup sync . WaitGroup
terminateWaitGroup sync . WaitGroup
2020-01-29 19:54:34 +01:00
shutdownRequested chan struct { }
2021-05-15 15:22:26 +01:00
toRunAtShutdown [ ] func ( )
toRunAtTerminate [ ] func ( )
2019-11-21 18:32:02 +00:00
}
2019-12-15 09:51:28 +00:00
func newGracefulManager ( ctx context . Context ) * Manager {
manager := & Manager {
2019-11-21 18:32:02 +00:00
isChild : false ,
lock : & sync . RWMutex { } ,
2019-11-30 08:40:22 -06:00
ctx : ctx ,
2019-11-21 18:32:02 +00:00
}
manager . createServerWaitGroup . Add ( numberOfServersToCreate )
2019-12-15 09:51:28 +00:00
manager . start ( )
2019-11-21 18:32:02 +00:00
return manager
}
2019-12-15 09:51:28 +00:00
func ( g * Manager ) start ( ) {
2021-05-15 15:22:26 +01:00
// Make contexts
g . terminateCtx , g . terminateCtxCancel = context . WithCancel ( g . ctx )
g . shutdownCtx , g . shutdownCtxCancel = context . WithCancel ( g . ctx )
g . hammerCtx , g . hammerCtxCancel = context . WithCancel ( g . ctx )
2022-03-25 12:47:12 +00:00
g . managerCtx , g . managerCtxCancel = context . WithCancel ( g . ctx )
// Next add pprof labels to these contexts
g . terminateCtx = pprof . WithLabels ( g . terminateCtx , pprof . Labels ( "graceful-lifecycle" , "with-terminate" ) )
g . shutdownCtx = pprof . WithLabels ( g . shutdownCtx , pprof . Labels ( "graceful-lifecycle" , "with-shutdown" ) )
g . hammerCtx = pprof . WithLabels ( g . hammerCtx , pprof . Labels ( "graceful-lifecycle" , "with-hammer" ) )
g . managerCtx = pprof . WithLabels ( g . managerCtx , pprof . Labels ( "graceful-lifecycle" , "with-manager" ) )
// Now label this and all goroutines created by this goroutine with the graceful-lifecycle manager
pprof . SetGoroutineLabels ( g . managerCtx )
defer pprof . SetGoroutineLabels ( g . ctx )
2021-05-15 15:22:26 +01:00
2019-12-15 09:51:28 +00:00
// Make channels
2020-01-29 01:01:06 +00:00
g . shutdownRequested = make ( chan struct { } )
2019-12-15 09:51:28 +00:00
// Set the running state
2019-11-21 18:32:02 +00:00
g . setState ( stateRunning )
if skip , _ := strconv . ParseBool ( os . Getenv ( "SKIP_MINWINSVC" ) ) ; skip {
2021-03-31 20:48:48 +01:00
log . Trace ( "Skipping SVC check as SKIP_MINWINSVC is set" )
2019-11-21 18:32:02 +00:00
return
}
2019-12-15 09:51:28 +00:00
// Make SVC process
2019-11-21 18:32:02 +00:00
run := svc . Run
2021-05-07 10:27:31 +01:00
//lint:ignore SA1019 We use IsAnInteractiveSession because IsWindowsService has a different permissions profile
2023-04-20 04:08:01 +02:00
isAnInteractiveSession , err := svc . IsAnInteractiveSession ( ) //nolint:staticcheck
2019-11-21 18:32:02 +00:00
if err != nil {
2021-03-31 20:48:48 +01:00
log . Error ( "Unable to ascertain if running as an Windows Service: %v" , err )
2019-11-21 18:32:02 +00:00
return
}
2021-05-07 10:27:31 +01:00
if isAnInteractiveSession {
2021-03-31 20:48:48 +01:00
log . Trace ( "Not running a service ... using the debug SVC manager" )
2019-11-21 18:32:02 +00:00
run = debug . Run
}
2021-01-06 09:38:00 +08:00
go func ( ) {
_ = run ( WindowsServiceName , g )
} ( )
2019-11-21 18:32:02 +00:00
}
2019-12-15 09:51:28 +00:00
// Execute makes Manager implement svc.Handler
func ( g * Manager ) Execute ( args [ ] string , changes <- chan svc . ChangeRequest , status chan <- svc . Status ) ( svcSpecificEC bool , exitCode uint32 ) {
2019-11-21 18:32:02 +00:00
if setting . StartupTimeout > 0 {
2019-11-22 22:00:01 +01:00
status <- svc . Status { State : svc . StartPending , WaitHint : uint32 ( setting . StartupTimeout / time . Millisecond ) }
2022-08-20 22:09:41 +01:00
} else {
status <- svc . Status { State : svc . StartPending }
2019-11-21 18:32:02 +00:00
}
2021-03-31 20:48:48 +01:00
log . Trace ( "Awaiting server start-up" )
2019-11-21 18:32:02 +00:00
// Now need to wait for everything to start...
if ! g . awaitServer ( setting . StartupTimeout ) {
2021-03-31 20:48:48 +01:00
log . Trace ( "... start-up failed ... Stopped" )
2019-11-21 18:32:02 +00:00
return false , 1
}
2021-03-31 20:48:48 +01:00
log . Trace ( "Sending Running state to SVC" )
2019-11-21 18:32:02 +00:00
// We need to implement some way of svc.AcceptParamChange/svc.ParamChange
status <- svc . Status {
State : svc . Running ,
Accepts : svc . AcceptStop | svc . AcceptShutdown | acceptHammerCode ,
}
2021-03-31 20:48:48 +01:00
log . Trace ( "Started" )
2019-11-21 18:32:02 +00:00
waitTime := 30 * time . Second
loop :
2019-11-30 08:40:22 -06:00
for {
select {
case <- g . ctx . Done ( ) :
2021-03-31 20:48:48 +01:00
log . Trace ( "Shutting down" )
2020-01-29 01:01:06 +00:00
g . DoGracefulShutdown ( )
waitTime += setting . GracefulHammerTime
break loop
case <- g . shutdownRequested :
2021-03-31 20:48:48 +01:00
log . Trace ( "Shutting down" )
2019-11-21 18:32:02 +00:00
waitTime += setting . GracefulHammerTime
break loop
2019-11-30 08:40:22 -06:00
case change := <- changes :
switch change . Cmd {
case svc . Interrogate :
2021-03-31 20:48:48 +01:00
log . Trace ( "SVC sent interrogate" )
2019-11-30 08:40:22 -06:00
status <- change . CurrentStatus
case svc . Stop , svc . Shutdown :
2021-03-31 20:48:48 +01:00
log . Trace ( "SVC requested shutdown - shutting down" )
2020-01-29 01:01:06 +00:00
g . DoGracefulShutdown ( )
2019-11-30 08:40:22 -06:00
waitTime += setting . GracefulHammerTime
break loop
case hammerCode :
2021-03-31 20:48:48 +01:00
log . Trace ( "SVC requested hammer - shutting down and hammering immediately" )
2020-01-29 01:01:06 +00:00
g . DoGracefulShutdown ( )
g . DoImmediateHammer ( )
2019-11-30 08:40:22 -06:00
break loop
default :
log . Debug ( "Unexpected control request: %v" , change . Cmd )
}
2019-11-21 18:32:02 +00:00
}
}
2021-03-31 20:48:48 +01:00
log . Trace ( "Sending StopPending state to SVC" )
2019-11-21 18:32:02 +00:00
status <- svc . Status {
2019-11-22 22:00:01 +01:00
State : svc . StopPending ,
WaitHint : uint32 ( waitTime / time . Millisecond ) ,
2019-11-21 18:32:02 +00:00
}
hammerLoop :
for {
select {
case change := <- changes :
switch change . Cmd {
case svc . Interrogate :
2021-03-31 20:48:48 +01:00
log . Trace ( "SVC sent interrogate" )
2019-11-21 18:32:02 +00:00
status <- change . CurrentStatus
case svc . Stop , svc . Shutdown , hammerCmd :
2021-03-31 20:48:48 +01:00
log . Trace ( "SVC requested hammer - hammering immediately" )
2020-01-29 01:01:06 +00:00
g . DoImmediateHammer ( )
2019-11-21 18:32:02 +00:00
break hammerLoop
default :
log . Debug ( "Unexpected control request: %v" , change . Cmd )
}
2021-05-15 15:22:26 +01:00
case <- g . hammerCtx . Done ( ) :
2019-11-21 18:32:02 +00:00
break hammerLoop
}
}
2021-03-31 20:48:48 +01:00
log . Trace ( "Stopped" )
2019-11-21 18:32:02 +00:00
return false , 0
}
2020-01-29 01:01:06 +00:00
// DoImmediateHammer causes an immediate hammer
func ( g * Manager ) DoImmediateHammer ( ) {
g . doHammerTime ( 0 * time . Second )
}
// DoGracefulShutdown causes a graceful shutdown
func ( g * Manager ) DoGracefulShutdown ( ) {
g . lock . Lock ( )
select {
case <- g . shutdownRequested :
g . lock . Unlock ( )
default :
close ( g . shutdownRequested )
g . lock . Unlock ( )
g . doShutdown ( )
}
}
2019-12-15 09:51:28 +00:00
// RegisterServer registers the running of a listening server.
// Any call to RegisterServer must be matched by a call to ServerDone
func ( g * Manager ) RegisterServer ( ) {
2019-11-21 18:32:02 +00:00
g . runningServerWaitGroup . Add ( 1 )
}
2019-12-15 09:51:28 +00:00
func ( g * Manager ) awaitServer ( limit time . Duration ) bool {
2019-11-21 18:32:02 +00:00
c := make ( chan struct { } )
go func ( ) {
defer close ( c )
g . createServerWaitGroup . Wait ( )
} ( )
if limit > 0 {
select {
case <- c :
return true // completed normally
case <- time . After ( limit ) :
return false // timed out
case <- g . IsShutdown ( ) :
return false
}
} else {
select {
case <- c :
return true // completed normally
case <- g . IsShutdown ( ) :
return false
}
}
}