Laravel 4存储库验证和测试

时间:2013-07-30 22:24:30

标签: phpunit laravel laravel-4

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

2 个答案:

答案 0 :(得分: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-&gt; 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自动加载,依赖注入和创建自定义服务提供程序。

答案 1 :(得分: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()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);
    }
}