Объект типа «Пользователь» не сериализуем JSON (родительский класс)

Я пытаюсь вставить запись в свою базу данных через API Flask. Связь между классами User и Project — один ко многим (у пользователя может быть много проектов). У меня нет проблем с вставкой нового пользователя, но когда я пытаюсь вставить новый проект в базу данных, он действительно работает, но атрибут пользователя в проекте не может быть преобразован в JSON. Вот мой код:

Файл модели пользователя:

class Usuario(db.Model):
 id = db.Column(db.BigInteger, primary_key=True)
 login = db.Column(db.String(80), unique=True, nullable=False)
 email = db.Column(db.String(120), unique=True, nullable=False)
 nome = db.Column(db.String(80), nullable=False)
 senha = db.Column(db.String(80), nullable=False)
 telefone = db.Column(db.String(80))
 receivedNotification = db.Column(
    db.Boolean, default=False, nullable=False)
 receivedEmail = db.Column(db.Boolean, default=False, nullable=False)
 emailVerificated = db.Column(db.Boolean, default=False, nullable=False)
 imagem = db.Column(db.LargeBinary(length=(2 ** 32) - 1))

 def __init__(self, login='', email='', nome='', senha='',
             telefone='', received_notification=False,
             received_email=False, email_verificated=False, imagem=None, init_dict=None):
    if init_dict is None:
        self.login = login
        self.email = email
        self.nome = nome
        self.senha = senha
        self.telefone = telefone
        self.receivedNotification = received_notification
        self.receivedEmail = received_email
        self.emailVerificated = email_verificated
        self.imagem = imagem
    else:
        for key in init_dict:
            setattr(self, key, init_dict[key])
class UsuarioSchema(ma.Schema):
class Meta:
    fields = ('id', 'login', 'email', 'nome', 'telefone',
              'receivedNotification', 'receivedEmail', 'emailVerificated', 'imagem')

user_schema = UsuarioSchema()
users_schema = UsuarioSchema(many=True)

Файл модели проекта:

class Projeto(db.Model):
  codigo = db.Column(db.BigInteger, primary_key=True)
  nome = db.Column(db.String(100))
  descricao = db.Column(db.String(200))
  dataCriacao = db.Column(db.Date)
  dataPrevFinalizacao = db.Column(db.Date)
  dataFinalizacao = db.Column(db.Date)
  usuarioAdmId = db.Column(db.BigInteger,
                         db.ForeignKey('usuario.id'), nullable=False)
  usuarioAdm = db.relationship('Usuario',
                             backref=db.backref('projetos', lazy=True))

  def __init__(self, nome, descricao, data_criacao,
             data_prev_finalizacao, usuario_adm_id, data_finalizacao=None):
    self.nome = nome
    self.descricao = descricao
    self.dataCriacao = data_criacao
    self.dataPrevFinalizacao = data_prev_finalizacao
    self.dataFinalizacao = data_finalizacao
    self.usuarioAdmId = usuario_adm_id

class ProjetoSchema(ma.Schema):
class Meta:
    fields = ('nome', 'descricao', 'dataCriacao',
              'dataPrevFinalizacao', 'dataFinalizacao', 'usuarioAdm')

projeto_schema = ProjetoSchema()
projetos_schema = ProjetoSchema(many=True)

Мой код маршрута для добавления проекта:

@projeto_controller.route(mapping, methods=['POST'])
def add_projeto():
  nome = request.json['nome']
  descricao = request.json['descricao']
  data_criacao = request.json['dataCriacao']
  data_criacao = datetime.strptime(
    data_criacao, '%b %d, %Y %I:%M:%S %p').date()
  data_prev_finalizacao = request.json['dataPrevFinalizacao']
  data_prev_finalizacao = datetime.strptime(
    data_prev_finalizacao, '%b %d, %Y %I:%M:%S %p').date()
  usuario_adm_id = request.json['usuarioAdm']['id']
  usuario_adm = request.json['usuarioAdm']

  usuario = mUsuario.Usuario.query.get(usuario_adm_id)
  usuarioAdm = usuario

  new_projeto = mProjeto.Projeto(nome=nome, descricao=descricao,
                               data_criacao=data_criacao,

  data_prev_finalizacao=data_prev_finalizacao,
                               usuario_adm_id=usuario_adm_id)

  new_projeto.usuarioAdm = usuarioAdm
  usuarioAdm.projetos.append(new_projeto)

  db.session.commit()
  return mProjeto.projeto_schema.jsonify(new_projeto) 

