Docker CLI разрешает тег, но Docker Python API вызывает APIError - PullRequest
0 голосов
/ 20 января 2019

Я пытаюсь отправить локальный образ Docker в ECR с помощью Docker Python API.В рамках этого процесса мне нужно пометить изображение определенным образом.Когда я делаю это на CLI, он работает:

docker tag foo/bar '{user_id}.dkr.ecr.us-east-1.amazonaws.com/foo/bar'

Однако, когда я пытаюсь сделать то же самое, используя функцию docker.images.Image.tag в Docker Python SDK, происходит сбой:

import docker
(docker.client.from_env().images.get('foo/bar')
 .tag('foo/bar', 
      '{user-id}.dkr.ecr.us-east-1.amazonaws.com/foo/bar'
     )
)

(замените user_id в приведенных выше примерах кода на значение идентификатора пользователя AWS, например, 717171717171; для целей этого вопроса я запутал его здесь)

следующая ошибка:

In [10]: docker.client.from_env().images.get('foo/bar').ta
    ...: g('foo/bar', '{user_id}.dkr.ecr.us-east-1.amaz
    ...: onaws.com/foo/bar')                              
---------------------------------------------------------------------------
HTTPError                                 Traceback (most recent call last)
~/miniconda3/envs/alekseylearn-dev/lib/python3.6/site-packages/docker/api/client.py in _raise_for_status(self, response)
    255         try:
--> 256             response.raise_for_status()
    257         except requests.exceptions.HTTPError as e:

~/miniconda3/envs/alekseylearn-dev/lib/python3.6/site-packages/requests/models.py in raise_for_status(self)
    939         if http_error_msg:
--> 940             raise HTTPError(http_error_msg, response=self)
    941 

HTTPError: 500 Server Error: Internal Server Error for url: http+docker://localhost/v1.35/images/sha256:afe07035bce72b6c496878a7e3960bedffd46c1bedc79f1bd2b89619e8457194/tag?tag={user_id}.dkr.ecr.us-east-1.amazonaws.com%2Ffoo%2Fbar&repo=foo%2Fbar&force=0

During handling of the above exception, another exception occurred:

APIError                                  Traceback (most recent call last)
<ipython-input-10-5bb015d17409> in <module>
----> 1 docker.client.from_env().images.get('alekseylearn-example/build').tag('foo/bar', '{user_id}.dkr.ecr.us-east-1.amazonaws.com/foo/bar')

~/miniconda3/envs/alekseylearn-dev/lib/python3.6/site-packages/docker/models/images.py in tag(self, repository, tag, **kwargs)
    120             (bool): ``True`` if successful
    121         """
--> 122         return self.client.api.tag(self.id, repository, tag=tag, **kwargs)
    123 
    124 

~/miniconda3/envs/alekseylearn-dev/lib/python3.6/site-packages/docker/utils/decorators.py in wrapped(self, resource_id, *args, **kwargs)
     17                     'Resource ID was not provided'
     18                 )
---> 19             return f(self, resource_id, *args, **kwargs)
     20         return wrapped
     21     return decorator

~/miniconda3/envs/alekseylearn-dev/lib/python3.6/site-packages/docker/api/image.py in tag(self, image, repository, tag, force)
    531         url = self._url("/images/{0}/tag", image)
    532         res = self._post(url, params=params)
--> 533         self._raise_for_status(res)
    534         return res.status_code == 201
    535 

~/miniconda3/envs/alekseylearn-dev/lib/python3.6/site-packages/docker/api/client.py in _raise_for_status(self, response)
    256             response.raise_for_status()
    257         except requests.exceptions.HTTPError as e:
--> 258             raise create_api_error_from_http_exception(e)
    259 
    260     def _result(self, response, json=False, binary=False):

~/miniconda3/envs/alekseylearn-dev/lib/python3.6/site-packages/docker/errors.py in create_api_error_from_http_exception(e)
     29         else:
     30             cls = NotFound
---> 31     raise cls(e, response=response, explanation=explanation)
     32 
     33 

APIError: 500 Server Error: Internal Server Error ("invalid tag format")

Почему команда CLI завершается успешно, а команда Python API не выполняется?

1 Ответ

0 голосов
/ 21 января 2019

В подробном жаргоне Docker API имя изображения, например 123456789012.dkr.ecr.us-east-1.amazon.aws.com/foo/bar:baz, разделяется на хранилище (перед двоеточием) и тег (после двоеточия).Часть имени хоста имени хранилища - это Registry .Значение тега по умолчанию, если ничего не указано, является литералом latest.

. В вашем случае у вас уже есть объект Image, поэтому вам нужно применить две "половины" второго аргумента:

docker.client.from_env().images.get('foo/bar')
 .tag('{user-id}.dkr.ecr.us-east-1.amazonaws.com/foo/bar',
      'latest'
     )

(Во многих практических случаях использование тега latest не является хорошей идеей; что-то вроде отметки времени или идентификатора коммита контроля источника лучше идентифицирует изображение и помогает указывать таким службам, как ECS или EKS или обычныйОбращаю ваше внимание на то, что им нужно сделать обновление.Кроме того, хотя идентификаторы изображений ECR непрактично длинные, в контексте сценариев ничто не мешает вам использовать их напрямую, например, вы можете docker build -t 12345...amazonaws.com/foo/bar:abcdef0 и пропустить промежуточный docker tagшаг, если хотите.)

...