volume.py 4.2 KB

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