Laravel 5.2 validation errors not appearing - php

I am trying to get validation errors to show up in Laravel.
I have a UserController set up like so:
<?php
namespace App\Http\Controllers;
use App\User;
use App\Http\Controllers\Controller;
//Use Request;
Use Flash;
Use Illuminate\Http\Request;
class UserController extends Controller
{
/**
* Show the profile for the given user.
*
* #param int $id
* #return Response
*/
public function showProfile($id)
{
return view('user.profile', ['user' => User::findOrFail($id)]);
}
public function store(Request $request) {
$this->validate($request, [
'email' => 'required|unique:users|email|max:255',
]);
if($this) {
$input = Request::all();
User::create($input);
return redirect('/');
}
else {
return redirect('/')->withErrors($validator);
}
}
}
In my view (layout.blade.php), I have included:
#if (count($errors) > 0)
#foreach ($errors->all() as $error)
{{!! $errors !!}}
#endforeach
#endif
To account for the route, I have:
Route::group(['middleware' => ['web']], function () {
Route::get('/', function (){
return view('home');
});
});
Unfortunately, when I enter "bad" data that shouldn't be verified, I am not seeing any error (but it is not being stored in the db, so there's that).
One other note, when the blade template is rendered, I am seeing an extra "}" bracket, which I'm not sure why that is there.

