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

267 lines
7.5 KiB
Python

import logging
import uuid
from django.contrib.auth.models import AbstractUser
from django.db import models
logger = logging.getLogger(__name__)
SITE_CHOICES = (
("5HT2C", "5-HT2C"),
("5HT2A", "5-HT2A"),
("GABAB", "GABAB"),
("NMDA", "NMDA"),
)
MECHANISM_CHOICES = (
("AGONISM", "Agonism"),
("ANTAGONISM", "Antagomism"),
("MODULATION", "Modulation"),
)
SOURCE_TYPE_CHOICES = (
("PSITE", "Professional pharmaceutical data repository"),
("DWIKI", "Dedicated peer-reviewed community wiki"),
("CWIKI", "Peer-reviewed community wiki"),
("WIKI", "Private wiki"),
("DFORUM", "Dedicated community forum"),
("FORUM", "Community forum"),
)
DOSAGE_UNIT_CHOICES = (
("mg", "mg"),
("g", "g"),
("ug", "ug"),
)
DOSAGE_TIMING_CHOICES = (
("SECONDS", "Seconds"),
("MINUTES", "Minutes"),
("HOURS", "Hours"),
("DAYS", "Days :D"),
("WEEKS", "Weeks :O"),
("MONTHS", "Months :-|"),
("YEARS", "Years x_X"),
)
SEI_TYPE_CHOICES = (
("PHYSICAL", "Physical"),
("COGNITIVE", "Cognitive"),
("VISUAL", "Visual"),
("AUDITORY", "Auditory"),
("MULTISENSORY", "Multi-sensory"),
("TRANSPERSONAL", "Transpersonal"),
)
class User(AbstractUser):
# Stripe customer ID
stripe_id = models.CharField(max_length=255, null=True, blank=True)
customer_id = models.UUIDField(default=uuid.uuid4, null=True, blank=True)
billing_provider_id = models.CharField(max_length=255, null=True, blank=True)
email = models.EmailField(unique=True)
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
self._original = self
def get_notification_settings(self):
return NotificationSettings.objects.get_or_create(user=self)[0]
class NotificationSettings(models.Model):
user = models.OneToOneField(User, on_delete=models.CASCADE)
ntfy_topic = models.CharField(max_length=255, null=True, blank=True)
ntfy_url = models.CharField(max_length=255, null=True, blank=True)
def __str__(self):
return f"Notification settings for {self.user}"
class Source(models.Model):
"""
A source of information. Any information.
"""
name = models.CharField(max_length=255, unique=True)
type = models.CharField(max_length=255, choices=SOURCE_TYPE_CHOICES)
# Base endpoint for automated crawler processes
endpoint = models.CharField(max_length=1024, blank=True, null=True)
# Score, affects ordering
score = models.IntegerField(blank=True)
class Entry(models.Model):
"""
A snippet of information located on a Source.
Used to gather conflicting information and store it coherently.
"""
source = models.ForeignKey(Source)
# Slug of the article on the Source
slug = models.CharField(max_length=1024, null=True, blank=True)
# Extra information can be added
description = models.CharField(max_length=1024, null=True, blank=True)
class Dosage(models.Model):
"""
Registers the correlation between dose and intensity.
Linked to Entry to analyse conflicting records.
"""
entry = models.ForeignKey("core.Entry")
# Unit of mass as drugs are diverse
# Dosages varying between micrograms and grams
unit = models.CharField(max_length=255, choices=DOSAGE_UNIT_CHOICES)
# I can no longer say I am sober, but it is slight
threshold = models.IntegerField()
# Light
light = models.IntegerField()
# Average dose for a user
common = models.IntegerField()
# Strong intensity, many sober activities may become impossible
strong = models.IntegerField()
# Highest intensity
heavy = models.IntegerField()
class Timing(models.Model):
"""
Registers the time between administration to various experience levels.
Linked to Entry to analyse conflicting records.
"""
entry = models.ForeignKey("core.Entry")
# Unit of time as drugs are diverse
# Half-lives varying between seconds and months
unit = models.CharField(
max_length=255, choices=DOSAGE_TIMING_CHOICES, default="HOURS"
)
# It has just now begin, I can no longer say I am sober
onset = models.IntegerField()
# The intensity is accelerating
comeup = models.IntegerField()
# The maximum intensity has been reached
# How long this state occurs
peak = models.IntegerField()
# How long it takes to get back to baseline
offset = models.IntegerField(null=True, blank=True)
total = models.IntegerField()
class SEI(models.Model):
"""
Subjective Effect Index from Psychonaut Wiki.
Registers a subjective effect in a category, allowing a description to be
specified.
"""
# PHYSICAL, COGNITIVE, etc
type = models.CharField(
max_length=255, choices=SEI_TYPE_CHOICES, default="PHYSICAL"
)
# WIP: consider euphoric, depressant, relaxant
# Specify how
description = models.CharField(max_length=4096, blank=True, null=True)
class Effect(models.Model):
"""
Registers subjective effects for a drug.
Linked to multiple subjective effect indexes all from the same entry.
Linked to Entry to analyse conflicting records.
"""
entry = models.ForeignKey("core.Entry")
# List of subjective effects, since they would likely be from the same entry
subjective_effects = models.ManyToManyField(SEI)
class Action(models.Model):
"""
Site action record for a drug.
Registers a certain kind of action from a list of receptor sites with a given
affinity.
Linked to Entry to analyse conflicting records.
"""
entry = models.ForeignKey("core.Entry")
# Site - like 5HT2A for LSD
site = models.CharField(max_length=255, choices=SITE_CHOICES)
# Usually agonist or antagonist
mechanism = models.CharField(max_length=255, choices=MECHANISM_CHOICES)
# Free integer for binding affinity
affinity = models.IntegerField(blank=True)
class Drug(models.Model):
"""
Model of a drug. Not open to interpretation.
"""
# Lysergic acid diethylamide, Phenibut
name = models.CharField(max_length=255, unique=True)
# Psychedelic, Sedative
drug_class = models.CharField(max_length=255)
# LSD
common_name = models.CharField(max_length=1024, unique=True)
# Factsheets, posts
links = models.ManyToManyField(Entry)
# Dosages, how much to take to get a certain effect
dosages = models.ManyToManyField(Dosage)
# Timings, how long to wait to reach maximum intensity (and others)
timings = models.ManyToManyField(Timing)
# Effects, what does it do on a subjective level?
effects = models.ManyToManyField(Effect)
# Actions, what does it do on an objective level?
actions = models.ManyToManyField(Action)
# class Perms(models.Model):
# class Meta:
# permissions = (
# ("bypass_hashing", "Can bypass field hashing"), #
# ("bypass_blacklist", "Can bypass the blacklist"), #
# ("bypass_encryption", "Can bypass field encryption"), #
# ("bypass_obfuscation", "Can bypass field obfuscation"), #
# ("bypass_delay", "Can bypass data delay"), #
# ("bypass_randomisation", "Can bypass data randomisation"), #
# ("post_irc", "Can post to IRC"),
# ("post_discord", "Can post to Discord"),
# ("query_search", "Can search with query strings"), #
# ("use_insights", "Can use the Insights page"),
# ("index_int", "Can use the internal index"),
# ("index_meta", "Can use the meta index"),
# ("restricted_sources", "Can access restricted sources"),
# )