Today, I found myself in a horrible situation. I was heading down i-235 and I became stricken with an insatiable burrito urge. All the apps on my phone were borked and all I had was my trusty Postgres database. I needed to find a burritos shop and I needed to find one fast.
Where oh where to begin
Luckily, I happen to know the latitude and longitude of all my favorite burrito places in Oklahoma City. I quickly add them to my database:
--Hello Table! CREATE TABLE burritoplaces (id serial PRIMARY KEY, establishment varchar(50) not null, lat double precision, lon double precision); --Hello Data! INSERT INTO burritoplaces(establishment, lat,lon) VALUES ('Verde Bueno Burrito', 35.484388,-97.505035) ,('Locos Lobos', 35.496025,-97.510185) ,('Captain Crustacean', 35.51468,-97.524347) ,('Double Stuffers Cafe', 35.523762,-97.508039) ,('Cafe Truncate of Watonga', 35.866413,-98.473206);
The data checks out:
-- all 5 entries SELECT * FROM burritoplaces;
Perfect! All my favorite places are there but sadly, I can’t do geospatial math in my head. I can’t even really do it outside of my head so I’m going to have to get some help. Enter the earthdistance postgres extension and its helpful sibling ‘cube’:
--turn on the magic CREATE EXTENSION cube; CREATE EXTENSION earthdistance;
I’m becoming so weak… I need to wrap this up. Since I live in one of the four countries in the world that uses the Imperial system of measurement, I need to figure out how to do this whole thing in miles. Skimming over the earthdistance documentation, it looks like I just need to use the point commands. I whip up a query that is as tech tasty as some carne asada:
-- my location on i-235 is a very hungry (35.512363,-97.515678) SELECT *, point(-97.515678, 35.512363) <@> point(lon, lat)::point AS burrito_distance FROM burritoplaces WHERE (point(-97.515678, 35.512363) <@> point(lon, lat)) < 10 --feel free to play this! ORDER by burrito_distance;
This returns me everything that is less than 10 miles away and orders them by distance. I see that Captain Crustacian is a half mile away and they have amazing guacamole. Thank you Postgres, cube, and earthdistance. You have saved my life once again.
If I lived anywhere else in the world, I would likely use meters and write the query as such:
SELECT *, earth_distance(ll_to_earth(35.512363,-97.515678), ll_to_earth(lat, lon)) as burrito_distance FROM burritoplaces WHERE earth_box(ll_to_earth(35.512363,-97.515678), 4000) @> ll_to_earth(lat, lon) ORDER by burrito_distance;