In laravel version 5.2.41, the middleware web is thrown out.
Means adding the routes inside Route::group(['middleware' => ['web']], function () { will make the validation not work.

There are a couple things wrong or that can be improved here. The store method on the UserController has a lot of weird issues. $this will always be true because objects are true in php. Also, you pass in $validator into withErrors which doesn't make sense because there's no variable validator.
public function store(Request $request) {
$this->validate($request, [
'email' => 'required|unique:users|email|max:255',
]);
User::create(Request::all());
return redirect('/');
}
The validate method will throw an Illuminate\Foundation\Validation\ValidationException if there is a validation error. This exception should be listed in the $dontReport instance variable in App\Exceptions\Handler as seen below:
protected $dontReport = [
AuthorizationException::class,
HttpException::class,
ModelNotFoundException::class,
ValidationException::class,
];
If you have changed these values, removed, or modified the ValidatesRequest trait you may have broken this functionality.
Your error reporting code is not correct either:
#foreach ($errors->all() as $error)
{!! $errors->first() !!}
#endforeach
There are 3 changes here. First I removed the outer errors size check, this doesn't really get you anything. Next, I fixed your extra } error, the syntax for un-escaping data is {!! $errors->first() !!}. Lastly, I called ->first() this returns the first error associated with that particular field.
I think it's important to note that the validation exception will create a redirect response to the previous page. The logic for determining the previous page can be found in Illuminate\Routing\UrlGenerator::previous().

The errors block should be:
#if (count($errors) > 0)
<div class="alert alert-danger">
<ul>
#foreach ($errors->all() as $error)
<li>{{ $error }}</li>
#endforeach
</ul>
</div>
#endif
Assuming you're using Bootstrap for the alerts.
You also don't have $validator defined. You need to do something like this:
$validator = Validator::make($request->all(), [
'email' => 'required|unique:users|email|max:255',
]);
Instead of $this->validate().
That should do it.

If it is still not working and you have tried several things like
Moving ShareErrorsFromSession::class from $middlewareGroups to $middleware
Removing the web.php to web middleware
Try modifying the request headers to
Accept: application/json
Or you can also create a new middleware that will modify the request header like this:
<?php
namespace App\Http\Middleware;
use Closure;
use Illuminate\Http\Request;
class ModifyAcceptHeaderMiddleware
{
public function handle(Request $request, Closure $next)
{
$request->headers->set('Accept', 'application/json');
return $next($request);
}
}

Try to remove "\Illuminate\Session\Middleware\StartSession::class" from 'middleware' in Kernel.php, it must be only in the 'web' group.

#if(count(($errors->all()) > 0))
#foreach($errors->all() as $error)
<div class="alert alert-danger">
{{$error}}
</div>
#endforeach
#endif

Related

Laravel withErrors 5.4.36 is empty within view

Strange issue here! My return code within the controller is as follows:
return back()->withErrors([ 'not_updated' => 'Unable to update record or no changes made' ]);
And then I display the errors within blade:
#if ($errors->any())
<article class="message is-danger">
<div class="message-body">
<ul>
#foreach ($errors->all() as $error)
<li>{!! $error !!}</li>
#endforeach
</ul>
</div>
</article>
#endif
However this doesn't appear to be working at all, $errors is empty for some reason, however this works fine from another controller!
This is the method where this works, I have included the use classes.
namespace App\Http\Controllers;
use App\Pages;
use App\PlannerStatus;
use App\SubPages;
use App\VehicleMake;
use App\Website;
use App\WebsiteRedirects;
use Illuminate\Http\Request;
use Redirect;
class RedirectsController extends Controller
{
public function store(Request $request, Website $website)
{
$error = [ 'test' => 'test error' ];
if (!empty($error)) {
return back()->withErrors($error)->withInput();
}
return back();
}
}
And this is the controller where this does NOT work, as you can see they are the same, more or less!
namespace App\Http\Controllers;
use App\ResultsText;
use App\VehicleMake;
use App\VehicleModel;
use App\VehicleType;
use App\SearchPath;
use App\Website;
use Illuminate\Http\Request;
use Illuminate\Support\Facades\Cache;
use Illuminate\Support\MessageBag;
use Redirect;
class ResultsTextController extends Controller
{
public function update(Website $website, ResultsText $resultsText, Request $request)
{
$data = request()->except(['_token','id']);
$result = ResultsText::where('id', $resultsText->id)->update($data);
if (!$result) {
return back()->withErrors([ 'not_updated' => 'Unable to update record or no changes made' ]);
}
return Redirect::action('ResultsTextController#index', $website);
}
}
Also here are my Routes, just so you can see they are pretty much identical:
Route::prefix('/redirects')->group(function () {
Route::get('/', 'RedirectsController#index')->middleware('SettingStatus:redirect');
Route::patch('/update', 'RedirectsController#update');
});
Route::prefix('/results-text')->group(function () {
Route::post('{resultsText}/update', 'ResultsTextController#update');
});
Inside your blade try this
#if ($errors->any())
<article class="message is-danger">
<div class="message-body">
<ul>
#foreach ($errors as $error)
<li>{{ $error }}</li>
#endforeach
</ul>
</div>
</article>
#endif
It's easily overlooked in your question. The problem is not what it seems. You will probably find it funny why it doesn't work.
Replace
return back()->withErrors([ 'not_updated' => 'Unable to update record or no changes made' ]);
with
return redirect()->back()->withErrors([ 'not_updated' => 'Unable to update record or no changes made' ]);

Laravel Default Validaion error message not work

I am new to laravel. I want to set laravel validation on form submit. I have write below code for validation validation works but on view it does not show me any error message
use Illuminate\Http\Request;
use App\Http\Controllers\Controller;
use App\User;
use DB;
use Session;
use Illuminate\Foundation\Validation\ValidatesRequests;
use Illuminate\Support\Facades\Hash;
use App\Http\Requests;
use Illuminate\Support\Facades\Validator;
use Illuminate\Support\Facades\Input;
use Illuminate\Support\Facades\Redirect;
use Illuminate\Support\Facades\Auth;
use Illuminate\Foundation\Auth\RegistersUsers;
use Illuminate\Support\MessageBag;
use Illuminate\Pagination\LengthAwarePaginator;
and blow code in my register function
$input = Input::all();
$rules = array(
'first_name' => array('required'),
'last_name' => 'required');
$validator = Validator::make($input, $rules);
if ($validator->fails()) {
$messages = $validator->messages();
// echo '<pre>';
// print_r($messages);
// exit;
// return redirect('admin/user/add')
// ->withErrors($validator)
// ->withInput();
return Redirect::back()->withErrors($messages);
}
to display I have use following code in my view
#if ($errors->any())
#foreach ($errors->all() as $error)
{{ $error }}
#endforeach
#endif
can anyone help me to solve this issue.
If there are validation errors, you should use $errors in your view to get them. The $errors variable is an instance of Illuminate\Support\MessageBag and Laravel always makes it available to your views. You can use it like this:
#if ($errors->any())
<div class="errors">
#foreach ($errors->all() as $message)
<p>{{ $message }}</p>
#endforeach
</div>
#endif
That will output all of your validation messages (if there are any) to your view. If you want to check to see if there's an error message for a specific field, you can use the has method. You can then use the first message to retrieve the first message for that field or the get method to retrieve them all. An example using first:
#if ($errors->has('your_field'))
<div class="error">{{ $errors->first('your_field') }}</div>
#endif
See here for more information
Validation messages are stored in $errors, which is globally passed to all views, and not in session()->get("message"), unless you specifically used that. With the above, you'd use
#if($errors->has("first_name"))
<p class="alert alert-info">{{ $errors->first("first_name") }}</p>
#endif
Do the same for all fields being validated and you should start seeing your error messages.
Do you override the function "showRegistrationForm"?
If Yes, you should override the function "register":
public function register(Request $request) {
$validation = $this->validator($request->all());
if ($validation->fails()) {
$errors = $validation->errors();
return view('auth.register', compact('errors'));
}
else {
$user = $this->create($request->all());
Auth::login($user);
return redirect('/')->with(['message'=>'Account Successfully Created.']);
}
}

Laravel 5.6, relationship path is returning array instead of object

Longtime googler, first time asker here. Hi, folks.
I am debugging and updating my app after updating it from Laravel 5.1.10 to 5.6
and this bug is proving hard to google.
Exploring my error message “Trying to get property of non-object” I think what is happening is that the nested relationship path that used to work just fine to give me the object, is now instead giving me an array of its attributes.
More code below, but here is the snippet from my view:
#section('content')
<h2>
<?php // Header: project number and title ?>
#if ($bookingDefault->project->courseNumber)
{{ $bookingDefault->project->courseNumber }}:
#endif
This results in error:
Trying to get property 'courseNumber' of non-object
It’s not returning a null; the data is there and it works perfectly fine if I access the project as an array thus:
#section('content')
<h2>
<?php // Header: project number and title ?>
#if ($bookingDefault->project['courseNumber'])
{{ $bookingDefault->project['courseNumber'] }}:
#endif
So I know that the relationships are defined okay because it is finding the project. It’s just not giving a proper object to the view anymore.
I could change a large number of views to access attributes as an array instead, but that is a lot of code to comb through and change, and doesn’t give me access to the object’s methods. I would rather fix why I am getting an array instead of the object I was getting before.
CODE THAT MAY BE RELEVANT:
from app/Http/Kernel.php (partial) - I checked that SubstituteBindings::class is there.
/**
* The application's route middleware groups.
*
* #var array
*/
protected $middlewareGroups = [
'web' => [
\App\Http\Middleware\EncryptCookies::class,
\Illuminate\Cookie\Middleware\AddQueuedCookiesToResponse::class,
\Illuminate\Session\Middleware\StartSession::class,
\Illuminate\Session\Middleware\AuthenticateSession::class,
\Illuminate\View\Middleware\ShareErrorsFromSession::class,
\App\Http\Middleware\VerifyCsrfToken::class,
\Illuminate\Routing\Middleware\SubstituteBindings::class,
],
'api' => [
'throttle:60,1',
'bindings',
],
];
/**
* The application's route middleware.
*
* These middleware may be assigned to groups or used individually.
*
* #var array
*/
protected $routeMiddleware = [
'auth' => \Illuminate\Auth\Middleware\Authenticate::class,
'auth.basic' => \Illuminate\Auth\Middleware\AuthenticateWithBasicAuth::class,
'bindings' => \Illuminate\Routing\Middleware\SubstituteBindings::class,
'cache.headers' => \Illuminate\Http\Middleware\SetCacheHeaders::class,
'can' => \Illuminate\Auth\Middleware\Authorize::class,
'guest' => \App\Http\Middleware\RedirectIfAuthenticated::class,
'signed' => \Illuminate\Routing\Middleware\ValidateSignature::class,
'throttle' => \Illuminate\Routing\Middleware\ThrottleRequests::class,
];
from routes/web.php (partial) - all my old Route::model and Route::bind are still there. I wasn’t sure if I was supposed to take them out or put them somewhere else but fiddling with it didn’t change anything. I tried moving them to RouteServiceProvider’s boot() function but that didn’t change anything so I put them back into web.php.
Route::model('bookingdefaults', 'BookingDefault');
Route::model('bookings', 'Booking');
Route::model('people', 'User');
Route::model('projects', 'Project');
Route::bind('bookingdefaults', function($value, $route) {return App\BookingDefault::where('id', $value)->first();});
Route::bind('bookings', function($value, $route) {return App\Booking::where('id', $value)->first();});
Route::bind('people', function($value, $route) {return App\User::where('id', $value)->first();});
Route::bind('projects', function($value, $route) {return App\Project::where('id', $value)->first();});
In the models - again, not the complete code which is long:
<?php
namespace App;
use Illuminate\Database\Eloquent\Model;
class BookingDefault extends Model
{
protected $guarded = [];
// RELATIONSHIPS
// always belongs to a certain project
public function project()
{
return $this->belongsTo('App\Project');
}
// always happens a certain place
public function location()
{
return $this->belongsTo('App\Location');
}
// many bookings could be made from this default
public function bookings()
{
return $this->hasMany('App\Booking');
}
// each booking default will suggest several roles that might be filled in a booking
public function bookingDefaultRoleAssignments()
{
return $this->hasMany('App\BookingDefaultRoleAssignment');
}
// somtimes it is defining a default of a certain type, but if this is a
// customized default then it may not belong to a bookingType
public function bookingType()
{
return $this->belongsTo('App\BookingType');
}
}
<?php
namespace App;
use Illuminate\Database\Eloquent\Model;
use Mail;
use App\Booking;
class Project extends Model
{
protected $guarded = [];
// RELATIONSHIPS
public function bookings()
{
return $this->hasMany('App\Booking');
}
// a project can have defaults for several types of booking
public function bookingDefaults()
{
return $this->hasMany('App\BookingDefault');
}
// there will be many assignments of users to this project in various roles
public function projectRoleAssignments()
{
return $this->hasMany('App\ProjectRoleAssignment');
}
public function projectType()
{
return $this->belongsTo('App\ProjectType');
}
}
from BookingDefaultsController.php (partial - actual controller is over 1000 lines)
namespace App\Http\Controllers;
use Illuminate\Http\Request;
use App\Http\Requests;
use App\Http\Controllers\Controller;
use Illuminate\Database\Eloquent\Collection;
use App\User;
use App\Booking;
use App\BookingDefault;
use App\BookingDefaultRoleAssignment;
use App\BookingRoleAssignment;
use App\BookingRole;
use App\BookingType;
use App\Location;
use App\Project;
use App\Qualification;
use Illuminate\Support\Facades\DB;
use Input;
use Redirect;
use Mail;
class BookingDefaultsController extends Controller
{
/**
* Show the form for editing the specified resource.
*/
public function edit(BookingDefault $bookingDefault)
{
// We'll need all the options for form dropdowns
$locations = Location::where('status', 'active')->orderby('location_name')->get();
$bookingTypes = BookingType::all();
$bookingRoles = BookingRole::all();
$qualifications = Qualification::all();
return view('bookingdefaults.edit',
compact('bookingDefault', 'locations', 'bookingTypes', 'bookingRoles', 'qualifications'));
}
}
And finally the view, from /resources/views/bookingdefaults/edit.blade.php
#extends('layout')
#section('title')
Staffstuff Edit Booking Default
#stop
#section('php')
{{ $user = Auth::user() }}
#stop
#section('navtag')
<div id="projpage">
#stop
#section('javascript')
#stop
#section('content')
<h2>
<?php // Header: project number and title ?>
#if ($bookingDefault->project->courseNumber)
{{ $bookingDefault->project->courseNumber }}:
#endif
#if ($bookingDefault->project->shortTitle) {{ $bookingDefault->project->shortTitle }}
#elseif ($bookingDefault->project->title) {{ $bookingDefault->project->title }}
#endif
<br />Booking Default:
</h2>
<?php // Form to edit the basic booking info, time, place ?>
{!! Form::model($bookingDefault, ['method' => 'PATCH', 'route' => ['bookingdefaults.update', $bookingDefault->id]]) !!}
{!! Form::hidden('id', $bookingDefault->id) !!}
{!! Form::hidden('project_id', $bookingDefault->project_id) !!}
#include('/partials/bookingdefaultform', ['submit_text' => 'Update Default'])
<div class="form-group">
{!! Form::label('update_existing', 'Update existing bookings (see below): ') !!}
{!! Form::checkbox('update_existing', 'value', TRUE) !!}
<br />Note that NOT updating existing bookings will decouple them from this default and they will need to be updated individually.
</div>
{!! Form::close() !!}
<p>
DONE EDITING
</p>
#stop
The exact error:
ErrorException (E_ERROR)
Trying to get property 'courseNumber' of non-object (View: /Users/redacted/Sites/testproject032418/resources/views/bookingdefaults/edit.blade.php)
This should work:
In your RouteServiceProvider (might also work in routes) change from plural to singular and add the full namespace to the 2nd argument:
Route::model('bookingdefault', App\BookingDefault::class);
// or: Route::model('bookingdefault', 'App\BookingDefault');
The rest is fine.
You'll probably need to do this for other models as well.
Edit
This is exactly what I tried on my testing project. GrandChild is just some random model I had set up:
RouteServiceProvider:
public function boot()
{
parent::boot();
Route::model('grandchild', \App\GrandChild::class);
}
Routes:
Route::resource('grandchildren', 'GrandChildController');
GrandChildController:
public function show(GrandChild $canBeWhatever)
{
return $canBeWhatever;
}
And it works:

How to get validated data from Validator instance in Laravel?

I manually created a Validator, but i can't find a method to get the validated data.
For Request, validated data return from $request->validate([...])
For FormRequest, it's return from $formRequest->validated()
But with Validator, i don't see a method like those 2 above.
Assuming that you're using Validator facade:
use Illuminate\Support\Facades\Validator;
$validator = Validator::make($request->all(), $rules, $messages, $attributes);
if ($validator->fails()) {
return $validator->errors();
}
//If validation passes, return valid fields
return $validator->valid();
https://laravel.com/api/5.5/Illuminate/Validation/Validator.html#method_valid
If you use the Validator facade with make it will return a validator instance. This validator instance has methods like validate(), fails() and so on. You can look those methods up in the validator class or in the laravel api-documentation.
Writing The Validation Logic
public function store(Request $request)
{
$validatedData = $request->validate([
'title' => 'required|unique:posts|max:255',
'body' => 'required',
]);
// The blog post is valid...
}
Displaying The Validation Errors
<!-- /resources/views/post/create.blade.php -->
<h1>Create Post</h1>
#if ($errors->any())
<div class="alert alert-danger">
<ul>
#foreach ($errors->all() as $error)
<li>{{ $error }}</li>
#endforeach
</ul>
</div>
#endif
<!-- Create Post Form -->

Laravel 5.2 Displaying Validation Error

I am trying to validate a simple form by using Laravel's validator. Looks like validation works fine but i am unable to display errors. Form and controller looks like this.
Form
<h3>Add a New Team</h3>
<form method="POST" action="/teams">
<input type="hidden" name="_token" value="{{ csrf_token() }}">
<div class="form-group">
<input class="form-control" name="team_name" value="{{ old('team_name') }}" />
</div>
<div class="form-group">
<button type="submit" class="btn bg-primary">Add Team</button>
</div>
</form>
#if(count($errors))
<div class="alert alert-danger">
<ul>
#foreach($errors->all() as $error)
<li>{{ $error }}</li>
#endforeach
</ul>
</div>
#endif
Controller Method
public function store(Request $request) {
$this->validate($request, [
'team_name' => 'required|min:10'
]);
$team = new Team;
$team->team_name = $request->team_name;
$team->save();
return back();
}
If i remove web middleware group from my routes, errors displays fine.
Currently my routes.php file looks like this
Route::group(['middleware' => ['web']], function () {
Route::get('/teams', 'TeamsController#create');
Route::post('/teams', 'TeamsController#store');
});
How do i fix this problem ? Any help would be appreciated.
why do use the validation looks like laravel 4 while you are using laravel 5!!
in laravel 5 you need first to make Request class that handle your validation
php artisan make:request RequestName
you will find the request class that you make in
'app/http/Requests/RequestName.php'
and in the rules function you can handle your validation
public function rules()
{
return [
// put your validation rules here
'team_name' => 'required|min:10'
];
}
finally in your controller
use App\Http\Requests\RequestName;
public function store(RequestName $request) {
Team::create($request->all());
return Redirect::back();
}
for more illustration here
I recommend you to use Laravel Form Request
run
artisan make:request TeamRequest
add some logic and rules
class TeamRequest extends Request {
/**
* Determine if the user is authorized to make this request.
*
* #return bool
*/
public function authorize()
{
return true; //you can put here any other variable or condition
}
/**
* Get the validation rules that apply to the request.
*
* #return array
*/
public function rules()
{
return [
// put your validation rules here
];
}
}
then your contorller code will be like so:
public function store(TeamRequest $request)
{
$team = Team::create($request->all());
return back();
}
you no longer need to validate request and redirect back with errors and other stuff, laravel will do it for you
And you code looks more clean and neat, isn't it?
Write below code in your controller :
// define rules
$rules = array(
'team_name' => 'required|min:10'
);
$validator = Validator::make(Input::all(), $rules);
if ($validator->fails())
{
// something
return Redirect::back()
->withErrors($validator) // send back all errors to the login form
->withInput();
}
else
{
// something
// save your data
$team = new Team;
$team->team_name = $request->team_name;
$team->save();
}
change in View File :
#if (count($errors) > 0)
<div class="alert alert-danger">
<ul>
#foreach ($errors->all() as $error)
<li>{{ $error }}</li>
#endforeach
</ul>
</div>
#endif
After a little bit research, i have found that Laravel 5.2 has a RouteServiceProvider and it includes web middleware group for all routes. So i do not have to add the web middleware group to my routes manually. I just removed it from routes.php and problem solved.
If i remove web middleware group from my routes, errors displays fine.
In Laravel 5.2 the web midddleware is automatically applied to your routes in routes.php so no need to apply web middleware again. It is defined in mapWebRoutes() method of RouteServiceProvider.

Categories