Featured image of post Laravel中优雅的验证日期需要大于今天

Laravel中优雅的验证日期需要大于今天

表单验证有时候需要有一些特殊的验证需求

  • 我们有时会碰到一个场景,比如买车票,需要进行查询车票剩余,这时候需要一个日期范围,开始日期必须大于等于今天,结束日期必须大于等于开始日期。

Version

  • Laravel 5.5.40
  • PHP 7.1

开始之前,为方便查看结果,在/Exceptions/Handler.php文件捕获了异常


那么我们开始做一个验证吧:

  • 新建一个request并注入到需要验证控制器的方法
  • 下图圈出的红色部分,需改成return true因为我们身份验证一般不在这里验证
  • 如上图,end_date可以很简单的通过start_date来限制时间范围,那么start_date该怎么验证呢?我本来想的是这样的。
  • 通过增加一个额外的字段curr_date,然后start_date通过这个字段限制范围。测试发现实际并不生效。直接进入了控制器打印的数据
  • 调试了一下,我在Illuminate\Validation\Concerns、ValidatesAttributes::validateAfterOrEqual方法跳到compareDates方法,并尝试打印了$this->getValue($parameters[0])这时候的$paramters[0]curr_date总是null

  • 然后查阅了一下源码,知道了原因,直接说能正确验证的方法,再说详细的过程。
  • 正确的方法是这样的
  • 重写父类的prepareForValidation方法,之后再测试
  • 这时候得到想要的验证了

  • 分析一下过程
  • 能通过依赖注入实例化我们想要的对象,这一切都得感谢LaravelIoC容器
  • 容器每次解析完新对象之后,总是会释放一些绑定的事件
  • 这些事件很多都是通过服务提供者来绑定的,我们直接查看config/app.php里有关表单验证的服务提供者


  • 我们可以看到当实现ValidatesWhenResolved对象的类在解析之后会自动调用validate方法
  • 回到新建的IndexTicketRequest类,没有这个方法,去到父类,发现实现了ValidatesWhenResolved,而validate是在traitValidatesWhenResolvedTrait
  • 这个类找到了答案,在validate方法里分三步主要的
  • $this->prepareForValidation() 在验证之前的准备
  • 新建一个验证实例
  • 开始验证
  • 之所以是需要在验证之前设置curr_date,我们来看看新建验证实例便知道答案

  • 这里有一个地方导致我们的问题出现,就是先调用了$this->validationData()方法拿到request的数据,然后再通过$this->container->call([$this, 'rules'])拿到验证规则,所以我们在rules方法写的自然不生效了,之后的验证便无法继续进行

经网友指点,已经修改使用after_or_equal:today,因为today这个关键字在PHPstrtotime中是合法的关键字,可以成功转换