virtualmachines.py 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427
  1. from django.contrib.contenttypes.fields import GenericRelation
  2. from django.core.exceptions import ValidationError
  3. from django.core.validators import MinValueValidator
  4. from django.db import models
  5. from django.db.models import Q, Sum
  6. from django.db.models.functions import Lower
  7. from django.urls import reverse
  8. from django.utils.translation import gettext_lazy as _
  9. from dcim.models import BaseInterface
  10. from dcim.models.mixins import RenderConfigMixin
  11. from extras.models import ConfigContextModel
  12. from extras.querysets import ConfigContextModelQuerySet
  13. from netbox.config import get_config
  14. from netbox.models import NetBoxModel, PrimaryModel
  15. from netbox.models.features import ContactsMixin
  16. from utilities.fields import CounterCacheField, NaturalOrderingField
  17. from utilities.ordering import naturalize_interface
  18. from utilities.query_functions import CollateAsChar
  19. from utilities.tracking import TrackingModelMixin
  20. from virtualization.choices import *
  21. __all__ = (
  22. 'VirtualDisk',
  23. 'VirtualMachine',
  24. 'VMInterface',
  25. )
  26. class VirtualMachine(ContactsMixin, RenderConfigMixin, ConfigContextModel, PrimaryModel):
  27. """
  28. A virtual machine which runs inside a Cluster.
  29. """
  30. site = models.ForeignKey(
  31. to='dcim.Site',
  32. on_delete=models.PROTECT,
  33. related_name='virtual_machines',
  34. blank=True,
  35. null=True
  36. )
  37. cluster = models.ForeignKey(
  38. to='virtualization.Cluster',
  39. on_delete=models.PROTECT,
  40. related_name='virtual_machines',
  41. blank=True,
  42. null=True
  43. )
  44. device = models.ForeignKey(
  45. to='dcim.Device',
  46. on_delete=models.PROTECT,
  47. related_name='virtual_machines',
  48. blank=True,
  49. null=True
  50. )
  51. tenant = models.ForeignKey(
  52. to='tenancy.Tenant',
  53. on_delete=models.PROTECT,
  54. related_name='virtual_machines',
  55. blank=True,
  56. null=True
  57. )
  58. platform = models.ForeignKey(
  59. to='dcim.Platform',
  60. on_delete=models.SET_NULL,
  61. related_name='virtual_machines',
  62. blank=True,
  63. null=True
  64. )
  65. name = models.CharField(
  66. verbose_name=_('name'),
  67. max_length=64
  68. )
  69. _name = NaturalOrderingField(
  70. target_field='name',
  71. max_length=100,
  72. blank=True
  73. )
  74. status = models.CharField(
  75. max_length=50,
  76. choices=VirtualMachineStatusChoices,
  77. default=VirtualMachineStatusChoices.STATUS_ACTIVE,
  78. verbose_name=_('status')
  79. )
  80. role = models.ForeignKey(
  81. to='dcim.DeviceRole',
  82. on_delete=models.PROTECT,
  83. related_name='virtual_machines',
  84. limit_choices_to={'vm_role': True},
  85. blank=True,
  86. null=True
  87. )
  88. primary_ip4 = models.OneToOneField(
  89. to='ipam.IPAddress',
  90. on_delete=models.SET_NULL,
  91. related_name='+',
  92. blank=True,
  93. null=True,
  94. verbose_name=_('primary IPv4')
  95. )
  96. primary_ip6 = models.OneToOneField(
  97. to='ipam.IPAddress',
  98. on_delete=models.SET_NULL,
  99. related_name='+',
  100. blank=True,
  101. null=True,
  102. verbose_name=_('primary IPv6')
  103. )
  104. vcpus = models.DecimalField(
  105. max_digits=6,
  106. decimal_places=2,
  107. blank=True,
  108. null=True,
  109. verbose_name=_('vCPUs'),
  110. validators=(
  111. MinValueValidator(0.01),
  112. )
  113. )
  114. memory = models.PositiveIntegerField(
  115. blank=True,
  116. null=True,
  117. verbose_name=_('memory (MB)')
  118. )
  119. disk = models.PositiveIntegerField(
  120. blank=True,
  121. null=True,
  122. verbose_name=_('disk (GB)')
  123. )
  124. # Counter fields
  125. interface_count = CounterCacheField(
  126. to_model='virtualization.VMInterface',
  127. to_field='virtual_machine'
  128. )
  129. virtual_disk_count = CounterCacheField(
  130. to_model='virtualization.VirtualDisk',
  131. to_field='virtual_machine'
  132. )
  133. objects = ConfigContextModelQuerySet.as_manager()
  134. clone_fields = (
  135. 'site', 'cluster', 'device', 'tenant', 'platform', 'status', 'role', 'vcpus', 'memory', 'disk',
  136. )
  137. prerequisite_models = (
  138. 'virtualization.Cluster',
  139. )
  140. class Meta:
  141. ordering = ('_name', 'pk') # Name may be non-unique
  142. constraints = (
  143. models.UniqueConstraint(
  144. Lower('name'), 'cluster', 'tenant',
  145. name='%(app_label)s_%(class)s_unique_name_cluster_tenant'
  146. ),
  147. models.UniqueConstraint(
  148. Lower('name'), 'cluster',
  149. name='%(app_label)s_%(class)s_unique_name_cluster',
  150. condition=Q(tenant__isnull=True),
  151. violation_error_message=_("Virtual machine name must be unique per cluster.")
  152. ),
  153. )
  154. verbose_name = _('virtual machine')
  155. verbose_name_plural = _('virtual machines')
  156. def __str__(self):
  157. return self.name
  158. def get_absolute_url(self):
  159. return reverse('virtualization:virtualmachine', args=[self.pk])
  160. def clean(self):
  161. super().clean()
  162. # Must be assigned to a site and/or cluster
  163. if not self.site and not self.cluster:
  164. raise ValidationError({
  165. 'cluster': _('A virtual machine must be assigned to a site and/or cluster.')
  166. })
  167. # Validate site for cluster & device
  168. if self.cluster and self.site and self.cluster.site != self.site:
  169. raise ValidationError({
  170. 'cluster': _(
  171. 'The selected cluster ({cluster}) is not assigned to this site ({site}).'
  172. ).format(cluster=self.cluster, site=self.site)
  173. })
  174. # Validate assigned cluster device
  175. if self.device and not self.cluster:
  176. raise ValidationError({
  177. 'device': _('Must specify a cluster when assigning a host device.')
  178. })
  179. if self.device and self.device not in self.cluster.devices.all():
  180. raise ValidationError({
  181. 'device': _(
  182. "The selected device ({device}) is not assigned to this cluster ({cluster})."
  183. ).format(device=self.device, cluster=self.cluster)
  184. })
  185. # Validate aggregate disk size
  186. if self.pk:
  187. total_disk = self.virtualdisks.aggregate(Sum('size', default=0))['size__sum']
  188. if total_disk and self.disk != total_disk:
  189. raise ValidationError({
  190. 'disk': _(
  191. "The specified disk size ({size}) must match the aggregate size of assigned virtual disks "
  192. "({total_size})."
  193. ).format(size=self.disk, total_size=total_disk)
  194. })
  195. # Validate primary IP addresses
  196. interfaces = self.interfaces.all() if self.pk else None
  197. for family in (4, 6):
  198. field = f'primary_ip{family}'
  199. ip = getattr(self, field)
  200. if ip is not None:
  201. if ip.address.version != family:
  202. raise ValidationError({
  203. field: _(
  204. "Must be an IPv{family} address. ({ip} is an IPv{version} address.)"
  205. ).format(family=family, ip=ip, version=ip.address.version)
  206. })
  207. if ip.assigned_object in interfaces:
  208. pass
  209. elif ip.nat_inside is not None and ip.nat_inside.assigned_object in interfaces:
  210. pass
  211. else:
  212. raise ValidationError({
  213. field: _("The specified IP address ({ip}) is not assigned to this VM.").format(ip=ip),
  214. })
  215. def save(self, *args, **kwargs):
  216. # Assign site from cluster if not set
  217. if self.cluster and not self.site:
  218. self.site = self.cluster.site
  219. super().save(*args, **kwargs)
  220. def get_status_color(self):
  221. return VirtualMachineStatusChoices.colors.get(self.status)
  222. @property
  223. def primary_ip(self):
  224. if get_config().PREFER_IPV4 and self.primary_ip4:
  225. return self.primary_ip4
  226. elif self.primary_ip6:
  227. return self.primary_ip6
  228. elif self.primary_ip4:
  229. return self.primary_ip4
  230. else:
  231. return None
  232. #
  233. # VM components
  234. #
  235. class ComponentModel(NetBoxModel):
  236. """
  237. An abstract model inherited by any model which has a parent VirtualMachine.
  238. """
  239. virtual_machine = models.ForeignKey(
  240. to='virtualization.VirtualMachine',
  241. on_delete=models.CASCADE,
  242. related_name='%(class)ss'
  243. )
  244. name = models.CharField(
  245. verbose_name=_('name'),
  246. max_length=64
  247. )
  248. _name = NaturalOrderingField(
  249. target_field='name',
  250. naturalize_function=naturalize_interface,
  251. max_length=100,
  252. blank=True
  253. )
  254. description = models.CharField(
  255. verbose_name=_('description'),
  256. max_length=200,
  257. blank=True
  258. )
  259. class Meta:
  260. abstract = True
  261. ordering = ('virtual_machine', CollateAsChar('_name'))
  262. constraints = (
  263. models.UniqueConstraint(
  264. fields=('virtual_machine', 'name'),
  265. name='%(app_label)s_%(class)s_unique_virtual_machine_name'
  266. ),
  267. )
  268. def __str__(self):
  269. return self.name
  270. def to_objectchange(self, action):
  271. objectchange = super().to_objectchange(action)
  272. objectchange.related_object = self.virtual_machine
  273. return objectchange
  274. @property
  275. def parent_object(self):
  276. return self.virtual_machine
  277. class VMInterface(ComponentModel, BaseInterface, TrackingModelMixin):
  278. virtual_machine = models.ForeignKey(
  279. to='virtualization.VirtualMachine',
  280. on_delete=models.CASCADE,
  281. related_name='interfaces' # Override ComponentModel
  282. )
  283. _name = NaturalOrderingField(
  284. target_field='name',
  285. naturalize_function=naturalize_interface,
  286. max_length=100,
  287. blank=True
  288. )
  289. untagged_vlan = models.ForeignKey(
  290. to='ipam.VLAN',
  291. on_delete=models.SET_NULL,
  292. related_name='vminterfaces_as_untagged',
  293. null=True,
  294. blank=True,
  295. verbose_name=_('untagged VLAN')
  296. )
  297. tagged_vlans = models.ManyToManyField(
  298. to='ipam.VLAN',
  299. related_name='vminterfaces_as_tagged',
  300. blank=True,
  301. verbose_name=_('tagged VLANs')
  302. )
  303. ip_addresses = GenericRelation(
  304. to='ipam.IPAddress',
  305. content_type_field='assigned_object_type',
  306. object_id_field='assigned_object_id',
  307. related_query_name='vminterface'
  308. )
  309. vrf = models.ForeignKey(
  310. to='ipam.VRF',
  311. on_delete=models.SET_NULL,
  312. related_name='vminterfaces',
  313. null=True,
  314. blank=True,
  315. verbose_name=_('VRF')
  316. )
  317. fhrp_group_assignments = GenericRelation(
  318. to='ipam.FHRPGroupAssignment',
  319. content_type_field='interface_type',
  320. object_id_field='interface_id',
  321. related_query_name='+'
  322. )
  323. l2vpn_terminations = GenericRelation(
  324. to='ipam.L2VPNTermination',
  325. content_type_field='assigned_object_type',
  326. object_id_field='assigned_object_id',
  327. related_query_name='vminterface',
  328. )
  329. class Meta(ComponentModel.Meta):
  330. verbose_name = _('interface')
  331. verbose_name_plural = _('interfaces')
  332. def get_absolute_url(self):
  333. return reverse('virtualization:vminterface', kwargs={'pk': self.pk})
  334. def clean(self):
  335. super().clean()
  336. # Parent validation
  337. # An interface cannot be its own parent
  338. if self.pk and self.parent_id == self.pk:
  339. raise ValidationError({'parent': _("An interface cannot be its own parent.")})
  340. # An interface's parent must belong to the same virtual machine
  341. if self.parent and self.parent.virtual_machine != self.virtual_machine:
  342. raise ValidationError({
  343. 'parent': _(
  344. "The selected parent interface ({parent}) belongs to a different virtual machine "
  345. "({virtual_machine})."
  346. ).format(parent=self.parent, virtual_machine=self.parent.virtual_machine)
  347. })
  348. # Bridge validation
  349. # An interface cannot be bridged to itself
  350. if self.pk and self.bridge_id == self.pk:
  351. raise ValidationError({'bridge': _("An interface cannot be bridged to itself.")})
  352. # A bridged interface belong to the same virtual machine
  353. if self.bridge and self.bridge.virtual_machine != self.virtual_machine:
  354. raise ValidationError({
  355. 'bridge': _(
  356. "The selected bridge interface ({bridge}) belongs to a different virtual machine "
  357. "({virtual_machine})."
  358. ).format(bridge=self.bridge, virtual_machine=self.bridge.virtual_machine)
  359. })
  360. # VLAN validation
  361. # Validate untagged VLAN
  362. if self.untagged_vlan and self.untagged_vlan.site not in [self.virtual_machine.site, None]:
  363. raise ValidationError({
  364. 'untagged_vlan': _(
  365. "The untagged VLAN ({untagged_vlan}) must belong to the same site as the interface's parent "
  366. "virtual machine, or it must be global."
  367. ).format(untagged_vlan=self.untagged_vlan)
  368. })
  369. @property
  370. def l2vpn_termination(self):
  371. return self.l2vpn_terminations.first()
  372. class VirtualDisk(ComponentModel, TrackingModelMixin):
  373. size = models.PositiveIntegerField(
  374. verbose_name=_('size (GB)'),
  375. )
  376. class Meta(ComponentModel.Meta):
  377. verbose_name = _('virtual disk')
  378. verbose_name_plural = _('virtual disks')
  379. def get_absolute_url(self):
  380. return reverse('virtualization:virtualdisk', args=[self.pk])