mirror of https://github.com/docker/docker-py.git
257 lines
7.7 KiB
Python
257 lines
7.7 KiB
Python
import io
|
|
import os
|
|
import shutil
|
|
import tempfile
|
|
|
|
from docker import errors
|
|
|
|
import six
|
|
|
|
from .base import BaseAPIIntegrationTest
|
|
from ..helpers import requires_api_version
|
|
|
|
|
|
class BuildTest(BaseAPIIntegrationTest):
|
|
def test_build_streaming(self):
|
|
script = io.BytesIO('\n'.join([
|
|
'FROM busybox',
|
|
'RUN mkdir -p /tmp/test',
|
|
'EXPOSE 8080',
|
|
'ADD https://dl.dropboxusercontent.com/u/20637798/silence.tar.gz'
|
|
' /tmp/silence.tar.gz'
|
|
]).encode('ascii'))
|
|
stream = self.client.build(fileobj=script, stream=True, decode=True)
|
|
logs = []
|
|
for chunk in stream:
|
|
logs.append(chunk)
|
|
assert len(logs) > 0
|
|
|
|
def test_build_from_stringio(self):
|
|
if six.PY3:
|
|
return
|
|
script = io.StringIO(six.text_type('\n').join([
|
|
'FROM busybox',
|
|
'RUN mkdir -p /tmp/test',
|
|
'EXPOSE 8080',
|
|
'ADD https://dl.dropboxusercontent.com/u/20637798/silence.tar.gz'
|
|
' /tmp/silence.tar.gz'
|
|
]))
|
|
stream = self.client.build(fileobj=script, stream=True)
|
|
logs = ''
|
|
for chunk in stream:
|
|
if six.PY3:
|
|
chunk = chunk.decode('utf-8')
|
|
logs += chunk
|
|
self.assertNotEqual(logs, '')
|
|
|
|
@requires_api_version('1.8')
|
|
def test_build_with_dockerignore(self):
|
|
base_dir = tempfile.mkdtemp()
|
|
self.addCleanup(shutil.rmtree, base_dir)
|
|
|
|
with open(os.path.join(base_dir, 'Dockerfile'), 'w') as f:
|
|
f.write("\n".join([
|
|
'FROM busybox',
|
|
'ADD . /test',
|
|
]))
|
|
|
|
with open(os.path.join(base_dir, '.dockerignore'), 'w') as f:
|
|
f.write("\n".join([
|
|
'ignored',
|
|
'Dockerfile',
|
|
'.dockerignore',
|
|
'!ignored/subdir/excepted-file',
|
|
'', # empty line
|
|
]))
|
|
|
|
with open(os.path.join(base_dir, 'not-ignored'), 'w') as f:
|
|
f.write("this file should not be ignored")
|
|
|
|
subdir = os.path.join(base_dir, 'ignored', 'subdir')
|
|
os.makedirs(subdir)
|
|
with open(os.path.join(subdir, 'file'), 'w') as f:
|
|
f.write("this file should be ignored")
|
|
|
|
with open(os.path.join(subdir, 'excepted-file'), 'w') as f:
|
|
f.write("this file should not be ignored")
|
|
|
|
tag = 'docker-py-test-build-with-dockerignore'
|
|
stream = self.client.build(
|
|
path=base_dir,
|
|
tag=tag,
|
|
)
|
|
for chunk in stream:
|
|
pass
|
|
|
|
c = self.client.create_container(tag, ['find', '/test', '-type', 'f'])
|
|
self.client.start(c)
|
|
self.client.wait(c)
|
|
logs = self.client.logs(c)
|
|
|
|
if six.PY3:
|
|
logs = logs.decode('utf-8')
|
|
|
|
self.assertEqual(
|
|
sorted(list(filter(None, logs.split('\n')))),
|
|
sorted(['/test/ignored/subdir/excepted-file',
|
|
'/test/not-ignored']),
|
|
)
|
|
|
|
@requires_api_version('1.21')
|
|
def test_build_with_buildargs(self):
|
|
script = io.BytesIO('\n'.join([
|
|
'FROM scratch',
|
|
'ARG test',
|
|
'USER $test'
|
|
]).encode('ascii'))
|
|
|
|
stream = self.client.build(
|
|
fileobj=script, tag='buildargs', buildargs={'test': 'OK'}
|
|
)
|
|
self.tmp_imgs.append('buildargs')
|
|
for chunk in stream:
|
|
pass
|
|
|
|
info = self.client.inspect_image('buildargs')
|
|
self.assertEqual(info['Config']['User'], 'OK')
|
|
|
|
@requires_api_version('1.22')
|
|
def test_build_shmsize(self):
|
|
script = io.BytesIO('\n'.join([
|
|
'FROM scratch',
|
|
'CMD sh -c "echo \'Hello, World!\'"',
|
|
]).encode('ascii'))
|
|
|
|
tag = 'shmsize'
|
|
shmsize = 134217728
|
|
|
|
stream = self.client.build(
|
|
fileobj=script, tag=tag, shmsize=shmsize
|
|
)
|
|
self.tmp_imgs.append(tag)
|
|
for chunk in stream:
|
|
pass
|
|
|
|
# There is currently no way to get the shmsize
|
|
# that was used to build the image
|
|
|
|
@requires_api_version('1.23')
|
|
def test_build_labels(self):
|
|
script = io.BytesIO('\n'.join([
|
|
'FROM scratch',
|
|
]).encode('ascii'))
|
|
|
|
labels = {'test': 'OK'}
|
|
|
|
stream = self.client.build(
|
|
fileobj=script, tag='labels', labels=labels
|
|
)
|
|
self.tmp_imgs.append('labels')
|
|
for chunk in stream:
|
|
pass
|
|
|
|
info = self.client.inspect_image('labels')
|
|
self.assertEqual(info['Config']['Labels'], labels)
|
|
|
|
@requires_api_version('1.25')
|
|
def test_build_with_cache_from(self):
|
|
script = io.BytesIO('\n'.join([
|
|
'FROM busybox',
|
|
'ENV FOO=bar',
|
|
'RUN touch baz',
|
|
'RUN touch bax',
|
|
]).encode('ascii'))
|
|
|
|
stream = self.client.build(fileobj=script, tag='build1')
|
|
self.tmp_imgs.append('build1')
|
|
for chunk in stream:
|
|
pass
|
|
|
|
stream = self.client.build(
|
|
fileobj=script, tag='build2', cache_from=['build1'],
|
|
decode=True
|
|
)
|
|
self.tmp_imgs.append('build2')
|
|
counter = 0
|
|
for chunk in stream:
|
|
if 'Using cache' in chunk.get('stream', ''):
|
|
counter += 1
|
|
assert counter == 3
|
|
self.client.remove_image('build2')
|
|
|
|
counter = 0
|
|
stream = self.client.build(
|
|
fileobj=script, tag='build2', cache_from=['nosuchtag'],
|
|
decode=True
|
|
)
|
|
for chunk in stream:
|
|
if 'Using cache' in chunk.get('stream', ''):
|
|
counter += 1
|
|
assert counter == 0
|
|
|
|
@requires_api_version('1.29')
|
|
def test_build_container_with_target(self):
|
|
script = io.BytesIO('\n'.join([
|
|
'FROM busybox as first',
|
|
'RUN mkdir -p /tmp/test',
|
|
'RUN touch /tmp/silence.tar.gz',
|
|
'FROM alpine:latest',
|
|
'WORKDIR /root/'
|
|
'COPY --from=first /tmp/silence.tar.gz .',
|
|
'ONBUILD RUN echo "This should not be in the final image"'
|
|
]).encode('ascii'))
|
|
|
|
stream = self.client.build(
|
|
fileobj=script, target='first', tag='build1'
|
|
)
|
|
self.tmp_imgs.append('build1')
|
|
for chunk in stream:
|
|
pass
|
|
|
|
info = self.client.inspect_image('build1')
|
|
self.assertEqual(info['Config']['OnBuild'], [])
|
|
|
|
def test_build_stderr_data(self):
|
|
control_chars = ['\x1b[91m', '\x1b[0m']
|
|
snippet = 'Ancient Temple (Mystic Oriental Dream ~ Ancient Temple)'
|
|
script = io.BytesIO(b'\n'.join([
|
|
b'FROM busybox',
|
|
'RUN sh -c ">&2 echo \'{0}\'"'.format(snippet).encode('utf-8')
|
|
]))
|
|
|
|
stream = self.client.build(
|
|
fileobj=script, stream=True, decode=True, nocache=True
|
|
)
|
|
lines = []
|
|
for chunk in stream:
|
|
lines.append(chunk.get('stream'))
|
|
expected = '{0}{2}\n{1}'.format(
|
|
control_chars[0], control_chars[1], snippet
|
|
)
|
|
self.assertTrue(any([line == expected for line in lines]))
|
|
|
|
def test_build_gzip_encoding(self):
|
|
base_dir = tempfile.mkdtemp()
|
|
self.addCleanup(shutil.rmtree, base_dir)
|
|
|
|
with open(os.path.join(base_dir, 'Dockerfile'), 'w') as f:
|
|
f.write("\n".join([
|
|
'FROM busybox',
|
|
'ADD . /test',
|
|
]))
|
|
|
|
stream = self.client.build(
|
|
path=base_dir, stream=True, decode=True, nocache=True,
|
|
gzip=True
|
|
)
|
|
|
|
lines = []
|
|
for chunk in stream:
|
|
lines.append(chunk)
|
|
|
|
assert 'Successfully built' in lines[-1]['stream']
|
|
|
|
def test_build_gzip_custom_encoding(self):
|
|
with self.assertRaises(errors.DockerException):
|
|
self.client.build(path='.', gzip=True, encoding='text/html')
|