SQLalchemy AttributeError: объект 'str' не имеет атрибута '_sa_instance_state'

Я пытаюсь добавить элемент в свою базу данных с помощью SQLAlchemy + Python, но продолжаю получать сообщение об ошибке.

Моя database_setup.py:

class company(Base):
    __tablename__ = 'company'
    compID = Column(Integer, primary_key = True)
    name = Column(String(80), nullable = False)

class item(Base):
    __tablename__ = 'items'
    itemID = Column(Integer, primary_key = True)
    name = Column(String(80), nullable = False)
    category = Column(String(250))
    description = Column(String(250))
    price = Column(String(8))
    compID = Column(Integer, ForeignKey('company.compID'))
    company = relationship(company)

после импорта sqlalchemy в мой терминал, я определяю элемент для вставки:

JawboneUP3 = item(
    itemID="1",
    name="Jawbone UP3",
    description="The latest UP!", 
    category="tracker",
    price="$174.99",
    company="Jawbone"
)

и нарисуйте сеанс для добавления и фиксации:

session.add(JawboneUP3)
session.commit()

Когда я отправляю сообщение, я продолжаю получать эту ошибку:

Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
  File "/usr/lib/python2.7/dist-packages/sqlalchemy/orm/session.py", line 1399, in add
    self._save_or_update_state(state)
  File "/usr/lib/python2.7/dist-packages/sqlalchemy/orm/session.py", line 1417, in _save_or_update_state
    halt_on=self._contains_state):
  File "/usr/lib/python2.7/dist-packages/sqlalchemy/orm/mapper.py", line 2037, in cascade_iterator
    parent_dict, visited_states, halt_on))
  File "/usr/lib/python2.7/dist-packages/sqlalchemy/orm/properties.py", line 932, in cascade_iterator
    get_all_pending(state, dict_)
  File "/usr/lib/python2.7/dist-packages/sqlalchemy/orm/attributes.py", line 761, in get_all_pending
   ret = [(instance_state(current), current)]
AttributeError: 'str' object has no attribute '_sa_instance_state'

Я добавил объект "Jawbone" в таблицу своей компании, и я понимаю, что мой "JawboneUP3" должен вернуться. Этот объект был добавлен правильно через форму браузера, которую я включил через мой веб-сервер script. Я считаю, что должен иметь возможность добавлять элементы прямо из терминала.

Ответ 1

Я думаю, что проблема заключается в том, как вы определяете связанную схему компании:

JawboneUP3 = item(itemID = "1", name = "Jawbone UP3", description = "The latest UP!", 
                  category = "tracker", price = "$174.99", company = "Jawbone")
                                                           # HERE^

Конструктор item ожидает экземпляр company, но вы передаете строковое значение. Исправьте это:

JawboneUP3 = item(itemID="1", 
                  name="Jawbone UP3", 
                  description="The latest UP!", 
                  category="tracker", 
                  price="$174.99", 
                  company=company(name="Jawbone"))

Ответ 2

Прежде всего, при определении класса в Python полезно начинать имена с прописными буквами следующим образом:

class Company(Base):
    __tablename__ = 'company'
    compID = Column(Integer, primary_key = True)
    name = Column(String(80), nullable = False)

class Item(Base):
    __tablename__ = 'items'
    itemID = Column(Integer, primary_key = True)
    name = Column(String(80), nullable = False)
    category = Column(String(250))
    description = Column(String(250))
    price = Column(String(8))
    compID = Column(Integer, ForeignKey('company.compID'))
    company = relationship(company)

При этом не то, почему ваш код вызывает ошибку.:)

Причина

Конструктор Item ожидает, что экземпляр объекта Company будет передан как значение переменной Company

Решение 1: компания Jawbone не существует в вашей базе данных

Здесь ответ @alecxe действителен.

Вы должны заменить свой код на:

JawboneUP3 = Item(itemID="1", 
                  name="Jawbone UP3", 
                  description="The latest UP!", 
                  category="tracker", 
                  price="$174.99", 
                  company=company(name="Jawbone"))

Добавление этого объекта в сеанс и внесение изменений фактически приведет к двум записям в вашей базе данных:

  • Запись элемента с именем "Jawbone UP3"
  • Строка в базе данных компании, компания с именем "Jawbone"

Решение 2: Компания Jawbone существует в вашей базе данных

Здесь вы должны получить компанию Jawbone из своей таблицы "компания" и передать ее в качестве аргумента конструктору Item, например:

jawbone = session.query(Company).filter_by(name="Jawbone").first()

JawboneUP3 = Item(itemID="1", 
                  name="Jawbone UP3", 
                  description="The latest UP!", 
                  category="tracker", 
                  price="$174.99", 
                  company=jawbone)

Для параметра session проверьте this

Ответ 3

from flask import Flask
from flask import request,redirect,render_template
from flask_sqlalchemy import SQLAlchemy

app=Flask(__name__)
app.config['SQLALCHEMY_DATABASE_URI'] = 'sqlite:///students1.sqlite3'
db=SQLAlchemy(app)

class Categ(db.Model):
    id=db.Column(db.Integer,primary_key=True)
    name=db.Column(db.String(100))
    product=db.relationship('Product',backref='owner',lazy=True)

class Product(db.Model):
    id=db.Column(db.Integer,primary_key=True)
    pname=db.Column(db.String(100))
    price=db.Column(db.Integer)
    categ_id=db.Column(db.Integer,db.ForeignKey('categ.id'))

@app.route('/')
def home():
    return 'home'

@app.route('/post',methods=['POST','GET'])
def Productform():
    ob=Categ.query.all()
    if request.method=='POST':
        owner=request.form['ca']
        user = Categ.query.filter_by(name=owner).first()

        user=Product(pname=request.form['pname'],price=request.form['price'],owner=user)
        db.session.add(user)
        db.session.commit()

        return 'submit'

    return render_template('product.html',ob=ob)



@app.route('/categ',methods=['POST','GET'])
def Categoryform():
    if request.method=='POST':
        user=Categ(name=request.form['cname'])
        db.session.add(user)
        db.session.commit()

        return 'submit'

    return render_template('categ.html')


if __name__ == '__main__':
   app.run(debug=True)
   db.create_all()