2

我正在尝试编写一个可测试的 Laravel 4 应用程序。在 Taylor Otwells 关于 Laravel 的书https://leanpub.com/laravel中,他写道,我们应该考虑在 Validation 命名空间中创建一个 UserValidator 类,并将该验证器注入到您的存储库中。您能否提供一个代码示例,该代码在控制器、存储库和测试中看起来像。如果用户输入不满意,我应该在存储库中抛出验证异常并在控制器中捕获错误吗?http://jasonlewis.me/article/laravel-advanced-validation

4

2 回答 2

3

我最近编写了类似的代码,这是 laracasts ( http://laracasts.com ) 推荐的(参考了 Taylor Otwell 的书)。请注意,您不必遵循所使用的应用程序结构。

<?php namespace ACME\Services\Validation;

use Validator as V;

abstract class Validator {

    protected $errormessages;
    protected $rules;

    public function validate($input, $rules)
    {

        $validator = V::make($input, $rules);
        $this->rules = $rules;

        if ($validator->fails()) {

            $this->errormessages = $validator->messages();

            return false;
        }

        return true;

    }

    public function getErrorMessages()
    {
        return $this->errormessages;
    }

    public function getValidationRules()
    {
        return $this->rules;
    }
}

这是应该为各种验证目的而扩展的类,如下面的示例代码片段所示。

<?php namespace ACME\Services\Validation;

use Auth;

class UserValidator extends Validator {

    protected $create_rules = [
        'firstname' => 'required|min:3|max:64|alpha-dash',
        'lastname' => 'required|min:2|max:64|alpha-dash',
        'account' => 'required|min:4|max:15|alpha_num',
        'email' => 'required|between:3,254|email|unique:users',
        'description' => 'max:500'
    ];

    protected $edit_rules = [
        'firstname' => 'required|min:3|max:64|alpha-dash',
        'lastname' => 'required|min:2|max:64|alpha-dash',
        'description' => 'max:500'
    ];

    protected $pass_edit_rules = [
        'oldpassword' => 'required',
        'password'  => 'required|min:5|confirmed',
        'password_confirmation'=> 'required|min:5'
    ];

    public function validateCreate($input)
    {


        return parent::validate($input, $this->create_rules);
    }

    public function validateEdit($input)
    {
        $newRules = $this->edit_rules;

        if ($this->validatePasswordChanged($input))
            $newRules = array_merge($newRules, $this->pass_edit_rules);

        return parent::validate($input, $newRules);
    }

    public function validatePasswordChanged($input)
    {
        return $input['password'] != '' || $input['oldpassword'] != '' || $input['password_confirmation'] != '' ? true : false;
    }

}

$this->validate 可以像在类中使用 parent::validate 一样容易。

另一个例子,由于 laracasts 提供课程,他们可能有一个 LessonValidator.php 文件,其中包含class LessonValidator extends Validator {}和一组不同的规则。

在存储库中使用?不包括存储库接口和服务提供者

<?php namespace ACME\Repositories;

use User;
use UserController;
use ACME\Services\Validation\UserValidator;

class DatabaseUserRepository extendes UserRepositoryInterface {
    protected $validator;

    public function __construct(UserValidator $validator, UserController $listener)
    {
        $this->validator = $validator;
        $this->listener = $listener;
    }

    public function createUser(User $user)
    {
        if ($this->validator->validateCreate($input))
            return $this->listener->withErrors('/',$this->validator->getErrorMessages());

        /* Validation passed, create user with User::create() */

        $this->listener->withView('usercreatedview');
    }
}

然后,您的控制器将包含类似的内容。

<?php

use ACME\Repositories\UserRepositoryInterface;

class UserController extends BaseController {

    protected $repository;

    public function __construct(UserRepositoryInterface $repository)
    {
        $this->repository = $repository;
    }

    public function withErrors($path, MessageBag $errors)
    {
        return Redirect::to($path)->withErrors($errors);
    }

    public function withView($view)
    {
        return View::make($view);
    }

    /* Other controller stuff... */
}

请注意,如果您不将 DatabaseUserRepository 绑定到 UserRepositoryInterface (以及我可能犯的任何错字),这将出错。您可以将验证类注入控制器,但我相信您想合并一个存储库。我倾向于在事件处理程序内部进行验证和存储库调用。

我会阅读 psr 自动加载、依赖注入和创建自定义服务提供程序。

于 2014-02-02T08:55:53.257 回答
1

就个人而言,我更喜欢直接在模型中进行验证。也就是说,我将为每个模型提供getValidator()如下方法:

class User extends Eloquent
{
    public function getValidator()
    {
        $params = array(
            'username' => $this->username,
            'password' => $this->password,
        );

        $rules = array(
            'username' => ['required', 'unique:users'],
            'password' => ['required', 'min:6'],
        );

        return Validator::make($params, $rules);
    }
}

然后在我的控制器、命令或测试中,我只需调用该方法以获得一个验证器实例,然后我会调用我需要的方法,可以是passes()or fails()

下面说明了我如何在控制器中实际使用它。

class UserController extends BaseController
{
    public function processCreateUser()
    {
        // Retrieve user input.
        $user = new User(Input::all());

        // Validate input.
        $validator = $user->getValidator();

        if ($validator->passes()) 
        {
            // Hash the password.
            $user->password = Hash::make($user->password);

            // Save the new user.
            $user->save();

            return Redirect::to('users')
                ->with('success', 'User created!');
        }

        return Redirect::route('users.create')
            ->withInput()
            ->with('error', 'Cannot create user, please double check the form.')
            ->withErrors($validator);
    }
}
于 2013-07-31T04:22:15.550 回答