You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 

753 lines
30 KiB

from tabnanny import verbose
from django.db import models
from django.utils.translation import gettext_lazy as _
from django.utils.text import slugify
from typing import Optional
from utils.slug import generate_smart_slug
from django.conf import settings
class NarratorLayer(models.Model):
"""
Model for narrator layers/classes (Tabaqat)
Represents the classification level of narrators in hadis chains
"""
name = models.JSONField(default = list , verbose_name=_('Name'))
number = models.PositiveIntegerField(verbose_name=_('layer number'), unique=True)
description = models.JSONField(default = list , verbose_name=_('Description'))
slug = models.SlugField(max_length=255,unique=True, verbose_name=_('slug'), blank=True)
created_at = models.DateTimeField(auto_now_add=True, verbose_name=_('created at'))
updated_at = models.DateTimeField(auto_now=True, verbose_name=_('updated at'))
class Meta:
verbose_name = _('Narrator Layer')
verbose_name_plural = _('Narrator Layers')
ordering = ['number']
def __str__(self):
return f"{_('Layer')} {self.number} - {self.name[0]['text']}"
def get_description(self,lang):
"""
Get title for a specific language
"""
if not self.description or not isinstance(self.description, list):
return None
for tr in self.description:
if isinstance(tr, dict) and tr.get('language_code') == lang:
return tr.get('text', '')
for tr in self.description:
if isinstance(tr, dict) and tr.get('language_code') == 'en':
return tr.get('text', '')
return None
def get_name(self,lang):
"""
Get title for a specific language
"""
if not self.name or not isinstance(self.name, list):
return None
for tr in self.name:
if isinstance(tr, dict) and tr.get('language_code') == lang:
return tr.get('text', '')
for tr in self.name:
if isinstance(tr, dict) and tr.get('language_code') == 'en':
return tr.get('text', '')
return None
def save(self, *args, **kwargs):
if not self.slug or (isinstance(self.slug, str) and self.slug.strip() == ''):
# Try to get text from name field with robust error handling
try:
if self.name and isinstance(self.name, list) and len(self.name) > 0:
first_item = self.name[0]
if isinstance(first_item, dict):
text = first_item.get('text', '').strip()
if text:
slug = slugify(text)
# Ensure uniqueness
counter = 1
base_slug = slug
while NarratorLayer.objects.filter(slug=slug).exclude(pk=self.pk).exists():
slug = f"{base_slug}-{counter}"
counter += 1
self.slug = slug
else:
# Fallback to layer number if text is empty
base_slug = f"layer-{self.number}"
slug = base_slug
counter = 1
while NarratorLayer.objects.filter(slug=slug).exclude(pk=self.pk).exists():
slug = f"{base_slug}-{counter}"
counter += 1
self.slug = slug
else:
# Fallback to layer number if name structure is invalid
base_slug = f"layer-{self.number}"
slug = base_slug
counter = 1
while NarratorLayer.objects.filter(slug=slug).exclude(pk=self.pk).exists():
slug = f"{base_slug}-{counter}"
counter += 1
self.slug = slug
else:
# Fallback to layer number if name structure is invalid
base_slug = f"layer-{self.number}"
slug = base_slug
counter = 1
while NarratorLayer.objects.filter(slug=slug).exclude(pk=self.pk).exists():
slug = f"{base_slug}-{counter}"
counter += 1
self.slug = slug
except (IndexError, KeyError, AttributeError, TypeError):
# Fallback to layer number on any error
base_slug = f"layer-{self.number}"
slug = base_slug
counter = 1
while NarratorLayer.objects.filter(slug=slug).exclude(pk=self.pk).exists():
slug = f"{base_slug}-{counter}"
counter += 1
self.slug = slug
super().save(*args, **kwargs)
class TransmitterReliability(models.Model):
class ColorChoices(models.TextChoices):
RED = 'red', _('Red')
GREEN = 'green', _('Green')
BLUE = 'blue', _('Blue')
YELLOW = 'yellow', _('Yellow')
ORANGE = 'orange', _('Orange')
PURPLE = 'purple', _('Purple')
GRAY = 'gray', _('Gray')
title = models.JSONField(default = list , verbose_name=_('Title'))
slug = models.SlugField(max_length=255, verbose_name=_('slug'), blank=True,null=True)
color = models.CharField(max_length=20, choices=ColorChoices.choices, verbose_name=_('color'))
def save(self, *args, **kwargs):
if not self.slug or (isinstance(self.slug, str) and self.slug.strip() == ''):
# Try to get text from title field with robust error handling
try:
if self.title and isinstance(self.title, list) and len(self.title) > 0:
first_item = self.title[0]
if isinstance(first_item, dict):
text = first_item.get('text', '').strip()
if text:
slug = slugify(text)
# Ensure uniqueness
counter = 1
base_slug = slug
while TransmitterReliability.objects.filter(slug=slug).exclude(pk=self.pk).exists():
slug = f"{base_slug}-{counter}"
counter += 1
self.slug = slug
else:
# Fallback to a timestamp-based slug
import time
suffix = int(time.time() * 1000) % 1000000
base_slug = f"reliability-{suffix}"
counter = 1
while TransmitterReliability.objects.filter(slug=base_slug).exclude(pk=self.pk).exists():
base_slug = f"reliability-{suffix}-{counter}"
counter += 1
self.slug = base_slug
else:
import time
suffix = int(time.time() * 1000) % 1000000
base_slug = f"reliability-{suffix}"
counter = 1
while TransmitterReliability.objects.filter(slug=base_slug).exclude(pk=self.pk).exists():
base_slug = f"reliability-{suffix}-{counter}"
counter += 1
self.slug = base_slug
else:
import time
suffix = int(time.time() * 1000) % 1000000
base_slug = f"reliability-{suffix}"
counter = 1
while TransmitterReliability.objects.filter(slug=base_slug).exclude(pk=self.pk).exists():
base_slug = f"reliability-{suffix}-{counter}"
counter += 1
self.slug = base_slug
except (IndexError, KeyError, AttributeError, TypeError):
import time
suffix = int(time.time() * 1000) % 1000000
base_slug = f"reliability-{suffix}"
counter = 1
while TransmitterReliability.objects.filter(slug=base_slug).exclude(pk=self.pk).exists():
base_slug = f"reliability-{suffix}-{counter}"
counter += 1
self.slug = base_slug
super().save(*args, **kwargs)
def __str__(self):
return self.title[0]['text']
def get_title(self,lang):
"""
Get title for a specific language
"""
if not self.title or not isinstance(self.title, list):
return None
for tr in self.title:
if isinstance(tr, dict) and tr.get('language_code') == lang:
return tr.get('text', '')
for tr in self.title:
if isinstance(tr, dict) and tr.get('language_code') == 'en':
return tr.get('text', '')
return None
class Meta:
verbose_name = _('Transmitter Reliability')
verbose_name_plural = _('Transmitter Reliabilities')
class Transmitters(models.Model):
# class ReliabilityLevel(models.TextChoices):
# VERY_RELIABLE = 'very_reliable', _('Very Reliable')
# RELIABLE = 'reliable', _('Reliable')
# ACCEPTABLE = 'acceptable', _('Acceptable')
# WEAK = 'weak', _('Weak')
# VERY_WEAK = 'very_weak', _('Very Weak')
# UNKNOWN = 'unknown', _('Unknown')
class MadhhabChoices(models.TextChoices):
SHIA = 'shia', _('Shia')
SUNNI = 'sunni', _('Sunni')
# HANAFI = 'hanafi', _('Hanafi')
# MALIKI = 'maliki', _('Maliki')
# SHAFII = 'shafii', _('Shafi\'i')
# HANBALI = 'hanbali', _('Hanbali')
OTHER = 'other', _('Other')
UNKNOWN = 'unknown', _('Unknown')
# Basic Information
full_name = models.JSONField(default = list , verbose_name=_('Full Name'))
kunya = models.JSONField(default = list , verbose_name=_('Kunya'))
known_as = models.JSONField(default = list , verbose_name=_('Known as'))
nickname = models.JSONField(default = list , verbose_name=_('Nick Name'))
slug = models.SlugField(max_length=255, verbose_name=_('slug'), blank=True,unique=True)
# Geographic Information
origin = models.JSONField(default = list , verbose_name=_('Origin'))
lived_in = models.JSONField(default = list , verbose_name=_('Lived in'))
died_in = models.JSONField(default = list , verbose_name=_('Died in'))
# Date Information
birth_year_hijri = models.IntegerField(verbose_name=_("Birth Year (Hijri)"), null=True, blank=True)
death_year_hijri = models.IntegerField(verbose_name=_("Death Year (Hijri)"), null=True, blank=True)
age_at_death = models.PositiveIntegerField(verbose_name=_('Age at Death'), blank=True, null=True)
generation = models.PositiveIntegerField(verbose_name=_('Generation'), blank=True, null=True)
# Religious & Academic Information
reliability = models.ForeignKey(
TransmitterReliability,
on_delete=models.CASCADE,
verbose_name=_('reliability'),
related_name='transmitters',
default=12 # ID of 'Unknown' reliability
)
madhhab = models.CharField(
max_length=20,
choices=MadhhabChoices.choices,
default=MadhhabChoices.UNKNOWN,
verbose_name=_('Madhhab/School of Thought')
)
# Presence in Famous Collections
in_sahih_muslim = models.BooleanField(
default=False,
verbose_name=_('In Sahih Muslim'),
help_text=_('Is this narrator present in Sahih Muslim?')
)
in_sahih_bukhari = models.BooleanField(
default=False,
verbose_name=_('In Sahih Bukhari'),
help_text=_('Is this narrator present in Sahih Bukhari?')
)
# Additional Information
description = models.JSONField(default = list , verbose_name=_('Description'))
thumbnail = models.ImageField(upload_to='hadis/transmitter_thumbnails/', null=True, blank=True, help_text=_('image allowed'))
created_at = models.DateTimeField(auto_now_add=True, verbose_name=_('created at'))
updated_at = models.DateTimeField(auto_now=True, verbose_name=_('updated at'))
class Meta:
indexes = [
# For ordering in sync API
models.Index(fields=['id']),
]
verbose_name = _('Transmitter')
verbose_name_plural = _('Transmitters')
ordering = ('full_name',)
@property
def share_link(self):
if self.slug:
return f"{settings.DOVODI_DOMAIN}/arguments/narrators/{self.slug}"
return None
def save(self, *args, **kwargs):
if not self.slug or (isinstance(self.slug, str) and self.slug.strip() == ''):
# Try to get text from full_name field with robust error handling
try:
if self.full_name and isinstance(self.full_name, list) and len(self.full_name) > 0:
first_item = self.full_name[0]
if isinstance(first_item, dict):
name_text = first_item.get('text', '').strip()
if name_text:
base_slug = slugify(name_text, allow_unicode=True)
slug = base_slug
counter = 1
while Transmitters.objects.filter(slug=slug).exclude(pk=self.pk).exists():
slug = f"{base_slug}-{counter}"
counter += 1
self.slug = slug
else:
# Fallback if text is empty
import time
suffix = int(time.time() * 1000) % 1000000
base_slug = f"transmitter-{suffix}"
counter = 1
while Transmitters.objects.filter(slug=base_slug).exclude(pk=self.pk).exists():
base_slug = f"transmitter-{suffix}-{counter}"
counter += 1
self.slug = base_slug
else:
# Fallback if structure is invalid
import time
suffix = int(time.time() * 1000) % 1000000
base_slug = f"transmitter-{suffix}"
counter = 1
while Transmitters.objects.filter(slug=base_slug).exclude(pk=self.pk).exists():
base_slug = f"transmitter-{suffix}-{counter}"
counter += 1
self.slug = base_slug
else:
# Fallback if full_name is empty or invalid
import time
suffix = int(time.time() * 1000) % 1000000
base_slug = f"transmitter-{suffix}"
counter = 1
while Transmitters.objects.filter(slug=base_slug).exclude(pk=self.pk).exists():
base_slug = f"transmitter-{suffix}-{counter}"
counter += 1
self.slug = base_slug
except (IndexError, KeyError, AttributeError, TypeError):
# Fallback on any error
import time
suffix = int(time.time() * 1000) % 1000000
base_slug = f"transmitter-{suffix}"
counter = 1
while Transmitters.objects.filter(slug=base_slug).exclude(pk=self.pk).exists():
base_slug = f"transmitter-{suffix}-{counter}"
counter += 1
self.slug = base_slug
super().save(*args, **kwargs)
def _get_json_field(self, field_name: str, lang: Optional[str]=None , fallback: str = "en"):
"""
Generic getter for JSONField in our [{text, language_code}] format.
Usage: self._get_json_field('title', 'fa')
"""
if lang is None:
lang = fallback
value = getattr(self, field_name, None)
if not value or not isinstance(value, list):
return None
# 1) exact language
for item in value:
if isinstance(item, dict) and item.get("language_code") == lang:
return item.get("text", "")
# 2) fallback language
if fallback and fallback != lang:
for item in value:
if isinstance(item, dict) and item.get("language_code") == fallback:
return item.get("text", "")
# 3) first available
item = value[0]
print(item)
return item.get("text", "") if isinstance(item, dict) else None
def get_full_name(self, lang):
return self._get_json_field("full_name" , lang)
def get_kunya(self, lang):
return self._get_json_field("kunya" , lang)
def get_nickname(self, lang):
return self._get_json_field("nickname" , lang)
def get_origin(self, lang):
return self._get_json_field("origin" , lang)
def get_lived_in(self,lang):
return self._get_json_field("lived_in" , lang)
def get_died_in(self, lang):
return self._get_json_field("died_in" , lang)
def get_description(self, lang):
return self._get_json_field("description" , lang)
def __str__(self):
name = self.full_name[0]
return name.get('text')
class HadisTransmitter(models.Model):
hadis = models.ForeignKey(
"hadis.Hadis",
on_delete=models.CASCADE,
verbose_name=_('hadis'),
related_name='transmitters'
)
transmitter = models.ForeignKey(
Transmitters,
on_delete=models.CASCADE,
verbose_name=_('transmitter'),
related_name='hadises'
)
narrator_layer = models.ForeignKey(
NarratorLayer,
on_delete=models.SET_NULL,
verbose_name=_('narrator layer'),
related_name='transmitters',
null=True,
blank=True,
help_text=_('The layer/class (Tabaqah) this narrator belongs to')
)
status = models.ForeignKey(
TransmitterReliability,
on_delete=models.SET_NULL,
verbose_name=_('reliability status'),
related_name='hadis_transmitters',
null=True,
blank=True,
help_text=_('Reliability status of the narrator')
)
order = models.PositiveIntegerField(
default=0,
verbose_name=_('Order'),
help_text=_('Order in the chain of transmission')
)
is_gap = models.BooleanField(default=False, verbose_name=_('is gap'))
created_at = models.DateTimeField(auto_now_add=True, verbose_name=_('created at'))
class Meta:
indexes = [
# Speeds up fetching transmitters for a specific hadis in order
models.Index(fields=['hadis', 'order']),
]
verbose_name = _('Hadis Transmitter')
verbose_name_plural = _('Hadis Transmitters')
ordering = ('hadis', 'order')
unique_together = ('hadis', 'transmitter', 'order')
def __str__(self):
layer_info = f" - {self.narrator_layer}" if self.narrator_layer else ""
return f'{self.hadis.number} - {self.transmitter.full_name} ({self.order}){layer_info}'
class OpinionStatus(models.Model):
class ColorChoices(models.TextChoices):
RED = 'red', _('Red')
GREEN = 'green', _('Green')
BLUE = 'blue', _('Blue')
YELLOW = 'yellow', _('Yellow')
ORANGE = 'orange', _('Orange')
PURPLE = 'purple', _('Purple')
GRAY = 'gray', _('Gray')
title = models.JSONField(default = list , verbose_name=_('Title'))
slug = models.SlugField(max_length=255, verbose_name=_('slug'), blank=True,null=True)
color = models.CharField(max_length=20, choices=ColorChoices.choices, verbose_name=_('color'))
def save(self, *args, **kwargs):
if not self.slug or (isinstance(self.slug, str) and self.slug.strip() == ''):
# Try to get text from title field with robust error handling
try:
if self.title and isinstance(self.title, list) and len(self.title) > 0:
first_item = self.title[0]
if isinstance(first_item, dict):
text = first_item.get('text', '').strip()
if text:
slug = slugify(text)
# Ensure uniqueness
counter = 1
base_slug = slug
while OpinionStatus.objects.filter(slug=slug).exclude(pk=self.pk).exists():
slug = f"{base_slug}-{counter}"
counter += 1
self.slug = slug
else:
# Fallback to a timestamp-based slug
import time
suffix = int(time.time() * 1000) % 1000000
base_slug = f"opinion-{suffix}"
counter = 1
while OpinionStatus.objects.filter(slug=base_slug).exclude(pk=self.pk).exists():
base_slug = f"opinion-{suffix}-{counter}"
counter += 1
self.slug = base_slug
else:
import time
suffix = int(time.time() * 1000) % 1000000
base_slug = f"opinion-{suffix}"
counter = 1
while OpinionStatus.objects.filter(slug=base_slug).exclude(pk=self.pk).exists():
base_slug = f"opinion-{suffix}-{counter}"
counter += 1
self.slug = base_slug
else:
import time
suffix = int(time.time() * 1000) % 1000000
base_slug = f"opinion-{suffix}"
counter = 1
while OpinionStatus.objects.filter(slug=base_slug).exclude(pk=self.pk).exists():
base_slug = f"opinion-{suffix}-{counter}"
counter += 1
self.slug = base_slug
except (IndexError, KeyError, AttributeError, TypeError):
import time
suffix = int(time.time() * 1000) % 1000000
base_slug = f"opinion-{suffix}"
counter = 1
while OpinionStatus.objects.filter(slug=base_slug).exclude(pk=self.pk).exists():
base_slug = f"opinion-{suffix}-{counter}"
counter += 1
self.slug = base_slug
super().save(*args, **kwargs)
def __str__(self):
return self.title[0]['text']
def get_title(self,lang):
"""
Get title for a specific language
"""
if not self.title or not isinstance(self.title, list):
return None
for tr in self.title:
if isinstance(tr, dict) and tr.get('language_code') == lang:
return tr.get('text', '')
for tr in self.title:
if isinstance(tr, dict) and tr.get('language_code') == 'en':
return tr.get('text', '')
return None
class Meta:
verbose_name = _('Opinion Status')
verbose_name_plural = _('Opinion Statuses')
class TransmitterOpinion(models.Model):
"""
Model for scholarly opinions about transmitters
"""
# class OpinionStatus(models.TextChoices):
# CONFIRMED = 'confirmed', _('Confirmed')
# MIXED = 'mixed', _('Mixed')
# REJECTED = 'rejected', _('Rejected')
transmitter = models.ForeignKey(
Transmitters,
on_delete=models.CASCADE,
verbose_name=_('transmitter'),
related_name='opinions'
)
scholar_name = models.JSONField(default = list , verbose_name=_('Scholar Name'))
opinion_text = models.JSONField(default = list , verbose_name=_('Opinion Text'))
status = models.ForeignKey(
OpinionStatus,
on_delete=models.CASCADE,
verbose_name=_('opinion status'),
related_name='opinions',
# default=1,
blank = True,
null=True
)
created_at = models.DateTimeField(auto_now_add=True, verbose_name=_('created at'))
updated_at = models.DateTimeField(auto_now=True, verbose_name=_('updated at'))
class Meta:
indexes = [
# For filtering by transmitter + ordering
models.Index(fields=['transmitter']),
]
verbose_name = _('Transmitter Opinion')
verbose_name_plural = _('Transmitter Opinions')
ordering = ('-created_at',)
def __str__(self):
return f"{self.scholar_name[0]['text']}'s opinion on {self.transmitter.full_name[0]['text']} ({self.status})"
def get_scholar_name(self,lang):
"""
Get title for a specific language
"""
if not self.scholar_name or not isinstance(self.scholar_name, list):
return None
for tr in self.scholar_name:
if isinstance(tr, dict) and tr.get('language_code') == lang:
return tr.get('text', '')
for tr in self.scholar_name:
if isinstance(tr, dict) and tr.get('language_code') == 'en':
return tr.get('text', '')
return None
def get_opinion_tex(self,lang):
"""
Get title for a specific language
"""
if not self.opinion_text or not isinstance(self.opinion_text, list):
return None
for tr in self.opinion_text:
if isinstance(tr, dict) and tr.get('language_code') == lang:
return tr.get('text', '')
for tr in self.opinion_text:
if isinstance(tr, dict) and tr.get('language_code') == 'en':
return tr.get('text', '')
return None
class TransmitterOriginalText(models.Model):
transmitter = models.ForeignKey(
Transmitters,
on_delete=models.CASCADE,
verbose_name=_('transmitter'),
related_name='originaltexts'
)
slug = models.SlugField(max_length=255, verbose_name=_('slug'), blank=True,unique=True)
title = models.JSONField(default = list , verbose_name=_('Title'))
text = models.JSONField(default = list , verbose_name=_('Text'))
translation = models.JSONField(verbose_name=_('translation'), default=list)
share_link = models.CharField(max_length=255, verbose_name=_('share link'), null=True, blank=True)
embedded_in = models.JSONField(default=list, blank=True)
class Meta:
indexes = [
# For filtering by transmitter + ordering
models.Index(fields=['transmitter']),
]
verbose_name = _('Transmitter Original Text')
verbose_name_plural = _('Transmitter Original Text')
def __str__(self):
return f"{self.title[0]['text']} by {self.transmitter.full_name[0]['text']}"
def save(self, *args, **kwargs):
"""
Override save to automatically generate smart slugs.
"""
# Generate slug if not already set
if not self.slug and self.title:
# Extract title text
title_text = None
if isinstance(self.title, list) and self.title:
first_item = self.title[0]
if isinstance(first_item, dict):
title_text = first_item.get("text")
# Generate smart slug
if title_text:
self.slug = generate_smart_slug(
text=title_text,
model_class=TransmitterOriginalText,
max_length=100, # ← Adjust max length here
keep_words=8, # ← Limit to 8 words (your requirement)
instance=self,
)
else:
# Fallback if title is empty - use timestamp for uniqueness
import time
suffix = int(time.time() * 1000) % 1000000
transmitter_slug = self.transmitter.slug if self.transmitter and self.transmitter.slug else 'unknown'
base_slug = f"original-text-{transmitter_slug}-{suffix}"
# Ensure uniqueness
counter = 1
while TransmitterOriginalText.objects.filter(slug=base_slug).exclude(pk=self.pk).exists():
base_slug = f"original-text-{transmitter_slug}-{suffix}-{counter}"
counter += 1
self.slug = base_slug
# Generate/update share_link before saving
if self.slug and self.transmitter and self.transmitter.slug:
self.share_link = f"{settings.DOVODI_DOMAIN}/arguments/narrators/{self.transmitter.slug}/original-texts/{self.slug}"
# Reset embedded_in if text or translation changes
if self.pk:
old_instance = TransmitterOriginalText.objects.get(pk=self.pk)
if (old_instance.text != self.text or
old_instance.translation != self.translation):
self.embedded_in = [] # Reset!
super().save(*args, **kwargs)
def get_title(self,lang):
"""
Get title for a specific language
"""
if not self.title or not isinstance(self.title, list):
return None
for tr in self.title:
if isinstance(tr, dict) and tr.get('language_code') == lang:
return tr.get('text', '')
for tr in self.title:
if isinstance(tr, dict) and tr.get('language_code') == 'en':
return tr.get('text', '')
return None
def get_opinion_tex(self,lang):
"""
Get title for a specific language
"""
if not self.opinion_text or not isinstance(self.opinion_text, list):
return None
for tr in self.opinion_text:
if isinstance(tr, dict) and tr.get('language_code') == lang:
return tr.get('text', '')
for tr in self.opinion_text:
if isinstance(tr, dict) and tr.get('language_code') == 'en':
return tr.get('text', '')
return None