如何在 Sails 中使用 through 实现多对多关联?

How to implement many to many association using through in Sails?

我在 Mac [=78= 上使用 Sails v0.11.2MongoDB 3.2 ] X El Capitan 和我正在尝试使用尚不支持的 Through 选项实现 Many-To-Many 关联.

然而,谷歌搜索我发现这个 Waterline Github Issueelennaro,一个 github 用户,给了我一些链接和一些例子:

  1. First one
  2. Second one

我已经尝试将它们调整到我自己的 Sails 应用程序中,但我无法让它工作。我在控制台上没有收到任何错误,但记录或 document on the intermediary table is not created 只有 Form 文档在 table.

这些是我的模型:

User.js

module.exports = {
    schema: true,
    tableName: 'users',
    autoCreatedAt: false,
    autoUpdatedAt: false,

    attributes:
    {
        email               : { type: 'email', required: true, unique: true },
        encrypted_password  : { type: 'string' },
        reset_password_token: { type: 'string', defaultsTo: null},
        permission_level    : { type: 'integer', required: true, min: 1, max: 3, defaultsTo: 0 },
        belongs_to          : { type: 'string', required: true, defaultsTo: 0 },
        signin_count        : { type: 'integer', required: true, defaultsTo: 1 },
        status_active       : { type: 'boolean', required: true, defaultsTo: false },
        last_signin_at      : { type: 'datetime', defaultsTo: function (){ return new Date(); } },
        last_signin_ip      : { type: 'string', defaultsTo: '0.0.0.0' },

        // Add a reference to Person
        person_id:
        {
            model: 'person'
        },
        // Add a reference to Forms collection
        forms:
        {
            collection: 'form',
            via: 'user_id',
            through: 'userhasform'
        },
        has:
        {
            collection: 'userhasform',
            via: 'form_id'
        }
    }

};

Form.js

module.exports = {
    schema: true,
    tableName: 'forms',

    attributes:
    {
        name    : { type: 'string', required: true, unique: true },
        creator : { type: 'string', unique: false },
        sequence: { type: 'integer', autoIncrement: true },

        // Add a reference to Questions collection
        questions:
        {
            collection: 'question',
            via: 'form_id'
        },
        // Add a reference to the owners Users
        owners: {
            collection: 'user',
            via: 'form_id',
            through: 'userhasform'
        }
    }

};

UserHasForm.js

module.exports = {
    schema: true,
    tableName: 'users_have_forms',

    attributes:
    {
        to_edit     : { type: 'boolean' },
        to_delete   : { type: 'boolean' },
        user_id     : { model: 'user' },
        form_id     : { model: 'form' }
    }

};

我在其中创建表单的控制器,假设中间文档是在连接处创建的 table 是:

FormController.js

module.exports = {
    create: function (req, res)
    {
        var ownerJson = {},
            tmpFolio;

        // Get the logged user to make the Folio and then create the form
        SessionService.getUser(req, createForm);
        // Callback function
        function createForm (err, session)
        {
            // If there's no logged user or any error
            if (err || !session)
            {
                console.log(err);
                return res.json(err.status, {error: err});
            }

            console.log('User to create Folio: ', session.id);

            ownerJson.owner_a = session.first_name;
            ownerJson.owner_b = session.second_name;
            ownerJson.owner_c = session.last_name;
            // Construct the Folio creator part like AVC
            tmpFolio = FolioService.generateFolio(ownerJson);

            Form.create({
                name: req.body.name,
                creator: tmpFolio
            })
            .then(function (form){
                if (err)
                {
                    console.log(err);
                    return res.json(err.status, {error: err});
                }

                // Create the jointable record
                var createdRecord = UserHasForm.create({
                        to_edit: true,
                        to_delete: true,
                        user_id: session.id,
                        form_id: form.id
                    })
                    .then(function (createdRecord){
                        if (err)
                        {
                            console.log(err);
                            return res.json(err.status, {error: err});
                        }

                        return createdRecord;
                    });

                return [form, createdRecord];
            })
            .spread(function (form, createdRecord){
                return res.json(200,
                {
                    message: 'The form was created successfuly!',
                    data: form,
                    sharing: createdRecord
                });
            })
            .fail(function (err){
                if (err)
                {
                    console.log(err);
                    res.json(err.status, {error: err});
                }
            });
        }
    },

};

当我 运行 这段代码时,我得到了下一个错误:

[ReferenceError: UserHasForm is not defined]
Unhandled rejection TypeError: Cannot read property 'toString' of undefined

所以我想它找不到模型,所以我在模型的开头添加了下一行:

var UserHasForm = require('../models/UserHasForm');

现在我得到下一个错误:

[TypeError: UserHasForm.create is not a function]

所有这些都在列表中的 the first example 之后。

知道我为什么会收到此错误吗?

欢迎任何形式的帮助!

好吧,在尝试了很多示例之后,我终于找到了解决方案,感谢 @elennaro 他的所有支持。整个对话可以在 link 到我们在主要问题评论下开始的聊天中找到。

另外我可以告诉你,他提供的 link 中的示例(在上面的问题中)工作正常,问题是我使用的版本不支持这些功能这些例子表明。

基本上我要做的是为 NodeJSSailsJSWaterline[ 安装最新版本=48=].

就我而言,我实际上有下一个:

  • 节点 v5.3.0
  • Sails v0.11.3
  • 水线 v0.10.30

之后我必须对我的模型进行一些更改,最后它们看起来像这样:

User.js

module.exports = {

    schema: true,
    tableName: 'users',
    autoCreatedAt: false,
    autoUpdatedAt: false,

    attributes:
    {
        // username         : { type: 'string', unique: true, minLength: 5, maxLength: 15 },
        email               : { type: 'email', required: true, unique: true },
        encrypted_password  : { type: 'string' },
        reset_password_token: { type: 'string', defaultsTo: null},
        permission_level    : { type: 'integer', required: true, min: 1, max: 3, defaultsTo: 0 },
        belongs_to          : { type: 'string', required: true, defaultsTo: 0 },
        signin_count        : { type: 'integer', required: true, defaultsTo: 1 },
        status_active       : { type: 'boolean', required: true, defaultsTo: false },
        last_signin_at      : { type: 'datetime', defaultsTo: function (){ return new Date(); } },
        last_signin_ip      : { type: 'string', defaultsTo: '0.0.0.0' },

        // Add a reference to Forms collection
        forms:
        {
            collection: 'form',
            via: 'user',
            through: 'userhasform'

            // dominant: true
        }
    }

};

Form.js

module.exports = {

    schema: true,
    tableName: 'forms',

    attributes:
    {
        name    : { type: 'string', required: true, unique: true },
        creator : { type: 'string', unique: false },
        sequence: { type: 'integer', autoIncrement: true },

        // Add a reference to the owners Users
        owners: {
            collection: 'user',
            via: 'form',
            through: 'userhasform'
        }
    }

};

UserHasForm.js

module.exports = {

    schema: true,
    tableName: 'users_have_forms',

    attributes:
    {
        to_edit     : { type: 'boolean' },
        to_delete   : { type: 'boolean' },
        user        : { model: 'User', foreignKey: true, columnName: 'user_id' },
        form        : { model: 'Form',  foreignKey: true, columnName: 'form_id' }
    }

};

FormController.js

Still the same as in the question

我希望它对任何人都有用。再次感谢@Alexander Arutinyants 的支持!

有什么问题欢迎留言!