volume.py 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  1. from __future__ import absolute_import
  2. from __future__ import unicode_literals
  3. import logging
  4. from docker.errors import NotFound
  5. from docker.utils import version_lt
  6. from .config import ConfigurationError
  7. from .const import LABEL_PROJECT
  8. from .const import LABEL_VOLUME
  9. log = logging.getLogger(__name__)
  10. class Volume(object):
  11. def __init__(self, client, project, name, driver=None, driver_opts=None,
  12. external=False, labels=None, custom_name=False):
  13. self.client = client
  14. self.project = project
  15. self.name = name
  16. self.driver = driver
  17. self.driver_opts = driver_opts
  18. self.external = external
  19. self.labels = labels
  20. self.custom_name = custom_name
  21. def create(self):
  22. return self.client.create_volume(
  23. self.full_name, self.driver, self.driver_opts, labels=self._labels
  24. )
  25. def remove(self):
  26. if self.external:
  27. log.info("Volume %s is external, skipping", self.full_name)
  28. return
  29. log.info("Removing volume %s", self.full_name)
  30. return self.client.remove_volume(self.full_name)
  31. def inspect(self):
  32. return self.client.inspect_volume(self.full_name)
  33. def exists(self):
  34. try:
  35. self.inspect()
  36. except NotFound:
  37. return False
  38. return True
  39. @property
  40. def full_name(self):
  41. if self.custom_name:
  42. return self.name
  43. return '{0}_{1}'.format(self.project, self.name)
  44. @property
  45. def _labels(self):
  46. if version_lt(self.client._version, '1.23'):
  47. return None
  48. labels = self.labels.copy() if self.labels else {}
  49. labels.update({
  50. LABEL_PROJECT: self.project,
  51. LABEL_VOLUME: self.name,
  52. })
  53. return labels
  54. class ProjectVolumes(object):
  55. def __init__(self, volumes):
  56. self.volumes = volumes
  57. @classmethod
  58. def from_config(cls, name, config_data, client):
  59. config_volumes = config_data.volumes or {}
  60. volumes = {
  61. vol_name: Volume(
  62. client=client,
  63. project=name,
  64. name=data.get('name', vol_name),
  65. driver=data.get('driver'),
  66. driver_opts=data.get('driver_opts'),
  67. custom_name=data.get('name') is not None,
  68. labels=data.get('labels'),
  69. external=bool(data.get('external', False))
  70. )
  71. for vol_name, data in config_volumes.items()
  72. }
  73. return cls(volumes)
  74. def remove(self):
  75. for volume in self.volumes.values():
  76. try:
  77. volume.remove()
  78. except NotFound:
  79. log.warn("Volume %s not found.", volume.full_name)
  80. def initialize(self):
  81. try:
  82. for volume in self.volumes.values():
  83. volume_exists = volume.exists()
  84. if volume.external:
  85. log.debug(
  86. 'Volume {0} declared as external. No new '
  87. 'volume will be created.'.format(volume.name)
  88. )
  89. if not volume_exists:
  90. raise ConfigurationError(
  91. 'Volume {name} declared as external, but could'
  92. ' not be found. Please create the volume manually'
  93. ' using `{command}{name}` and try again.'.format(
  94. name=volume.full_name,
  95. command='docker volume create --name='
  96. )
  97. )
  98. continue
  99. if not volume_exists:
  100. log.info(
  101. 'Creating volume "{0}" with {1} driver'.format(
  102. volume.full_name, volume.driver or 'default'
  103. )
  104. )
  105. volume.create()
  106. else:
  107. driver = volume.inspect()['Driver']
  108. if volume.driver is not None and driver != volume.driver:
  109. raise ConfigurationError(
  110. 'Configuration for volume {0} specifies driver '
  111. '{1}, but a volume with the same name uses a '
  112. 'different driver ({3}). If you wish to use the '
  113. 'new configuration, please remove the existing '
  114. 'volume "{2}" first:\n'
  115. '$ docker volume rm {2}'.format(
  116. volume.name, volume.driver, volume.full_name,
  117. volume.inspect()['Driver']
  118. )
  119. )
  120. except NotFound:
  121. raise ConfigurationError(
  122. 'Volume %s specifies nonexistent driver %s' % (volume.name, volume.driver)
  123. )
  124. def namespace_spec(self, volume_spec):
  125. if not volume_spec.is_named_volume:
  126. return volume_spec
  127. volume = self.volumes[volume_spec.external]
  128. return volume_spec._replace(external=volume.full_name)