.near: show closest users to you

This commit is contained in:
Ryan Hitchman 2014-01-07 13:41:07 -08:00
parent aca9ba54ff
commit 6ae70537c0
2 changed files with 63 additions and 10 deletions

View File

@ -1,5 +1,6 @@
# -*- coding: utf-8 -*- # -*- coding: utf-8 -*-
import math
import random import random
import re import re
import threading import threading
@ -128,7 +129,7 @@ def get_nicks_by_tagset(db, chan, tagset):
nicks = [munge(x[0], 1) for x in sorted(nicks)] nicks = [munge(x[0], 1) for x in sorted(nicks)]
if not nicks: if not nicks:
return 'no tags found in intersection of "%s"' % tagset return 'no nicks found with tags "%s"' % tagset
return 'nicks tagged "%s": ' % tagset + winnow(nicks) return 'nicks tagged "%s": ' % tagset + winnow(nicks)
@ -173,6 +174,40 @@ def tagged(inp, chan='', db=None):
return get_nicks_by_tagset(db, chan, inp) return get_nicks_by_tagset(db, chan, inp)
def distance(lat1, lon1, lat2, lon2):
deg_to_rad = math.pi / 180
lat1 *= deg_to_rad
lat2 *= deg_to_rad
lon1 *= deg_to_rad
lon2 *= deg_to_rad
R = 6371 # km
d = math.acos(math.sin(lat1)*math.sin(lat2) +
math.cos(lat1)*math.cos(lat2) *
math.cos(lon2-lon1)) * R
return d
@hook.command(autohelp=False)
def near(inp, nick='', chan='', db=None):
init_db(db)
try:
loc = db.execute("select lat, lon from location where chan=? and lower(nick)=lower(?)", (chan, nick)).fetchone()
except db.OperationError:
loc = None
if loc is None:
return 'use .weather <loc> first to set your location'
lat, lon = loc
db.create_function('distance', 4, distance)
nearby = db.execute("select nick, distance(lat, lon, ?, ?) as dist from location where chan=?"
" limit 20", (lat, lon, chan)).fetchall()
return nearby
character_replacements = { character_replacements = {
'a': 'ä', 'a': 'ä',

View File

@ -1,28 +1,41 @@
"weather, thanks to wunderground" "weather, thanks to wunderground"
import math
from util import hook, http from util import hook, http
@hook.api_key('wunderground') @hook.api_key('wunderground')
@hook.command(autohelp=False) @hook.command(autohelp=False)
def weather(inp, nick='', server='', reply=None, db=None, api_key=None): def weather(inp, chan='', nick='', reply=None, db=None, api_key=None):
".weather <location> [dontsave] -- gets weather data from Wunderground "\ ".weather <location> [dontsave] -- gets weather data from Wunderground "\
"http://wunderground.com/weather/api" "http://wunderground.com/weather/api"
if not api_key: if not api_key:
return None return None
# this database is used by other plugins interested in user's locations,
# like .near in tag.py
db.execute("create table if not exists location(chan, nick, loc, lat, lon, primary key(chan, lower(nick)))")
loc = inp loc = inp
dontsave = loc.endswith(" dontsave") dontsave = loc.endswith(" dontsave")
if dontsave: if dontsave:
loc = loc[:-9].strip().lower() loc = loc[:-9].strip().lower()
db.execute("create table if not exists weather(nick primary key, loc)")
if not loc: # blank line if not loc: # blank line
loc = db.execute("select loc from location where chan=? and lower(nick)=lower(?)",
(chan, nick)).fetchone()
if not loc:
try:
# grab from old-style weather database
loc = db.execute("select loc from weather where nick=lower(?)", loc = db.execute("select loc from weather where nick=lower(?)",
(nick,)).fetchone() (nick,)).fetchone()
except db.OperationalError:
pass # no such table
if not loc: if not loc:
return weather.__doc__ return weather.__doc__
loc = loc[0] loc = loc[0]
@ -97,7 +110,12 @@ def weather(inp, nick='', server='', reply=None, db=None, api_key=None):
'(H:{h_f}F/{h_c}C L:{l_f}F/{l_c}C)' \ '(H:{h_f}F/{h_c}C L:{l_f}F/{l_c}C)' \
', Humidity: {humid}, {wind}'.format(**info)) ', Humidity: {humid}, {wind}'.format(**info))
lat = float(obs['display_location']['latitude'])
lon = float(obs['display_location']['longitude'])
if inp and not dontsave: if inp and not dontsave:
db.execute("insert or replace into weather(nick, loc) values (?,?)", db.execute("insert or replace into location(chan, nick, loc, lat, lon) "
(nick.lower(), inp)) "values (?, ?, ?, ?,?)", (chan, nick, inp, lat, lon))
db.commit() db.commit()