-
Notifications
You must be signed in to change notification settings - Fork 84
Flask 3 dalis
Flask leidžia mums dirbti su duomenų bazėmis, praktiškai nesitepant rankų į SQL užklausas. Viskuo pasirūpina modulis Flask-SQLAlchemy. Iš principo tai yra SQLAlchemy, optimizuota flaskui. Diegiasi pip install Flask-SQLAlchemy.
Susikurkime pirmą duomenų bazę:
import os
from flask import Flask
from flask_sqlalchemy import SQLAlchemy
basedir = os.path.abspath(os.path.dirname(__file__))
# pilnas kelias iki šio failo.
app = Flask(__name__)
app.config['SQLALCHEMY_DATABASE_URI'] = 'sqlite:///'+os.path.join(basedir, 'data.sqlite')
# nustatėme, kad mūsų duomenų bazė bus šalia šio failo esants data.sqlite failas
app.config['SQLALCHEMY_TRACK_MODIFICATIONS'] = False
# neseksime kiekvienos modifikacijos
db = SQLAlchemy(app)
# sukuriame duomenų bazės objektą
# sukurkime modelį užklausos formai, kuris sukurs duomenų bazėje lentelę
class Query(db.Model):
# DB lentelei priskiria pavadinimą, jei nenurodysite, priskirs automatiškai pagal klasės pavadinimą.
__tablename__ = 'messages'
id = db.Column(db.Integer, primary_key=True) # stulpelis, kurio reikšmės integer. Taip pat jis bus primary_key.
name = db.Column(db.String(80), nullable=False)
email = db.Column(db.String(120), unique=True, nullable=False)
message = db.Column(db.Text, nullable=False)
def __init__(self, name, email, message):
self.name = name
self.email = email
self.message = message
def __repr__(self):
return f'{self.name} - {self.email}'
class Query yra aprašyta mūsų duomenų bazės lentelė. Paleidus šį failą, duomenų bazė nesusikurs, reikia inicijuoti šį veiksmą iš išorės (konsolės arba kito .py skripto). Susikurkime dar vieną python failą:
from app import db, Message
db.create_all() # sukurs mūsų lentelę DB
# Iš karto inicijuosime testams keletą įrašų:
jonas = Message('Jonas', '[email protected]', 'Kažkoks labai rimtas atsiliepimas.')
antanas = Message('Antanas', '[email protected]', 'Antano nuomonė labai svarbi.')
juozas = Message('Juozas', '[email protected]', 'Aš labai piktas, nes blogai.')
bronius = Message('Bronius', '[email protected]', 'Aš tai linksmas esu, man patinka.')
# Pridėsime šiuos veikėjus į mūsų DB
db.session.add_all([jonas, antanas, juozas, bronius])
# .commit išsaugo pakeitimus
db.session.commit()
print(jonas.id)
print(antanas.id)
print(bronius.id)
print(juozas.id)
# 1
# 2
# 4
# 3
.create_all() įrašė lentelę į DB, inicijuoti testiniai duomenys taip pat sėkmingai nukeliavo į lentelę, atsispausdinome jų ID, kurie buvo sugeneruoti automatiškai.
Susikurkime dar vieną failą, crud operacijų demonstracijai. Atsispausdinkime visus lentelėje esančius objektus:
from app import db, Message
all_messages = Message.query.all()
print(all_messages)
# [Jonas - [email protected], Antanas - [email protected], Juozas - [email protected], Bronius - [email protected]]
Atsispausdinkime vieną iš objektų:
message_1 = Message.query.get(1)
print(message_1)
# Jonas - [email protected]
Išfiltruokime objektą pagal nurodytą požymį:
message_antanas = Message.query.filter_by(name='Antanas')
print(message_antanas.all())
# [Antanas - [email protected]]
filter_by išrinks mums visus įrašus, kuriuose name='Antanas'
Pakeiskime Antano el. paštą:
antanas = Message.query.get(2)
antanas.email = '[email protected]'
db.session.add(antanas)
db.session.commit()
print(Message.query.all())
# [Jonas - [email protected], Antanas - [email protected], Juozas - [email protected], Bronius - [email protected]]
Ištrinkime Joną:
jonas = Message.query.get(1)
db.session.delete(jonas)
db.session.commit()
print(Message.query.all())
# [Antanas - [email protected], Juozas - [email protected], Bronius - [email protected]]
Jeigu mums prireiktų papildyti savo lentelę papildomu stulpeliu, tiesiog papildžius klasę Message nauja eilute mums nepavyktų, kadangi duomenų bazė jau inicijuota tokia, kokią nurodėme pirmą kartą. Tą reikia turėti omenyje, kas kartą, kuriant duomenų bazę stengtis pasidaryti ją kuo išbaigtesnę. Tuomet ateityje, norint ją papildyti naujais stulpeliais, ar pakeisti esamų nustatymus, reikės mažiau migracijos procesų. Susitvarkykime savo projektą taip, kad galėtumėm vykdyti migracijas:
-
nustatykime FLASK-APP aplinkos kintamąjį (environment variable). Tą reikės padaryti Windows komandinėje eilutėje, arba Linux/MacOS terminale:
-
windows - set FLASK_APP=failas_kuriame_musu_db_modelis.py
-
linux/macOS - export FLASK_APP=failas_kuriame_musu_db_modelis.py
-
įsitikinkite, kad komandą leidžiate iš to paties katalogo, kuriame failas su jūsų DB modeliu.
-
įdiekime Flask-Migrate paketą (pip install Flask-Migrate)
-
pertvarkykime savo .py failą:
import os
from flask import Flask
from flask_sqlalchemy import SQLAlchemy
from flask_migrate import Migrate # importuojame migracijas
basedir = os.path.abspath(os.path.dirname(__file__))
app = Flask(__name__)
app.config['SQLALCHEMY_DATABASE_URI'] = 'sqlite:///'+os.path.join(basedir, 'data.sqlite')
app.config['SQLALCHEMY_TRACK_MODIFICATIONS'] = False
db = SQLAlchemy(app)
Migrate(app, db) # Susiejame app ir db.
class Message(db.Model):
__tablename__ = 'messages'
id = db.Column(db.Integer, primary_key=True)
name = db.Column(db.String(80), nullable=False)
email = db.Column(db.String(120), unique=True, nullable=False)
phone = db.Column(db.String(40), unique=True) # Papildome duomenų bazės modelį nauju stulpeliu.
message = db.Column(db.Text, nullable=False)
# prie konstruktoriaus irgi nepamirštame pridėti:
def __init__(self, name, email, message, phone):
self.name = name
self.email = email
self.message = message
self.phone = phone
def __repr__(self):
return f'{self.name} - {self.email}'
-
Importus papildėme migracijos 'tarnyba':) Jai nurodėme, kokią aplikaciją susieti su kokia duomenų baze. Papildėme duomenų bazės modelį nauju stulpeliu (phone).
-
Inicijuokime migracijas mūsų projektui su komanda flask db init:
(flask-kursui) robotautas@robotautas-MS-7A34:~/Dropbox/Flask 3 dalis/Code$ flask db init
Creating directory /home/robotautas/Dropbox/Flask 3 dalis/Code/migrations ... done
Creating directory /home/robotautas/Dropbox/Flask 3 dalis/Code/migrations/versions ... done
Generating /home/robotautas/Dropbox/Flask 3 dalis/Code/migrations/alembic.ini ... done
Generating /home/robotautas/Dropbox/Flask 3 dalis/Code/migrations/README ... done
Generating /home/robotautas/Dropbox/Flask 3 dalis/Code/migrations/env.py ... done
Generating /home/robotautas/Dropbox/Flask 3 dalis/Code/migrations/script.py.mako ... done
Please edit configuration/connection/logging settings in '/home/robotautas/Dropbox/Flask 3 dalis/Code/migrations/alembic.ini' before proceeding.
(flask-kursui) robotautas@robotautas-MS-7A34:~/Dropbox/Flask 3 dalis/Code$ ls
app.py data.sqlite migrations __pycache__ setupdb.py simple_crud.py test.py
- matome, kad sukurtas migracijų katalogas. Dabar paruoškime savo pirmą migraciją, flask db migrate -m "žinutė atminčiai" (jei tai darome pakartotinai ir matome klaidą ERROR [flask_migrate] Error: Target database is not up to date., reikės prieš tai paleisti "flask db stamp head"):
(flask-kursui) robotautas@robotautas-MS-7A34:~/Dropbox/Flask 3 dalis/Code$ flask db migrate -m "pridėtas stulpelis phone"
INFO [alembic.runtime.migration] Context impl SQLiteImpl.
INFO [alembic.runtime.migration] Will assume non-transactional DDL.
INFO [alembic.autogenerate.compare] Detected added column 'messages.phone'
INFO [alembic.autogenerate.compare] Detected added unique constraint 'None' on '['phone']'
Generating /home/robotautas/Dropbox/Flask 3 dalis/Code/migrations/versions/d31d8cda085d_pridėtas_stulpelis_phone.py ... done
- matome, kad aptikti pakeitimai. Dabar įvykdykime pačią migraciją, flask db upgrade:
(flask-kursui) robotautas@robotautas-MS-7A34:~/Dropbox/Flask 3 dalis/Code$ flask db upgrade
INFO [alembic.runtime.migration] Context impl SQLiteImpl.
INFO [alembic.runtime.migration] Will assume non-transactional DDL.
INFO [alembic.runtime.migration] Running upgrade -> d31d8cda085d, pridėtas stulpelis phone
ERROR [root] Error: No support for ALTER of constraints in SQLite dialect
# klaidelę galime ignoruoti, ne visiškas dialekto palaikymas..
- patikrinkime, ar suveikė:
messages = Message.query.all()
for i in messages:
random_phone = randint(999999, 10000000)
i.phone = str(random_phone)
db.session.add(i)
db.session.commit()
for x in messages:
print (f'{x.id}, {x.name}, {x.email}, {x.phone}, {x.message}')
# 2, Antanas, [email protected], 9033639, Antano nuomonė labai svarbi.
# 3, Juozas, [email protected], 2233484, Aš labai piktas, nes blogai.
# 4, Bronius, [email protected], 4211290, Aš tai linksmas esu, man patinka.
Jeigu atkreipėte dėmesį, migracijų procesas labai panašus į GIT procesus.
© Donatas Noreika ir Jotautas Treigys
Dekoratoriai
Iteratoriai ir generatoriai
RegEx
Pillow
NumPy
Pandas
- Pandas I
- Užduotys I
- Atsakymai I
- Pandas II
- Užduotys II
- Atsakymai II
- Pandas III
- Užduotys III
- Atsakymai III
Seaborn
Mašininis mokymasis
- 1 Tiesinės regresijos modelis
- 1 Užduotis
- 1 Atsakymas
- 2 Modeliai - klasifikatoriai
- 2 Užduotis
- 2 Atsakymas
- 3 Modeliai praktikoje
- 3 Užduotis (atnaujinta), atsakymas
- 3 Užduotis
- 3 Atsakymas (kodas su komentarais)
Requests, JSON, API
Web Scraping (Beautiful Soup)
Duomenų bazės
- SQL 1
- Užduotys
- Atsakymai
- SQL 2
- Užduotys
- Atsakymai
- SQL 3
- Užduotys
- Atsakymai
- SQL 4
- Užduotis
- SQL per Python
- Užduotis
- ORM 1
- Užduotys
- ORM 2
- Užduotis
- Atsakymas
Flask
- Įžanga
- Užduotys
- I dalis
- Užduotis
- I dalies kodas (atsakymas)
- II dalis
- Užduotis
- Atsakymas (kodas)
- III dalis
- Užduotis
- Atsakymas
- IV dalis
- Užduotis
- Atsakymas
- Flask Many2one, CRUD
- Flask One2many, many2many CRUD
- Flask: REST API kūrimas
- Flask: API su One2many (kodas)
- Flask: vartotojai
- Flask: nuotraukos pridėjimas, admin puslapis, puslapiavimas
- Flask: password reset, error pages
- Flask: projekto sutvarkymas
- Flask diegimas į serverį
Django
- 1. Įžanga, Modeliai
- 2. Administratoriaus svetainė
- 3. Šablonai
- 4. Views
- 5. Puslapiavimas, Paieška, Nuotraukos
- 6. Autorizacija
- 7. Vartotojai II, HTML laukai
- 8. Registracija, Formos
- 9. Vartotojo profilis
- 10. Create, Update, Delete rodinių klasės
- 11. Vertimai
- Užduotis: Autoservisas
- Biblioteka: kodas iki 6-tos dalies
- Biblioteka: kodas nuo 6-tos dalies
- Django diegimas į serverį
- Django diegimas į serverį 2
Django REST
Odoo
- Kurso programa: projektų valdymas
- 1 pamoka
- 2 pamoka
- 3 pamoka
- 4 pamoka
- 5 pamoka
- 6 pamoka
- 7 pamoka
- 8 pamoka
- 9 pamoka
Linux