[go: nahoru, domu]

Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Unable to react to graceful shutdown of (Windows) container #25982

Open
godefroi opened this issue Aug 24, 2016 · 98 comments
Open

Unable to react to graceful shutdown of (Windows) container #25982

godefroi opened this issue Aug 24, 2016 · 98 comments

Comments

@godefroi
Copy link
Contributor

Output of docker version:

Client:
 Version:      1.12.0
 API version:  1.24
 Go version:   go1.6.3
 Git commit:   8eab29e
 Built:        Thu Jul 28 21:15:28 2016
 OS/Arch:      windows/amd64

Server:
 Version:      1.12.0
 API version:  1.24
 Go version:   go1.6.3
 Git commit:   8eab29e
 Built:        Thu Jul 28 21:15:28 2016
 OS/Arch:      windows/amd64

Output of docker info:

Containers: 1
 Running: 0
 Paused: 0
 Stopped: 1
Images: 86
Server Version: 1.12.0
Storage Driver: windowsfilter
 Windows:
Logging Driver: json-file
Plugins:
 Volume: local
 Network: nat null overlay
Swarm: inactive
Security Options:
Kernel Version: 10.0 14300 (14300.1045.amd64fre.rs1_release_svc.160705-1059)
Operating System: Windows Server 2016 Standard Technical Preview 5
OSType: windows
Architecture: x86_64
CPUs: 2
Total Memory: 8 GiB
Name: slc-dev-s16p-2
ID: H3YG:MO32:XSEU:NGD4:Z7FC:4SMS:EYPL:RGHO:DGWY:W767:O3L2:HNXZ
Docker Root Dir: C:\ProgramData\docker
Debug Mode (client): false
Debug Mode (server): false
Registry: https://index.docker.io/v1/
Insecure Registries:
 127.0.0.0/8

I am unable to react to a graceful shutdown of my application running inside a (Windows) container. I have tried SetConsoleCtrlHandler(), but my handler is never called. I have tried signal(), but no SIGTERM is received. I have tried running a message loop, but WM_CLOSE is never received.

The work of shutting down the container is (apparently) done by the ShutdownComputeSystem routine from vmcompute.dll (this is from zhcsshim.go), but I cannot find any documentation or other information on what ShutdownComputeSystem does. It has been suggested that @jhowardmsft would know what's going on.

Please help!

@godefroi
Copy link
Contributor Author

There is some discussion related to this here and here, for what it's worth.

@thaJeztah
Copy link
Member

ping @jhowardmsft ptal

@lowenna
Copy link
Member
lowenna commented Sep 27, 2016

The call into HCSShim will differ under the covers depending on whether this is a Windows Server Container, or a Hyper-V container, and depending on not whether the initiator of the shutdown is a forced shutdown or graceful shutdown (eg from docker stop -f container vs docker stop container). In a forced case, for either container type, no notification will be expected. It is possible that there is a kernel issue in the Windows Server Container case, but I'm told by the kernel folks that the way it works is that the cexecsvc calls an initiatesystemshutdown, and the processes in the job object/silo should be notified as per a regular system shutdown.

Can you confirm what container type this is? And if you see the same for both types. If you run the app outside of a container just on the host and run "shutdown /t 0 /r" (effectively an InitateSystemShutdown call), does your app get notified?

@lowenna
Copy link
Member
lowenna commented Sep 27, 2016

(Note also this was on TP5 - I would strongly recommend seeing if the behaviour is different on 14393/RTM)

@godefroi
Copy link
Contributor Author
godefroi commented Sep 27, 2016

@jhowardmsft This is a Windows Server container, and a graceful shutdown (docker stop). I installed today the evaluation version that was released yesterday (which I assume is 14393, but am not at my machine currently to verify), but I have not re-tested. I will in the morning, and update.

@godefroi
Copy link
Contributor Author

@jhowardmsft I gave it a shot this morning, and this is what I found. First, the versions I'm now working with:

docker info says this about my Windows version:

Kernel Version: 10.0 14393 (14393.206.amd64fre.rs1_release.160915-0644)
Operating System: Windows Server 2016 Standard Evaluation

docker version says this:

Client:
 Version:      1.12.2-cs2-ws-beta-rc1
 API version:  1.25
 Go version:   go1.7.1
 Git commit:   62d9ff9
 Built:        Fri Sep 23 20:50:29 2016
 OS/Arch:      windows/amd64

