web-dev-qa-db-ja.com

sqlalchemy.orm.exc.UnmappedInstanceError in flask

SQLAlchemyのドキュメントを読んでいますが、理解できません。エラー(UnmappedInstanceError)は、何かがマップされていないことを示しています。マッピングされていないものは何ですか?私は本当にsqlalchemyを取得していないので、裸のsqliteの使用に戻りたいのですが、多くの人がこれを推奨しているので、学ぶべきだと思いました。ここにトレースバックがあります:

File "C:\Users\Me\repos\mandj\venv\lib\site-packages\flask\app.py", line 1836, in __call__
return self.wsgi_app(environ, start_response)

File "C:\Users\Me\repos\mandj\venv\lib\site-packages\flask\app.py", line 1820, in wsgi_app
response = self.make_response(self.handle_exception(e))

File "C:\Users\Me\repos\mandj\venv\lib\site-packages\flask\app.py", line 1403, in handle_exception
reraise(exc_type, exc_value, tb)

File "C:\Users\Me\repos\mandj\venv\lib\site-packages\flask\app.py", line 1817, in wsgi_app
response = self.full_dispatch_request()

File "C:\Users\Me\repos\mandj\venv\lib\site-packages\flask\app.py", line 1477, in full_dispatch_request
rv = self.handle_user_exception(e)

File "C:\Users\Me\repos\mandj\venv\lib\site-packages\flask\app.py", line 1381, in handle_user_exception
reraise(exc_type, exc_value, tb)

File "C:\Users\Me\repos\mandj\venv\lib\site-packages\flask\app.py", line 1475, in full_dispatch_request
rv = self.dispatch_request()

File "C:\Users\Me\repos\mandj\venv\lib\site-packages\flask\app.py", line 1461, in dispatch_request
return self.view_functions[rule.endpoint](**req.view_args)

File "C:\Users\Me\repos\mandj2\app\views.py", line 170, in add_manentry
db.session.add(q)

File "C:\Users\Me\repos\mandj\venv\lib\site-packages\sqlalchemy\orm\scoping.py", line 149, in do
return getattr(self.registry(), name)(*args, **kwargs)

File "C:\Users\Me\repos\mandj\venv\lib\site-packages\sqlalchemy\orm\session.py", line 1452, in add
raise exc.UnmappedInstanceError(instance)

UnmappedInstanceError: Class '__builtin__.unicode' is not mapped

該当するコードは次のとおりです。

@app.route('/addm', methods=['POST'])
def add_mentry():
    if not session.get('logged_in'):
        abort(401)
    form = MForm(request.form)
    filename = ""
    if request.method == 'POST':
        cover = request.files['cover']
        if cover and allowed_file(cover.filename):
            filename = secure_filename(cover.filename)
            cover = cover.save(os.path.join(app.config['UPLOAD_FOLDER'], filename))

    q = request.form['name']
    # do for 12 more fields
    db.session.add(q)
    db.session.commit()
    flash('New entry was successfully posted')
    return redirect(url_for('moutput'))
19
user2986242
_q = request.form['name']
# do for 12 more fields
db.session.add(q)
_

_request.form['name']_はUnicode値を返します。その後、あなたは...

_db.session.add(q)
_

セッションの目的は、エンティティ(Pythonオブジェクト)を追跡することであり、実行しようとしているように見える個々のUnicode値ではありません( こちら を参照してください)セッションの詳細については)。したがって、マッピングのあるオブジェクト(Userオブジェクトなどのオブジェクトを "Mapping"セクション に示すように追加する必要があります。 ORMチュートリアル )、しかし実際には単純なユニコード値を渡しています

使用しているのは、SQLAlchemyの一部であるORM(オブジェクトリレーショナルマッパー)です。 ORMは、新しいpythonオブジェクトを作成し、そのオブジェクトをセッションに「追加」することでSQLを自動的に生成できるようにすることを試みます。

_a = MyEntity()
session.add(a)
session.commit() # Generates SQL to do an insert for the table that MyEntity is for
_

ORM機能を使用せずにSQLAlchemyを使用できることに注意してください。 db.execute('INSERT...', val1, val2)を実行して、すでに「裸の」SQLを置き換えることができます。 SQLAlchemyは接続プーリングなどを提供します(ただし、SQLiteを使用している場合は、おそらく接続プーリングは気にしません)。

Flask-SQLAlchemyを理解したい場合は、まず( tutorials 。次に、Flask-SQLAlchemyがどのように機能するかを理解します。

8
Mark Hildreth

非モデルオブジェクトをセッションに追加すると、UnmappedInstanceErrorが取得されます。

あなたの場合、qはおそらくモデルオブジェクトではなくUnicode文字列でした。 (Python 2.xを使用しているようですPython 3.xではstrと表示されます)、これが根本原因の行ですために UnmappedInstanceError: Class '__builtin__.unicode' is not mapped

  File "C:\Users\Me\repos\mandj2\app\views.py", line 170, in add_manentry
  db.session.add(q)
9
Devy

データベースを変更する場合は、以下を参照してください。変更するオブジェクトのアイテムに対処する必要があります。下記参照:

client = session.query(Clients).filter_by(id=client_id).one()
if request.method == 'POST':
    new_name = request.form['form_name']
    client.name = new_name
    session.add(client)
    session.commit()

「クライアント」オブジェクトを見るとわかるように、オブジェクト内の情報の中に「名前」があります。 「名前」のみを直接変更したいので、それに対処する必要があります。 (client.name)


データベースに新しいものを追加する場合:ここでormを使用してデータベースに新しい値を追加する場合、データを受信するオブジェクト内のアイテムを指定する必要があります。この場合(Client.name)

    if request.method == 'POST':
        new_name = request.form['form_name']
        name = Clients(name=new_name)
        session.add(name)
        session.commit()

お役に立てば幸いです。

0
Andy