Sequelize JSON data type

前端 未结 5 710
北荒
北荒 2021-02-02 14:22

I have defined a model as

module.exports = function (sequelize, DataTypes) {
  const MyModel = sequelize.define(\'MyModel\', {
    data: {
      type: DataTypes         


        
相关标签:
5条回答
  • 2021-02-02 14:26

    JSON data types aren't supported for MySQL.

    See the docs here http://docs.sequelizejs.com/en/v3/docs/models-definition/#data-types

    A work around could be to use text and stringify/parse when querying it

    0 讨论(0)
  • 2021-02-02 14:28

    No need to use getters and setters as JSON is now supported by sequelize.See sequelize api

    0 讨论(0)
  • 2021-02-02 14:39

    Jalal's answer was great but didn't work for me unless I tweaked it a little. Here's what worked for me:

    First: create a migration that adds the field you want as type TEXT. example - I want to add a field called address to the Stores table:

    module.exports = {
      up: (queryInterface, Sequelize) => {
        return queryInterface.addColumn("Stores", "address", Sequelize.TEXT);
      },
    
      down: (queryInterface, Sequelize) => {
        return queryInterface.removeColumn("Stores", "address");
      }
    };
    

    Next: Inside your model, add the field with getters and setters to your list (object) of fields and datatypes:

    address: {
      type: DataTypes.TEXT,
        get: function() {
          return JSON.parse(this.getDataValue("address"));
        },
        set: function(value) {
          return this.setDataValue("address", JSON.stringify(value));
        }
    },
    

    So my entire model looks like this:

    module.exports = (sequelize, DataTypes) => {
      const Store = sequelize.define(
        "Store",
        {
          name: DataTypes.STRING,
          isActive: DataTypes.BOOLEAN,
          address: {
            type: DataTypes.TEXT,
            get: function() {
              return JSON.parse(this.getDataValue("address"));
            },
            set: function(value) {
              return this.setDataValue("address", JSON.stringify(value));
            }
          }
        },
        {}
      );
      Store.associate = function(models) {
        // associations can be defined here
        Store.hasMany(models.Order, {
          foreignKey: "id",
          targetKey: "storeId"
        });
      };
      return Store;
    };
    

    Now you can create and retrieve records just like you would with a JSON type field in the db.

    For example: const store = await Store.create({name: "Joe's corner store", address: {address1: "123 test street", city: "Los Angeles", state: "CA"}})

    Some notes:

    In the above code blocks...

    • Replace address with the field name you want to use.
    • Replace Stores with your model/table name.
    • I added return on the setter, otherwise it was erroring out when trying to create a new record (the way Jalal has it).
    0 讨论(0)
  • 2021-02-02 14:41

    The easiest solution is to use this little library called sequelize-json

    Create a database and a Schema:

    var Sequelize = require('sequelize'),
      JsonField = require('sequelize-json'),
      db,
      User;
    
    db = new Sequelize('database', 'username', 'password', {
      dialect: 'sqlite',
      logging: false
    });
    
    User = db.define('User', {
      username: Sequelize.STRING,
      jsonField: JsonField(db, 'User', 'jsonField')
    });
    

    Note the parameters of JsonField, you pass your Sequelize instance, the name of the model, and the name of the field. A little awkard, but this is needed in order to add the proper hooks to the model instance.

    Now, you can always treat that field as a json object:

    User.create({
          username: 'Scott',
          jsonField: {
            likes: ['running', 'node']
          }
        })
        .then(function(user) {
          user.jsonField.likes.push('tests');
          return user.save();
        })
        .then(function(user) {
          expect(user.jsonField).to.be.a('object');
          expect(user.jsonField.likes).to.have.length(3);
        });
    It will work with normal save commands, as well as updateAttribute commands.

    0 讨论(0)
  • 2021-02-02 14:43

    It's not supported yet MySQL JSON Data Type #4727. But you can do something like this:

     module.exports = function (sequelize, DataTypes) {
          const MyModel = sequelize.define('MyModel', {
            data: {
              type: Sequelize.TEXT,
               get: function () {
                    return JSON.parse(this.getDataValue('value'));
                },
                set: function (value) {
                    this.setDataValue('value', JSON.stringify(value));
                },
        ,
              ...
            },
            ...
          });
    
          return MyModel;
    };
    

    I also found this package on github sequelize-json you can give it a try if you don't want to use getters and setters.

    0 讨论(0)
提交回复
热议问题