places
This commit is contained in:
parent
6eff556204
commit
d1480f550e
4 changed files with 153 additions and 56 deletions
|
@ -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'}]
|
||||||
|
|
|
@ -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)
|
||||||
|
|
|
@ -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)
|
||||||
|
|
|
@ -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)
|
||||||
|
|
Loading…
Reference in a new issue