Spaces:
Sleeping
Sleeping
File size: 6,936 Bytes
b39c0ba d25fc26 b39c0ba d25fc26 b39c0ba 1c14126 b39c0ba 015e632 0767396 015e632 b39c0ba 0767396 b39c0ba 69beac6 b39c0ba 69beac6 b39c0ba 69beac6 b39c0ba 69beac6 b39c0ba 69beac6 b39c0ba 69beac6 b39c0ba |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 |
from sqlalchemy import select, delete, update
from sqlalchemy.exc import SQLAlchemyError
from utils.error_handlers import handle_error, not_found_error, no_entries_found, handle_exception
from fastapi.responses import JSONResponse
from typing import List, Type, Optional
class BaseQuery:
def __init__(self, user):
self.user = user
self.user_id = user.get("id")
def _fetch(self, db, query, not_found_message, multiple: bool = False):
"""Fetch a single or multiple results based on the 'multiple' flag."""
try:
if multiple:
results = db.execute(query).all()
if not results:
return []
return results
else:
result = db.execute(query).scalar_one_or_none()
if not result:
return []
return result
except Exception as e:
return handle_error(
e,
"Failed to fetch entry" if not multiple else "Failed to fetch entries",
)
def _handle_commit(self, db):
try:
db.commit()
except SQLAlchemyError as e:
db.rollback()
return handle_exception(e)
except Exception as e:
db.rollback()
return handle_error(e, "Operation failed")
def add(self, db, instance):
"""Add a new entry."""
db.add(instance)
return self._handle_commit(db)
def insert_entries(self, db, entries):
"""Insert multiple entries."""
db.add_all(entries)
return self._handle_commit(db)
def delete(self, db, model, id=None, filter_conditions=None):
"""Delete an entry by ID with optional filter conditions."""
# Build the query to select the entry
query = select(model)
if id:
query = query.where(model.id == id)
if filter_conditions:
query = query.where(*filter_conditions)
# Fetch the entry
entry = self._fetch(db, query, f"Entry with ID {id} not found.", multiple=False)
if isinstance(entry, JSONResponse):
return entry
# Build the delete query
delete_query = delete(model).where(model.id == id)
if filter_conditions:
delete_query = delete_query.where(*filter_conditions)
# Execute the delete query and commit
db.execute(delete_query)
return self._handle_commit(db)
def delete_all(self, db, model, filter_conditions=None):
"""Delete all entries or based on filters."""
query = delete(model)
if filter_conditions:
query = query.where(*filter_conditions)
db.execute(query)
return self._handle_commit(db)
def update(self, db, model, id, update_data, filter_conditions=None):
"""Update an entry by ID."""
# Define the initial query to fetch the entry
query = select(model).where(model.id == id)
# Append additional filter conditions if provided
if filter_conditions:
query = query.where(model.id == id, *filter_conditions)
# Attempt to fetch the entry
not_found_message = f"Entry with ID {id} not found."
entry = self._fetch(db, query, not_found_message, multiple=False)
# Check if the entry was found
if isinstance(entry, JSONResponse):
return entry
# Prepare the update statement
stmt = update(model).where(model.id == id).values(update_data)
db.execute(stmt)
# If filter conditions were provided, apply them to the update as well
if filter_conditions:
filter_stmt = update(model).where(model.id == id, *filter_conditions).values(update_data)
db.execute(filter_stmt)
return self._handle_commit(db)
def update_entries(self, db, model, update_data, filter_conditions=None):
"""Update multiple entries with optional filtering."""
query = select(model)
if filter_conditions:
query = query.where(*filter_conditions)
not_found_message = "No entries found matching the filter conditions."
results = self._fetch(
db, query, not_found_message , multiple=True
)
if isinstance(results, JSONResponse):
return results
db.execute(update(model).where(*filter_conditions).values(update_data))
return self._handle_commit(db)
def get(
self,
db,
model: Type = None,
id: Optional[int] = None,
filter_conditions=None,
columns: Optional[List[str]] = None,
multiple: bool = False,
):
"""Get one or multiple entries, filtered by ID or conditions."""
if columns:
query = select(*columns)
else:
query = select(model)
if id:
query = query.where(model.id == id)
if filter_conditions:
query = query.where(*filter_conditions)
return self._fetch(
db,
query,
"Entry not found." if not multiple else "No entries found.",
multiple=multiple,
)
def get_with_joins(
self,
db,
join_models: List[Type],
join_conditions: List=None,
model: Type=None,
filter_conditions=None,
columns: Optional[List[str]] = None,
multiple: bool = False,
):
"""Get one or multiple entries with joins and optional filters."""
if columns:
query = select(*columns)
else:
query = select(model, *join_models).select_from(model)
# Apply joins
if join_conditions:
for join_model, join_condition in zip(join_models, join_conditions):
query = query.join(join_model, join_condition)
else:
query = query.join(*join_models)
# Apply filtering by user ID and optional conditions
if filter_conditions:
query = query.where(*filter_conditions)
return self._fetch(
db,
query,
"Entry not found." if not multiple else "No entries found.",
multiple=multiple,
)
def get_columns(
self,
db,
columns: List[str],
model=None,
filter_conditions=None,
id: Optional[int] = None,
multiple: bool = False,
):
"""Get specific columns by ID or filtering."""
query = select(*columns).select_from(model)
if id:
query = query.where(model.id == id)
if filter_conditions:
query = query.where(*filter_conditions)
return self._fetch(
db,
query,
"Entry not found." if not multiple else "No entries found.",
multiple=multiple,
)
|