我正在尝试编写一个可测试的 Laravel 4 应用程序。在 Taylor Otwells 关于 Laravel 的书https://leanpub.com/laravel中,他写道,我们应该考虑在 Validation 命名空间中创建一个 UserValidator 类,并将该验证器注入到您的存储库中。您能否提供一个代码示例,该代码在控制器、存储库和测试中看起来像。如果用户输入不满意,我应该在存储库中抛出验证异常并在控制器中捕获错误吗?http://jasonlewis.me/article/laravel-advanced-validation
2 回答
我最近编写了类似的代码,这是 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 自动加载、依赖注入和创建自定义服务提供程序。
就个人而言,我更喜欢直接在模型中进行验证。也就是说,我将为每个模型提供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);
}
}