O que você precisa saber é se a garantia ainda é válida ou expirou e a data da garantia, então primeiro você precisa construir uma tabela que reflita isso:
select product
, IF( warranty >= NOW(), 1, 0 ) as valid
, IF( warranty < NOW(), 1, 0 ) as expired
, warranty as last
from (
select product
, ADDDATE( purchased, INTERVAL 5 YEAR ) as warranty
from productWarranty
) w
group by product
;
Isso te daria algo assim:
+---------+-------+---------+---------------------+
| product | valid | expired | warranty |
+---------+-------+---------+---------------------+
| Hammer | 1 | 0 | 2017-01-01 00:00:00 |
| Nipper | 1 | 0 | 2017-01-01 00:00:00 |
| Nipper | 1 | 0 | 2017-01-01 00:00:00 |
| Nipper | 1 | 0 | 2017-01-01 00:00:00 |
| Saw | 1 | 0 | 2017-01-01 00:00:00 |
| Saw | 0 | 1 | 2011-01-01 00:00:00 |
| Saw | 1 | 0 | 2017-01-01 00:00:00 |
| Saw | 1 | 0 | 2017-01-01 00:00:00 |
+---------+-------+---------+---------------------+
Em seguida, use funções agregadas para filtrar e resumir as informações que você está procurando:
select product
, SUM( IF( warranty >= NOW(), 1, 0 ) ) as valid
, SUM( IF( warranty < NOW(), 1, 0 ) ) as expired
, MAX( warranty ) as last
from (
select product
, adddate( purchased, interval 5 year ) as warranty
from productWarranty
) w
group by product
;
+---------+-------+---------+---------------------+
| product | valid | expired | last |
+---------+-------+---------+---------------------+
| Hammer | 1 | 0 | 2017-01-01 00:00:00 |
| Nipper | 3 | 0 | 2017-01-01 00:00:00 |
| Saw | 3 | 1 | 2017-01-01 00:00:00 |
+---------+-------+---------+---------------------+