更新belongsToMany关联数据

时间:2017-08-18 07:36:35

标签: cakephp cakephp-3.x

我正在设置一个表单,用于更新person和相关的person_to_role表,person_to_role是一个中间表,可将personrole相关联n..n关系。 role有一个预定义的角色列表,不应该从某个人的范围进行修改。

我只需更新role_id表格中的descriptionperson_to_role,然后添加/删除记录

SQL

--person
+-------------+-------------+------+-----+---------+----------------+
| Field       | Type        | Null | Key | Default | Extra          |
+-------------+-------------+------+-----+---------+----------------+
| ID          | int(11)     | NO   | PRI | NULL    | auto_increment |
| NICK_NAME   | varchar(16) | YES  |     | NULL    |                |
| FIRST_NAME  | varchar(24) | NO   |     | NULL    |                |
| MIDDLE_NAME | varchar(8)  | YES  |     | NULL    |                |
| LAST_NAME   | varchar(24) | NO   |     | NULL    |                |
| BIRTH_DATE  | date        | NO   |     | NULL    |                |
| GENDER      | varchar(8)  | NO   |     | NULL    |                |
+-------------+-------------+------+-----+---------+----------------+

-- role
+-------------+--------------+------+-----+---------+----------------+
| Field       | Type         | Null | Key | Default | Extra          |
+-------------+--------------+------+-----+---------+----------------+
| ID          | int(11)      | NO   | PRI | NULL    | auto_increment |
| NAME        | varchar(24)  | NO   |     | NULL    |                |
| DESCRIPTION | varchar(100) | YES  |     | NULL    |                |
+-------------+--------------+------+-----+---------+----------------+

--person_to_role
+-------------+--------------+------+-----+---------+----------------+
| Field       | Type         | Null | Key | Default | Extra          |
+-------------+--------------+------+-----+---------+----------------+
| ID          | int(11)      | NO   | PRI | NULL    | auto_increment |
| PERSON_ID   | int(11)      | NO   | MUL | NULL    |                |
| ROLE_ID     | int(11)      | NO   | MUL | NULL    |                |
| DESCRIPTION | varchar(100) | YES  |     | NULL    |                |
+-------------+--------------+------+-----+---------+----------------+

PersonTable Model

public function initialize(array $ config)     {         父::初始化($配置);

    $this->setTable('person');
    $this->setDisplayField('ID');
    $this->setPrimaryKey('ID');

    // define relations
    $this->belongsToMany('Role', [
        'joinTable'=> 'person_to_role',
        'foreignKey' => 'PERSON_ID'
    ]);        
}

人员控制器 - >编辑()

public function edit($id = null)
{
    $person = $this->Person->get($id, [
        'contain' => ['role']
    ]);

    $this->set('roles', $this->Person->Role->find('list'));

    if ($this->request->is(['patch', 'post', 'put'])) {            
        $person = $this->Person->patchEntity($person, $this->request->getData());
        if ($this->Person->save($person)) {
            $this->Flash->success(__('The person has been saved.'));

            return $this->redirect(['action' => 'index']);
        }
        $errors = print_r($person->errors(),1);
        $this->Flash->error(__('The person could not be saved. Please, try again.<br><pre>'. $errors .'</pre>'),['escape'=> false]);
    }

    $this->set(compact('person'));
    $this->set('_serialize', ['person']);
}

edit.ctp表单字段

echo $this->Form->control('NICK_NAME');
echo $this->Form->control('FIRST_NAME');
echo $this->Form->control('MIDDLE_NAME');
echo $this->Form->control('LAST_NAME');
echo $this->Form->control('BIRTH_DATE');
echo $this->Form->control('GENDER',['type'=>'select','options'=>[''=> '- 
Please Select -', 'Male'=>'Male','Female'=>'Female']]);

Assigned roles

...
<?php foreach ($person->role as $k=>$role) { ?>
    <tr><td><?php 

   echo $this->Form->input("role.$k._joinData.ID",['type'=>'hidden','value'=>$person->role[$k]['_joinData']['ID']]);

    echo $this->Form->input("role.$k._joinData.ROLE_ID",['value'=>$role-
>ID,'options'=>$roles,'templates'=>['formGroup' =>'{{input}}']]) 

            ?></td><td>
<?php echo $this->Form->input("role.$k._joinData.DESCRIPTION",    ['value'=>$person->role[$k]['_joinData']['DESCRIPTION'],'templates'=>
['formGroup' =>'{{input}}']]);  ?></td></tr>
        <?php } ?>
...

这给了我以下POST数据:

'NICK_NAME' => 'Johny',
    'FIRST_NAME' => 'John',
    'MIDDLE_NAME' => 'J.',
    'LAST_NAME' => 'Smith',
    'BIRTH_DATE' => '1961-01-01',
    'GENDER' => 'Male',
    'role' => [
        (int) 0 => [
            '_joinData' => [
                'ID' => '1',
                'ROLE_ID' => '5',
                'DESCRIPTION' => 'person role description'
            ]
        ]
    ]
]

但更新失败并出现以下错误:

该人无法得救。请再试一次。

  Array
  (
  [role] => Array
  (
    [0] => Array
    (
      [NAME] => Array
      (
        [_required] => This field is required
      )

      [_joinData] => Array
      (
        [PERSON_ID] => Array
        (
          [_required] => This field is required
        )
      )
    )
  )
)

它要求提供person_to_role.person_id哪个是foreignKey(它应该知道当前的人ID),并且需要role.name的值。 我的关联设置错了吗?任何帮助表示赞赏。

更新2017-08-20

仍然没有去,尝试了所有可能的文档和其他Internet资源的变化。目前,我可以将验证传递给save操作,但会生成INSERT个查询而不是UPDATE,并且会在唯一约束违规时出错。

我可以访问person.IDperson_to_role.ID

protected $_accessible = [
    '*' => true,
    'ID' => true
];

我的POST数据如下所示:

[
  'NICK_NAME' => '',
  'FIRST_NAME' => 'test',
  'MIDDLE_NAME' => '',
  'LAST_NAME' => 'user',
  'BIRTH_DATE' => '1996-10-01',
  'GENDER' => 'Male',
  'role' => [
    (int) 0 => [
      '_joinData' => [
        'ID' => '153',
        'DESCRIPTION' => 'test edited text'
      ],
      'ID' => '2'
    ]
  ]
]

我在person_to_role中同时使用_joinData个记录ID尝试了两个,结果相同:

INSERT INTO person_to_role (PERSON_ID, ROLE_ID, DESCRIPTION) 
VALUES (129, 2, 'test edited text')

1 个答案:

答案 0 :(得分:0)

在尝试使用'through' association的不同方法时,我在PersonToRoleTable.php模型中添加了以下行:

$this->belongsTo('Person');
$this->belongsTo('Role');

在对这些内容进行评论后,一切都按预期工作,我可以保存更新,并在连接表中添加新的和删除现有记录。