0

我有 2 个模型所有者和用户关联为用户属于所有者。

我只有一个表格来收集两个模型/表格的数据。

表单布局在下面的 add.ctp 中声明:

<div class="container-fluid">
    <div class="row">
        <div class="col-md-9">
            <h1><?= __("Création d'un nouveau propriétaire") ?></h1>
            <p>&nbsp;</p>
            <div class="col-md-19 col-md-offset-2 container-fluid well">
                <?= $this->Form->create($owner, array('novalidate' => true)) ?>
                    <fieldset>
                        <div class="form-group">
                            <?= $this->Form->input('username', array('label' => __("Nom d'utilisateur :"), 'class' => 'form-control', 'placeholder' => __("Nom d'utilisateur"), 'autocomplete' => 'off')); ?>
                        </div>
                        <div class="form-group">
                            <?=  $this->Form->input('password', array('label' => __("Mot de passe :"), 'class' => 'form-control', 'placeholder' => __("Mot de passe"))); ?>
                        </div>

                        // I cut some fields but username and password belong to User, postcode and city belong to Owner

                        <div class="form-group">
                            <?= $this->Form->input('postcode', array('label' => __("Code postal :"), 'class' => 'form-control', 'placeholder' => __("Code postal"))); ?>
                        </div>
                        <div class="form-group">
                            <?= $this->Form->input('city', array('label' => __("Ville :"), 'class' => 'form-control', 'placeholder' => __("ville"))); ?>
                        </div>

                    </fieldset>
                <?= $this->Form->button(__('Enregistrer')) ?>
                <?= $this->Form->end() ?>
            </div>
        </div>
    </div>
</div>

此表单由 OwnersController.php 控制

public function add() {

    $this->loadModel('Users');

    $user = $this->Users->newEntity($this->request->data);
    $owner = $this->Owners->newEntity($this->request->data);

    $addDatas = [ 
            'owner_id' => 'id',
            'email'     => $owner['email'],
            'role'  => 'owner',
            'token'     => md5(time() . '-' . uniqid()),
    ];

    $user = $this->Users->patchEntity($user, $addDatas);

    $owner->users = [$user];

    if ($this->request->is('post')) {
        if ($this->Owners->validate($owner)) {
            if ($this->Owners->save($owner)) {
                $user = $this->Users->get($owner->users[0]['id']);
                $email = new Email('gmail');
                $email->template('activationLink')
                    ->emailFormat('text')
                    ->to($owner['email'])
                        ->from('myemail@monsite.com')
                    ->subject(__('Votre inscription sur monsite.com'))
                    ->viewVars(['user' => $user])
                    ->send();

                $this->Flash->success(__("Merci de vous être enregistré. un email a été envoyé à {0} pour activer votre compte", $owner['email']));
                return $this->redirect(['action' => 'index']);
            }
            $this->Flash->error(__("Impossible de vous enregistrer, veuillez corriger les erreurs"));
        } else {
            $this->Flash->error(__("Impossible de vous enregistrer, veuillez corriger les erreurs"));
        }
    }

    $this->set(compact('owner'));
}

一切正常,我可以输入我的数据,验证并神奇地保存所有者和用户的数据,这要归功于神话般的 ORM。

但是我对来自验证器的错误消息有疑问。

无论错误来自所有者的验证器还是用户的验证器,我都会在表单中看到我的 flash 消息告诉您有问题,但我只看到所有者数据的字段错误消息,而不是用户数据的错误消息。

如果我在提交空白表单后调试所有者,我可以看到:

object(App\Model\Entity\Owner) {

    'new' => true,
    'accessible' => [
        'email' => true,
        'phone' => true,
        'company' => true,
        'tva_number' => true,
        'address' => true,
        'postcode' => true,
        'city' => true,
        'users' => true
    ],
    'properties' => [
        'company' => '',
        'email' => '',
        'phone' => '',
        'address' => '',
        'postcode' => '',
        'city' => '',
        'users' => [
            (int) 0 => object(App\Model\Entity\User) {

                'new' => true,
                'accessible' => [
                    'username' => true,
                    'password' => true,
                    'first_name' => true,
                    'last_name' => true,
                    'email' => true,
                    'role' => true,
                    'owner' => true,
                    'sites_user' => true,
                    'active' => true,
                    'token' => true
                ],
                'properties' => [
                    'username' => '',
                    'password' => '$2y$10$EtGitvmw7CeIOJOeA4eNKuZBifd97f165O9I5gidpiC7p1NYRGsE.',
                    'first_name' => '',
                    'last_name' => '',
                    'email' => '',
                    'role' => 'owner',
                    'token' => '4f842f347e06aff9a3778199c9a8d20a'
                ],
                'dirty' => [
                    'username' => true,
                    'password' => true,
                    'first_name' => true,
                    'last_name' => true,
                    'email' => true,
                    'role' => true,
                    'token' => true
                ],
                'original' => [],
                'virtual' => [],
                'errors' => [
                    'username' => [
                        (int) 0 => 'Vous devez indiquer un nom d'utilisateur'
                    ],
                    'first_name' => [
                        (int) 0 => 'Vous devez indiquer votre prénom'
                    ],
                    'last_name' => [
                        (int) 0 => 'Vous devez indiquer votre nom'
                    ],
                    'email' => [
                        (int) 0 => 'This field cannot be left empty'
                    ]
                ],
                'repository' => 'Users'

            }
        ]
    ],
    'dirty' => [
        'company' => true,
        'email' => true,
        'phone' => true,
        'address' => true,
        'postcode' => true,
        'city' => true,
        'users' => true
    ],
    'original' => [],
    'virtual' => [],
    'errors' => [
        'email' => [
            (int) 0 => 'Vous devez saisir votre email'
        ],
        'phone' => [
            (int) 0 => 'Vous devez saisir votre téléphone'
        ],
        'address' => [
            (int) 0 => 'Vous devez saisir votre adresse'
        ],
        'postcode' => [
            (int) 0 => 'Vous devez saisir votre code postal'
        ],
        'city' => [
            (int) 0 => 'vous devez saisir votre ville'
        ]
    ],
    'repository' => 'Owners'

}

