Форум программистов, компьютерный форум, киберфорум
Python: Базы данных
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
 Аватар для GennDALF
15 / 14 / 2
Регистрация: 24.09.2009
Сообщений: 68

SQLAlchemy: добавление объектов в session

26.08.2023, 21:52. Показов 1409. Ответов 1

Студворк — интернет-сервис помощи студентам
Во время экспериментов с SQLAlchemy 2.0 столкнулся с непонятным мне поведением:
Code
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
# загрузка объектов из базы данных
>>> student1 = session.execute(...)
>>> ex1 = session.execute(...)
>>> ex2 = session.execute(...)
 
# новые объекты
>>> hw1 = Homework(student=student1)
>>> t1 = Task(exercise=ex1, homework=hw1)
>>> t2 = Task(exercise=ex2, homework=hw1)
 
>>> session.add(hw1)
...
sqlalchemy.exc.InvalidRequestError: Can't attach instance
    <Task at 0x1d9f9df8750>; another instance with key 
    (<class 'model.orm.Task'>, (1,), None) is already 
    present in this session.
Сначала подумал на проблемы с каскадным выполнением операций. Но на соответствующей странице документации не нашёл ничего похожего, и никаких упоминаний выброшенного исключения. Возможно, плохо искал.

Ниже код используемых ORM классов.

Python
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
class Student(abstract.Base, abstract.Model):
    id = mapped_column(sqltypes.Integer(), primary_key=True)
    ...
    homeworks = relationship('Homework', back_populates='student', lazy='selectin')
 
class Homework(abstract.Base, abstract.Model):
    id = mapped_column(sqltypes.Integer(), primary_key=True)
    ...
    student_id = mapped_column(schema.ForeignKey('students.id'), nullable=False)
    student = relationship('Student', back_populates='homeworks')
    tasks = relationship('Task', back_populates='homework', lazy='selectin')
 
class Exercise(abstract.Base, abstract.Model):
    id = mapped_column(sqltypes.Integer(), primary_key=True)
    ...
    tasks = relationship('Task', back_populates='exercise', lazy='selectin')
 
class Task(abstract.Base, abstract.Model):
    id = mapped_column(sqltypes.Integer(), primary_key=True)
    ...
    exercise_id = mapped_column(schema.ForeignKey('exercises.id'), nullable=False)
    homework_id = mapped_column(schema.ForeignKey('homeworks.id'), nullable=False)
    exercise = relationship('Exercise', back_populates='tasks')
    homework = relationship('Homework', back_populates='tasks')
Может ли кто-то по этой информации подсказать, что происходит? Был бы весьма признателен за пояснения!
0
IT_Exp
Эксперт
34794 / 4073 / 2104
Регистрация: 17.06.2006
Сообщений: 32,602
Блог
26.08.2023, 21:52
Ответы с готовыми решениями:

Добавление данных в таблицу БД в цикле for. sqlalchemy PYTHON
Помогите с кодом Python. Нужно скопировать записи с БД с заменой одного значения. Есть модель БД: lass Group(db.Model): ...

Что твориться с сессиями ? (strListingID = Session('ListingID'); strAccountID = Session('AccountID'); strActive = Session('Active'); )
-----------------------page1.asp-------------------------- Basicrs = Server.CreateObject('ADODB.Recordset'); theSQL='....'; ...

Почему session('aaa') обнуляется при session.Abandon, а session.sessionid остается таким же ?
Народ помогите &lt;% response.write(session.sessionid) response.write(session('aaa')) session.Abandon ...

1
 Аватар для GennDALF
15 / 14 / 2
Регистрация: 24.09.2009
Сообщений: 68
27.08.2023, 19:58  [ТС]
Поправка: при прямой работе с объектом session – как я написал сообщением выше – ошибка не воспроизводится, это я погорячился.)


Ошибка имеет место при использовании собственного класса для доступа к БД.

Code
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
>>> app = application.Application()
 
>>> student1 = app.pm.read_model(orm.Student, 'id', 1)
>>> ex1 = app.pm.read_model(orm.Exercise, 'id', 1)
>>> ex2 = app.pm.read_model(orm.Exercise, 'id', 2)
 
>>> hw2 = orm.Homework(student=student1)
>>> t1 = orm.Task(exercise=ex1, homework=hw2)
>>> t2 = orm.Task(exercise=ex2, homework=hw2)
 
>>> app.pm.write_model(hw2)
... 
sqlalchemy.exc.InvalidRequestError: Can't attach instance 
    <Task at 0x1db45f2ded0>; another instance with key 
    (<class 'ego.model.orm.Task'>, (1,), None) is already 
    present in this session.
Но этот факт лично у меня вызывает только большее недоумение.
Потому что в используемом классе отдельные объекты сессии создаются для каждой операции (см. код ниже)
да, я знаю, что это неоптимально с точки зрения управления транзакциями – но, во-первых, лучше пока не умею, во-вторых, в данном проекте значения не имеет
А если сессии отдельные, то откуда в них возьмётся "лишний" объект Task?

