SQLAlchemy ORM - Menghapus Objek Terkait

Mudah untuk melakukan operasi hapus pada satu tabel. Yang harus Anda lakukan adalah menghapus objek kelas yang dipetakan dari sesi dan melakukan tindakan. Namun, operasi hapus pada beberapa tabel terkait sedikit rumit.

Dalam database sales.db kami, kelas Pelanggan dan Faktur dipetakan ke tabel pelanggan dan faktur dengan satu jenis hubungan ke banyak. Kami akan mencoba untuk menghapus objek Pelanggan dan melihat hasilnya.

Sebagai referensi cepat, di bawah ini adalah definisi kelas Pelanggan dan Faktur -

from sqlalchemy import create_engine, ForeignKey, Column, Integer, String
engine = create_engine('sqlite:///sales.db', echo = True)
from sqlalchemy.ext.declarative import declarative_base
Base = declarative_base()
from sqlalchemy.orm import relationship
class Customer(Base):
   __tablename__ = 'customers'

   id = Column(Integer, primary_key = True)
   name = Column(String)
   address = Column(String)
   email = Column(String)
   
class Invoice(Base):
   __tablename__ = 'invoices'

   id = Column(Integer, primary_key = True)
   custid = Column(Integer, ForeignKey('customers.id'))
   invno = Column(Integer)
   amount = Column(Integer)
   customer = relationship("Customer", back_populates = "invoices")
   
Customer.invoices = relationship("Invoice", order_by = Invoice.id, back_populates = "customer")

Kami menyiapkan sesi dan mendapatkan objek Pelanggan dengan menanyakannya dengan ID utama menggunakan program di bawah ini -

from sqlalchemy.orm import sessionmaker
Session = sessionmaker(bind=engine)
session = Session()
x = session.query(Customer).get(2)

Dalam tabel contoh kami, x.name kebetulan adalah 'Gopal Krishna'. Mari kita hapus x ini dari sesi dan hitung kemunculan nama ini.

session.delete(x)
session.query(Customer).filter_by(name = 'Gopal Krishna').count()

Ekspresi SQL yang dihasilkan akan mengembalikan 0.

SELECT count(*) 
AS count_1
FROM (
   SELECT customers.id 
   AS customers_id, customers.name 
   AS customers_name, customers.address 
   AS customers_address, customers.email 
   AS customers_email
   FROM customers
   WHERE customers.name = ?) 
AS anon_1('Gopal Krishna',) 0

Namun, objek Faktur terkait x masih ada. Itu dapat diverifikasi dengan kode berikut -

session.query(Invoice).filter(Invoice.invno.in_([10,14])).count()

Di sini, 10 dan 14 adalah nomor faktur milik pelanggan Gopal Krishna. Hasil query di atas adalah 2 yang berarti objek terkait belum dihapus.

SELECT count(*) 
AS count_1
FROM (
   SELECT invoices.id 
   AS invoices_id, invoices.custid 
   AS invoices_custid, invoices.invno 
   AS invoices_invno, invoices.amount 
   AS invoices_amount
   FROM invoices
   WHERE invoices.invno IN (?, ?)) 
AS anon_1(10, 14) 2

Ini karena SQLAlchemy tidak mengasumsikan penghapusan cascade; kita harus memberi perintah untuk menghapusnya.

Untuk mengubah perilaku, kita mengkonfigurasi opsi bertingkat pada hubungan User.addresses. Mari kita tutup sesi yang sedang berlangsung, gunakan new declarative_base () dan deklarasikan ulang kelas User, tambahkan dalam hubungan alamat termasuk konfigurasi cascade.

Atribut kaskade dalam fungsi hubungan adalah daftar aturan kaskade yang dipisahkan koma yang menentukan bagaimana operasi Sesi harus "bertingkat" dari induk ke turunan. Secara default, ini adalah False, yang berarti "simpan-perbarui, gabungkan".

Kaskade yang tersedia adalah sebagai berikut -

  • save-update
  • merge
  • expunge
  • delete
  • delete-orphan
  • refresh-expire

Opsi yang sering digunakan adalah "semua, hapus-orphan" untuk menunjukkan bahwa objek terkait harus mengikuti objek induk dalam semua kasus, dan dihapus saat tidak terkait.

Karenanya kelas Pelanggan yang dideklarasikan ulang ditunjukkan di bawah ini -

class Customer(Base): 
   __tablename__ = 'customers'
   
   id = Column(Integer, primary_key = True) 
   name = Column(String) 
   address = Column(String) 
   email = Column(String) 
   invoices = relationship(
      "Invoice", 
      order_by = Invoice.id, 
      back_populates = "customer",
      cascade = "all, 
      delete, delete-orphan" 
   )

Mari kita hapus Pelanggan dengan nama Gopal Krishna menggunakan program di bawah ini dan melihat hitungan objek Faktur terkait -

from sqlalchemy.orm import sessionmaker
Session = sessionmaker(bind = engine)
session = Session()
x = session.query(Customer).get(2)
session.delete(x)
session.query(Customer).filter_by(name = 'Gopal Krishna').count()
session.query(Invoice).filter(Invoice.invno.in_([10,14])).count()

Hitungannya sekarang 0 dengan mengikuti SQL yang dipancarkan oleh skrip di atas -

SELECT customers.id 
AS customers_id, customers.name 
AS customers_name, customers.address 
AS customers_address, customers.email 
AS customers_email
FROM customers
WHERE customers.id = ?
(2,)
SELECT invoices.id 
AS invoices_id, invoices.custid 
AS invoices_custid, invoices.invno 
AS invoices_invno, invoices.amount
AS invoices_amount
FROM invoices
WHERE ? = invoices.custid 
ORDER BY invoices.id (2,)
DELETE FROM invoices 
WHERE invoices.id = ? ((1,), (2,))
DELETE FROM customers 
WHERE customers.id = ? (2,)
SELECT count(*) 
AS count_1
FROM (
   SELECT customers.id 
   AS customers_id, customers.name 
   AS customers_name, customers.address 
   AS customers_address, customers.email 
   AS customers_email
   FROM customers
   WHERE customers.name = ?) 
AS anon_1('Gopal Krishna',)
SELECT count(*) 
AS count_1
FROM (
   SELECT invoices.id 
   AS invoices_id, invoices.custid 
   AS invoices_custid, invoices.invno 
   AS invoices_invno, invoices.amount 
   AS invoices_amount
   FROM invoices
   WHERE invoices.invno IN (?, ?)) 
AS anon_1(10, 14)
0