質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

88.80%

Cakephp3系におけるHabtmの保存・更新

解決済

回答 2

投稿

  • 評価
  • クリップ 0
  • VIEW 1,227

kenkbou

score 159

お世話になります。

Cakephp3系でHabtmデータの追加・更新・削除をしたいと思っております。
追加、削除は実装出来るのですが、更新がうまくできずに困っております。

UsersTable.php

        $this->belongsToMany('Labels', [
            'foreignKey' => 'user_id',
            'targetForeignKey' => 'label_id',
            'joinTable' => 'labels_users'
        ]);

UsersController.php

        if ($this->request->is(['patch', 'post', 'put'])) {
            $data = $this->Users->patchEntity($data, $this->request->data, ['associated' => ['LabelsUsers']]);
            if ($this->Users->save($data)) {
                $this->Flash->success(__('保存完了しました。'));
                return $this->redirect(['action' => 'index']);
            }
            $this->Flash->error(__('保存に失敗しました。もう一度お試しください。'));
        }


template

foreach ($indicators as $i => $indicator) {
    $label = $data->get_label($data->id, $indicator->id);
    if (!empty($label)) {
        echo $this->Form->input('labels.' . $i . '.id', [
            'type' => 'hidden',
            'value' => $label->id
        ]);
    }
    echo $this->Form->input('labels.' . $i . '.user_id', [
        'type' => 'hidden',
        'value' => $data->id
    ]);
    echo $this->Form->input('labels.' . $i . '.label_id',[
        'type' => 'select',
        'options' => $list
    ]);
}

saveとdeleteはうまくできているのですが、
idを入れた際に、更新されず、エラーがかえってきます。

Error: SQLSTATE[23000]: Integrity constraint violation: 1062 Duplicate entry '***' for key 'user_id'
  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 過去に投稿した質問と同じ内容の質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 2

checkベストアンサー

0

Error: SQLSTATE[23000]: Integrity constraint violation: 1062 Duplicate entry '***' for key 'user_id'


ということは、重複する[user_id]をさらに登録しようとしていると思います。

$data = $this->Users->patchEntity($data, $this->request->data, ['associated' => ['LabelsUsers']]);


という形でエンティティを作成しているので、バリデーションチェックも行っているはずなので、
まずは、debug($data); exit; を上記のコード直後に書いて、何が返ってきているかを確かめる必要があります。

その出力がわかりましたら、再度コメントをお寄せ下さい。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2017/06/04 10:05

    ありがとうございます。
    debugをする以前にえらーになってしまいます、、、
    if saveのところでデバッガーを当ててみましたが、errors配列には何も入っておりませんでした。

    キャンセル

  • 2017/06/04 10:12

    今、一度全てのlabels_usersを削除して、saveしなおすという方法で解決しましたmm
    お助けありがとうございました。

    キャンセル

0

ちなみに、私はpatchEntityを使わないようにしています。

それは、レコードの更新を行うたびにSQLを

  • 更新対象レコードの検索(SELECT文)
  • レコードの更新(UPDATE文)

の2回発行しており、楽観ロックなどの処理を考慮する際に非常に難しくなるからです。

更新処理の場合は、クエリビルダを用いて
UPDATE文を必要な分だけ書いて実行すると、いろいろ楽になるのでそちらも検討してくださると幸いです。

INSERT、DELETEは今までどおりで大丈夫ですよ^^

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2017/06/04 10:13

    ありがとうございます!
    こちらの方法でも試してみます^^

    キャンセル

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 88.80%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る