This commit is contained in:
j 2011-02-24 19:39:58 +01:00
parent 6eff556204
commit d1480f550e
4 changed files with 153 additions and 56 deletions

View file

@ -31,7 +31,7 @@ def _order_query(qs, sort):
qs = qs.order_by(*order_by) qs = qs.order_by(*order_by)
return qs return qs
def _parse_query(data, user): def parse_query(data, user):
query = {} query = {}
query['range'] = [0, 100] query['range'] = [0, 100]
query['sort'] = [{'key':'user', 'operator':'+'}, {'key':'name', 'operator':'+'}] query['sort'] = [{'key':'user', 'operator':'+'}, {'key':'name', 'operator':'+'}]
@ -69,14 +69,14 @@ def findLists(request):
} }
return {status: {code: int, text: string}, return {status: {code: int, text: string},
data: { data: {
lists: [ items: [
{name:, user:, featured:, public...} {name:, user:, featured:, public...}
] ]
} }
} }
''' '''
data = json.loads(request.POST['data']) data = json.loads(request.POST['data'])
query = _parse_query(data, request.user) query = parse_query(data, request.user)
#order #order
is_section_request = query['sort'] == [{u'operator': u'+', u'key': u'position'}] is_section_request = query['sort'] == [{u'operator': u'+', u'key': u'position'}]

View file

@ -8,13 +8,13 @@ class PlaceManager(Manager):
def get_query_set(self): def get_query_set(self):
return super(PlaceManager, self).get_query_set() return super(PlaceManager, self).get_query_set()
def find(self, q='', f="globe", sw_lat=-180.0, sw_lng=-180.0, ne_lat=180.0, ne_lng=180.0): def find(self, q='', south=-180.0, west=-180.0, north=180.0, east=180.0):
qs = self.get_query_set() qs = self.get_query_set()
qs = qs.filter(Q( qs = qs.filter(Q(
Q(Q(sw_lat__gt=sw_lat)|Q(sw_lat__lt=ne_lat)|Q(sw_lng__gt=sw_lng)|Q(sw_lng__lt=ne_lng)) & Q(Q(south__gt=south)|Q(south__lt=north)|Q(west__gt=west)|Q(west__lt=east)) &
Q(Q(sw_lat__gt=sw_lat)|Q(sw_lat__lt=ne_lat)|Q(sw_lng__lt=ne_lng)|Q(ne_lng__gt=ne_lng)) & Q(Q(south__gt=south)|Q(south__lt=north)|Q(west__lt=east)|Q(east__gt=east)) &
Q(Q(ne_lat__gt=sw_lat)|Q(ne_lat__lt=ne_lat)|Q(sw_lng__gt=sw_lng)|Q(sw_lng__lt=ne_lng)) & Q(Q(north__gt=south)|Q(north__lt=north)|Q(west__gt=west)|Q(west__lt=east)) &
Q(Q(ne_lat__gt=sw_lat)|Q(ne_lat__lt=ne_lat)|Q(ne_lng__gt=sw_lng)|Q(ne_lng__lt=ne_lng)) Q(Q(north__gt=south)|Q(north__lt=north)|Q(east__gt=west)|Q(east__lt=east))
)) ))
if q: if q:
qs = qs.filter(name_find__icontains=q) qs = qs.filter(name_find__icontains=q)

View file