如您所见,该对象包含有关所有者和关联用户的错误,但由于我无法猜测的原因,cakePHP 似乎无法从用户那里获取错误。

所以我想知道它是否应该工作,如果我在某个地方犯了错误,可以解释它无法检索用户的错误。

4

2 回答 2

1

如前所述,这里的主要问题是您的输入缺少适当的关联。


在实体之前检查请求数据

输入获取输入数据的事实(这可能是您根据评论判断的困惑)与实体上设置或未设置的内容无关!

这是由于实体上下文是如何工作的(上下文是位于表单助手和数据之间的层),在它尝试检查实体的数据之前,它会检查请求,这就是它选择的地方,例如username字段,如果它会在实体中检查它,那么它不会找到它,因为Owner由于字段名称中缺少关联,它会在实体上查找它。

https://github.com/cakephp/cakephp/blob/3.0.0-beta3/src/View/Form/EntityContext.php#L201-L204


验证错误只能在实体上找到

因此,虽然您的输入可以获取数据,因为它以“错误”字段名存储在请求中,但验证错误只能在实体上找到,这就是您的代码最终出错的地方,Owner实体没有username字段,但这是实体上下文查找错误的地方。

https://github.com/cakephp/cakephp/blob/3.0.0-beta3/src/View/Form/EntityContext.php#L468

它不会在那里找到任何东西,因为错误位于第一个User实体中,存储在Owner实体users属性 ( Owner->users[0]->errors) 中,因此为了能够获取这些错误,您的字段名必须如下所示

users.0.username

它反映了实体中的结构。

http://book.cakephp.org/3.0/en/views/helpers/form.html#field-naming-conventions


关联模型只需要创建一个新的主实体

在您的控制器中,您正在创建一个User实体和一个Owner实体,这不是必需的,因为您的模型是关联的,newEntity()对表的调用Owners将创建所需的一切,一个Owner实体及其数据及其所有关联的实体数据,那是User实体。

http://book.cakephp.org/3.0/en/orm/table-objects.html#converting-request-data-into-entities

如果您需要为User实体设置其他数据,只需修补创建并放置在Owner实体users属性中的实体,如下所示:

public function add() {
    $this->loadModel('Users');

    // Creates an entity with all necessary data
    $owner = $this->Owners->newEntity($this->request->data);

    if ($this->request->is('post')) {
        if(isset($owner->users[0])) {
            // Patch the existing (new) entity with some 
            // additional data
            $this->Users->patchEntity($owner->users[0], [
                'email' => $owner->email,
                'role'  => 'owner',
                'token' => md5(time() . '-' . uniqid()),
            ]);
        } else {
            // Bail out here or check for the existence of
            // a user using validation.
            //
            // You should have some kind of validation somewhere
            // in order to make sure that a user (only 
            // one (1)) was submitted, and the controller
            // mostly isn't really the right place for such logic
        }

        if ($this->Owners->save($owner)) {
            $this->Flash->success(__("Merci de vous être enregistré. un email a été envoyé à {0} pour activer votre compte", $owner['email']));
            return $this->redirect(['action' => 'index']);
        } else {
            $this->Flash->error(__("Impossible de vous enregistrer, veuillez corriger les erreurs"));
        }
    }

    $this->set(compact('owner'));
}

请注意,我已经从示例中删除了validate()andget()调用,不一定需要显式调用验证,根据您的评论,您可能不需要它,如果您愿意,您现在需要它,如果您愿意,那么您d 禁用由save()调用触发的验证,因为不需要验证两次。

http://book.cakephp.org/3.0/en/orm/table-objects.html#validating-entities

该示例应该可以正常工作,但请确保您已阅读有关如何避免大规模分配攻击的部分!

于 2014-11-24T14:28:09.197 回答
0

您可以使用 set 方法轻松地将错误踢到视图中

if ($Models->errors()) {
$errors=$Models->errors();
 $this->set(compact('errors'));

}

于 2015-08-18T00:34:07.217 回答