Server:
 Version:      1.12.2-cs2-ws-beta-rc1
 API version:  1.25
 Go version:   go1.7.1
 Git commit:   62d9ff9
 Built:        Fri Sep 23 20:50:29 2016
 OS/Arch:      windows/amd64

My test application is a C# application, and I'm using the following methods to detect system shutdown:

  • I'm hooking the Console.CancelKeyPress event.
  • I'm hooking the AppDomain.CurrentDomain.ProcessExit and AppDomain.CurrentDomain.DomainUnload events.
  • I'm hooking the SessionEnding, SessionEnded, SessionSwitch, EventsThreadShutdown, and PowerModeChanged events from the Microsoft.Win32.SystemEvents class.
  • I'm setting up a handler for the SIGTERM signal.
  • I'm setting a handler using SetConsoleCtrlHandler.
  • I'm running a form that overrides WndProc and watches for the WM_CLOSE message.

Of these methods, when I run the application and shut it down using shutdown /t 0 /r as you suggested, only the Microsoft.Win32.SystemEvents.SessionEnding and Microsoft.Win32.SystemEvents.SessionEnded events are raised (which surprised me; maybe I should trap WM_QUERYENDSESSION too). When I shut down a Windows Container using docker stop, none of the events are raised.

I am happy to provide the complete source to the app I'm using to test, if you'd like.

@lowenna
Copy link
Member
lowenna commented Sep 28, 2016

Getting the app would be useful, although I'd probably just be a go-between for the kernel team who would need to dig into what is going on.

@godefroi
Copy link
Contributor Author

Here's my test app: show_stop.txt

Hopefully it provides some helpful insight.

@lowenna
Copy link
Member
lowenna commented Oct 17, 2016

@darrenstahlmsft FYI

@sandersaares
Copy link

Do I understand it right that there is currently no way to get a notification that a Windows container is shutting down? How are containerized apps intended to avoid incomplete writes in such a situation? Or do I miss something here?

@godefroi
Copy link
Contributor Author
godefroi commented Nov 14, 2016

@sandersaares That is my understanding with current versions, yes.

@PatrickLang
Copy link

This is still in progress. Changes are needed to the Windows platform. MS#8633377. Will share more info as things change.

@marcosnils
Copy link

@PatrickLang is that MS#8633377 case public accessible?

@PatrickLang
Copy link

It's internal. I referenced it so my team could find it and you would know it's being worked on. It's still on backlog for a future release.

@rgl
Copy link
Contributor
rgl commented Sep 3, 2017

@jhowardmsft @PatrickLang any news about this?

I've tested the current behavior from plain C applications and posted the results at rgl/docker-windows-2016-vagrant, essentially:

Windows containers cannot be gracefully shutdown, either there is no shutdown notification or they are forcefully terminated after a while.

The next table describes whether a docker stop --time 30 <container> will graceful shutdown a container that is running a console, gui, or service app.

