Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Laravel Eloquent compare date from datetime field

I want to get all the rows from a table through an expression:

table.date <= 2014-07-10

But if the column contains a datetime let's say:

2014-07-10 12:00:00

But if I do:

where('date', '<=', $date)

it won't get the row.

I guess this is because $date = 2014-07-10 which makes MySQL assume that it is 2014-07-10 00:00:00.

In regular MySQL I would just do

where DATE(date) <= $date

What would be the equivalent using Laravel's Eloquent?

like image 206
almo Avatar asked Aug 05 '14 13:08

almo


Video Answer


7 Answers

Laravel 4+ offers you these methods: whereDay(), whereMonth(), whereYear() (#3946) and whereDate() (#6879).

They do the SQL DATE() work for you, and manage the differences of SQLite.

Your result can be achieved as so:

->whereDate('date', '<=', '2014-07-10')

For more examples, see first message of #3946 and this Laravel Daily article.


Update: Though the above method is convenient, as noted by Arth it is inefficient on large datasets, because the DATE() SQL function has to be applied on each record, thus discarding the possible index.

Here are some ways to make the comparison (but please read notes below):

->where('date', '<=', '2014-07-10 23:59:59')

->where('date', '<', '2014-07-11')

// '2014-07-11'
$dayAfter = (new DateTime('2014-07-10'))->modify('+1 day')->format('Y-m-d');

->where('date', '<', $dayAfter)

Notes:

  • 23:59:59 is okay (for now) because of the 1-second precision, but have a look at this article: 23:59:59 is not the end of the day. No, really!
  • Keep in mind the "zero date" case ("0000-00-00 00:00:00"). Though, these "zero dates" should be avoided, they are source of so many problems. Better make the field nullable if needed.
like image 52
Gras Double Avatar answered Oct 16 '22 20:10

Gras Double


Have you considered using:

where('date', '<', '2014-08-11')

You should avoid using the DATE() function on indexed columns in MySQL, as this prevents the engine from using the index.

UPDATE

As there seems to be some disagreement about the importance of DATE() and indexes, I have created a fiddle that demonstrates the difference, see POSSIBLE KEYS.

like image 27
Arth Avatar answered Oct 16 '22 22:10

Arth


You can get the all record of the date '2016-07-14' or before '2016-07-14' by choosing one syntax from follows:

->whereDate('date','=','2014-07-10')

->whereDate('date', '<=', '2014-07-10')

Or use the another code for dynamic date

whereDate('date',$date)
like image 29
Majbah Habib Avatar answered Oct 16 '22 21:10

Majbah Habib


You can use this

whereDate('date', '=', $date)

If you give whereDate then compare only date from datetime field.

like image 35
Jakir Hossain Avatar answered Oct 16 '22 22:10

Jakir Hossain


use Carbon\Carbon;

public function scopePublished($query)
{
  $now = Carbon::now();
  $date = Carbon::parse($now)->toDateString();
  $time = Carbon::parse($now)->toTimeString();
  return $query->whereDate('published_at', '<', $date)
  ->orWhere(function($query) use ($date, $time) {
    $query->whereDate('published_at', '=', $date)
    ->whereTime('published_at', '<=', $time);
  });
}
like image 34
Vantoose Avatar answered Oct 16 '22 22:10

Vantoose


Here is my logic: if you are comparing date then your method should be whereDate and if your comparing complete datetime then your method will be only where:

$calendar_alert = DB::table('calendar_alerts')->whereDate('when', '=', now()->format('Y-m-d'))->where('when', '>', now()->format('H:i:s'))->get();
like image 37
fahdshaykh Avatar answered Oct 16 '22 22:10

fahdshaykh


If you're still wondering how to solve it.

I use

protected $dates = ['created_at', 'updated_at', 'aired'];

In my model and in my where i do

where('aired', '>=', time());

So just use the unix to compare in where.

In views on the other hand you have to use the date object.

like image 27
pixie_frigo Avatar answered Oct 16 '22 20:10

pixie_frigo