Python: Django
Primjeri web sjedišta u Djangu:
- Instagram (velikim dijelom)
- RiTeh (shoutout: Zavod za računarstvo)
Postavljanje razvojnog okruženja web aplikacija na operacijskim sustavima sličnim Unixu i Windowsima. Stvaranje projekta i organizacija koda. Korištenje dokumentacije
Stvaranje prvog projekta
Stvaranje direktorija gdje ćemo stvoriti naš projekt:
$ mkdir moja_prva_stranica
(...)
Zatim idemo u taj direktorij naredbom:
$ cd moja_prva_stranica
(...)
U njemu stvaramo Django projekt naredbom:
$ django-admin startproject mysite
(...)
Ostale raspoložive naredbe možemo vidjeti s komandom django-admin
, a za detaljnije o pojedinim naredbama koristite django-admin help
, npr. za naredbu startproject
na način:
$ django-admin help startproject
(...)
Naredba startproject
kreira novi direktorij naziva kojega smo proslijedili, u našem slučaju to je mysite
.
Prebacimo se u direktorij mysite
pomoću naredbe cd mysite
i ispišimo sadržaj direktorija u obiliku stabla s naredbom tree mysite
, možete koristiti i naredbu ls mysite
.
$ tree mysite
mysite
├── manage.py
└── mysite
├── asgi.py
├── __init__.py
├── settings.py
├── urls.py
└── wsgi.py
Direktorij mysite
je središnji direktorij naše stranice koju ćemo graditi. Ovo je "glavna" tj. središnja aplikacija. Njena uloga je usmjeravanje Djanga na druge aplikacije koje ćemo kreirati. Ono što ćemo tu razvijati je samo usmjeravanje na druge aplikacije koje razvijamo i podešavanje postavki.
Projektni direktorij i njegov sadržaj
Sadržaj koji se nalazi u novo kreiranom projektu:
Izvan direktorija mysite/
: korijenski je direktorij koji ima ulogu mjesta gdje se nalaze sve datoteke i direktoriji projekta.
Unutar direktorija mysite/
: nalazi se Python package projekta, a njegov sadržaj je prikazan iznad.
Datoteka mysite/__init__.py
: Govori interpreteru da sve unutar ovoga direktorija bude tretirano kao Python package.
Datoteka mysite/settings.py
: Postavke/konfiguracija za Django projekt. Obratiti pažnju na INSTALLED_APPS
koji nam je trenutno najbitniji dio u ovoj datoteci koji će nam trebati. Svaki puta kada se dodaje nova aplikacija na našu web stranicu, moramo ju ovdje ručno dodavati.
Danger
Paziti na SECRET_KEY
da nije vidljiv svima, npr. na GitHubu-u. Zlonamjerni pojedinac može to iskoristiti i upravljati vašom stranicom kao administrator.
Datoteka mysite/urls.py
: Sadrži deklarirane URLove vezane za Django projekt, služi Django aplikaciji kao "kazalo sadržaja". Njegova uloga je 'kontroliranje' naše stranice/aplikacije. Pomoću njega 'pokazujemo/usmjeravamo' program na naše aplikacije.
Tip
Osvježite si znanje regularnih izraza prije nastavka.
Datoteke mysite/wsgi.py i mysite/asgi.py
: ASGI je nasljednik WSGI, dugogodišnjeg Python standarda za kompatibilnost između web poslužitelja, frameworka i aplikacija. WSGI povećava mogućnosti u radu na webu pomoću Pythona, dok je cilj ASGI-a produžiti to za područje asinkronog Pythona.
Datoteka manage.py
: Koristimo pri radu u terminalu za prosljeđivanje raznih naredbi koje prosljeđujemo programu da se izvrše nad projektom, više o tome u nastavku.
Kreiranje prve aplikacije
Započnimo s kreiranjem prve aplikacije. Za početak dodat ćemo aplikacije na naš projekt mysite
, to činimo naredbom: ./manage.py startapp main
. Primijetit ćete da u direktoriju mysite
je stvoren novi poddirektorij naziva main
.
Zadatak
Provjerite sadržaj direktorija mysite/mysite
i direktorija mysite/main
s naredbama tree
i ls
.
Lokalni poslužitelj za razvoj
Pokrenimo sada naš lokalni poslužitelj (eng.server) na kojemu ćemo razvijati aplikaciju.
Tip
Za pokretanje poslužitelja koristite zasebni terminal. Poslužitelj aktivno radi za vrijeme razvoja i nije potrebno ponovno pokretati poslužitelj nakon svake promjene.
U zasebnom terminalu za pokretanje poslužitelja koristi se naredba:
$ ./manage.py runserver
(...)
Zadatak
Provjerite rad poslužitelja posjetom adrese http://127.0.0.1:8000/
u web pregledniku.
Ovo je poslužitelj na kojem ćemo razvijati Django aplikacije.
U terminalu možemo vidjeti HTTP zahtjeve na poslužitelj i statusni kod odgovora na njih.
Zadatak
Provjerite ispis u terminalu prilikom osvježavanja stranice pritiskom na tipki Ctrl+R ili F5.
Dodatak: Čitanje dokumentacije i StackOverflow
Tko radi taj i griješi, a iz grešaka se najbolje uči.
Djangovu službenu dokumentaciju možete pronaći ovdje.
Stack Overflow je mjesto za pitanja i odgovore za profesionalne programere i entuzijaste. Sadrži pitanja i odgovore na širok raspon tema u računalnom programiranju. Više na Wikipediji.
Primjeri pitanja postavljenih na Stack Overflowu vezanih za probleme u Pythonu i Djangu:
Korištenje baze podataka. Stvaranje modela i objektno-relacijsko preslikavanje
Paradigma model-view-controller (MVC)
Django koristi paradigmu model-view-controller (MVC) koju naziva model-template-view (MTV). Svaka stranica kreirana pomoću Django frameworka vjerojatno koristi ove tri stvari kako bi vam prikazala podatke.
Iako ga nazivamo MVC (model-view-controller), način na koji radi ide u obrnutom slijedu. Korisnik će posjetiti URL, vaš kontroler (/urls.py
) ukazat će na određeni prikaz (/views.py
). Taj se prikaz tada može (ili ne mora) povezati s vašim modelima. Drugim riječima, prvo ide Controller, zatim View i na posljetku Model po potrebi.
Unutar Djanga koristi se djelomično izmijenjena terminologija. Stoga od sad nadalje koristit ćemo Django terminologiju (model, template i view). U tablici u nastavku detaljnije su objašnjeni pojmovi i termini koji se koriste.
Korišteni naziv | Django naziv | Značenje |
---|---|---|
Model | Model | Sloj modela u Djangu odnosi se na bazu podataka plus Python kôd koji je izravno koristi. Modelira stvarnost. Pohranjuje sve potrebne vrijednosti unutar baze podataka koje su potrebne web aplikaciji. Django vam omogućuje pisanje Python klasa koje nazivamo modeli, koje se vezuju za tablice baze podataka. Stvoreni modeli nisu trajno zadani, nego se mogu izmjenjivati i dopunjavati. Izmjene su dostupne odmah nakon primjene migracija o kojima detaljnije kasnije na ovim vježbama u poglavlju ORM. |
View | Template | View sloj u Djangu odnosi se na korisničko sučelje (UI). Funkcija pregleda ili skraćeno prikaz (eng. view) je Python funkcija koja je zadužena za generiranje HTMLa i ostalih UI elemenata. Pomoću Python kôda renderiraju se pogledi. Django uzima web zahtjev i vraća web odgovor. Ovaj odgovor može biti bilo što, npr. HTML sadržaj web stranice, preusmjeravanje, statusni kod, XML dokument ili slika... |
Controller | View | Središnji dio sustava, sadržava logiku koja povezuje cjeline da bi se pružio odgovor korisniku na traženi zahtjev. Upravlja zahtjevima i odgovorima na njih. uspostavlja vezu s bazom podataka i učitavanjem datoteka. |
Uvod u Django: Hello world
U nastavku ovog poglavlja prikazano je kako kreirati Django aplikaciju za teksta na početnoj stranici. Za ovo nije potreban Model nego samo View u kojemu je definiran tekst koji želimo prikazati na stranici.
Zadatak
Otvorite datoteku i provjerite što je zapisano u mysite/mysite/url.py
.
Za definiranje putanje na koju će Django primati HTTP zahtjeve iskoristit ćemo funkciju django.urls.path()
(dokumentacija) i usmjerit ćemo na administratorsko sučelje (dokumentacija).
Datoteka mysite/mysite/url.py
ima sadržaj
from django.contrib import admin
from django.urls import path
urlpatterns = [
path('admin/', admin.site.urls),
]
Vidimo da imamo samo jedan URL koji se tiče stranice za administraciju koji nam trenutno ne treba. Sljedeće što je potrebno učiniti je usmjeriti view na URL. Razlog tome je što Django web stranice vidi kao kolekciju aplikacija koje 'ispisuje' pomoću danih URLova koji 'pokazuju' Djangu gdje tražiti. Datoteka urls.py
unutar naše glavne aplikacije obično samo usmjerava na aplikacije.
Pa krenimo sa kreiranjem aplikacije, i to na način da ju kreiramo pomoću naredbe ./manage.py startapp main
.
Idemo sada usmjeriti naš glavni dio Django aplikacije da provjerava, odnosno poziva našu novostvorenu aplikaciju. To radimo unutar mysite/mysite/url.py
:
from django.contrib import admin
from django.urls import path, include #importamo include
urlpatterns = [
path('', include('main.urls')), #dodajemo urls
path('admin/', admin.site.urls),
]
Ako pogledamo u direktorij aplikacije main
vidjet ćemo da datoteka urls.py
ne postoji. Kreirajmo datoteku mysite/main/urls.py
i dodajmo sljedeći sadržaj.
from django.urls import path
from . import views
app_name = 'main' # here for namespacing of urls.
urlpatterns = [
path('', views.homepage, name='homepage'),
]
Prođimo ukratko kroz logiku rada našeg programa. Prvo se posjećuje mysite/url.py
, u URL-u nije prosljeđeno ništa, stoga odgovara ''
iz path('', include('main.urls'))
. Program ovo tumači tako da uključuje main.urls
.
Program zatim pronalazi i učitava main.urls
koji se nalazi na lokaciji mysite/main/urls.py
i čita njegov sadržaj. Upravo smo izmjenili sadržaj tako da smo dodali uzorak ''
koji odgovara path('', views.homepage, name='homepage')
. Ovime smo usmjerili aplikaciju da pokrene funkciju homepage()
unutar main/views.py
koji još nismo izmjenili stoga ćemo to sada učiniti rako da dodamo funkciju naziva homepage()
.
Datoteka mysite/main/views.py
ima sadržaj:
from django.shortcuts import render
from django.http import HttpResponse
## Create your views here.
def homepage(request):
return HttpResponse('Welcome to homepage! <strong>#samoOIRI</strong>')
# primjetiti korištenje HTML-a
Osvježimo stranicu http://127.0.0.1:8000/
u browseru.
Modeli: Uvod
Nastavak na program iz uvoda. U ovom poglavlju detaljnije je objasnjeno stvaranje modela u Djangu.
Stvoreni model se mapira na tablicu u bazi podataka. Baza poidataka je već stvorena i možete ju vidjeti na lokaciji mysite/mysite
. Po defaultu tip baze podataka je SQLite 3 pa je ekstenzija datoteke .sqlite3
. Tip baze podataka možete mijenjati u settings.py
pod DATABASES
.
Stvaranje modela i polja modela
Ovime stvaramo novu tablicu u bazi podataka a zadana polja postaju stupci u toj tablici. Automatski se po defaultu kreira primarni ključ iz toga, ali po želji možemo i proizvoljno odrediti da neka od zadanih vrijednosti to bude.
Zadatak
Otvorite main/models.py
i definirajte klasu imena Predmet. U klasi definirajte stupce naziva:predmet_naslov
, predmet_sadrzaj
i predmet_vrijeme_objave
. Njihovi tipovi neka budu CharField()
koji ima zadan parametar max_length
na 100
, TextField()
i DateTimeField()
koji ima naziv postavljen na 'date published'
.
Rješenje
U datoteku main/models.py
dodajemo:
class Predmet(models.Model):
predmet_naslov = models.CharField(max_length=100)
predmet_sadrzaj = models.TextField()
predmet_vrijeme_objave = models.DateTimeField('date published')
Zadatak
Definirajte funkciju __str__()
unutar klase Predmet
koja vraća naziv predmeta predmet_naziv
.
Rješenje
U datoteku main/models.py
unutar klase Predmet
dodajemo:
def __str__(self):
return self.predmet_naslov
Službena Django dokumentacija o svim poljima unutar modela.
Objektno-relacijsko preslikavanje
Svaki novi model je nova tablica u bazi podataka. Zbog toga moramo napraviti dvije stvari. Prva je pripremiti za migraciju naredbom makemigrations
, a zatim napraviti migraciju nredbom migrate
.
Pokrenimo naš lokalni server naredbom ./manage.py runserver
. Primijetite u outputu konzole sljedeću poruku You have 18 unapplied migration(s). Your project may not work properly until you apply the migrations for app(s): admin, auth, contenttypes, sessions Run './manage.py migrate' to apply them
.
Unesite naredbu kako biste pripremili migraciju.
$ ./manage.py makemigrations
No changes detected
Razlog tomu je što još nismo povezali main
aplikaciju. Način na koji ćemo povezati aplikaciju je taj da ju pozovemo unutar INSTALLED_APPS
unutar mysite/settings.py
.
Možemo vidjeti da je unutar main/apps.py
definirana kalsa MainConfig
.
Dopunimo sada mysite/settings.py
sa pozivom klase MainConfig
iz aplikacije main
. To radimo na način da pod INSTALLED_APPS
dodamo 'main.apps.MainConfig'
.
Datoteka mysite/mysite/settings.py
sad ima sadržaj:
## Application definition
INSTALLED_APPS = [
'django.contrib.admin',
'django.contrib.auth',
'django.contrib.contenttypes',
'django.contrib.sessions',
'django.contrib.messages',
'django.contrib.staticfiles',
'main.apps.MainConfig' # ovu liniju dodajemo
]
Unesite ponovo naredbu ./manage.py makemigrations
da biste pripremili migraciju.
$ ./manage.py makemigrations
Migrations for 'main':
main\migrations\0001_initial.py
- Create model Predmet
Zadatak
Pročitajte sadržaj unutar datoteke main\migrations\0001_initial.py
.
Zatim unesimo naredbu ./manage.py migrate
da bi se migracija izvršila.
Nakon svakog rada sa modelima, bile to izmjene ili stvaranje novih modela potrebno je napraviti migraciju na načn makemigrations
a zatim migrate
.
Pogledajmo output u terminalu koji nam vraća naredba ./manage.py migrate
:
$ ./manage.py migrate
Operations to perform:
Apply all migrations: admin, auth, contenttypes, sessions
Running migrations:
Applying contenttypes.0001_initial... OK
Applying auth.0001_initial... OK
Applying admin.0001_initial... OK
Applying admin.0002_logentry_remove_auto_add... OK
Applying admin.0003_logentry_add_action_flag_choices... OK
Applying contenttypes.0002_remove_content_type_name... OK
Applying auth.0002_alter_permission_name_max_length... OK
Applying auth.0003_alter_user_email_max_length... OK
Applying auth.0004_alter_user_username_opts... OK
Applying auth.0005_alter_user_last_login_null... OK
Applying auth.0006_require_contenttypes_0002... OK
Applying auth.0007_alter_validators_add_error_messages... OK
Applying auth.0008_alter_user_username_max_length... OK
Applying auth.0009_alter_user_last_name_max_length... OK
Applying auth.0010_alter_group_name_max_length... OK
Applying auth.0011_update_proxy_permissions... OK
Applying auth.0012_alter_user_first_name_max_length... OK
Applying sessions.0001_initial... OK
Interakcija sa modelom
Za interakciju sa modelom koristimo naredbu ./manage.py shell
koja pokreće Python shell.
Za početak uvezimo potrebne modele i pakete naredbama:
from main.models import Predmet
from django.utils import timezone
Za dohvaćanje svih objekata koristimo Predmet.objects.all()
što vraća QuerrySet []
ondosno praznu listu.
Dodajmo vrijednosti u novu instancu klase Predmet nazvia novi_predmet
pomoću:
novi_predmet = Predmet()
novi_predmet.predmet_naslov = 'DWA2'
novi_predmet.predmet_sadrzaj = 'ovo je opis predmeta'
novi_predmet.predmet_vrijeme_objave = timezone.now()
Pohranimo promjene:
novi_predmet.save()
Isprobajmo ponovno naredbu Predmet.objects.all()
i uočimo novi predmet u listi.
Kroz predmete se može i iterirati, primjerice korištenjem petlje for
:
for p in Predmet.objects.all():
print(p.predmet_naslov)
Kreiranje superusera i povezivanje modela
Administratora kreiramo naredbom:
$ ./manage.py createsuperuser
Username (leave blank to use 'korisnik'):
Email address:
Password:
Password (again):
Dodjelite proizvoljno ime i lozinku, mail adresu trenutno možemo ostaviti praznu.
Zadatak
Posjetite na lokalnom serveru adresu http://127.0.0.1:8000/admin/
.
Idemo sada nadopuniti kod i povezati stvoreni model pomoću mysite/main/admin.py
.
Sadržaj datoteke mysite/main/admin.py
je oblika:
from django.contrib import admin
from .models import Predmet
## Register your models here.
admin.site.register(Predmet)
Zadatak
Osvježite stranicu na adresi http://127.0.0.1:8000/admin/
.
Relacije među modelima. Upiti
Modeli: Relacije među modelima
Relacije su detaljno opisane u Djangovoj dokumnetaciji:
Many-to-many
Na prethodnim vježbama stvorili smo model koji sadržava klasu Predmet, dodajmo sada klasu Student koja će biti povezana s Predmetom relacijom ManyToMany.
Zadatak
- Model iz Vježbi2
main/models.py
nadopunite tako da stvorite novu klasuStudent
. Klasa student sadrži stupce nazivastudent_ime
,student_prezime
,student_broj_xice
istudent_prvi_upis_predmeta
. Tipovi podataka neka buduCharField()
zastudent_ime
samax_length
na25
,student_prezime
samax_length
na50
. Vrijednoststudent_broj_xice
postavite naCharField()
zastudent_ime
samax_length
na10
. - Dodajte vrijednost
student_predmeti
koja će biti povezan s klasomPredmet
, tip veze neka budeManyToMany
. Unutar klasePredmet
izmijenite vrijednostpredmet_vrijeme_objave
tako da joj postavite zadanu vrijednost natimezone.now
. Nakon kreirane klase pokrenite naredbemakemigrations
imigrate
.
Rješenje
Uredit ćemo datoteku main/models.py
tako da bude oblika:
from django.db import models
from django.utils import timezone
class Predmet(models.Model):
predmet_naslov = models.CharField(max_length=100)
predmet_sadrzaj = models.TextField()
predmet_vrijeme_objave = models.DateTimeField(default=timezone.now)
def __str__(self):
return self.predmet_naslov
class Student(models.Model):
student_ime = models.CharField(max_length=25)
student_prezime = models.CharField(max_length=50)
student_broj_xice = models.CharField(max_length=10)
student_predmeti = models.ManyToManyField(Predmet)
Zadatak
Definirajte funkciju __str__()
unutar klase Student koja vraća student_broj_xice
. Zatim dodajte klasu Student
unutar main/admin.py
tako da ona postane vidljiva u admin panelu.
Rješenje
U datoteci main/models.py
unutar klase Student
dodajemo:
def __str__(self):
return self.student_broj_xice
U datoteci main/admin.py
:
from django.contrib import admin
from .models import *
model_list = [Predmet, Student]
admin.site.register(model_list)
Nakon što ste nadopunili main/models.py
primjenite pripremu i nakon toga migraciju s naredbama makemigrations
i migrate
. Provjerite radi li vam sve tako da posjetite http://127.0.0.1:8000/admin/
.
Many-to-one
Zadatak
- Definirajte klasu
Profesor
koja sadrži vrijednosti,prof_ime
iprof_prezime
koji suCharField
duljine 30, zatim definirajteprof_email
koji je tipaEmailField
. - Unutar klase zadajte funkciju
__str__()
koja vraća email adresu od profesora. - Nakon kreirane klase pokrenite naredbe
makemigrations
imigrate
.
Rješenje
U datoteci main/models.py
:
class Profesor(models.Model):
prof_ime = models.CharField(max_length=30)
prof_prezime = models.CharField(max_length=30)
prof_email = models.EmailField()
def __str__(self):
return self.prof_email
Zadatak
- Izmijenite klasu
Predmet
tako da joj dodate nositelja, vrijednostnositelj
neka bude tip veze One to many. Za definiranje veze koristiteForeignKey
. - Nakon kreirane klase pokrenite naredbe
makemigrations
imigrate
.
Rješenje
U datoteci main/models.py
:
class Predmet(models.Model):
predmet_naslov = models.CharField(max_length=100)
predmet_sadrzaj = models.TextField()
predmet_vrijeme_objave = models.DateTimeField(default=timezone.now)
predmet_nositelj = models.ForeignKey(Profesor, on_delete=models.CASCADE)
def __str__(self):
return self.predmet_naslov
One-to-one
Student i Profesor povezat ćemo u klasi ZavrsniRad, na kojem radi student, dok mu je profesor mentor. Svaki završni rad ima Studenta koji ga piše i profesora koji mu je mentor. Ovo ćemo ostvariti vezama one-to-one.
Zadatak
- Definirajte klasu
ZavrsniRad
.ZavrsniRad
neka ima zadanog nositeljamentor
koji je povezan saProfesor
pomoćuOneToOne
veze, dodatni parametar koje zadajete u definiciji veze jeprimary_key=True
. - Klasu
ZavrsniRad
zatim povežite saStudent
, tip veze neka budeOne-to-one
, dodatni parametri koje zadajete suon_delete=models.CASCADE
iprimary_key=True
. - Zatim dodajte vrijednosti
rad_naslov
irad_zadatak
koji su CharField duljine 25 i 50 i bool vrijednostrad_prvi_upis
koja po defaultu ima vrijednostTrue
. - Nakon kreirane klase pokrenite naredbe
makemigrations
imigrate
.
Rješenje
U datoteci main/models.py
:
class ZavrsniRad(models.Model):
mentor = models.OneToOneField(
Profesor,
on_delete=models.CASCADE,
)
student = models.OneToOneField(
Student,
on_delete=models.CASCADE,
primary_key=True,
)
rad_naslov = models.CharField(max_length=50)
rad_zadatak = models.CharField(max_length=50)
rad_prvi_upis = models.BooleanField(default=True)
Zadatak
Definirajte funkciju __str__()
unutar klase ZavrsniRad
koja vraća student_broj_xice
. Zatim dodajte klasu ZavrsniRad
unutar main/admin.py
tako da ona postane vidljiva u admin panelu.
Rješenje
Unutar klase ZavrsniRad
u main/models.py
dopunjavamo:
def __str__(self):
return 'Završni rad studenta s brojem X-ice {}'.format(self.student.student_broj_xice)
U main/admin.py
samo dopunjavamo model_list
na način:
model_list = [Predmet, Student, ZavrsniRad]
Korištenje naredbe ./manage.py
:
- Nakon svake novo stvorene klase u modelu pokrenite naredbe
./manage.py makemigrations
i./manage.py migrate
. - Naredbu
./manage.py flush
koristite za očistiti bazu padatak od prethodno unešenih vrijednosti. - Naredba
./manage.py dbshell
omogućuje unos SQL naredbi, primjericeALTER TABLE main_predmet ADD COLUMN "predmet_nositelj_id" integer;
.
Upiti
Naredba za pokretanje Pythonove ljuske specifične za Django:
$ ./manage.py shell
>>>
Za definiranje instance klase:
>>> profesor = Profesor()
>>> predmet = Predmet()
Povezivanje instanci Profesor
i Predmet
pomoću vanjskog ključa, odnosno dodavanje nositelja na predmet:
>>> predmet.predmet_nositelj = profesor
Instanci klase Student
možemo dodati n predmeta:
>>> student.student_predmeti.add(predmet1, predmet2)
Kreiranje instance klase ZavrsniRad
:
>>> zr = ZavrsniRad()
Dodavanje veze tipa one-to-one, na instancu klase ZavrsniRad
:
>>> zr.mentor=profesor
>>> zr.student=student
Upit za koji vraća sve instance tražene klase:
profesors = Profesor.objects.all()
Pretraga po zadanoj vrijednosti:
email = Profesor.objects.get(prof_email='prof_mail@uniri.hr')
Pretraga svih instanci koji imaju traženo ime:
prof_peros = Profesor.objects.filter(prof_ime__contains='Pero')
Uzimanje prvih 5 zapisa:
Profesor.objects.all()[:5]
Sortiranje i dohvaćanje prvog u listi:
Profesor.objects.order_by('prof_ime')[0]
Cjelovit kod današnjih vježbi
from django.db import models
from django.utils import timezone
## Create your models here.
class Profesor(models.Model):
prof_ime = models.CharField(max_length=30)
prof_prezime = models.CharField(max_length=30)
prof_email = models.EmailField()
def __str__(self):
return self.prof_email
class Predmet(models.Model):
predmet_naslov = models.CharField(max_length=100)
predmet_sadrzaj = models.TextField()
predmet_vrijeme_objave = models.DateTimeField(default=timezone.now)
predmet_nositelj = models.ForeignKey(Profesor, default=1, on_delete=models.CASCADE)
def __str__(self):
return self.predmet_naslov
class Student(models.Model):
student_ime = models.CharField(max_length=25)
student_prezime = models.CharField(max_length=50)
student_broj_xice = models.CharField(max_length=10)
student_predmeti = models.ManyToManyField(Predmet)
def __str__(self):
return self.student_broj_xice
class ZavrsniRad(models.Model):
mentor = models.OneToOneField(
Profesor,
on_delete=models.CASCADE,
)
student = models.OneToOneField(
Student,
on_delete=models.CASCADE,
primary_key=True
)
rad_naslov = models.CharField(max_length=50)
rad_zadatak = models.CharField(max_length=150)
rad_prvi_upis = models.BooleanField(default=True)
def __str__(self):
return 'Završni rad studenta s brojem X-ice {}'.format(self.student.student_broj_xice)
Usmjeravanje i URL-i. Stvaranje pogleda kao odgovora na HTTP zahtjeve
Na današnjim vježbama čeka nas gradivo vezano za usmjeravanje pomoću URL-ova. Zatim ćemo vidjeti par primjera kako se izgleda odgovor na poslani HTTP zahtjev.
Usmjeravanje pomoću urls.py
Potrebno je prvo stvoriti novi projekt i unutar njega aplikaciju koju ćemo povezati.
Za kreiranje projekta koristi se naredba:
$ django-admin startproject <project_name>
(...)
Za kreiranje aplikacije unutar projekta koristi se naredba:
$ django-admin startapp <app_name>
(...)
Nakon što su projekt i aplikacija unutar njega kreirani, potrebno ih je povezati. Ovo se radi unutar datoteka urls.py
koja se nalazi u projektnom direktoriju.
Na ovim vježbama kreirati će se projekt naziva vj4
unutar kojeg je stvorena aplikacija naziva main
.
Zadatak
Povežite kreiranu aplikaciju main
s glavnim djelom aplikacije unutar main/urls.py
.
Rješenje
Datoteka urls.py
:
from django.contrib import admin
from django.urls import path, include
urlpatterns = [
path('admin/', admin.site.urls),
path('main/', include('main.urls')),
]
Warning
Potrebno je povezati novostvorenu aplikaciju main
i unutar setting.py
. UnutarINSTALLED_APPS
potrebno je dodati MainConfig
iz apps.py
koja se nalazi unutar aplikacije main
.
Program smo usmjerili na main/urls.py
koji trenutno ne postoji. Iz toga razloga, potrebno ga je stvoriti.
Zadatak
- Stvorite datoteku
main/urls.py
. Odmah importajte sve iz datotekemain/views.py
i neka ime aplikacije bude zadano naapp_name = 'main'
. - Zatim definirajte uzorak URL-a neka upućuje na
homepage
, odnosno na funkciju unutarmain/views.py
koja se zovehomepage
.
Rješenje
Datoteka main/urls.py
:
from django.urls import path
from . import views
app_name = 'main'
urlpatterns = [
path('homepage', views.homepage, name='homepage'),
]
Definirali smo poveznice unutar datoteka main/urls.py
. Sada je potrebno kreirati funkciju homepage()
unutar main/views.py
koju smo pozvali unutar main/urls.py
.
Slanje zahtjeva
Zadatak
Definirajte funkciju homepage()
unutar main/views.py
koja će vraćati HTTP odgovor na zahtjev. Za vraćanje HTTP odgovora koristite funkciju HttpResponse
koju uvozite kodom from django.http import HttpResponse
.
Rješenje
U datoteci main/views.py
:
from django.shortcuts import render
from django.http import HttpResponse
## Create your views here.
def homepage(request):
return HttpResponse('<html><body><strong>Homepage</strong> i još neki tekst na homepage.</body></html>')
Pohranite sve promjene i pokrenite server.
Zadatak
Definirajte funkciju current_datetime()
unutar datoteke main/views.py
koja će vraćati HTTP odgovor na zahtjev. Neka vrijednost koju funkcija vraća budu datum i trenutno vrijeme.
Rješenje
U datoteci main/views.py
:
from django.shortcuts import render
from django.http import HttpResponse
import datetime
## Create your views here.
def current_datetime(request):
now = datetime.datetime.now()
html = '<html><body>Trenutno vrijeme: {}.</body></html>'.format(now)
return HttpResponse(html)
Vraćajne grešaka u odgovorima na zahtjeve
Zadatak
Definirajte funkciju not_found()
unutar main/views.py
. Funkcija neka vraća HttpResponseNotFound
. Vratite proizvoljni sadržaj odgovora.
Rješenje
U datoteci main/views.py
:
from django.http import HttpResponse, HttpResponseNotFound
def not_found(request):
return HttpResponseNotFound('<h1>Page not found</h1>')
Vraćanje zapisa iz baze
U nastavku je prikazano kako se mogu dohvaćati vrijednosti iz baze podataka i kako ih možemo prikazivati na stranici.
Zadatak
Kreirajte klasu Student
, neka sadrži, ime prezime i broj xice kao atribute. Dodajte ju zatim unutar admin.py
da bi se mogle unositi vrijednosti. Za kraj pokrenite naredbe za migraciju da se kreira baza.
Rješenje
U datoteci models.py
:
class Student(models.Model):
ime = models.CharField(max_length=25)
prezime = models.CharField(max_length=50)
broj_xice = models.CharField(max_length=10)
def __str__(self):
return str(self.broj_xice)
U datoteci admin.py
:
from django.contrib import admin
from main.models import *
## Register your models here.
admin.site.register(Student)
Nakon što je baza pomoću modela kreirana, potrebno je unijeti u nju vrijednosti da se može izvršiti tražene upite.
Zadatak
Kreirajte administratora i dodajte u bazu podataka 5 studenata. Od 5 studenata, 3 neka imaju isto ime, primjerice: Marko, Marko, Marko, Ivan i Ana. Prezime i broj X-ice zadajte proizvoljno.
Kada smo popunili bazu, idemo kreirati i upite.
Zadatak
Definirajte funkciju koja u bazi pronalazi sve studente zadanog imena, listu pronađenih imena proslijedite funkciji render
. Proslijeđena rješenja neka se prikazuju unutar students.html
. Za početak, neka vaša funkcija vraća render(request, 'students.html', context=context)
, a datoteku students.html
ćemo stvoriti u nastavku rješavanja ovoga zadatka.
U datoteci views.py
:
def all_peros(request):
peros = Student.objects.filter(ime__contains='pero')
context = {'peros': peros}
return render(request, 'students.html', context=context)
Za prikaz rješenja prethodnog zadatka, potrebna nam je HTML datoteka, koji će prikazati rezultate upita nad bazom. Kreirajte unutar direktorija main
datoteku main/templates
, unutar koje pohranjujete students.html
.
Datoteka students.html
ima sadržaj:
<ul>
{% for p in peros %}
<li>
Ime: {{ p.ime }}<br>
Prezime: {{ p.prezime }}<br>
Broj xice:{{ p.broj_xice }}
</li>
{% endfor %}
</ul>
Zadatak
Dodajte unutar datoteke main/urls.py
putanju koja nas vodi na prethodno kreiranu funkciju i zatim provjerite prikaz rezultata na serveru.
Idemo još prikazati ukupan broj studenata u našoj bazi. Ovaj broj ćemo zatim proslijediti funkciji render()
koja će ispisati ukupan broj studenata u za to kreiranom HTML-u.
Zadatak
Definirajte funkciju koja u bazi pronalazi ukupan broj studenata, broj studenata funkciji render
. Proslijeđena rješenja neka se prikazuju unutar datoteke index.html
. Dodajte grešku u slučaju da Student
u bazi nije pronađen.
Rješenje
U datoteci views.py
:
def detail(request):
try:
num_students = Student.objects.all().count()
context = {'num_students': num_students}
except Student.DoesNotExist:
raise Http404('Student does not exist')
return render(request, 'detail.html', context=context)
Datoteka: detail.html
:
{% block content %}
<h1>Dobrodosli na UNIRI</h1>
<p>Na faxu je upisano:</p>
<ul>
<li><strong>Studenata:</strong> {{ num_students }}</li>
</ul>
{% endblock %}
Izrada generičkih pogleda
Na današnjim vježbama radit će se generički pregledi.
Priprema i postavljanje projekta
Prije početka rada potrebno je kreirati novi Django projekt vj5
unutar kojeg kreirate aplikaciju main
.
Povežite projekt i aplikaciju:
- Dodati
main
aplikaciju podINSTALLED_APPS
unutarvj5/settings.py
. - Unutar
vj5/urls.py
dodati usmjeravanje namain/urls.py
,main/urls.py
još nije stvoren, stoga ga je potrebno kreirati.
Datoteka vj5/main/urls.py
je oblika:
from django.urls import path
urlpatterns = [
]
Za potrebe ovih vježbi koristit će se gotov model koji je zadan u nastavku.
Datoteka vj5/main/models.py
je oblika:
from django.db import models
## Create your models here.
class Publisher(models.Model):
name = models.CharField(max_length=30)
address = models.CharField(max_length=50)
city = models.CharField(max_length=60)
state_province = models.CharField(max_length=30)
country = models.CharField(max_length=50)
website = models.URLField()
class Meta:
ordering = ['-name']
def __str__(self):
return self.name
class Author(models.Model):
salutation = models.CharField(max_length=10)
name = models.CharField(max_length=200)
email = models.EmailField()
headshot = models.ImageField(upload_to='author_headshots')
def __str__(self):
return self.name
class Book(models.Model):
title = models.CharField(max_length=100)
authors = models.ManyToManyField('Author')
publisher = models.ForeignKey(Publisher, on_delete=models.CASCADE)
publication_date = models.DateField()
Nakon što je model kreiran unutar vj5/main/models.py
potrebno je provesti migraciju. Naredbe za migraciju su:
$ ./manage.py makemigrations
(...)
$ ./manage.py migrate
(...)
Napravite migraciju i zatim pokrenite server.
Generički pogledi
Kreirajte prvi generički pogled nad stvorenim modelom.
Datoteka vj5/main/views.py
ima sadržaj:
from django.views.generic import ListView
from main.models import Publisher
class PublisherList(ListView):
model = Publisher
A zatim ga povežite unutar main/urls.py
na način:
from django.urls import path
from main.views import PublisherList
urlpatterns = [
path('publishers/', PublisherList.as_view()),
]
Kada smo kreirali pogled i pozvali ga unutar urls.py
potreban nam je predložak unutar kojeg će se prikazati odgovor.
Sve predloške koje ćemo koristiti organizirat ćemo tako da se nalaze u zajedničkom direktoriju templates
, koji se nalazi u korijenskom direktoriju.
Kreirajte direktorij ./templates
, unutar kojeg kreirate direktorij main
, dakle ./templates/main
, a unutar njega kreirajte datoteku publisher_list.html
.
Datoteka ./templates/main/publisher_list.html
ima sadržaj:
{% block content %}
<h2>Publishers</h2>
<ul>
{% for publisher in object_list %}
<li>
Name: {{ publisher.name }}<br>
City: {{ publisher.city }}
</li>
{% endfor %}
</ul>
{% endblock %}
Potrebno je još zadati putanju za predloške unutar settings.py
.
Za dodavanje putanje, pod TEMPLATES
dodajte putanju do templates
direktorija (./templates
), odnosno 'DIRS': ['./templates'],
.
Zadatak
Kreirajte administratora i dodajte u bazu podataka 3 izdavača. Sve vrijednosti proizvoljno zadajte. Provjerite ispis izdavača koji su dodani u bazu na adresi http://127.0.0.1/main/publishers/
.
Dinamičko filtriranje
U nastavku je prikazan način na koji se omogućava dinamička pretraga pomoću URL-a. Za zadani naziv izdavača vraćat će se sve knjige koje je taj izdavač objavio. U zadanom URL uzorku u aplikaciji neće statično biti definirati naziv, nego će se on dinamično generirati.
Za početak potrebno je definirati prikaz unutar ./main/views.py
koji će vraćati sve knjige od zadanog izdavača.
Datoteka vj5/main/views.py
:
from django.shortcuts import get_object_or_404
from django.views.generic import ListView
from main.models import Book, Publisher
class PublisherBookList(ListView):
template_name = 'main/books_by_publisher.html'
def get_queryset(self):
self.publisher = get_object_or_404(Publisher, name=self.kwargs['publisher'])
return Book.objects.filter(publisher=self.publisher)
Zatim unutar ./main/urls.py
povezujemo s traženim pogledom. U ovom slučaju ne koristi se statično zadani uzorak Umjesto da svakog pojedinog izdavača zadajemo pojedinačno, koristimo <publisher>
.
Datoteka vj5/main/urls.py
:
from django.urls import path
from main.views import PublisherList, PublisherBookList
urlpatterns = [
path('publishers/', PublisherList.as_view()),
path('<publisher>/', PublisherBookList.as_view()),
]
I za zadnji dio potrebno je kreirati prikaz unutar ./templates
koji će nam prikazivati rezultate pretrage za zadanog izdavača.
Zadatak
Kreirajte books_by_publisher.html
unutar ./templates/main
koji će ispisati sve knjige od traženog izdavača. Neka se ispisuje samo naslov svake knjige.
Rješenje
{% block content %}
<h2>Books list: </h2>
<ul>
{% for book in object_list %}
<li>Book title: {{ book.title }}</li>
{% endfor %}
</ul>
{% endblock %}
Pokrenite server i provjerite pretraživanje po izdavaču.
Predaja obrazaca HTTP metodama GET i POST. Provjera unosa i prikaz poruka o greškama
Na današnjim vježbama radit će se generičko popunjavanje baze i obrasci.
Postavljanje projekta
Zadatak
Kreirajte projekt naziva vj6
i unutar njega aplikaciju naziva main
. Povežite aplikaciju sa projektom: dodajte aplikaciju unutar settings.py
i putanju main/urls.py
unutar urls.py
, a zatim kreirajte main/urls.py
.
Generičko popunjavanje baze podataka
Model koji se koristi sadrži dvije klase, Author
i Book
.
Sadržaj datoteke vj6/main/models.py
:
from django.db import models
## Create your models here.
class Author(models.Model):
name = models.CharField(max_length=30)
address = models.CharField(max_length=50)
city = models.CharField(max_length=60)
country = models.CharField(max_length=50)
def __str__(self):
return self.name
class Book(models.Model):
title = models.CharField(max_length=100)
abstract = models.TextField()
author = models.ForeignKey(Author, on_delete=models.CASCADE)
publication_date = models.DateField()
def __str__(self):
return self.title
Kreirani model potrebno je popuniti podacima, za to će se koristiti naredba ./manage.py setup_test_data.py
. Prilikom pokretanja naredbe, program vraća grešku jer naredba još nije kreirana.
Instalacija potrebnih Python paketa:
$ pip3 install factory_boy
(...)
Kada je instaliran paket factory_boy, potrebno je kreirati klase koje će automatski popunjavati bazu sa tzv. dummy data, odnostno nasumično generiranim podacima koji će nam pojednostaviti proces popunjavanja baze nad kojom želimo izvršavati upite. Detaljnije o njegovoj funkcionalnosti možete pronaći u službenoj dokumentaciji.
Zadatak
Unutar vj6/main
stvorite datoteku factory.py
Stvorena datoteka vj6/main/factory.py
koristit će se kao predložak za popunjavanje modela definiranog unutarvj6/main/models.py
. Primjetit ćete sličnost u stilu pisanja klasa. Dakle, potrebno je definirati klase, sukladno klasama koje su definirane unutar vj6/main/models.py
.
Datoteka vj6/main/factory.py
:
## factories.py
import factory
from factory.django import DjangoModelFactory
from main.models import *
## Defining a factory
class AuthorFactory(DjangoModelFactory):
class Meta:
model = Author
name = factory.Faker("first_name")
address = factory.Faker("address")
city = factory.Faker("city")
country = factory.Faker("country")
class BookFactory(DjangoModelFactory):
class Meta:
model = Book
title = factory.Faker("sentence", nb_words=4)
abstract = factory.Faker("sentence", nb_words=50)
author = factory.Iterator(Author.objects.all())
publication_date = factory.Faker("date_time")
Zadatak
Nakon što su klase definirane unutar factory.py
, isprobajte njihovu funkcionalnost. Prije pokretanja ljuske primjenite migraciju na bazu.
$ ./manage.py shell
(...)
>>> from main.factories import *
>>> a = AuthorFactory()
>>> b = BookFactory()
>>> a
>>> b.title
>>> b.author
Zadatak
Kreirajte administratnora, zatim unutar admin.py
registrirajte modele Book
i Author
. Provjerite ako su podaci generirani sa factory.py
uneseni u bazu.
Rješenje
U datoteci admin.py
:
from django.contrib import admin
from main.models import *
models_list = [Author, Book]
## Register your models here.
admin.site.register(models_list)
Kreiranje naredbe u manage.py
Kada je kreiran i testiran factory.py
, slijedi kreiranje naredbe koja će se prosljeđivati ./manage.py
.
Za početak porenite naredbu:
$ ./manage.py
(...)
Izlistao nam se trenutni popis opcija koje možemo izvršavati.
Kreirajte direktorij commands
, unutar kojeg će se nalaziti skripta. Zatim se pozicionirajte u njega.
$ mkdir main/management/commands
$ cd main/management/commands
(...)
A zatim, unutar direktorija commands
kreirajte setup_test_data.py
.
$ touch setup_test_data.py
(...)
Otvorite kreirani setup_test_data.py
unutar kojeg će se kreirati vlastita upravljačka naredba (detaljnije o upravljačkim naredbama koje su kreirane od strane korisnika).
Sadržaj datoteke main/management/commands/setup_test_data.py
:
import random
from django.db import transaction
from django.core.management.base import BaseCommand
from main.models import Author, Book
from main.factories import (
AuthorFactory,
BookFactory
)
NUM_AUTHORS = 10
NUM_BOOKS = 100
class Command(BaseCommand):
help = "Generates test data"
@transaction.atomic
def handle(self, *args, **kwargs):
self.stdout.write("Deleting old data...")
models = [Author, Book]
for m in models:
m.objects.all().delete()
self.stdout.write("Creating new data...")
for _ in range(NUM_AUTHORS):
author = AuthorFactory()
for _ in range(NUM_BOOKS):
book = BookFactory()
Svojstvo funkcije handle()
je postavljeno na transaction.atomic
, što označava da ako je blok koda uspješno izvršen, promjene se pohranjuju u bazu podataka (detaljnije objašnjenje o modulu transaction korištenom u prethodnom primjeru).
Zadatak
Isprobajte funkcionalnost kreirane naredbe, a zatim provjerite ako su uneseni podaci unutar admin sučelja.
$ ./manage.py setup_test_data
(...)
Predlošci obrazaca. Stvaranje obrazaca iz modela
Kreiranje predložaka
Nakon što je baza kreirana i popunjena teestim podacima, sljedeći korak je definiranje predložaka pomoću kojih će se ti podaci prikazivati.
Sintaksa
Varijable
Pomoću varijabli ispisujemo sadržaj koji je prosljeđen iz konteksta. Objekt je sličan Python riječniku (engl. dictionary) gdje je sadržaj mapiran u odnosu key-value.
Varijable pišemo unutar vitičastih zagrada {{ varijabla }}
:
<p>Ime: {{ first_name }}, Prezime: {{ last_name }}.</p>
Što će iz konteksta {'first_name': 'Ivo', 'last_name': 'Ivić'}
biti prikazano kao:
<p>Ime: Ivo, Prezime: Ivić.</p>
Oznake
Oznake se pišu unutar {% oznaka %}
. Označavaju proizvoljnu logiku unutar prikaza. Oznaka može biti ispis sadržaja, ili logička cjelina ili pak pristup drugim oznakama iz predloška, Primjerice: {% tag %} ... sadržaj ... {% endtag %}
ili {% block title %}{{ object.title }}{% endblock %}
.
Filteri
Filtere koristimo da bi transformirali vrijednosti varijabli. Neki od primjera korištenja filtera mogu biti za:
- Pretvaranje u mala slova:
{{ name|lower }}
- Uzimanje prvih 30 riječi:
{{ bio|truncatewords:30 }}
- Dohvaćanje duljine varijable:
{{ value|length }}
Komentari
Za komentiranje dijelova koristimo #
što unutar predloška izgleda: {# Ovo neće biti prikazano #}
.
Petlje
Petlje se pišu unutar {% %}
odnosno definiraju se njihove oznake. Primjer korištenja petlje for
:
<ul>
{% for author in author_list %}
<li>{{ author.name }}</li>
{% endfor %}
</ul>
Primjer provjere autentifikacije korisnika naredbom if
:
{% if user.is_authenticated %}
<p>Hello, {{ user.username }}.</p>
{% endif %}
Primjer naredbi if
i else
:
{% if author_list %}
Number of athletes: {{ athlete_list|length }}
{% else %}
No authors.
{% endif %}
Zadatak
Kreirajte direktorij ./templates
unutar kojeg će biti pohranjeni predlošci koji se koriste, npr. base_generic.html
. Ne zaboravite definirati putanju unutar settings.py
.
Rješenje
Datoteka ./templates/base_generic.html
:
<!DOCTYPE html>
<html lang="en">
<head>
</head>
<body>
<h1>My library</h1>
<div id="sidebar">
{% block sidebar %}
<ul>
<li><a href="/main">Home</a></li>
<li><a href="/main/books">Books</a></li>
<li><a href="/main/authors">Authors</a></li>
</ul>
{% endblock %}
</div>
<div class="content">
{% block content %}
{% endblock %}
</div>
</body>
</html>
Nasljeđivanje u predlošcima
Datoteka ./template/main/book_list.html
:
{% extends "base_generic.html" %}
{% block content %}
<br>
<h2> Books </h2>
<br>
{% for book in book_list %}
<div class="book">
<h4>{{ book.title }}</h4>
Author: {{book.author}}
<p>{{ book.abstract }}</p>
</div>
{% endfor %}
{% endblock %}
Unutar main/urls.py
dodajte:
path('books', BookList.as_view())
Datoteka main/views.py
:
from django.shortcuts import get_object_or_404
from django.views.generic import ListView
from main.models import Author, Book
class BookList(ListView):
model = Book
Zadatak
Sukladno prethodnom primjeru kreirajte prikaz za sve autore unutar baze.
CSS
Dodavanje CSS-a u HTML predložak.
Uvoz
<link rel="stylesheet" href="https://www.w3schools.com/html/styles.css">
<head>
<link rel="stylesheet" href="https://www.w3schools.com/html/styles.css">
<title>{% block title %}Knjižnica{% endblock %}</title>
</head>
Direktorij za statičke datoteke
Unutar aplikacije main
potrebno je stvoriti direktorij static
a unutar njega style.css
.
Referenciranje na style.css
unutar aplikacije:
{% load static %}
<link rel="stylesheet" type="text/css" href="{% static 'style.css' %}">
Prikaz unutar HTML templatea:
<head>
{% load static %}
<link rel="stylesheet" type="text/css" href="{% static 'style.css' %}">
<title>{% block title %}Knjiznica{% endblock %}</title>
</head>
Sadržaj datoteke style.css
možete proizvoljno zadati i prilagođavati vlastitim željama. Primjerice, datoteka main/static/style.css
može biti oblika:
h1 {
color: blue;
text-align: center;
}
h2 {
text-align: center;
}
li {
list-style-type: none;
float: left;
}
li a {
padding: 16px;
}
Zadatak
Dopunite prikaz na autora tako da njegovo ime bude link. Link neka vodi na prikaz svih knjiga od odabranog autora. Za tu svrhu prvo je potrebno kreirati pretraživanje po traženom autoru, a zatim vratiti sve knjige koje su pronađene za traženog autora.
Rješenje
Uredit ćemo datoteku main/views.py
da bude oblika:
from django.shortcuts import get_object_or_404
from django.views.generic import ListView
from main.models import Author, Book
class AuthorBookList(ListView):
template_name = 'main/book_list.html'
def get_queryset(self):
self.author = get_object_or_404(Author, name=self.kwargs['author'])
return Book.objects.filter(author=self.author)
Potrebno je dodati putanju unutar views.py
. Datoteka main/views.py
je oblika:
path('<author>', AuthorBookList.as_view())
Zatim je potrebno izmjeniti predložak, odnosno dodati linkove koji vode na autore. Datoteka author_list.html
:
{% extends "base_generic.html" %}
{% block content %}
<h2>Books</h2>
{% for book in book_list %}
<div class="book">
<h4>{{ book.title }}</h4>
<p>Author: <a href="{{book.author}}"> {{book.author}}</a></p>
<p>{{ book.abstract }}</p>
</div>
{% endfor %}
{% endblock %}
Izmjenimo i predložak za prikaz knjiga. Datoteka book_list.html
:
{% extends "base_generic.html" %}
{% block content %}
<h2>Authors</h2>
{% for author in author_list %}
<div class="author">
<h4>{{author.name}}</h4>
<p>City: {{author.city}}</p>
<p>Country: {{author.country}}</p>
<p><a href="{{author.name}}"> All books by {{author.name}}</a></p>
</div>
{% endfor %}
{% endblock %}
Zadatak
Dopunite style.css
tako da dodate obrub na elemente Knjige i Autora.
Rješenje
U datoteku style.css
dodajemo:
.book {
border-color: cyan;
border-style: double;
padding-left: 20px;
padding-right: 20px;
padding-bottom: 20px;
}
.author {
border-color: olive;
border-style: double;
padding-left: 20px;
padding-right: 20px;
padding-bottom: 20px;
}
Sijanje i migracije
Sijanje (seeding)
Zadatak
- Kreirajte projekt
vj9
i unutar njega aplikacijumain
. Unutar modela u aplikacijimain
kreirajte klasuStudent
. KlasaStudent
neka sadrži vrijednostfirst_name
. - Provedite potrebne naredbe za migraciju.
- Pokrenite
./manage.py shell
i kreirajte jednog studenta.
Naredbom dumpdata
izvozimo vrijednosti iz baze (detaljnije o naredbi dumpdata). Pokrenimo je na način:
$ ./manage.py dumpdata main.Student --pk 1 --indent 4 > 0001_student.json
(...)
Zadatak
Izbrišite iz baze zapis studenta kojeg ste prethodno unjeli.
Rješenje
>>> Student.objects.filter(pk=1).delete()
Za uvoz podataka u bazu koristimo naredbu loaddata
. Detaljnije o naredbi loaddata.
Zadatak
Uvezite prethodno kreirani 0001_student.json
u bazu.
Fixture
Fixture je zbirka podataka koje Django uvozi u bazu podataka. Najjednostavniji način rada sa podacima je pomoću naredbi dumpdata
i loaddata
.
Paket django-seed
$ pip3 install django-seed
(...)
Python modul pomoću kojeg se mogu generirati podaci za bazu podataka. U pozadini koristi biblioteku faker (dokumentacija) za generiranje testnih podataka. Detaljnije o django-seed možete pronaći u dokumentaciji.
Brisanje podataka sijanja
U nastavku je generirana skripta revert_seed.py
pomoću koje brišemo vrijednosti iz baze koje smo prethodno stvorili i unosili sijanjem.
import json
import glob
g = globals()
has_access = {}
fixtures = glob.glob("*.json")
fixtures.sort(reverse=True)
def get_access(model):
import importlib
mod = importlib.import_module(model)
names = getattr(mod, '__all__', [n for n in dir(mod) if not n.startswith('_')])
global g
for name in names:
g[name.lower()] = {
'var': getattr(mod, name),
'name': name
}
for fixture in fixtures:
msg = 'Reverting '+fixture+'\n'
with open(fixture) as json_file:
datas = json.load(json_file)
for data in datas:
app_name = data['model'].split('.')[0]
class_name = data['model'].split('.')[1]
if app_name not in has_access.keys():
get_access(app_name+'.models')
has_access[app_name] = True
class_model = g[class_name]['var']
class_model_name = g[class_name]['name']
pk = data['pk']
msg += '{}(pk={}): '.format(class_model_name, pk)
try:
class_model.objects.get(pk=pk).delete()
msg += 'deleted\n'
except:
msg += 'not deleted\n'
print(msg)
Skriptu pokrenite naredbom:
$ manage.py shell < revert_seed.py
(...)
Učitavanje podataka sijanja u testiranje
Testirajmo rad tako da dohvatimo podatke o studentu koji ima primarni ključ 1 i čije ime je Ivo.
## test1.py
from django.test import TestCase
from main.models import Student
class MyTest(TestCase):
# fixtures = ["0001_student.json"]
def test_should_create_group(self):
s = Student.objects.get(pk=1)
self.assertEqual(s.first_name, 'Ivo')
Kreirani test pokrenite u terminalu s naredbom:
$ ./manage.py test test1
(...)
>>> from django.contrib.auth.models import User, Group
>>> Group.objects.create(name='usergroup')
>>> usergroup = Group.objects.get(name='usergroup')
>>> ivo = User.objects.create_user('Ivo')
>>> ivo.pk
>>> ivo.groups.add(usergroup)
>>> python manage.py dumpdata auth.User --pk 1 --indent 4
>>> python manage.py dumpdata auth.User --pk 1 --indent 4 --natural-foreign
Django vježba 10: Autentifikacija. Autorizacija
Stvaranje projekta
Priprema za rad: stvorite projekt naziva vj10
, unutar njega aplikaciju naziva main
. Provedite migraciju. Zatim kreirajte administratora, za stvaranje korisnika sa administratorskim ovlastima koristite naredbu ./manage.py createsuperuser
.
Povezivanje projekta i aplikacije
Datoteka vj10/urls.py
:
urlpatterns = [
path('admin/', admin.site.urls),
path('', include('main.urls')),
path('accounts/', include('django.contrib.auth.urls'))
]
Homepage
Datoteka main/urls.py
:
from django.urls import path
from . import views
urlpatterns = [
path('', views.index, name='index'),
]
Stvaranje pogleda za index
:
def index(request):
return render(request, 'main/index.html')
Unutar aplikacije main
stvorite si direktorij templates
, unutar kojeg kreirate index.html
. HTML predložak:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title></title>
</head>
<body>
</body>
</html>
Kreiranje korisnika
Posjetite http://127.0.0.1:8000/accounts/
i http://127.0.0.1:8000/accounts/login/
. Prilikom posjete /accounts/login/
javila se greška TemplateDoesNotExist at /accounts/login/
, gdje možemo vidjeti iz poruke Exception Value: registration/login.html
da Django ne može pronaći traženi predložak.
Zadatak
Unutar templates/registration
stvorite login.html
.
Login
{% if form.errors %}
<h3>Unos nije ispravan.</h3>
{% endif %}
{% if next %}
{% if user.is_authenticated %}
<p>Your account doesn't have access to this page. To proceed,
please login with an account that has access.</p>
{% else %}
<p>Please login to see this page.</p>
{% endif %}
{% endif %}
<form method="post" action="{% url 'login' %}">
{% csrf_token %}
<table>
<tr>
<td>{{ form.username.label_tag }}</td>
<td>{{ form.username }}</td>
</tr>
<tr>
<td>{{ form.password.label_tag }}</td>
<td>{{ form.password }}</td>
</tr>
</table>
<input type="submit" value="login" />
<input type="hidden" name="next" value="{{ next }}" />
</form>
Detaljnije o CSRF tokenu
Postavljanje lokacije gdje želimo da korisnik bude usmjeren nakon uspješnog logina radimo unutar settings.py
, tako da dodamo npr. LOGIN_REDIRECT_URL = '/'
za usmjeravanje na index.html
.
Registracija
Za registraciju koristimo gotovu formu sa:
from django.contrib.auth.forms import UserCreationForm
I kreiramo funkciju register()
:
def register(request):
form = UserCreationForm()
context = {'form': form}
return render(request, 'registration/register.html', context)
Kreirajmo register.html
:
<form method="post" action="{% url 'register' %}">
{% csrf_token %}
{% if form.errors %}
<p>Greška.</p>
{% endif %}
{{ form }}
<input type="submit" value="Register" />
</form>
Izmjenimo funkciju register()
:
from django.contrib.auth import authenticate, login
def register(request):
if request.method == 'POST':
form = UserCreationForm(request.POST)
if form.is_valid():
form.save()
username = form.cleaned_data['username']
password = form.cleaned_data['password1']
user = authenticate(username=username, password=password)
login(request, user)
return redirect('index')
else:
form = UserCreationForm()
context = {'form': form}
return render(request, 'registration/register.html', context)
Izmjene na index.html
ako je korisnik ulogiran.
<h1>This is our homepage</h1>
{% if user.is_authenticated %}
<p>Vaše ime: {{ user.username }}</p>
{% else %}
<p>Niste prijavljeni.</p>
{% endif %}
Django vježba 11: Testiranje
Ovaj dio je sastavljen prema Django Tutorialu s MDN-a.
Test jedinke (engl. unit test) je najbrži za izvedbu, testiraju dio koda neovisno o drugim djelovima.
def zbroji(prvi, drugi):
return prvi + drugi
def test_zbroji():
assert zbroji(3, 6) == 9
Test integracije testira više dijelova zajedno kako bi se osiguralo da međusobno dobro surađuju/rade.
Funkcijsko testiranje je test koji radi na principu da osigurava funkcionalnos iz perspektive krajnjeg korisnika. Najsporiji za izvođenje.
Regresijsko testiranje reproducira greške koje su se prethodno događale u programskom kodu. Svaki se test u početku pokreće kako bi se provjerilo ako je greška u kodu ispravljena, a zatim se ponovno pokreće kako bi se osiguralo da nije ponovno uveden nakon kasnijih promjena koda.
Za testiranje projekta i programskog koda unutar Django-a koristiti ćemo osnovnu klasu za testiranje koja se zove django.test.TestCase. Najzastupljenija je klasa za testiranje, iako neke testove ne provodi "najbrže" (svaki test ne zahtjeva kreiranje baze podataka).
class Author(models.Model):
name = models.CharField(max_length=30)
address = models.CharField(max_length=50)
city = models.CharField(max_length=60)
country = models.CharField(max_length=50)
date_of_birth = models.DateField(null=True, blank=True)
date_of_death = models.DateField('Died', null=True, blank=True)
def __str__(self):
return self.name
Prije početka pisanja samih testova pokrenite si projekt koji smo radili na vježbama 6. Na Merlinu je dostupna arhiva s projektom ako ju nemate.
Zadatak
Preuzmite i pokrenite projekt vj6
zatim unutar aplikacije main kreirajte direktorij tests
.
Unutar direktorija tests
nalaze se testovi kojima će se testirati urls.py
, views.py
i models.py
.
Naredba koju koristite za pokretanje testova je:
$ ./manage.py test main.tests
(...)
Zadatak
Stvorite testne .py
datoteke unutar tests
koje koristite za testiranje rada urls.py
, views.py
i models.py
.
Testiranje urls.py
from django.test import SimpleTestCase
from django.urls import reverse, resolve
from main.views import homepage, BookList, AuthorList, AuthorBookList
class TestUrls(SimpleTestCase):
def test_homepage_url_is_resolved(self):
url = reverse('homepage')
# print(resolve(url))
self.assertEquals(resolve(url).func, homepage)
def test_books_url_is_resolved(self):
url = reverse('books')
self.assertEquals(resolve(url).func.view_class, BookList)
def test_authors_url_is_resolved(self):
url = reverse('authors')
self.assertEquals(resolve(url).func.view_class, AuthorList)
def test_authors_url_is_resolved(self):
url = reverse('author_q', args=['some-author'])
self.assertEquals(resolve(url).func.view_class, AuthorBookList)
Testiranje views.py
from django.test import TestCase, Client
from django.urls import reverse
from main.models import Author, Book
class TestViews(TestCase):
def setUp(self):
self.client = Client()
self.homepage_url = reverse('homepage')
self.authors_q_url = reverse('author_q', args=['some-author'])
self.author1 = Author.objects.create(
name = 'some-author',
address = 'TestAdress',
city = 'TestCity',
country = 'TestCountry'
)
def test_project_homepage_GET(self):
client = Client()
response = client.get(self.homepage_url)
self.assertEquals(response.status_code, 200)
self.assertTemplateUsed(response, 'base_generic.html')
def test_project_authors_GET(self):
client = Client()
response = client.get(self.authors_q_url)
self.assertEquals(response.status_code, 200)
self.assertTemplateUsed(response, 'main/book_list.html')
Testiranje models.py
from django.test import TestCase
from main.models import Author, Book
class Testmodels(TestCase):
def setUp(self):
self.author1 = Author.objects.create(
name = "some-author",
address = "TestAdress",
city = "TestCity",
country = "TestCountry"
)
def test_author(self):
self.assertEquals(self.author1.name, "some-author")
Author: Milan Petrović