리눅스 - "Docker Desktop for Windows" Container 환경에서 IPv6 Loopback Address 바인딩 오류
예를 들어, 다음과 같이 예제 코드를 작성한 다음,
using System.Net;
using System.Net.Sockets;
namespace ConsoleApp1;
internal class Program
{
static void Main(string[] args)
{
Socket socket = new Socket(AddressFamily.InterNetworkV6, SocketType.Stream, ProtocolType.Tcp);
IPEndPoint ep = new IPEndPoint(IPAddress.IPv6Loopback, 15000);
socket.Bind(ep);
Thread.Sleep(5000);
socket.Close();
}
}
윈도우 또는 리눅스 데스크톱에서 실행하면 잘 동작할 것입니다. 하지만, Visual Studio에서
"Add" / "Docker Support..." 메뉴를 선택해 Docker 환경에서 실행하도록 바꾸면 이제 Bind에서는 다음과 같은 오류가 발생합니다.
System.Net.Sockets.SocketException
HResult=0x80004005
Message=Cannot assign requested address
Source=System.Net.Sockets
StackTrace:
at System.Net.Sockets.Socket.DoBind(EndPoint endPointSnapshot, SocketAddress socketAddress)
at System.Net.Sockets.Socket.Bind(EndPoint localEP)
at ConsoleApp1.Program.Main(String[] args)
말 그대로 IPv6의 경우 Localhost Loopback 주소에 대해서는 컨테이너 환경에서 바인딩할 수가 없는 것입니다. 이에 대해서는 Any 주소를 직접 바인딩하는 식으로 우회해야 합니다.
// 아래의 코드는 Container 환경에서도 동작
IPEndPoint ep = new IPEndPoint(IPAddress.IPv6Any, 15000);
socket.Bind(ep); // IPv6Any == "::"
그런데, 저게 별 의미가 없습니다.
왜냐하면, 애당초 컨테이너 환경을 구성하는 네트워크 자체가 (IPv6가 아닌) IPv4 환경이기 때문입니다. (잘 동작합니다. ^^)
문서에 따르면,
Enable IPv6 support
; https://docs.docker.com/config/daemon/ipv6/
IPv6를 사용하기 위해 ("Docker Desktop for Windows"의 경우 %USERPROFILE%\.docker 디렉터리에 위치한) daemon.json 파일에 다음의 내용을 추가하면 된다고 합니다.
{
"builder": { "gc": { "defaultKeepStorage": "20GB", "enabled": true } },
"experimental": true,
"ip6tables": true,
"features": { "buildkit": true }
}
하지만, 제가 해보니까 저건 그때 당시에만 "experimental"일 뿐이었고, 현재는 통합이 된 듯합니다. 그래서 그냥 ipv6 네트워크를 곧장 생성할 수 있습니다.
c:\temp> docker network create --ipv6 --subnet 2001:0DB8::/112 ip6net
c:\temp> docker network ls
NETWORK ID NAME DRIVER SCOPE
1745376123f7 bridge bridge local
802205405723 host host local
10c5265fee2d ip6net bridge local
4ad2d57a8bca none null local
그다음, 컨테이너를 위에서 생성한 네트워크를 사용하도록 설정한 다음,
c:\temp> docker run --rm --network ip6net -p 15000:15000 --name ConsoleApp1 -it 2dee319b7378 /bin/bash
다시 AddressFamily.InterNetworkV6 + IPAddress.IPv6Loopback 바인딩을 시도하면 정상적으로 동작하는 것을 확인할 수 있습니다.
혹시나, 실수로라도 아래와 같이 바인딩을 시도하면,
Socket socket = new Socket(AddressFamily.InterNetworkV6,
SocketType.Stream, ProtocolType.Tcp);
IPEndPoint ep = new IPEndPoint(IPAddress.Loopback, 15000);
socket.Bind(ep);
이런 오류를 보게 될 것입니다. ^^
Unhandled exception. System.Net.Sockets.SocketException (10014): The system detected an invalid pointer address in attempting to use a pointer argument in a call.
at System.Net.Sockets.Socket.UpdateStatusAfterSocketErrorAndThrowException(SocketError error, Boolean disconnectOnFailure, String callerName)
at System.Net.Sockets.Socket.DoBind(EndPoint endPointSnapshot, SocketAddress socketAddress)
at System.Net.Sockets.Socket.Bind(EndPoint localEP)
at ConsoleApp1.Program.Main(String[] args)
[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]