@ -6,6 +6,7 @@ from django.db import models
import ox import ox
from ox.django import fields from ox.django import fields
from django.contrib.auth.models import User, Group
import managers import managers
@ -14,24 +15,26 @@ class Place(models.Model):
''' '''
Places are named locations, they should have geographical information attached to them. Places are named locations, they should have geographical information attached to them.
''' '''
created = models.DateTimeField(auto_now_add=True)
modified = models.DateTimeField(auto_now=True)
user = models.ForeignKey(User, null=True, related_name='items')
name = models.CharField(max_length=200, unique=True) name = fields.TupleField(default=[])
name_sort = models.CharField(max_length=200) name_sort = models.CharField(max_length=200)
name_find = models.TextField(default='', editable=False) name_find = models.TextField(default='', editable=False)
geoname = models.CharField(max_length=1024, unique=True) geoname = models.CharField(max_length=1024, unique=True)
geoname_reverse = models.CharField(max_length=1024, unique=True) geoname_sort = models.CharField(max_length=1024, unique=True)
wikipediaId = models.CharField(max_length=1000, blank=True) wikipediaId = models.CharField(max_length=1000, blank=True)
aliases = fields.TupleField(default=[])
sw_lat = models.FloatField(default=0) south = models.FloatField(default=0)
sw_lng = models.FloatField(default=0) west = models.FloatField(default=0)
ne_lat = models.FloatField(default=0) north = models.FloatField(default=0)
ne_lng = models.FloatField(default=0) east = models.FloatField(default=0)
center_lat = models.FloatField(default=0) lat = models.FloatField(default=0)
center_lng = models.FloatField(default=0) lng = models.FloatField(default=0)
area = models.FloatField(default=-1) size = models.FloatField(default=-1)
objects = managers.PlaceManager() objects = managers.PlaceManager()
@ -41,25 +44,38 @@ class Place(models.Model):
def __unicode__(self): def __unicode__(self):
return self.name return self.name
def editable(self, user):
if user.is_staff or self.user == user:
return True
return False
def get_id(self):
return ox.to32(self.id)
def json(self): def json(self):
j = {} j = {
for key in ('name', 'name_sort', 'aliases', 'geoname', 'geoname_reversed', 'id': self.get_id(),
'sw_lat', 'sw_lng', 'ne_lat', 'ne_lng', 'user': self.user.username,
'center_lat', 'center_lng'): }
for key in ('created', 'modified',
'name', 'geoname',
'south', 'west', 'north', 'east',
'lat', 'lng', 'size'):
j[key] = getattr(self, key) j[key] = getattr(self, key)
return j
def save(self, *args, **kwargs): def save(self, *args, **kwargs):
if not self.name_sort: if not self.name_sort:
self.name_sort = self.name self.name_sort = ', '.join(self.name)
self.geoname_reverse = ', '.join(reversed(self.geoname.split(', '))) self.geoname_sort = ', '.join(reversed(self.geoname.split(', ')))
self.name_find = '|%s|'%'|'.join([self.name] + self.aliases) self.name_find = '|%s|'%'|'.join(self.name)
#update center #update center
self.lat_center = ox.location.center(self.lat_sw, self.lat_ne) #self.lat = ox.location.center(self.south, self.north)
self.lng_center = ox.location.center(self.lng_sw, self.lng_ne) #self.lng = ox.location.center(self.east, self.west)
#update area #update area
self.area = ox.location.area(self.lat_sw, self.lng_sw, self.lat_ne, self.lng_ne) #self.size = ox.location.area(self.south, self.west, self.north, self.east)
super(Place, self).save(*args, **kwargs) super(Place, self).save(*args, **kwargs)

View file

