volume.py 4.4 KB

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