volume.py 7.2 KB

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