base image app behaviour
nanoserver console does not receive the shutdown notification
windowsservercore console receives the shutdown notification but is killed after about 5 seconds
nanoserver gui fails to run RegisterClass (there's no GUI support in nano)
windowsservercore gui receives the shutdown notification but is killed after about 5 seconds
nanoserver service does not receive the shutdown notification
windowsservercore service does not receive the shutdown notification

@sandersaares
Copy link

@PatrickLang will this feature be improved in RS3?

@riverar
Copy link
riverar commented Sep 11, 2017

@rgl I suspect the reason your services don't get notifications is because

  • WaitToKillServiceTimeout defaults to ~20 seconds
  • Docker kills the container way before then

Can you re-test with SERVICE_ACCEPT_PRESHUTDOWN? Should work and align it with the others (but still die after ~5 seconds).

I presume this is still an open issue due to docker not supporting container shutdown deferral (for good reasons). Maybe an orchestrator-side flag to opt into this potentially unwanted behavior is in order (e.g. --allow-shutdown-deferral)?

@godefroi
Copy link
Contributor Author

@riverar It's an open issue because Docker on Windows doesn't ask nicely to shut down processes, it simply kills them. This behavior does not match Docker on other platforms. This is (per @PatrickLang) due to issues in Windows.

@riverar
Copy link
riverar commented Sep 11, 2017

That seems to jive with my experience, but runs counter to @jhowardmsft's statement about changes in Windows to call InitateSystemShutdown. I'm running latest insider bits, so if it's not in here, those changes probably never made it into RS3.

Maybe John or @PatrickLang can update us.

Update: I'm guessing this is a larger issue of non-hyperv containers not having a winlogon.exe to do all the maid work during shutdown.

@godefroi
Copy link
Contributor Author

@riverar All my containers are non-hyperv; @rgl were you testing in HyperV containers? Your results do not match my results (and that could absolutely be my fault).

@darstahl
Copy link
Contributor
darstahl commented Sep 11, 2017

Sorry for the delay updating this thread (Thanks @riverar for pinging me via email). Here's the current status on this.

In RS3 (currently available in insider preview builds) there is a partial fix available. The partial fix appears to be what @rgl is testing against. Processes started by Docker are able to register for console notifications via Kernel32.SetConsoleCtrlHandler. Prior to shutting down the container (when initiated by Docker) all applications tracked by Docker (applications started via docker run, docker exec etc.) are sent a CTRL_CLOSE_EVENT and given 5 seconds to shut down, after this, InitiateSystemShutdown is called, and it is up to the kernel to handle it from there (which currently terminates applications). Note that this notification does not currently propagate to child processes, so docker run image cmd.exe /c MyApp.exe will notify cmd.exe, but not MyApp.exe.

This means that most console and GUI applications will receive the notification, as most application runtimes register for these notifications and send them via the runtime specific shutdown mechanisms.

Services will not currently get the exit notification without a kernel fix which did not make RS3. It is possible to work around this by using a shim application which acts as the container entrypoint and manages starting the service at container start, and stopping the service when the shim receives the CTRL_CLOSE_EVENT. I can write a proof of concept for this if someone would like a starting point to work from.

I'm open to feedback on the current approach for future releases (though I hope to get kernel support for this so it works like regular Windows). Let me know what shutdown features are necessary for your application that are not possible to do with the console notification and 5 second timeout prior to kill.

@darstahl
Copy link
Contributor
darstahl commented Sep 11, 2017

@godefroi This fix is available starting in RS3, so the testing is probably on an insider preview build. This works in both HyperV containers and Windows Server containers running both nanoserver and windowsservercore.

@rgl The above chart looks right for the current RS3 state, except that in my testing nanoserver console apps correctly receive the notification exactly the same as windowsservercore. I'll try to take a look at your examples and see if I can see what is happening.

@matt-richardson
Copy link
Contributor

@darrenstahlmsft - Whats an RS3 when its around?

@darstahl
Copy link
Contributor
darstahl commented Sep 11, 2017

@matt-richardson Sorry, I was using Microsoft internal language 👼

RS3 is the internal name for the Fall 2017 Windows Semi-Annual Channel release for Windows Server 2016, or the Windows 10 Fall Creators Update depending if you are on Server or Client SKU respectively. It is currently available for preview in the Windows Insider program for the public to download and test, and will be going public some time this fall (I don't think I can share a date yet). For more info on the Windows Server side, see this link.

We generally refer to it as RS3 for brevity 😄

@rgl
Copy link
Contributor
rgl commented Sep 12, 2017

oh d'oh... I didn't actually implemented the (PRE)SHUTDOWN when I originally tested this! I've updated the source, and the service now receives the PRESHUTDOWN (but not SHUTDOWN) notification. Here's the updated table (only the service rows have changed).

The next table describes whether a docker stop --time 600 <container> will graceful shutdown a container that is running a console, gui, or service app.

base image app behaviour
nanoserver console does not receive the shutdown notification
windowsservercore console receives the shutdown notification but is killed after about 5 seconds
nanoserver gui fails to run RegisterClass (there's no GUI support in nano)
windowsservercore gui receives the shutdown notification but is killed after about 5 seconds
nanoserver service only receives the pre shutdown notification but is killed after about 195 seconds
windowsservercore service only receives the pre shutdown notification but is killed after about 195 seconds

NG setting WaitToKillServiceTimeout (e.g. Set-ItemProperty -Path HKLM:\SYSTEM\CurrentControlSet\Control -Name WaitToKillServiceTimeout -Value '450000') does not have any effect on extending the kill service timeout.

do you guys known from where that (approximate) 195 seconds timeout comes from? anyways, I would only expect the container to be killed after the timeout specified at the docker --time argument expires.

I'm testing this on a Windows Server 2016 (10.0.14393.1532) VM and with the following docker base images.

microsoft/windowsservercore:
==> default: be84290c2315 5 weeks ago Install update 10.0.14393.1593 2.63GB
==> default: 9 months ago Apply image 10.0.14393.0 7.68GB

microsoft/nanoserver:
==> default: 28dad12ef0bc 5 weeks ago Install update 10.0.14393.1593 368MB
==> default: 9 months ago Apply image 10.0.14393.0 701MB

@rgl
Copy link
Contributor
rgl commented Sep 12, 2017

@godefroi I'm not using Hyper-V. Please, see my previous comment to known what I'm using.

@cphillips83
Copy link

@OnurGumus Glad to have that confirmed! I do remember running into that difference when testing this, but couldn't root cause why terminal emulation behaved differently in nanoserver vs servercore. But since removing -t worked across both, it seemed like the right general solution. Hopefully this unblocks you from being able to rely on the correct shutdown behavior in your usage.

@OnurGumus @swernli I just wanted to comment in case this helps others too. If you don't add an event handler to CancelKeyPress, RunConsoleAsync just hangs after calling the hosted service StopAsync

System.Console.CancelKeyPress += (e, s) => { };

Here is how I FINALLY achieved graceful shutdown on windows containers after 2 days of beating my head against the wall.

Using the following...

USER ContainerAdministrator
RUN reg add hklm\system\currentcontrolset\services\cexecsvc /v ProcessShutdownTimeoutSeconds /t REG_DWORD /d 7200  
RUN reg add hklm\system\currentcontrolset\control /v WaitToKillServiceTimeout /t REG_SZ /d 7200000 /f

Starting the container with

docker run -id

Stopping the container with

docker stop -t 7200

This was all in conjuction with using SetConsoleCtrlHandler

sunghwan2789 added a commit to sunghwan2789/minecraft-server that referenced this issue Mar 9, 2020
@weijuans-msft
Copy link

@cphillips83 @OnurGumus can you confirm if this is still an issue, or now you have a workaround? Also do people need it for both Nano Server container and Server Core container?

Sorry @cphillips83 to see you have to bead your head against the wall for 2 days. ouch ...

For Microsoft folks, I created a new bug 25695040. I couldn't find the on Patrick gave 8633377.

@fedorbirjukov
Copy link

@weijuans-msft We need it in all windows images. So, yes in both Nano Server and in Server Core.

@timpclip
Copy link

I am trying something very similar to others here but am having an issue getting docker stop to wait longer than 5 minutes.
I created a scaled-down test program that just does Thread.Sleep for 20 minutes. I deployed it in a docker container with base image: windows servercore:1909

I'm using the lines:
USER ContainerAdministrator
RUN reg add hklm\system\currentcontrolset\services\cexecsvc /v ProcessShutdownTimeoutSeconds /t REG_DWORD /d 2700 /f
RUN reg add hklm\system\currentcontrolset\control /v WaitToKillServiceTimeout /t REG_SZ /d 2700000 /f

(This is set to 45 minutes which is what I actually want to use when I get this working)

I start it using: docker run -id
I stop it using: docker stop -t 400

I expect that it should wait 400 second (over 6 minutes) before killing the container.

When I run it on my desktop which is Windows 10 Professional version 2004, it works as expected.
But when I try it on hosts running Windows Server Datacenter version 1909, it stops after exactly 5 minutes.

I upgraded both Docker engines to version 19.03.12 but observe the same behavior as before.

Should this work as I am expecting? If so, any ideas what could be wrong?
Any help appreciated.

@zleight1
Copy link

Hello,

I am running the following powershell script in a Windows Container:

try
  {
    Write-Host "3. Configuring Azure Pipelines agent..." -ForegroundColor Cyan
  
    .\config.cmd --unattended `
      --agent "$(if (Test-Path Env:AZP_AGENT_NAME) { ${Env:AZP_AGENT_NAME} } else { ${Env:computername} })" `
      --url "$(${Env:AZP_URL})" `
      --auth PAT `
      --token "$(Get-Content ${Env:AZP_TOKEN_FILE})" `
      --pool "$(if (Test-Path Env:AZP_POOL) { ${Env:AZP_POOL} } else { 'Default' })" `
      --work "$(if (Test-Path Env:AZP_WORK) { ${Env:AZP_WORK} } else { '_work' })" `
      --replace
  
    Write-Host "4. Running Azure Pipelines agent..." -ForegroundColor Cyan
  
    .\run.cmd
  } 
  finally
  {
    Write-Host "Cleanup. Removing Azure Pipelines agent..." -ForegroundColor Cyan
  
    .\config.cmd remove --unattended `
      --auth PAT `
      --token "$(Get-Content ${Env:AZP_TOKEN_FILE})"
  }

My container is using the windowsservercore-1903 image.

Unfortunately when I stop the container using docker stop, with or without -t 60 , the finally block is never triggered.
Can someone help me with this?

@olandese Did you ever manage to get this to work?

@artlogic
Copy link

Just a note here for anyone who find this issue and was as confused as I was about all this:

  1. For Windows containers, the -t parameter of docker stop seems to do nothing. There seems to be no way to specify that value from outside the container. The command issues a CTRL_SHUTDOWN_EVENT immediately regardless of any --time set.
  2. The first registry value change, RUN reg add hklm\system\currentcontrolset\services\cexecsvc /v ProcessShutdownTimeoutSeconds /t REG_DWORD /d 7200, causes Windows to delay sending the shutdown event for the specified number of seconds. I'm not exactly sure how useful this is - unless you've used some other channel to tell your process to shut down.
  3. The second registry value change is more useful: RUN reg add hklm\system\currentcontrolset\control /v WaitToKillServiceTimeout /t REG_SZ /d 7200000 /f. This causes Windows to wait the specified number of milliseconds before killing a process after issuing a CTRL_SHUTDOWN_EVENT - essentially the rough equivalent of docker stop -t.
  4. If you don't delay while handling the CTRL_SHUTDOWN_EVENT your process will stop immediately, regardless of any delay set (much like how handling SIGTERM would work).

I hope this helps someone!

@jimmycartrette
Copy link

Are we sure that this setting ProcessShutdownTimeoutSeconds is actually doing anything? The first appearance of that key on the internet seems to be on a blog in 3 September 2019, but I can't find any information about where that came from. It may no be doing anything, it doesn't seem to have the functionality described by @artlogic . And it doesn't make sense that is it working, otherwise you'd just see the delay before your program gets the signal and then the "real" delay from the WaitToKillServiceTimeout starts.

The reason I'm finding this out is I'm using a different mechanism for signalling graceful shutdown (a file appears) but it doesn't seem to work because the CTRL_SHUTDOWN_EVENT is sent immediately and not delayed by ProcessShutdownTimeoutSeconds.

It seems SetConsoleCtrlHandler must be used to handle the CTRL_SHUTDOWN_EVENT and WaitToKillServiceTimeout provides time between that and service kill, and ProcessShutdownTimeoutSeconds isn't used.

@artlogic
Copy link
artlogic commented Nov 8, 2021

Are we sure that this setting ProcessShutdownTimeoutSeconds is actually doing anything? The first appearance of that key on the internet seems to be on a blog in 3 September 2019, but I can't find any information about where that came from. It may no be doing anything, it doesn't seem to have the functionality described by @artlogic . And it doesn't make sense that is it working, otherwise you'd just see the delay before your program gets the signal and then the "real" delay from the WaitToKillServiceTimeout starts.

I didn't test ProcessShutdownTimeoutSeconds extensively because it wasn't useful in my case. I should also mention that the testing I did was using the mcr.microsoft.com/dotnet/framework/sdk:4.8-20210525-windowsservercore-ltsc2019 image and a Python based service. It's entirely possible other images and runtimes could react differently.

@marcanpilami
Copy link

I can confirm that on host 21H2, with base image nanoserver 1809 and later, the --time in docker stop --time 123 containerid does not do anything. The ProcessShutdownTimeoutSeconds registry key is always respected for a console app regardless of --time.

Also, it seems that the CTRL_SHUTDOWN_EVENT is not propagated at all from a cmd batch script to the launched process, like it is in an interactive console. Since most containers have some kind of launcher script, this makes this functionality hard to use (we could as in a few example above actually do a manual register for the event, but we should not have to do this, and this requires access to the code being packaged in the image).

@david-garcia-garcia
Copy link

from @darstahl comments:

The second fix for this was in Windows 10 Build 17074, which changes two things. First, the notification to is changed to CTRL_SHUTDOWN_EVENT for both windowsservercore and nanoserver based images. Second, it extends the notification in windowsservercore based images to affect all process running in the container, including sending service shutdown notifications to services running in the container. This is currently available in insider builds, and will be available in Windows Version 1803, in the 1803 based container images.

I understand that this means the entrypoint has no control whatsoever in the shutdown sequence.

If this is right, this design decision does not seem to allow for orquestration of shutdown.

Let's say I have an entrypoint in a container running a MSSQL Server service, and I want to interact with MSSQL Server in the entrypoint shutdown. If they are both signaled at the same time, there is no way I can coordinate what happens inside the container from within the entrypoint shutdown logic.

Would it not make sense to first wait for the entrypoint to fully exist/finish, and then signal all processes and services to shutdown?

@david-garcia-garcia
Copy link
david-garcia-garcia commented Feb 20, 2024

@zleight1 I was also getting totally blocked trying to achieve this through powershell. Guess what. You cannot do it in a straighforward way.

# Define C# code with a method to handle console control events
$code = @"
using System;
using System.Runtime.InteropServices;
using System.Threading;

public class ConsoleCtrlHandler {
    [DllImport("Kernel32")]
    public static extern bool SetConsoleCtrlHandler(HandlerRoutine Handler, bool Add);

    public delegate bool HandlerRoutine(CtrlTypes CtrlType);

    public enum CtrlTypes {
        CTRL_C_EVENT = 0,
        CTRL_BREAK_EVENT,
        CTRL_CLOSE_EVENT,
        CTRL_LOGOFF_EVENT = 5,
        CTRL_SHUTDOWN_EVENT
    }

    private static bool _shutdownRequested = false;
    private static bool _shutdownAllowed = false;

    public static void SetShutdownAllowed(bool allowed) {
        _shutdownAllowed = allowed;
    }

    public static bool GetShutdownRequested() {
        return _shutdownRequested;
    }

    public static bool ConsoleCtrlCheck(CtrlTypes ctrlType) {
        switch (ctrlType) {
            case CtrlTypes.CTRL_CLOSE_EVENT:
            case CtrlTypes.CTRL_SHUTDOWN_EVENT:
                _shutdownRequested = true;
                System.Diagnostics.Stopwatch stopwatch = System.Diagnostics.Stopwatch.StartNew();
                // Wait until the PowerShell script sets _shutdownAllowed to true
                while (!_shutdownAllowed && stopwatch.Elapsed.TotalSeconds < 120) {
                    Thread.Sleep(1000); // Check every second
                }
                return true; // Indicate that the event has been handled
            default:
                return false;
        } 
    }
}
"@

# Add the C# type to the current PowerShell session
Add-Type -TypeDefinition $code -ReferencedAssemblies "System.Runtime.InteropServices"

# Create a delegate for the handler method
$handler = [ConsoleCtrlHandler+HandlerRoutine]::CreateDelegate([ConsoleCtrlHandler+HandlerRoutine], [ConsoleCtrlHandler], "ConsoleCtrlCheck");

# Register the handler
[ConsoleCtrlHandler]::SetConsoleCtrlHandler($handler, $true);

# Your script logic here
Write-Host "Waiting for console control event..."
while (-not [ConsoleCtrlHandler]::GetShutdownRequested()) {
    Start-Sleep -Seconds 1;
}

# Simulate a task that needs to be completed before shutdown
Write-Host "Shutdown requested, completing tasks..."
Start-Sleep -Seconds 3 # Replace with actual task
Write-Host "Tasks completed."

# Allow the shutdown to proceed
[ConsoleCtrlHandler]::SetShutdownAllowed($true)

Full working example here:

https://github.com/david-garcia-garcia/dockernosigterm

If you are preparing containers for kubernetes, i recommend using LifeCycleHooks for this:

https://kubernetes.io/docs/concepts/containers/container-lifecycle-hooks/

I am working on a kubernetes oriented container that must work on docker so that devs can use it locally. Production is K8S so this was already solved through LifeCycleHooks there.

Just a word of WARNING about what i posted here and the other solutions in this thread, they key to chaos is this:

The second fix for this was in Windows 10 Build 17074, which changes two things. First, the notification to is changed to CTRL_SHUTDOWN_EVENT for both windowsservercore and nanoserver based images. Second, it extends the notification in windowsservercore based images to affect all process running in the container, including sending service shutdown notifications to services running in the container. This is currently available in insider builds, and will be available in Windows Version 1803, in the 1803 based container images.

This effectively makes having any control on what is happening inside the container imposible. You can hold-up shutdown in your entrypoint, but the whole system is actually shutting down so you will face erratic behaviour as services start to shutdown without your control.

I have setup a docker image that wraps up all of this container lifecycle management for windows in an image: https://github.com/david-garcia-garcia/windowscontainers

Note that the script provided here was a POC iteration, and several considerations need to be made for this to be reliable and robust, see the linked repo for the final version and a better description on what approach was used to handle the complete container lifecycle in both Docker and K8S.

@ViliusS
Copy link
ViliusS commented Apr 5, 2024

@david-garcia-garcia your solution works, at least at some level, but did you find how to make Windows containers return 0 exit code instead of 0xC000013A ?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

No branches or pull requests