get distinct + group by then filter

From: Ivan Sergio Borgonovo <mail(at)webthatworks(dot)it>
To: pgsql-sql(at)postgresql(dot)org
Subject: get distinct + group by then filter
Date: 2009-12-17 15:28:09
Message-ID: 20091217162809.07f871e3@dawn.webthatworks.it
Views: Whole Thread | Raw Message | Download mbox | Resend email
Thread:
Lists: pgsql-sql

I've a web application and I'm trying to do some reporting on
affiliate commission

create table tracky_hit (
hitid serial,
esid varchar(32), -- related to browser session
track_time timestamp,
aid varchar(32), -- affiliate code
-- some other tracking stuff
);

create table tracky_event (
eventid serial,
esid varchar(32) references tracky_hit (esid)
);

create table tracky_ordergroup_event (
ordergroupid int references ...,
eventid int references tracky_event (eventid)
);

Now I'd like to pick up the first hit for each esid in a given
interval of time for a given aid and relate them with ordergroupid.

aid may change across the same esid.

Getting the first hit for each esid can be done:

select min(hitid) as h
from tracky_hit
group by esid;

or

select distinct on (esid) hitid
from tracky_hit
order by esid, track_time;

If I put a where aid='somestuff' right in the above query... I'm not
picking up the first hit in an esid.

The only way that comes to my mind to solve the problem is applying
the condition later in a subquery, but no conditions means a lot of
data returned.

I've a similar problem with the interval: if I chop in the middle of
a session I may not pick up the beginning of each session.
Furthermore I've to count session just once even if they cross the
boundary of an interval.

I could do something like:

select oe.ordergroupid from
tracky_ordergroup_event oe
join tracky_event e on e.eventid=oe.eventid
join tracky_hit th on th.esid=e.esid
where th.hitid in
(select distinct on (esid) hitid
from tracky_hit
where track_time between
('2009-12-01'::timestamp - interval '1 days')
and
('2009-12-01'::timestamp + interval '1 months' + interval '1
days')
order by esid, track_time
)
and th.aid='someaid'
and th.track_time between
('2009-12-01'::timestamp)
and
('2009-12-01'::timestamp + interval '1 months');

but this looks awful. Any better way?

I'm on 8.3 and no short term plan to move to 8.4

thanks

--
Ivan Sergio Borgonovo
http://www.webthatworks.it

Responses

Browse pgsql-sql by date

  From Date Subject
Next Message Filip Rembiałkowski 2009-12-18 12:20:00 Re: get distinct + group by then filter
Previous Message Tom Lane 2009-12-16 20:23:09 Re: 8.3/8.4 SQL incompatibility?