Совсем запутался. Объясните тугодуму, не оставьте в невежестве.

utils, abstract и exceptions – собственные модули проекта
Python
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
class PersistenceManager:
    __instance: Self = None
 
    def __new__(cls, *args, **kwargs):
        if cls.__instance is None:
            cls.__instance = super().__new__(cls)
            cls.__instance.__init__(*args, **kwargs)
        return cls.__instance
 
    _default_models_config: dict = utils.CONFIG['models-database']
 
    def __init__(self, models_config: dict = None):
        if not models_config:
            models_config = self._default_models_config
        self.models: Engine = create_engine(URL.create(**models_config))
 
    def write_model(self, obj):
        with Session(self.models) as session:
            session.add(obj)
            try:
                session.commit()
            except IntegrityError as exc:
                if 'UNIQUE' not in str(exc):
                    print(exc)
 
    def read_model(self, model: abstract.Model, primary_key: str, value):
        with Session(self.models) as session:
            stmt = select(model).where(getattr(model, primary_key) == value)
            result = session.execute(stmt).scalar()
        if not result:
            raise exceptions.NoResultsError()
        return result
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
BasicMan
Эксперт
29316 / 5623 / 2384
Регистрация: 17.02.2009
Сообщений: 30,364
Блог
27.08.2023, 19:58
Помогаю со студенческими работами здесь

Добавление get параметров в SESSION
if (empty($errorReg)) { $_SESSION = $_GET; $_SESSION = $_GET; $_SESSION = $_GET; $_SESSION = $_GET; $_SESSION =...

Добавление и удаление объектов. Рисование объектов на экране
Доброго времени суток. Есть очень интересующий вопрос, как сделать добавление, обновление, удаление объектов? Допустим есть класс...

session/session.php on line 423 / session/session.php on line 426
Установил джумлу и появилась вот такая ошибка: Warning: session_start() : Cannot send session cookie - headers already sent by (output...

Warning: session_start() [function.session-start]: Cannot send session cookie - headers already sent
Открыл движок а точне файл index.php та начало кода &lt;?php /** * Файл index.php расположен в корне CMS является единственной...

Warning: session_start() [function.session-start]: Cannot send session cookie - headers already sent
Пытаюсь сделать корзину товаров, проблема с session_start выдает вот такую ошибку: Warning: session_start() : Cannot send session cookie -...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
2
Ответ Создать тему
Новые блоги и статьи
Access
VikBal 11.12.2025
Помогите пожалуйста !! Как объединить 2 одинаковые БД Access с разными данными.
Новый ноутбук
volvo 07.12.2025
Всем привет. По скидке в "черную пятницу" взял себе новый ноутбук Lenovo ThinkBook 16 G7 на Амазоне: Ryzen 5 7533HS 64 Gb DDR5 1Tb NVMe 16" Full HD Display Win11 Pro
Музыка, написанная Искусственным Интеллектом
volvo 04.12.2025
Всем привет. Некоторое время назад меня заинтересовало, что уже умеет ИИ в плане написания музыки для песен, и, собственно, исполнения этих самых песен. Стихов у нас много, уже вышли 4 книги, еще 3. . .
От async/await к виртуальным потокам в Python
IndentationError 23.11.2025
Армин Ронахер поставил под сомнение async/ await. Создатель Flask заявляет: цветные функции - провал, виртуальные потоки - решение. Не threading-динозавры, а новое поколение лёгких потоков. Откат?. . .
Поиск "дружественных имён" СОМ портов
Argus19 22.11.2025
Поиск "дружественных имён" СОМ портов На странице: https:/ / norseev. ru/ 2018/ 01/ 04/ comportlist_windows/ нашёл схожую тему. Там приведён код на С++, который показывает только имена СОМ портов, типа,. . .
Сколько Государство потратило денег на меня, обеспечивая инсулином.
Programma_Boinc 20.11.2025
Сколько Государство потратило денег на меня, обеспечивая инсулином. Вот решила сделать интересный приблизительный подсчет, сколько государство потратило на меня денег на покупку инсулинов. . . .
Ломающие изменения в C#.NStar Alpha
Etyuhibosecyu 20.11.2025
Уже можно не только тестировать, но и пользоваться C#. NStar - писать оконные приложения, содержащие надписи, кнопки, текстовые поля и даже изображения, например, моя игра "Три в ряд" написана на этом. . .
Мысли в слух
kumehtar 18.11.2025
Кстати, совсем недавно имел разговор на тему медитаций с людьми. И обнаружил, что они вообще не понимают что такое медитация и зачем она нужна. Самые базовые вещи. Для них это - когда просто люди. . .
Создание Single Page Application на фреймах
krapotkin 16.11.2025
Статья исключительно для начинающих. Подходы оригинальностью не блещут. В век Веб все очень привыкли к дизайну Single-Page-Application . Быстренько разберем подход "на фреймах". Мы делаем одну. . .
Фото: Daniel Greenwood
kumehtar 13.11.2025
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2025, CyberForum.ru