models.py 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145
  1. # -*- coding: utf-8 -*-
  2. from __future__ import unicode_literals
  3. from django.db import models
  4. from polymorphic import PolymorphicModel
  5. from django.core.exceptions import ValidationError
  6. from django.core.urlresolvers import reverse
  7. from netfields import InetAddressField, NetManager
  8. from coin.configuration.models import Configuration
  9. # from coin.offers.backends import ValidateBackendType
  10. from coin import validation
  11. """BIG FAT WARNING
  12. Ce code requiert une sévère factorisation avec vpn/models.py et
  13. housing/models.py
  14. """
  15. FINGERPRINT_TYPES = (
  16. ('ED25519', 'ED25519'),
  17. ('RSA', 'RSA'),
  18. ('ECDSA', 'ECDSA')
  19. )
  20. PROTOCOLE_TYPES = (
  21. ('VNC', 'VNC'),
  22. )
  23. class VPSConfiguration(Configuration):
  24. url_namespace = "vps"
  25. activated = models.BooleanField(default=False, verbose_name='activé')
  26. ipv4_endpoint = InetAddressField(validators=[validation.validate_v4],
  27. verbose_name="IPv4", blank=True, null=True,
  28. help_text="Adresse IPv4 utilisée par "
  29. "défaut sur le VPS")
  30. ipv6_endpoint = InetAddressField(validators=[validation.validate_v6],
  31. verbose_name="IPv6", blank=True, null=True,
  32. help_text="Adresse IPv6 utilisée par "
  33. "défaut sur le VPS")
  34. objects = NetManager()
  35. def get_absolute_url(self):
  36. return reverse('vps:details', args=[str(self.pk)])
  37. # This method is part of the general configuration interface.
  38. def subnet_event(self):
  39. self.check_endpoints(delete=True)
  40. # We potentially changed the endpoints, so we need to save. Also,
  41. # saving will update the subnets in the LDAP backend.
  42. self.full_clean()
  43. self.save()
  44. def get_subnets(self, version):
  45. subnets = self.ip_subnet.all()
  46. return [subnet for subnet in subnets if subnet.inet.version == version]
  47. def generate_endpoints(self, v4=True, v6=True):
  48. """Generate IP endpoints in one of the attributed IP subnets. If there is
  49. no available subnet for a given address family, then no endpoint
  50. is generated for this address family. If there already is an
  51. endpoint, do nothing.
  52. Returns True if an endpoint was generated.
  53. TODO: this should be factored for other technologies (DSL, etc)
  54. """
  55. subnets = self.ip_subnet.all()
  56. updated = False
  57. if v4 and self.ipv4_endpoint is None:
  58. subnets_v4 = [s for s in subnets if s.inet.version == 4]
  59. if len(subnets_v4) > 0:
  60. self.ipv4_endpoint = subnets_v4[0].inet.ip
  61. updated = True
  62. if v6 and self.ipv6_endpoint is None:
  63. subnets_v6 = [s for s in subnets if s.inet.version == 6]
  64. if len(subnets_v6) > 0:
  65. # With v6, we choose the second host of the subnet (cafe::1)
  66. gen = subnets_v6[0].inet.iter_hosts()
  67. gen.next()
  68. self.ipv6_endpoint = gen.next()
  69. updated = True
  70. return updated
  71. def check_endpoints(self, delete=False):
  72. """Check that the IP endpoints are included in one of the attributed IP
  73. subnets.
  74. If [delete] is True, then simply delete the faulty endpoints
  75. instead of raising an exception.
  76. """
  77. error = "L'IP {} n'est pas dans un réseau attribué."
  78. subnets = self.ip_subnet.all()
  79. is_faulty = lambda endpoint : endpoint and not any([endpoint in subnet.inet for subnet in subnets])
  80. if is_faulty(self.ipv4_endpoint):
  81. if delete:
  82. self.ipv4_endpoint = None
  83. else:
  84. raise ValidationError(error.format(self.ipv4_endpoint))
  85. if is_faulty(self.ipv6_endpoint):
  86. if delete:
  87. self.ipv6_endpoint = None
  88. else:
  89. raise ValidationError(error.format(self.ipv6_endpoint))
  90. def clean(self):
  91. # If saving for the first time and IP endpoints are not specified,
  92. # generate them automatically.
  93. if self.pk is None:
  94. self.generate_endpoints()
  95. self.check_endpoints()
  96. def __unicode__(self):
  97. return 'VPS ' + str(self.offersubscription.member.username) + ' ' + self.offersubscription.member.last_name
  98. class Meta:
  99. verbose_name = 'VPS'
  100. class FingerPrint(PolymorphicModel):
  101. vps = models.ForeignKey(VPSConfiguration, verbose_name="vps")
  102. algo = models.CharField(max_length=256, verbose_name="algo",
  103. choices=FINGERPRINT_TYPES)
  104. fingerprint = models.CharField(max_length=256, verbose_name="empreinte")
  105. length = models.IntegerField(verbose_name="longueur de la clé", null=True)
  106. class Meta:
  107. verbose_name = 'Empreinte'
  108. class Console(models.Model):
  109. vps = models.OneToOneField(VPSConfiguration, verbose_name="vps")
  110. protocol = models.CharField(max_length=256, verbose_name="protocole",
  111. choices=PROTOCOLE_TYPES)
  112. domain = models.CharField(max_length=256, verbose_name="nom de domaine",
  113. blank=True, null=True)
  114. port = models.IntegerField(verbose_name="port", null=True)
  115. password_link = models.URLField(verbose_name="Mot de passe", blank=True,
  116. null=True, help_text="Lien à usage unique (détruit après ouverture)")
  117. class Meta:
  118. verbose_name = 'Console'