Неудачный сокет python (с использованием OPC UA) подключается к фиктивному серверу из Docker

Я почти уверен, что это проблема стандартного сетевого сокета. В настоящее время я докерирую свое приложение, в котором используется пакет python py-opcua (OPC UA). У вас есть идеи, как решить эту проблему, или какие-либо предложения, которые я мог бы попробовать?

Я установил фиктивный сервер с узлами, который работает локально на моем компьютере Mac (здесь нет Dockerization!):

        self.server = Server()
        self.server.set_endpoint("opc.tcp://0.0.0.0:50001/")

        # setup our own namespace, not really necessary but should as spec
        uri = "http://some-uri.com"
        self.idx = self.server.register_namespace(uri)

        # get Objects node, this is where we should put our custom stuff
        self.objects = self.server.get_objects_node()
        self.root = self.objects.add_object(self.idx, "Child")

        self._populate_server_with_variables()
        self.server.iserver.history_manager.set_storage(HistorySQLite("mock.db"))

        # Sample new values for all variables...
        self.start()

        self.make_all_variables_writable()

Теперь я пытаюсь подключиться к указанному выше серверу, используя класс py-opcua Client, который подключается следующим образом (эта часть запускается из контейнера докеров! Снова с моего локального компьютера):

        self.client = Client(
            "opc.tcp://0.0.0.0:50001/",
            timeout=60.
        )

        self.client.connect()  # Where my application fails from within Docker!

        self.root = self.client.get_root_node()

Когда я запускаю контейнер докеров, клиент не может подключиться к фиктивному серверу. В частности, когда соединение установлено (пытается быть) установлено, контейнер докеров аварийно завершает работу.

Я попытался запустить следующие docker run команды, которые все вызывают одну и ту же ошибку:

docker run --env-file env/.local-docker.env -i --rm -p 50001:50001 image_name
docker run --env-file env/.local-docker.env -i --rm --network host image_name
docker run --env-file env/.local-docker.env -i --rm --network=host image_name

Ошибка, которую я получаю, всегда является неудачным соединением сокета со стороны клиента. В частности, я получаю следующее сообщение об ошибке:

Logfile will be save in:  ./logs/log_2020-03-01 18:54:36.769166.log
Connecting to... "opc.tcp://0.0.0.0:50001/"
  File "/usr/local/lib/python3.6/runpy.py", line 193, in _run_module_as_main
    "__main__", mod_spec)
  File "/usr/local/lib/python3.6/runpy.py", line 85, in _run_code
    exec(code, run_globals)
  File "/app/deploy/middleware/loop.py", line 96, in <module>
    opc_reader_rt = RealTimeReader(opc_path)
  File "/app/deploy/opc_client/dataset.py", line 40, in __init__
    self.client.connect()
  File "/usr/local/lib/python3.6/site-packages/opcua/client/client.py", line 256, in connect
    self.connect_socket()
  File "/usr/local/lib/python3.6/site-packages/opcua/client/client.py", line 281, in connect_socket
    self.uaclient.connect_socket(self.server_url.hostname, self.server_url.port)
  File "/usr/local/lib/python3.6/site-packages/opcua/client/ua_client.py", line 256, in connect_socket
    return self._uasocket.connect_socket(host, port)
  File "/usr/local/lib/python3.6/site-packages/opcua/client/ua_client.py", line 155, in connect_socket
    sock = socket.create_connection((host, port), timeout=self.timeout)
  File "/usr/local/lib/python3.6/socket.py", line 704, in create_connection
    for res in getaddrinfo(host, port, 0, SOCK_STREAM):
  File "/usr/local/lib/python3.6/socket.py", line 745, in getaddrinfo
    for res in _socket.getaddrinfo(host, port, family, type, proto, flags):
socket.gaierror: [Errno -2] Name or service not known

person DaveTheAl    schedule 01.03.2020    source источник
comment
Docker на Mac использует виртуальную машину, поэтому вы не можете использовать традиционную сеть хоста - вам нужно использовать сеть, о которой обе среды знают и к которой могут подключаться. Самый простой способ временного тестирования - использовать локальный IP-адрес Mac для подключения из контейнера, например, 10.x.x.x (используйте ifconfig, чтобы найти фактическое значение). Другой вариант - создать мостовую сеть через сеть докеров, которую оба будут совместно использовать, если у вас есть динамический локальный IP.   -  person ldg    schedule 02.03.2020
comment
@ldg Итак, я попытался настроить общую сеть, составить докер и подключиться к указанному mac-IP (10.x.x.x), который я нашел с помощью ifconfig. Ни один из этих подходов не работает, все они приводят к одной и той же ошибке. Может быть, это из-за неправильной установки некоторых настроек docker-desktop?   -  person DaveTheAl    schedule 02.03.2020
comment
Я также создаю оболочку python, чтобы проверить, работает ли разрешение DNS. Оно делает. Также работает локально, без докера. Проверяю прямо сейчас, могу ли я использовать библиотеку без urllib для подключения к фиктивному серверу   -  person DaveTheAl    schedule 02.03.2020


Ответы (1)


Оказывается, переменная среды неправильно читала двойные кавычки а-ля https://github.com/docker/compose/issues/3702

person DaveTheAl    schedule 02.03.2020