1
0

volume.py 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218
  1. from __future__ import absolute_import
  2. from __future__ import unicode_literals
  3. import logging
  4. import re
  5. from docker.errors import NotFound
  6. from docker.utils import version_lt
  7. from . import __version__
  8. from .config import ConfigurationError
  9. from .config.types import VolumeSpec
  10. from .const import LABEL_PROJECT
  11. from .const import LABEL_VERSION
  12. from .const import LABEL_VOLUME
  13. log = logging.getLogger(__name__)
  14. class Volume(object):
  15. def __init__(self, client, project, name, driver=None, driver_opts=None,
  16. external=False, labels=None, custom_name=False):
  17. self.client = client
  18. self.project = project
  19. self.name = name
  20. self.driver = driver
  21. self.driver_opts = driver_opts
  22. self.external = external
  23. self.labels = labels
  24. self.custom_name = custom_name
  25. self.legacy = None
  26. def create(self):
  27. return self.client.create_volume(
  28. self.full_name, self.driver, self.driver_opts, labels=self._labels
  29. )
  30. def remove(self):
  31. if self.external:
  32. log.info("Volume %s is external, skipping", self.full_name)
  33. return
  34. log.info("Removing volume %s", self.full_name)
  35. try:
  36. return self.client.remove_volume(self.full_name)
  37. except NotFound:
  38. self.client.remove_volume(self.legacy_full_name)
  39. def inspect(self, legacy=None):
  40. if legacy:
  41. return self.client.inspect_volume(self.legacy_full_name)
  42. return self.client.inspect_volume(self.full_name)
  43. def exists(self):
  44. self._set_legacy_flag()
  45. try:
  46. self.inspect(legacy=self.legacy)
  47. except NotFound:
  48. return False
  49. return True
  50. @property
  51. def full_name(self):
  52. if self.custom_name:
  53. return self.name
  54. return '{0}_{1}'.format(self.project, self.name)
  55. @property
  56. def legacy_full_name(self):
  57. if self.custom_name:
  58. return self.name
  59. return '{0}_{1}'.format(
  60. re.sub(r'[_-]', '', self.project), self.name
  61. )
  62. @property
  63. def true_name(self):
  64. self._set_legacy_flag()
  65. if self.legacy:
  66. return self.legacy_full_name
  67. return self.full_name
  68. @property
  69. def _labels(self):
  70. if version_lt(self.client._version, '1.23'):
  71. return None
  72. labels = self.labels.copy() if self.labels else {}
  73. labels.update({
  74. LABEL_PROJECT: self.project,
  75. LABEL_VOLUME: self.name,
  76. LABEL_VERSION: __version__,
  77. })
  78. return labels
  79. def _set_legacy_flag(self):
  80. if self.legacy is not None:
  81. return
  82. try:
  83. data = self.inspect(legacy=True)
  84. self.legacy = data is not None
  85. except NotFound:
  86. self.legacy = False
  87. class ProjectVolumes(object):
  88. def __init__(self, volumes):
  89. self.volumes = volumes
  90. @classmethod
  91. def from_config(cls, name, config_data, client):
  92. config_volumes = config_data.volumes or {}
  93. volumes = {
  94. vol_name: Volume(
  95. client=client,
  96. project=name,
  97. name=data.get('name', vol_name),
  98. driver=data.get('driver'),
  99. driver_opts=data.get('driver_opts'),
  100. custom_name=data.get('name') is not None,
  101. labels=data.get('labels'),
  102. external=bool(data.get('external', False))
  103. )
  104. for vol_name, data in config_volumes.items()
  105. }
  106. return cls(volumes)
  107. def remove(self):
  108. for volume in self.volumes.values():
  109. try:
  110. volume.remove()
  111. except NotFound:
  112. log.warn("Volume %s not found.", volume.true_name)
  113. def initialize(self):
  114. try:
  115. for volume in self.volumes.values():
  116. volume_exists = volume.exists()
  117. if volume.external:
  118. log.debug(
  119. 'Volume {0} declared as external. No new '
  120. 'volume will be created.'.format(volume.name)
  121. )
  122. if not volume_exists:
  123. raise ConfigurationError(
  124. 'Volume {name} declared as external, but could'
  125. ' not be found. Please create the volume manually'
  126. ' using `{command}{name}` and try again.'.format(
  127. name=volume.full_name,
  128. command='docker volume create --name='
  129. )
  130. )
  131. continue
  132. if not volume_exists:
  133. log.info(
  134. 'Creating volume "{0}" with {1} driver'.format(
  135. volume.full_name, volume.driver or 'default'
  136. )
  137. )
  138. volume.create()
  139. else:
  140. check_remote_volume_config(volume.inspect(legacy=volume.legacy), volume)
  141. except NotFound:
  142. raise ConfigurationError(
  143. 'Volume %s specifies nonexistent driver %s' % (volume.name, volume.driver)
  144. )
  145. def namespace_spec(self, volume_spec):
  146. if not volume_spec.is_named_volume:
  147. return volume_spec
  148. if isinstance(volume_spec, VolumeSpec):
  149. volume = self.volumes[volume_spec.external]
  150. return volume_spec._replace(external=volume.true_name)
  151. else:
  152. volume_spec.source = self.volumes[volume_spec.source].true_name
  153. return volume_spec
  154. class VolumeConfigChangedError(ConfigurationError):
  155. def __init__(self, local, property_name, local_value, remote_value):
  156. super(VolumeConfigChangedError, self).__init__(
  157. 'Configuration for volume {vol_name} specifies {property_name} '
  158. '{local_value}, but a volume with the same name uses a different '
  159. '{property_name} ({remote_value}). If you wish to use the new '
  160. 'configuration, please remove the existing volume "{full_name}" '
  161. 'first:\n$ docker volume rm {full_name}'.format(
  162. vol_name=local.name, property_name=property_name,
  163. local_value=local_value, remote_value=remote_value,
  164. full_name=local.true_name
  165. )
  166. )
  167. def check_remote_volume_config(remote, local):
  168. if local.driver and remote.get('Driver') != local.driver:
  169. raise VolumeConfigChangedError(local, 'driver', local.driver, remote.get('Driver'))
  170. local_opts = local.driver_opts or {}
  171. remote_opts = remote.get('Options') or {}
  172. for k in set.union(set(remote_opts.keys()), set(local_opts.keys())):
  173. if k.startswith('com.docker.'): # These options are set internally
  174. continue
  175. if remote_opts.get(k) != local_opts.get(k):
  176. raise VolumeConfigChangedError(
  177. local, '"{}" driver_opt'.format(k), local_opts.get(k), remote_opts.get(k),
  178. )
  179. local_labels = local.labels or {}
  180. remote_labels = remote.get('Labels') or {}
  181. for k in set.union(set(remote_labels.keys()), set(local_labels.keys())):
  182. if k.startswith('com.docker.'): # We are only interested in user-specified labels
  183. continue
  184. if remote_labels.get(k) != local_labels.get(k):
  185. log.warn(
  186. 'Volume {}: label "{}" has changed. It may need to be'
  187. ' recreated.'.format(local.name, k)
  188. )