zip -r project.zip . -x "*.idea*" -x "*venv*" -x "*__pycache__*" tar.exe -a -cf project.zip main_app orm_skeleton caller.py manage.py requirements.txt tar.exe -a -cf project.zip main_app orm_skeleton caller.py manage.py requirements.txt; if (Test-Path project.zip) {echo "Archive created successfully - project.zip"} else {echo "Failed to create archive!"} python3 -m venv ./venv
source ./venv/bin/activate
venv\Scripts\activate docker exec -t <container_name> /bin/bash
psql -U your_user_me -d your_db_namepip install -r requirements.txtpip install --upgrade -r requirements.txtpip list --outdatedpip freezeORM - Object Relational Mapping
-
ORM - Предимства и недостатъци
- Pros:
- Не ни се налага писането на low level SQL
- По-лесна поддръжка
- Добър при CRUD операции
- Cons:
- Не много оптимизиран за по-сложни заявки
- Възможно е да влага излишна сложност в някои от заявките
- Pros:
-
Django models
- Всеки модел е отделна таблица
- Всяка променлова използваща поле от
modelsе колона в тази таблица
-
Създаване на модели
- Наследяваме
models.Model
- Наследяваме
-
Migrations
makemigrations- създава миграцииmigrate- прилага миграциите
-
Други команди
dbshell- отваря конзола, в която можем да пишем SQLCTRL + ALT + R- отваря manage.py console
-
Django Migrations Advanced
- Миграциите ни помагат да надграждаме промени в нашите модели
- Както и да можем да пазим предишни стейтове на нашата база
- Команди:
- makemigrations
- migrate
- Връщане до определена миграция - migrate main_app 0001
- Връщане на всички миграции - migrate main_app zero
- showmigrations - показва всички апове и миграциите, които имат
- showmigrations app_name - показва миграциите за един app
- showmigrations --list - showmigrations -l
- squashmigrations app_name migration_to_which_you_want_to_sqash - събира миграциите до определена миграция в една миграция
- sqlmigrate app_name migration_name - дава ни SQL-а на текущата миграция - използваме го, за да проверим дали миграцията е валидна
- makemigrations --empty main_app - прави празна миграция в зададен от нас app
-
Custom/Data migrations
-
Когато например добавим ново поле, искаме да го попълним с данни на база на вече съществуващи полета, използваме data migrations
-
RunPython
- викайки функция през него получаваме достъп до всички апове и техните модели (първи параметър), Scheme Editor (втори параметър)
- добра практика е да подаваме фунцкия и reverse функция, за да можем да връщаме безпроблемно миграции
-
Scheme Editor - клас, който превръща нашия пайтън код в SQL, ползваме го когато правим create, alter и delete на таблица
- използвайки RunPython в 95% от случаите няма да ни се наложи да ползавме Scheme Editor, освен, ако не правим някаква временна таблица индекси или промяна на схемата на таблицата
-
Стъпки:
2.1. Създаваме празен файл за миграция: makemigrations --empty main_app - прави празна миграция в зададен от нас app
2.2. Дефиниране на операции - Използваме RunPython за да изпълним data migrations
2.3. Прилагане на промените - migrate
-
Пример с временна таблица:
Да приемем, че имате модел с име „Person“ във вашето Django приложение и искате да създадете временна таблица, за да съхранявате някои изчислени данни въз основа на съществуващите данни в таблицата „Person“. В този случай можете да използвате мигриране на данни, за да извършите тази операция:
- Create the Data Migration:
Run the following command to create a data migration:
python manage.py makemigrations your_app_name --emptyThis will create an empty data migration file.
- Edit the Data Migration:
Open the generated data migration file and modify it to use RunPython with a custom Python function that utilizes the SchemaEditor to create a temporary table. Here's an example:
from django.db import migrations, models
def create_temporary_table(apps, schema_editor):
# Get the model class
Person = apps.get_model('your_app_name', 'Person')
# Access the SchemaEditor to create a temporary table
schema_editor.execute(
"CREATE TEMPORARY TABLE temp_person_data AS SELECT id, first_name, last_name FROM your_app_name_person"
)
def reverse_create_temporary_table(apps, schema_editor):
schema_editor.execute("DROP TABLE temp_person_data")
class Migration(migrations.Migration):
dependencies = [
('your_app_name', 'previous_migration'),
]
operations = [
migrations.RunPython(create_temporary_table, reverse_create_temporary_table),
]-
Django admin
- createsuperuser
- Register model, example:
@admin.register(OurModelName) class OurModelNameAdmin(admin.ModelAdmin): pass
-
Admin site customizations
-
str метод в модела, за да го визуализираме в админ панела по-достъпно
-
list_display - Показваме различни полета още в админа Пример:
class EmployeeAdmin(admin.ModelAdmin): list_display = ['job_title', 'first_name', 'email_address']
-
List filter - добавя страничен панел с готови филтри Пример:
class EmployeeAdmin(admin.ModelAdmin): list_filter = ['job_level']
-
Searched fields - казваме, в кои полета разрешаваме да се търси, по дефолт са всички Пример:
class EmployeeAdmin(admin.ModelAdmin): search_fields = ['email_address']
-
Layout changes - избираме, кои полета как и дали да се появяват при добавяне или промяна на запис Пример:
class EmployeeAdmin(admin.ModelAdmin): fields = [('first_name', 'last_name'), 'email_address']
-
list_per_page
-
fieldsets - променяме визуално показването на полетата Пример:
fieldsets = ( ('Personal info', {'fields': (...)}), ('Advanced options', {'classes': ('collapse',), 'fields': (...),}), )
-
CRUD overview
- CRUD - Create, Read, Update, Delete
- Използваме го при:
- Web Development
- Database Management
- Дава ни един консистентен начин, за това ние да създаваме фунцкионалност за CRUD
- Можем да го правим през ORM-a на Джанго
-
Мениджър в Django:
- Атрибут на ниво клас на модел за взаимодействия с база данни.
- Отговорен за CRUD
- Custom Manager: Подклас models.Model.
- Защо персонализирани мениджъри:
- Капсулиране на общи или сложни заявки.
- Подобрена четимост на кода.
- Избягвайме повторенията и подобряваме повторната употреба.
- Промяна наборите от заявки според нуждите.
- Защо персонализирани мениджъри:
-
Django Queryset
-
QuerySet - клас в пайтън, които изпозваме, за да пазим данните от дадена заявка
-
Данните не се взимат, докато не бъдат потърсени от нас
-
cars = Cars.objects.all() # <QuerySet []>
-
print(cars) # <QuerySet [Car object(1)]>
-
QuerySet Features:
- Lazy Evaluation - примера с колите, заявката не се вика, докато данните не потрябват
- Retrieving objects - можем да вземаме всички обекти или по даден критерии
- Chaining filters - MyModel.objects.filter(category='electronics').filter(price__lt=1000)
- query related objects - позволява ни да търсим в таблици, с които имаме релации, през модела: # Query related objects using double underscores related_objects = Order.objects.filter(customer__age__gte=18)
- Ordering - ordered_objects = Product.objects.order_by('-price')
- Pagination
from django.core.paginator import Paginator # Paginate queryset with 10 objects per page paginator = Paginator(queryset, per_page=10) page_number = 2 print([x for x in paginator.get_page(2)])
-
-
Django Simple Queries
- Object Manager - default Objects
- Methods:
all()first()get(**kwargs)create(**kwargs)filter(**kwargs)order_by(*fields)delete()
-
Django Shell and SQL Logging
- Django Shell
- Дава ни достъп до целия проект
- python manage.py shell
- SQL logging
- Enable SQL logging
- Добавя се в settings.py
- Django Shell
LOGGING = {
'version': 1,
'disable_existing_loggers': False,
'handlers': {
'console': {
'class': 'logging.StreamHandler',
},
},
'root': {
'handlers': ['console'],
'level': 'DEBUG', # Other levels CRITICAL, ERROR, WARNING, INFO, DEBUG
},
'loggers': {
'django.db.backends': { # responsible for the sql logs
'handlers': ['console'],
'level': 'DEBUG',
'propagate': False,
},
},
}- Без добавен SQL Logging можем да видим заявката чрез .query:
print(table_name.objects.all().query)-
Useful Methods
- filter() - връща subset от обекти; приема kwargs; връща queryset;
- exclude() - връща subset от обекти; приема kwargs; връща queryset;
- order_by() - връща сортираните обекти; - за desc;
- count() - като len, но по-бързо; count връща само бройката без да му трябвата реалните обекти;
- get() - взима един обект по даден критерии;
-
Chaning methods
- всеки метод работи с върнатия от предишния резултат
-
Lookup keys
-
Използват се във filter, exclude, get:
-
__exact (case sensitive) | __iexact (case insensitive) - матчва точно
--> SQL Equivalent: WHERE field LIKE 'value'; -
__contains (case sensitive) | __icontains (case insensitive) - проверява дали съдържа
--> SQL Equivalent: WHERE field = '%value%'; -
__startswith | __endswith
--> SQL Equivalent: WHERE field = 'value%';
--> SQL Equivalent: WHERE field = '%value'; -
__gt __gte
--> SQL Equivalent: WHERE field > INT;
--> SQL Equivalent: WHERE field >= INT; -
__lt __lte
--> SQL Equivalent: WHERE field < INT;
--> SQL Equivalent: WHERE field <= INT; -
__range=(X, Y) - both inclusive
--> SQL Equivalent: WHERE field BETWEEN X AND Y;
-
-
Bulk methods
- използват се, за да извършим операции върху много обекти едновременно
- bulk_create - създава множество обекти наведнъж;
- filter().update()
- filter().delete()
Django Models Relations
-
Database Normalization
-
Efficient Database Organization
- Data normalization - разбива големи таблици на по-малки такива, правейки данните по-организирани
- Пример: Все едно имаме онлайн магазин и вместо да пазим име, адрес и поръчка в една таблица, можем да разбием на 3 таблици и така да не повтаряме записи
-
Guidelines and Rules
-
First Normal Form (1NF):
- елеминираме поврарящите се записи, всяка таблица пази уникални стойности
-
Second Normal Form (2NF): извършваме първото като го правим зависимо на PK
- Пример: Онлайн магазин с данни и покупки Customers и Orders са свързани с PK, вместо всичко да е в една таблица
-
Third Normal Form (3NF):
- премахване на преходни зависимости
- Таблица служители пази id, служител, град, адрес => разделяме ги на 3 таблици и ги навързваме, без да е задължително по PK, може и по city_id вече employee е независимо
-
Boyce-Codd Normal Form (BCNF):
- По-строга версия на 3NF
- Тук правим да се навързват по PK
-
Fourth Normal Form (4NF):
- Ако данни от една таблица се използват в други две то това не е добре
- Пример: Имаме Курс X и Курс Y, на X Му трябват книгите A и B, на Y, A и C, това, което правим е да направим таблица с книгите А и таблица с Книгите Б
-
Fifth Normal Form (5NF) - Project-Join Normal Form or PJ/NF:
- Кратко казано да не ни се налага да минаваме през таблици с данни, които не ни трябват, за да достигнем до таблица с данни, която ни трябва
-
-
Database Schema Design
- Създаването на различни ключове и връзки между таблиците
-
Minimizing Data Redundancy
- Чрез разбиването на таблици бихме имали отново намалено повтаряне на информация
- Имаме книга и копия, копията са в отделна таблица, и са линкнати към оригинала
-
Ensuring Data Integrity & Eliminating Data Anomalies
- Това ни помага да update-ваме и изтриваме данните навсякъде еднакво
- отново благодарение на някакви constraints можем да променим една стойност в една таблица и тя да се отрази във всички
-
Efficiency and Maintainability
- Благодарение на по-малките таблици, ги query–ваме и update-ваме по-бързо
-
-
Релации в Django Модели
-
Получават се използвайки ForeignKey полета
-
related_name - можем да направим обартна връзка
- По дефолт тя е името + _set
-
Пример:
class Author(models.Model): name = models.CharField(max_length=100) class Post(models.Model): title = models.CharField(max_length=200) content = models.TextField() author = models.ForeignKey(Author, on_delete=models.CASCADE)
-
- Access all posts written by an author
author = Author.objects.get(id=1)
author_posts = author.post_set.all()- N.B.! _set работи само, ако няма предефинирано в модела related_name=
class Post(models.Model): title = models.CharField(max_length=200) content = models.TextField() author = models.ForeignKey(Author, on_delete=models.CASCADE, related_name='posts')
-
Types of relationships
-
Many-To-One (One-To-Many)
-
Many-To-Many
- Няма значение, в кой модел се слага
- Django автоматично създава join таблица или още наричана junction
- Но, ако искаме и ние можем да си създадем:
class Author(models.Model): name = models.CharField(max_length=100) class Book(models.Model): title = models.CharField(max_length=200) authors = models.ManyToManyField(Author, through='AuthorBook') class AuthorBook(models.Model): author = models.ForeignKey(Author, on_delete=models.CASCADE) book = models.ForeignKey(Book, on_delete=models.CASCADE) publication_date = models.DateField()
-
OneToOne, предимно се слага на PK
-
Self-referential Foreign Key
- Пример имаме работници и те могат да са мениджъри на други работници
class Employee(models.Model): name = models.CharField(max_length=100) supervisor = models.ForeignKey('self', on_delete=models.CASCADE, null=True, blank=True)
- Lazy Relationships - обекта от релацията се взима, чрез заявка, чак когато бъде повикан
-
-
Типове наследяване
- Multi-table
- Разширяваме модел с полетата от друг модел, като не копираме самите полета, а използваме създадения от django pointer, който прави One-To-One Relationship
- Пример:
class Person(models.Model): name = models.CharField(max_length=100) date_of_birth = models.DateField() def is_student(self): """Check if this person is also a student.""" return hasattr(self, 'student') class Student(Person): student_id = models.CharField(max_length=15) major = models.CharField(max_length=50)
-
Abstract Base Classes
- При това наследяване не се създават две нови таблици, а само една и тя е на наследяващия клас(Child), като абстрактния клас(Parent) е само шаблон
- Постигаме го чрез промяна на Meta класа:
class AbstractBaseModel(models.Model): common_field1 = models.CharField(max_length=100) common_field2 = models.DateField() def common_method(self): return "This is a common method" class Meta: abstract = True
-
Proxy Models
- Използваме ги, за да добавим функционалност към модел, който не можем да достъпим
- Можем да добавяме методи, но не и нови полета
- Пример:
class Article(models.Model): title = models.CharField(max_length=200) content = models.TextField() published_date = models.DateField() class RecentArticle(Article): class Meta: proxy = True def is_new(self): return self.published_date >= date.today() - timedelta(days=7) @classmethod def get_recent_articles(cls): return cls.objects.filter(published_date__gte=date.today() - timedelta(days=7))
- Multi-table
-
Основни Built-In Методи
save()- използва се за запазване на записи
def save(self, *args, **kwargs): # Check the price and set the is_discounted field if self.price < 5: self.is_discounted = True else: self.is_discounted = False # Call the "real" save() method super().save(*args, **kwargs)
clean()- използва се, когато искаме да валидираме логически няколко полета, например имаме тениска в 3 цвята, но ако е избран XXL цветовете са само 2.
-
Custom Model Properties
- Както и в ООП, можем чрез @property декоратора да правим нови атрибути, които в случая не се запазват в базата
- Използваме ги за динамични изчисления на стойностти
-
Custom Model Fields
- Ползваме ги когато, Django няма field, които ни върши работа
- Имаме методи като:
- from_db_value - извиква се, когато искаме да взмем стойността от базата в пайтън
- to_python - извиква се когато правим десериализация или clean
- get_prep_value - обратното на from_db_value, от Python към базата, предимно ползваме за сериализации
- pre_save - използва се за last minute changes, точно преди да запазим резултата в базата
class RGBColorField(models.TextField): # Convert the database format "R,G,B" to a Python tuple (R, G, B) def from_db_value(self, value, expression, connection): if value is None: return value return tuple(map(int, value.split(','))) # Convert any Python value to our desired format (tuple) def to_python(self, value): if isinstance(value, tuple) and len(value) == 3: return value if isinstance(value, str): return tuple(map(int, value.split(','))) raise ValidationError("Invalid RGB color format.") # Prepare the tuple format for database insertion def get_prep_value(self, value): # Convert tuple (R, G, B) to "R,G,B" for database storage return ','.join(map(str, value))
- Validation in Models
- Built-in Validators
- MaxValueValidator, MinValueValidator - приема два аргумета (limit, message)
- MaxLengthValidator, MinLengthValidator - приема два аргумета (limit, message)
- RegexValidator - приема два аргумета (regex, message)
class SampleModel(models.Model): name = models.CharField( max_length=50, validators=[MinLengthValidator(5)] # Name should have a minimum length of 5 characters ) age = models.IntegerField( validators=[MaxValueValidator(120)] # Assuming age shouldn't exceed 120 years ) phone = models.CharField( max_length=15, validators=[ RegexValidator( regex=r'^\+?1?\d{9,15}$', message="Phone number must be entered in the format: '+999999999'. Up to 15 digits allowed." )] # A simple regex for validating phone numbers )
- Built-in Validators
- Custom Validators - функции, които често пишем в отделен файл. При грешка raise-ваме ValidationError
-
Meta Options and Meta Inheritance
- В мета класа можем да променяме:
- Името на таблицата
- Подреждането на данните
- Можем да задаваме constraints
- Можем да задаваме типа на класа(proxy, abstract)
class SampleModel(models.Model): name = models.CharField(max_length=50) age = models.IntegerField() email = models.EmailField() class Meta: # Database table name db_table = 'custom_sample_model_table' # Default ordering (ascending by name) ordering = ['name'] - Случва се на SELECT, не на INSERT # Unique constraint (unique combination of name and email) unique_together = ['name', 'email']
- Meта наследяване:
- Ако наследим абстрактен клас и не презапишем мета класа, то наслеяваме мета класа на абстрактния клас
class BaseModel(models.Model): name = models.CharField(max_length=100) class Meta: abstract = True ordering = ['name'] class ChildModel(BaseModel): description = models.TextField() # ChildModel inherits the Meta options
- В мета класа можем да променяме:
-
Indexing
- Индексирането ни помага, подреждайки елементите в определен ред или създавайки друга структура, чрез, която да търсим по-бързо.
- Бързо взимаме записи, но ги запазваме по-бавно
- В Django можем да сложим индекс на поле, като добавим key-word аргумента db_index=True
- Можем да направим и индекс, чрез мета класа, като можем да правим и композитен индекс
class Meta: indexes=[ models.Index(fields=["title", "author"]), # прави търсенето по два критерия, по-бързо models.Index(fields=["publication_date"]) ]
-
Django Model Mixins
- Както знаем, миксините са класове, които използваме, за да отделим обща функционалност
class TimestampMixin(models.Model): created_at = models.DateTimeField(auto_now_add=True) updated_at = models.DateTimeField(auto_now=True) class Meta: abstract = True
-
Custom managers
- Използваме ги, за да изнсем бизнес логиката, за често използвани заявки на едно място
- Правим го наследявайки мениджъра по подразбиране.
class BookManager(models.Manager): def average_rating(self): # Calculate the average rating of all books return self.aggregate(avg_rating=models.Avg('rating'))['avg_rating'] def highest_rated_book(self): # Get the highest-rated book return self.order_by('-rating').first()
-
Annotations and Aggregations
- Анотации - използваме ги, за да добавяме нови полета във върнатия резултат, често на база някакви изчисления. Връща QuerySet.
- Пример:
# Annotating the queryset to get the count of books for each author authors_with_book_count = Book.objects.values('author').annotate(book_count=Count('id'))
- Агрегации - връщат едно поле(една стойност), често резултат от агрегиращи функции. Връща dict
# Annotating the queryset to get the average rating of all books average_rating = Book.objects.aggregate(avg_rating=Avg('rating'))
-
select_related & prefetch_related
- select_related - редуцира броя на заявките при One-To-One и Many-To-One заявки
- вместо lazily да взимаме свързаните обекти правим JOIN още при първата заявка
- Пример:
from django.db import models class Author(models.Model): name = models.CharField(max_length=100) class Book(models.Model): title = models.CharField(max_length=100) author = models.OneToOneField(Author, on_delete=models.CASCADE) books_with_authors = Book.objects.select_related('author') # SELECT * FROM "myapp_book" JOIN "myapp_author" ON ("myapp_book"."author_id" = "myapp_author"."id")
- prefetch_related - редуцира броя на заявките при Many-To-Many(не само) до броя на релациите + 1
- Пример:
class Author(models.Model): name = models.CharField(max_length=100) class Book(models.Model): title = models.CharField(max_length=100) authors = models.ManyToManyField(Author) authors_with_books = Author.objects.prefetch_related('book_set') # 1. SELECT * FROM "myapp_author" # 2. SELECT * FROM "myapp_book" INNER JOIN "myapp_book_authors" ON ("myapp_book"."id" = "myapp_book_authors"."book_id")
- select_related - редуцира броя на заявките при One-To-One и Many-To-One заявки
-
Q and F
-
Използваме Q object, за да правим заявки изискващи по-сложни условия
-
Пример:
q = Q(title__icontains='Django') & (Q(pub_year__gt=2010) | Q(author='John Doe')) books = Book.objects.filter(q)
-
Използваме F object, за да достъпваме, стойностите през, които итерираме на ниво SQL
from django.db.models import F Book.objects.update(rating=F('rating') + 1)
-
SQL Alchemy - ORM - Object Relational Mapper
- ORM - абстракция позволяваща ни да пишем SQL, чрез Python
- Core - грижи се за транзакциите, изпращането на заявки и database pooling
-
SetUp:
pip install sqlalchemypip install psycopg2
-
Модели
- Подобно на Django наследяваме базов клас,
Base, който взимаме като резултат от извикването наdeclarative_base()
from sqlalchemy.ext.declarative import declarative_base Base = declarative_base() class User(Base): __tablename__ = 'users' id = Column(Integer, primary_key=True) name = Column(String)
- Подобно на Django наследяваме базов клас,
-
Миграции
4.1 SetUp:
-
Не са включени в SQLAlchemy, за тях можем да използваме
Alembic -
pip install alembic -
alembic init alembic- създава ни файловата структура за миграциите<\br> -
sqlalchemy.url = postgresql+psycopg2://username:password@localhost/db_name- във файла alembic.ini -
py target_metadata = Base.metadata- във файла env.py, за да можем да поддържаме autogenerate
4.2 Команди:
-
alembic revision --autogenerate -m "Add User Table"- създава миграция със съобщени, кактоmakemigrations -
alembic upgrade head- прилага миграциите, кактоmigrate -
alembic downgrade -1- връща миграция
-
-
CRUD
- Отваряме връзка с базата, пускайки нова сесия
- Винаги затваряме сесията, след приключване на работа
- Трябва да комитнем резултата, подобно на Django, където ползвахме
save()
from sqlalchemy import create_engine from sqlalchemy.orm import sessionmaker engine = create_engine('sqlite:///example.db') Session = sessionmaker(bind=engine) session = Session() with Session() as session: # a good practice ...
5.1 Add:
new_user = User(username='john_doe', email='john@example.com') session.add(new_user)
5.2 Query
users = session.query(User).all()
5.3 Update
with engine.connect() as connection: # Create an update object upd = update(User).where(User.name == 'John').values(nickname='new_nickname') # Execute the update connection.execute(upd)
or
session.query(User).filter(User.name == 'John').update({"nickname": "new_nickname"}, synchronize_session=False) session.commit()
5.4 Delete ```py
del_stmt = delete(User).where(User.name == 'John') ``` -
Transactions
-
session.begin() -
session.commit() -
session.rollback()
-
-
Relationships
- Many to One
user_id = Column(Integer, ForeignKey('users.id')) user = relationship('User')
- One to One
uselist=false
class User(Base): __tablename__ = 'users' id = Column(Integer, primary_key=True) profile = relationship("UserProfile", back_populates="user", uselist=False) class UserProfile(Base): __tablename__ = 'profiles' id = Column(Integer, primary_key=True) user_id = Column(Integer, ForeignKey('users.id')) user = relationship("User", back_populates="profile")
- Many to many
user_group_association = Table('user_group', Base.metadata, Column('user_id', Integer, ForeignKey('users.id')), Column('group_id', Integer, ForeignKey('groups.id')) ) class Group(Base): __tablename__ = 'groups' id = Column(Integer, primary_key=True) users = relationship("User", secondary=user_group_association, back_populates="groups") class User(Base): __tablename__ = 'users' id = Column(Integer, primary_key=True) groups = relationship("Group", secondary=user_group_association, back_populates="users")
- Many to One
-
Database pooling
py engine = create_engine(DATABASE_URL, pool_size=10, max_overflow=20)- задава първоначални връзки и максимално създадени