Сообщение об ошибке:

  File "/anaconda3/envs/KeepSoft/lib/python3.7/site-packages/flask/app.py", line 2309, in __call__
return self.wsgi_app(environ, start_response)
 File "/anaconda3/envs/KeepSoft/lib/python3.7/site-packages/flask/app.py", line 2295, in wsgi_app
response = self.handle_exception(e)
  File "/anaconda3/envs/KeepSoft/lib/python3.7/site-packages/flask/app.py", line 1741, in handle_exception
reraise(exc_type, exc_value, tb)
  File "/anaconda3/envs/KeepSoft/lib/python3.7/site-packages/flask/_compat.py", line 35, in reraise
raise value
  File "/anaconda3/envs/KeepSoft/lib/python3.7/site-packages/flask/app.py", line 2292, in wsgi_app
response = self.full_dispatch_request()
  File "/anaconda3/envs/KeepSoft/lib/python3.7/site-packages/flask/app.py", line 1815, in full_dispatch_request
rv = self.handle_user_exception(e)
  File "/anaconda3/envs/KeepSoft/lib/python3.7/site-packages/flask/app.py", line 1718, in handle_user_exception
reraise(exc_type, exc_value, tb)
  File "/anaconda3/envs/KeepSoft/lib/python3.7/site-packages/flask/_compat.py", line 35, in reraise
raise value
  File "/anaconda3/envs/KeepSoft/lib/python3.7/site-packages/flask/app.py", line 1813, in full_dispatch_request
rv = self.dispatch_request()
  File "/anaconda3/envs/KeepSoft/lib/python3.7/site-packages/flask/app.py", line 1799, in dispatch_request
return self.view_functions[rule.endpoint](**req.view_args)
  File "/Users/user/Python/KeepSoftPythonAPI/controller/ProjetoController.py", line 93, in add_projeto
return mProjeto.projeto_schema.jsonify(new_projeto)
  File "/anaconda3/envs/KeepSoft/lib/python3.7/site-packages/flask_marshmallow/schema.py", line 42, in jsonify
return flask.jsonify(data, *args, **kwargs)
  File "/anaconda3/envs/KeepSoft/lib/python3.7/site-packages/flask/json/__init__.py", line 321, in jsonify
dumps(data, indent=indent, separators=separators) + '\n',
  File "/anaconda3/envs/KeepSoft/lib/python3.7/site-packages/flask/json/__init__.py", line 179, in dumps
rv = _json.dumps(obj, **kwargs)
  File "/anaconda3/envs/KeepSoft/lib/python3.7/json/__init__.py", line 238, in dumps
**kw).encode(obj)
  File "/anaconda3/envs/KeepSoft/lib/python3.7/json/encoder.py", line 201, in encode
chunks = list(chunks)
  File "/anaconda3/envs/KeepSoft/lib/python3.7/json/encoder.py", line 431, in _iterencode
yield from _iterencode_dict(o, _current_indent_level)
  File "/anaconda3/envs/KeepSoft/lib/python3.7/json/encoder.py", line 405, in _iterencode_dict
yield from chunks
  File "/anaconda3/envs/KeepSoft/lib/python3.7/json/encoder.py", line 438, in _iterencode
o = _default(o)
  File "/anaconda3/envs/KeepSoft/lib/python3.7/site-packages/flask/json/__init__.py", line 81, in default
return _json.JSONEncoder.default(self, o)
  File "/anaconda3/envs/KeepSoft/lib/python3.7/json/encoder.py", line 179, in default
