website/hub/services/models.py

598 lines
18 KiB
Python
Raw Normal View History

2025-01-27 14:58:23 +01:00
from django.db import models
2025-01-27 15:14:58 +01:00
from django.core.exceptions import ValidationError
2025-01-27 17:00:56 +01:00
from django.urls import reverse
2025-01-27 15:23:50 +01:00
from django.utils.text import slugify
2025-01-27 15:35:09 +01:00
from django_prose_editor.fields import ProseEditorField
2025-05-20 11:26:52 +02:00
from django.core.validators import URLValidator
2025-01-27 15:35:09 +01:00
2025-01-27 15:14:58 +01:00
def validate_image_size(value):
filesize = value.size
if filesize > 1 * 1024 * 1024: # 1MB
raise ValidationError("Maximum file size is 1MB")
2025-01-27 14:58:23 +01:00
2025-02-28 14:13:51 +01:00
class ReusableText(models.Model):
name = models.CharField(max_length=100)
textsnippet = models.ForeignKey(
"self",
on_delete=models.SET_NULL,
null=True,
blank=True,
related_name="children",
)
text = ProseEditorField()
class Meta:
ordering = ["name"]
def __str__(self):
return self.name
def get_full_text(self):
"""Returns the text with all nested textsnippet content recursively included in reverse order"""
text_parts = []
# Recursively collect all text parts
def collect_text(snippet):
if snippet is None:
return
collect_text(snippet.textsnippet) # Collect deepest snippets first
text_parts.append(snippet.text)
# Start collection with the deepest snippets
collect_text(self.textsnippet)
# Add the main text at the end
text_parts.append(self.text)
# Join all text parts
return "".join(text_parts)
2025-01-27 15:23:50 +01:00
class Category(models.Model):
name = models.CharField(max_length=100)
slug = models.SlugField(unique=True)
parent = models.ForeignKey(
"self", on_delete=models.CASCADE, null=True, blank=True, related_name="children"
)
description = models.TextField(blank=True)
order = models.IntegerField(default=0)
class Meta:
verbose_name_plural = "Categories"
ordering = ["order", "name"]
def __str__(self):
if self.parent:
return f"{self.parent} > {self.name}"
return self.name
def save(self, *args, **kwargs):
if not self.slug:
self.slug = slugify(self.name)
super().save(*args, **kwargs)
@property
def full_path(self):
path = [self.name]
parent = self.parent
while parent:
path.append(parent.name)
parent = parent.parent
return " > ".join(reversed(path))
2025-01-27 14:58:23 +01:00
class CloudProvider(models.Model):
name = models.CharField(max_length=100)
2025-01-27 15:54:37 +01:00
slug = models.SlugField(unique=True)
2025-01-28 13:55:43 +01:00
description = ProseEditorField()
website = models.URLField()
2025-03-03 09:41:54 +01:00
linkedin = models.URLField(blank=True)
phone = models.CharField(max_length=25, blank=True, null=True)
email = models.EmailField(max_length=254, blank=True, null=True)
address = models.TextField(max_length=250, blank=True, null=True)
2025-01-27 15:14:58 +01:00
logo = models.ImageField(
upload_to="cloud_provider_logos/",
validators=[validate_image_size],
null=True,
blank=True,
)
order = models.IntegerField(default=0)
2025-02-25 17:02:01 +01:00
is_featured = models.BooleanField(default=False)
2025-03-03 11:34:27 +01:00
disable_listing = models.BooleanField(default=False)
2025-01-27 14:58:23 +01:00
class Meta:
ordering = ["order"]
2025-01-27 14:58:23 +01:00
def __str__(self):
return self.name
2025-01-27 15:54:37 +01:00
def save(self, *args, **kwargs):
if not self.slug:
self.slug = slugify(self.name)
super().save(*args, **kwargs)
def get_absolute_url(self):
return reverse("services:provider_detail", kwargs={"slug": self.slug})
2025-01-27 14:58:23 +01:00
2025-01-28 13:55:43 +01:00
class Service(models.Model):
2025-01-28 08:53:00 +01:00
name = models.CharField(max_length=200)
2025-01-28 13:55:43 +01:00
slug = models.SlugField(max_length=250, unique=True)
description = ProseEditorField()
2025-03-03 17:54:47 +01:00
tagline = models.TextField(max_length=500, blank=True, null=True)
2025-01-28 08:53:00 +01:00
logo = models.ImageField(
2025-01-28 13:55:43 +01:00
upload_to="service_logos/",
2025-01-28 08:53:00 +01:00
validators=[validate_image_size],
null=True,
blank=True,
)
2025-01-28 13:55:43 +01:00
categories = models.ManyToManyField(Category, related_name="services")
features = ProseEditorField()
2025-02-25 17:02:01 +01:00
is_featured = models.BooleanField(default=False)
2025-02-26 15:27:56 +01:00
is_coming_soon = models.BooleanField(default=False)
2025-03-03 11:34:27 +01:00
disable_listing = models.BooleanField(default=False)
2025-01-28 08:53:00 +01:00
created_at = models.DateTimeField(auto_now_add=True)
updated_at = models.DateTimeField(auto_now=True)
def __str__(self):
return self.name
2025-02-26 15:27:56 +01:00
def clean(self):
if self.is_featured and self.is_coming_soon:
raise ValidationError(
"A service cannot be both featured and coming soon simultaneously."
)
super().clean()
2025-01-28 08:53:00 +01:00
def save(self, *args, **kwargs):
2025-02-26 15:27:56 +01:00
self.clean() # Ensure validation runs on save
2025-01-28 08:53:00 +01:00
if not self.slug:
self.slug = slugify(self.name)
2025-01-28 13:55:43 +01:00
counter = 1
while Service.objects.filter(slug=self.slug).exists():
self.slug = f"{slugify(self.name)}-{counter}"
counter += 1
2025-01-28 08:53:00 +01:00
super().save(*args, **kwargs)
def get_absolute_url(self):
2025-01-28 13:55:43 +01:00
return reverse("services:service_detail", kwargs={"slug": self.slug})
2025-01-28 08:53:00 +01:00
2025-01-28 13:55:43 +01:00
class ConsultingPartner(models.Model):
2025-01-27 14:58:23 +01:00
name = models.CharField(max_length=200)
2025-01-28 13:55:43 +01:00
slug = models.SlugField(unique=True)
2025-01-27 15:35:09 +01:00
description = ProseEditorField()
2025-01-27 15:14:58 +01:00
logo = models.ImageField(
2025-01-28 13:55:43 +01:00
upload_to="partner_logos/",
2025-01-27 15:14:58 +01:00
validators=[validate_image_size],
null=True,
blank=True,
)
2025-01-28 13:55:43 +01:00
website = models.URLField(blank=True)
2025-02-27 17:11:04 +01:00
linkedin = models.URLField(blank=True)
phone = models.CharField(max_length=25, blank=True, null=True)
email = models.EmailField(max_length=254, blank=True, null=True)
address = models.TextField(max_length=250, blank=True, null=True)
services = models.ManyToManyField(
Service, related_name="consulting_partners", blank=True
)
2025-01-28 13:55:43 +01:00
cloud_providers = models.ManyToManyField(
CloudProvider, related_name="consulting_partners", blank=True
)
2025-02-27 17:11:04 +01:00
order = models.IntegerField(default=0)
2025-02-25 17:02:01 +01:00
is_featured = models.BooleanField(default=False)
2025-03-03 11:34:27 +01:00
disable_listing = models.BooleanField(default=False)
2025-01-27 14:58:23 +01:00
created_at = models.DateTimeField(auto_now_add=True)
updated_at = models.DateTimeField(auto_now=True)
class Meta:
ordering = ["order"]
2025-01-27 14:58:23 +01:00
def __str__(self):
return self.name
2025-01-27 16:51:23 +01:00
2025-01-27 17:00:56 +01:00
def save(self, *args, **kwargs):
if not self.slug:
2025-01-28 13:55:43 +01:00
self.slug = slugify(self.name)
super().save(*args, **kwargs)
def get_absolute_url(self):
return reverse("services:partner_detail", kwargs={"slug": self.slug})
class ServiceOffering(models.Model):
service = models.ForeignKey(
Service, on_delete=models.CASCADE, related_name="offerings"
)
cloud_provider = models.ForeignKey(
CloudProvider, on_delete=models.CASCADE, related_name="offerings"
)
2025-02-28 14:13:51 +01:00
description = ProseEditorField(blank=True, null=True)
offer_description = models.ForeignKey(
ReusableText,
on_delete=models.PROTECT,
related_name="offer_descriptions",
blank=True,
null=True,
)
2025-03-03 11:34:27 +01:00
disable_listing = models.BooleanField(default=False)
2025-01-28 13:55:43 +01:00
created_at = models.DateTimeField(auto_now_add=True)
updated_at = models.DateTimeField(auto_now=True)
class Meta:
unique_together = ["service", "cloud_provider"]
ordering = ["service__name", "cloud_provider__name"]
2025-01-27 17:00:56 +01:00
2025-01-28 13:55:43 +01:00
def __str__(self):
return f"{self.service.name} on {self.cloud_provider.name}"
2025-01-27 17:00:56 +01:00
def get_absolute_url(self):
2025-02-28 14:25:35 +01:00
return reverse(
"services:offering_detail",
kwargs={
"provider_slug": self.cloud_provider.slug,
"service_slug": self.service.slug,
},
)
2025-01-27 17:00:56 +01:00
2025-01-28 10:41:39 +01:00
2025-01-28 13:55:43 +01:00
class Plan(models.Model):
name = models.CharField(max_length=100)
2025-02-28 14:27:19 +01:00
description = ProseEditorField(blank=True, null=True)
2025-03-03 08:28:21 +01:00
pricing = ProseEditorField(blank=True, null=True)
2025-02-28 14:13:51 +01:00
plan_description = models.ForeignKey(
ReusableText,
on_delete=models.PROTECT,
related_name="plan_descriptions",
blank=True,
null=True,
)
2025-01-28 13:55:43 +01:00
offering = models.ForeignKey(
ServiceOffering, on_delete=models.CASCADE, related_name="plans"
)
2025-02-28 14:13:51 +01:00
2025-01-27 16:51:23 +01:00
created_at = models.DateTimeField(auto_now_add=True)
2025-01-28 13:55:43 +01:00
updated_at = models.DateTimeField(auto_now=True)
class Meta:
2025-02-28 14:13:51 +01:00
ordering = ["name"]
2025-01-28 13:55:43 +01:00
unique_together = [["offering", "name"]]
2025-01-27 16:51:23 +01:00
def __str__(self):
2025-01-28 13:55:43 +01:00
return f"{self.offering} - {self.name}"
2025-02-28 14:13:51 +01:00
class ExternalLinkOffering(models.Model):
offering = models.ForeignKey(
ServiceOffering, on_delete=models.CASCADE, related_name="external_links"
)
url = models.URLField()
description = models.CharField(max_length=200)
order = models.IntegerField(default=0)
created_at = models.DateTimeField(auto_now_add=True)
updated_at = models.DateTimeField(auto_now=True)
2025-01-28 13:55:43 +01:00
class Meta:
2025-02-28 14:13:51 +01:00
ordering = ["order", "description"]
verbose_name = "External Link"
verbose_name_plural = "External Links"
2025-01-28 13:55:43 +01:00
def __str__(self):
2025-02-28 14:13:51 +01:00
return f"{self.description} ({self.url})"
def clean(self):
validate = URLValidator()
try:
validate(self.url)
except ValidationError:
raise ValidationError({"url": "Enter a valid URL."})
2025-01-28 11:00:15 +01:00
class ExternalLink(models.Model):
service = models.ForeignKey(
Service, on_delete=models.CASCADE, related_name="external_links"
)
url = models.URLField()
description = models.CharField(max_length=200)
order = models.IntegerField(default=0)
created_at = models.DateTimeField(auto_now_add=True)
updated_at = models.DateTimeField(auto_now=True)
class Meta:
ordering = ["order", "description"]
verbose_name = "External Link"
verbose_name_plural = "External Links"
def __str__(self):
return f"{self.description} ({self.url})"
def clean(self):
validate = URLValidator()
try:
validate(self.url)
except ValidationError:
raise ValidationError({"url": "Enter a valid URL."})
2025-01-28 13:55:43 +01:00
class Lead(models.Model):
2025-02-27 15:44:55 +01:00
name = models.CharField(max_length=200)
email = models.EmailField()
company = models.CharField(max_length=200, null=True, blank=True)
phone = models.CharField(max_length=50, null=True, blank=True)
message = models.TextField(blank=True, null=True, max_length=1000)
odoo_lead_id = models.IntegerField(null=True, blank=True)
service = models.ForeignKey(
Service, on_delete=models.SET_NULL, null=True, blank=True
)
2025-01-28 13:55:43 +01:00
offering = models.ForeignKey(
ServiceOffering, on_delete=models.SET_NULL, null=True, blank=True
)
plan = models.ForeignKey(Plan, on_delete=models.SET_NULL, null=True, blank=True)
2025-02-27 15:44:55 +01:00
2025-01-28 13:55:43 +01:00
created_at = models.DateTimeField(auto_now_add=True)
def __str__(self):
return f"{self.name} - {self.company} ({self.service})"
2025-03-06 11:35:55 +01:00
class WebsiteFaq(models.Model):
question = models.CharField(max_length=200)
answer = ProseEditorField()
order = models.IntegerField(default=0)
class Meta:
ordering = ["order"]
def __str__(self):
return self.question
2025-05-20 11:26:52 +02:00
2025-05-20 15:27:45 +02:00
class Currency(models.TextChoices):
CHF = "CHF", "Swiss Franc"
EUR = "EUR", "Euro"
USD = "USD", "US Dollar"
2025-05-22 16:34:15 +02:00
class Term(models.TextChoices):
MTH = "MTH", "per Month (30d)"
DAY = "DAY", "per Day"
HR = "HR", "per Hour"
MIN = "MIN", "per Minute"
class Unit(models.TextChoices):
GIB = "GIB", "GiB"
MIB = "MIB", "MiB"
CPU = "CPU", "vCPU"
2025-05-20 15:27:45 +02:00
class ComputePlanPrice(models.Model):
compute_plan = models.ForeignKey(
"ComputePlan", on_delete=models.CASCADE, related_name="prices"
)
currency = models.CharField(
max_length=3,
choices=Currency.choices,
)
amount = models.DecimalField(
max_digits=10,
decimal_places=2,
help_text="Price in the specified currency, excl. VAT",
)
class Meta:
unique_together = ("compute_plan", "currency")
ordering = ["currency"]
def __str__(self):
return f"{self.compute_plan.name} - {self.amount} {self.currency}"
2025-05-20 11:26:52 +02:00
class ComputePlan(models.Model):
name = models.CharField(max_length=200)
vcpus = models.FloatField(help_text="Number of available vCPUs")
ram = models.FloatField(help_text="Amount of RAM available")
cpu_mem_ratio = models.FloatField(
help_text="vCPU to Memory ratio. How much vCPU per GiB RAM is available?"
)
active = models.BooleanField(default=True, help_text="Is the plan active?")
2025-05-22 16:34:15 +02:00
term = models.CharField(
max_length=3,
default=Term.MTH,
choices=Term.choices,
)
2025-05-20 11:26:52 +02:00
cloud_provider = models.ForeignKey(
CloudProvider, on_delete=models.CASCADE, related_name="compute_plans"
)
valid_from = models.DateTimeField(blank=True, null=True)
valid_to = models.DateTimeField(blank=True, null=True)
class Meta:
2025-05-20 15:27:45 +02:00
ordering = ["name"]
2025-05-20 11:26:52 +02:00
def __str__(self):
return self.name
2025-05-20 15:27:45 +02:00
def get_price(self, currency_code: str):
try:
return self.prices.get(currency=currency_code).amount
except ComputePlanPrice.DoesNotExist:
return None
2025-05-22 16:34:15 +02:00
class StoragePlanPrice(models.Model):
storage_plan = models.ForeignKey(
"StoragePlan", on_delete=models.CASCADE, related_name="prices"
)
currency = models.CharField(
max_length=3,
choices=Currency.choices,
)
amount = models.DecimalField(
max_digits=10,
decimal_places=2,
help_text="Price in the specified currency, excl. VAT",
)
class Meta:
unique_together = ("storage_plan", "currency")
ordering = ["currency"]
def __str__(self):
return f"{self.storage_plan.name} - {self.amount} {self.currency}"
class StoragePlan(models.Model):
name = models.CharField(max_length=200)
cloud_provider = models.ForeignKey(
CloudProvider, on_delete=models.CASCADE, related_name="storage_plans"
)
term = models.CharField(
max_length=3,
default=Term.MTH,
choices=Term.choices,
)
unit = models.CharField(
max_length=3,
default=Unit.GIB,
choices=Unit.choices,
)
valid_from = models.DateTimeField(blank=True, null=True)
valid_to = models.DateTimeField(blank=True, null=True)
class Meta:
unique_together = ("cloud_provider", "term", "unit", "valid_from", "valid_to")
ordering = ["name"]
def __str__(self):
return self.name
def get_price(self, currency_code: str):
try:
return self.prices.get(currency=currency_code).amount
except ComputePlanPrice.DoesNotExist:
return None
2025-05-20 15:27:45 +02:00
class VSHNAppCatBaseFee(models.Model):
vshn_appcat_price_config = models.ForeignKey(
"VSHNAppCatPrice", on_delete=models.CASCADE, related_name="base_fees"
)
currency = models.CharField(
max_length=3,
choices=Currency.choices,
)
amount = models.DecimalField(
max_digits=10,
decimal_places=2,
help_text="Base fee in the specified currency, excl. VAT",
)
class Meta:
unique_together = ("vshn_appcat_price_config", "currency")
ordering = ["currency"]
def __str__(self):
return f"{self.vshn_appcat_price_config.service.name} Base Fee - {self.amount} {self.currency}"
class VSHNAppCatPrice(models.Model):
class VariableUnit(models.TextChoices):
RAM = "RAM", "Memory (RAM)"
CPU = "CPU", "CPU (vCPU)"
USER = "USR", "Users"
class ServiceLevel(models.TextChoices):
BEST_EFFORT = "BE", "Best Effort"
GUARANTEED = "GA", "Guaranteed Availability"
service = models.ForeignKey(
Service, on_delete=models.CASCADE, related_name="vshn_appcat_price"
)
variable_unit = models.CharField(
max_length=3,
choices=VariableUnit.choices,
default=VariableUnit.RAM,
)
ha_replica_min = models.IntegerField(
default=1, help_text="Minimum of replicas for HA"
)
ha_replica_max = models.IntegerField(
default=1, help_text="Maximum supported replicas"
)
2025-05-22 16:34:15 +02:00
term = models.CharField(
max_length=3,
default=Term.MTH,
choices=Term.choices,
)
2025-05-20 15:27:45 +02:00
def __str__(self):
return f"{self.service.name} - {self.get_variable_unit_display()} based pricing"
def get_base_fee(self, currency_code: str):
try:
return self.base_fees.get(currency=currency_code).amount
except VSHNAppCatBaseFee.DoesNotExist:
return None
def get_unit_rate(self, currency_code: str, service_level: str):
try:
return self.unit_rates.get(
currency=currency_code, service_level=service_level
).amount
except VSHNAppCatUnitRate.DoesNotExist:
return None
def calculate_final_price(
self, currency_code: str, service_level: str, number_of_units: int
):
base_fee = self.get_base_fee(currency_code)
unit_rate = self.get_unit_rate(currency_code, service_level)
if base_fee is None or unit_rate is None:
return None
if number_of_units < 0:
raise ValueError("Number of units cannot be negative")
total_price = base_fee + (unit_rate * number_of_units)
return total_price
class VSHNAppCatUnitRate(models.Model):
vshn_appcat_price_config = models.ForeignKey(
VSHNAppCatPrice, on_delete=models.CASCADE, related_name="unit_rates"
)
currency = models.CharField(
max_length=3,
choices=Currency.choices,
)
service_level = models.CharField(
max_length=2,
choices=VSHNAppCatPrice.ServiceLevel.choices,
)
amount = models.DecimalField(
max_digits=10,
decimal_places=4,
help_text="Price per unit in the specified currency and service level, excl. VAT",
)
class Meta:
unique_together = ("vshn_appcat_price_config", "currency", "service_level")
ordering = ["currency", "service_level"]
def __str__(self):
return f"{self.vshn_appcat_price_config.service.name} - {self.get_service_level_display()} Unit Rate - {self.amount} {self.currency}"