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