raise TypeError(f'Object of type {o.__class__.__name__} '
  TypeError: Object of type Usuario is not JSON serializable

Я ожидал, что будет возвращен Project JSON, поэтому я смогу преобразовать его в класс Project, который существует в моем мобильном приложении.


person Victor Yan    schedule 11.01.2019    source источник
comment
Возможный дубликат TypeError: объект Python не сериализуем JSON   -  person    schedule 12.01.2019
comment
Вы обеспечили это? Flask-SQLAlchemy должен быть инициализирован до Flask-Marshmallow.   -  person Arunmozhi    schedule 12.01.2019
comment
Вот мой ответ на Как jsonify объекты из sqlalchemy? Это работает и на этот вопрос.   -  person NicoNing    schedule 12.01.2019
comment
Для SQL Alchemy я думаю, что этот ответ самый чистый: stackoverflow.com/a/46180522/3559330   -  person mattyb    schedule 05.03.2021


Ответы (2)


У вас должна быть функция, которая возвращает объекты класса Projeto в формате словаря, а затем преобразует его в jsonify it. Что-то типа

def get_objects(self):
    """
    This function creates a dictionary and assigns
    the keys of the dictionary with the objects of
    the class and returns it in a json format. 
    """
    data = {
        'nome': self.nome,
        'descricao': self.descricao,
        'data_criacao': self.dataCriacao,
        'data_prev_finalizacao': self.dataPrevFinalizacao,
        'data_finalizacao': self.dataFinalizacao,
        'usuario_adm_id': self.usuarioAdmId
    }
    return jsonify(data)

Если вам нужно, чтобы данные для модели возвращались в формате JSON, вам нужно, чтобы данные возвращались в формате JSON. В настоящее время с этой строкой кода

return mProjeto.projeto_schema.jsonify(new_projeto)

что вы делаете, так это то, что вы пытаетесь преобразовать объект Usuario и Projeto в объект JSON, который не является допустимым типом данных, поддерживаемым функцией jsonify. Это то, что указано в сообщении

raise TypeError(f'Object of type {o.__class__.__name__} '
  TypeError: Object of type Usuario is not JSON serializable

То же самое можно сделать и для класса Usuario.

person the-chaos-magus    schedule 12.01.2019

После долгих поисков я смог решить свою проблему. Мне пришлось явно указать, что объект usuario, определенный в классе ProjetoSchema, был вложенным объектом, поэтому только при получении Projeto этот объект будет правильно загружен как JSON.

Итак, теперь это модифицированная схема:

class ProjetoSchema(ma.Schema):

  class Meta:
      fields = ('codigo', 'nome', 'descricao', 'dataCriacao',
              'dataPrevFinalizacao', 'dataFinalizacao', 'usuarioAdm')
      model = Projeto
  usuarioAdm = ma.Nested('UsuarioSchema', exclude=('projetos',))
  perfis = ma.Nested('PerfilSchema', many=True, exclude=('projeto',))


  projeto_schema = ProjetoSchema()
  projetos_schema = ProjetoSchema(many=True)

А это модифицированная UsuarioSchema:

class UsuarioSchema(ma.Schema):
  class Meta:
      fields = ('id', 'login', 'email', 'nome', 'telefone', 'receivedNotification',
              'receivedEmail', 'emailVerificated', 'imagem')
      model = Usuario
  projetos = ma.Nested('ProjetoSchema', many=True, exclude=('usuarioAdm',))
  perfis = ma.Nested('PerfilSchema', many=True, exclude=('usuario',))


  user_schema = UsuarioSchema()
  users_schema = UsuarioSchema(many=True)

Таким образом, мне не нужно будет иметь дело с бесконечными вложенными объектами, как это было до того, как я получил окончательный код. В конце концов, некоторые из этих строк не нужны, потому что, когда я получаю Usuario, я опускаю поле, которое будет содержать его дочерние Projeto, и когда я получаю Projeto, теперь он правильно показывает его родителя.

Простите мой английский, я бразилец.

Спасибо за все советы и решения.

person Victor Yan    schedule 18.01.2019