@ -2,35 +2,47 @@
# vi:si:et:sw=4:sts=4:ts=4 # vi:si:et:sw=4:sts=4:ts=4
from __future__ import division from __future__ import division
import ox
from ox.utils import json from ox.utils import json
from ox.django.decorators import login_required_json from ox.django.decorators import login_required_json
from ox.django.shortcuts import render_to_json_response, get_object_or_404_json, json_response from ox.django.shortcuts import render_to_json_response, get_object_or_404_json, json_response
import models
from api.actions import actions from api.actions import actions
from item import utils
import models
@login_required_json @login_required_json
def addPlace(request): def addPlace(request):
#FIXME: require admin #FIXME: require admin
''' '''
param data param data {
{ name: "",
'place': dict geoname: "",
south: float,
west: float,
north: float,
east: float,
lat: float,
lng: float,
size: float,
} }
place contains key/value pairs with place propterties
''' '''
data = json.loads(request.POST['data']) data = json.loads(request.POST['data'])
exists = False exists = False
names = [data['place']['name']] + data['place']['aliases'] names = data['name']
if isinstance(names, basestring):
names = [names]
for name in names: for name in names:
if models.Place.objects.filter(name_find__icontains=u'|%s|'%data['name']).count() != 0: if models.Place.objects.filter(name_find__icontains=u'|%s|'%name).count() != 0:
exists = True exists = True
if not exists: if not exists:
place = models.Place() place = models.Place()
for key in data['place']: place.user = request.user
setattr(place, key, data['place'][key]) for key in data:
setattr(place, key, data[key])
place.save() place.save()
response = json_response(status=200, text='created') response = json_response(status=200, text='created')
else: else:
@ -42,24 +54,27 @@ actions.register(addPlace, cache=False)
@login_required_json @login_required_json
def editPlace(request): def editPlace(request):
''' '''
param data param data {
{
'id': placeid, 'id': placeid,
'place': dict 'name': ...
'north': 0...
} }
place contains key/value pairs with place propterties can contain any of the allowed keys for place
''' '''
data = json.loads(request.POST['data']) data = json.loads(request.POST['data'])
place = get_object_or_404_json(models.Place, pk=data['id']) place = get_object_or_404_json(models.Place, pk=ox.from32(data['id']))
names = data.get('name', [])
if isinstance(names, basestring):
names = [names]
if place.editable(request.user): if place.editable(request.user):
conflict = False conflict = False
names = [data['place']['name']] + data['place']['aliases']
for name in names: for name in names:
if models.Place.objects.filter(name_find__icontains=u'|%s|'%data['name']).exclude(id=place.id).count() != 0: if models.Place.objects.filter(name_find__icontains=u'|%s|'%name).exclude(id=place.id).count() != 0:
conflict = True conflict = True
if not conflict: if not conflict:
for key in data['place']: for key in data:
setattr(place, key, data['place'][key]) if key != 'id':
setattr(place, key, data[key])
place.save() place.save()
response = json_response(status=200, text='updated') response = json_response(status=200, text='updated')
else: else:
@ -72,13 +87,68 @@ actions.register(editPlace, cache=False)
@login_required_json @login_required_json
def removePlace(request): def removePlace(request):
response = json_response(status=501, text='not implemented') '''
param data {
'id': placeid,
}
'''
data = json.loads(request.POST['data'])
if isinstance(data, dict):
data = data['id']
place = get_object_or_404_json(models.Place, pk=ox.from32(data))
if place.editable(request.user):
place.delete()
response = json_response(status=200, text='deleted')
else:
response = json_response(status=403, text='permission denied')
return render_to_json_response(response) return render_to_json_response(response)
actions.register(removePlace, cache=False) actions.register(removePlace, cache=False)
def parse_query(data, user):
query = {}
query['range'] = [0, 100]
query['sort'] = [{'key':'user', 'operator':'+'}, {'key':'name', 'operator':'+'}]
for key in ('keys', 'group', 'list', 'range', 'ids', 'sort'):
if key in data:
query[key] = data[key]
query['qs'] = models.Place.objects.all()
return query
def findPlace(request): def findPlaces(request):
''' '''
param data {
query: {
conditions: [
{
key: 'user',
value: 'something',
operator: '='
}
]
operator: ","
},
sort: [{key: 'name', operator: '+'}],
range: [0, 100]
keys: []
}
possible query keys:
name, geoname, user
possible keys:
name, geoname, user
return {
status: {
code: int,
text: string
},
data: {
items: [
{name:, user:, featured:, public...}
]
}
}
param data param data
{'query': query, 'sort': array, 'range': array, 'area': array} {'query': query, 'sort': array, 'range': array, 'area': array}
@ -96,7 +166,7 @@ def findPlace(request):
} }
] ]
range: result range, array [from, to] range: result range, array [from, to]
area: [sw_lat, sw_lng, ne_lat, ne_lng] only return places in that square area: [south, west, north, east] only return places in that square
with keys, items is list of dicts with requested properties: with keys, items is list of dicts with requested properties:
return {'status': {'code': int, 'text': string}, return {'status': {'code': int, 'text': string},
@ -111,10 +181,21 @@ Positions
ids: ids of places for which positions are required ids: ids of places for which positions are required
''' '''
data = json.loads(request.POST['data']) data = json.loads(request.POST['data'])
response = json_response(status=200, text='ok') response = json_response()
response['data']['places'] = []
query = parse_query(data, request.user)
qs = query['qs']
if 'keys' in data:
qs = qs[query['range'][0]:query['range'][1]]
response['data']['items'] = [p.json(request.user) for p in qs]
elif 'ids' in data:
ids = [i.get_id() for i in qs]
response['data']['positions'] = utils.get_positions(ids, query['ids'])
else:
response['data']['items'] = qs.count()
response['data']['items'] = []
#FIXME: add coordinates to limit search #FIXME: add coordinates to limit search
for p in models.Place.objects.find(data['query']): for p in models.Place.objects.find(data['query']):
response['data']['places'].append(p.json()) response['data']['items'].append(p.json())
return render_to_json_response(response) return render_to_json_response(response)
actions.register(findPlace) actions.register(findPlaces)