commit stuff
This commit is contained in:
parent
d63864a2fe
commit
0ed54a1d4f
7 changed files with 650 additions and 146 deletions
|
@ -59,11 +59,19 @@ def loadIMDb(imdbId):
|
||||||
setattr(movie, key, business[key])
|
setattr(movie, key, business[key])
|
||||||
|
|
||||||
movie.save()
|
movie.save()
|
||||||
|
models.AlternativeTitle.objects.filter(movie=movie).delete()
|
||||||
|
for i in oxweb.imdb.getMovieAKATitles(imdbId):
|
||||||
|
t = models.AlternativeTitle()
|
||||||
|
t.movie = movie
|
||||||
|
t.title = i[0]
|
||||||
|
t.type = i[1]
|
||||||
|
t.save()
|
||||||
|
|
||||||
#FIXME: related tables should be cleaned to not accumulate cruft
|
#FIXME: related tables should be cleaned to not accumulate cruft
|
||||||
#Country
|
#Country
|
||||||
models.MovieCountry.objects.filter(movie=movie).delete()
|
models.MovieCountry.objects.filter(movie=movie).delete()
|
||||||
position = 0
|
position = 0
|
||||||
|
if 'country' in info:
|
||||||
for i in info['country']:
|
for i in info['country']:
|
||||||
debug("add country", i)
|
debug("add country", i)
|
||||||
country = models.Country.get_or_create(i)
|
country = models.Country.get_or_create(i)
|
||||||
|
@ -73,6 +81,7 @@ def loadIMDb(imdbId):
|
||||||
#Language
|
#Language
|
||||||
models.MovieLanguage.objects.filter(movie=movie).delete()
|
models.MovieLanguage.objects.filter(movie=movie).delete()
|
||||||
position = 0
|
position = 0
|
||||||
|
if 'language' in info:
|
||||||
for i in info['language']:
|
for i in info['language']:
|
||||||
debug("add language", i)
|
debug("add language", i)
|
||||||
language = models.Language.get_or_create(i)
|
language = models.Language.get_or_create(i)
|
||||||
|
@ -89,6 +98,7 @@ def loadIMDb(imdbId):
|
||||||
|
|
||||||
#Genre
|
#Genre
|
||||||
movie.genres.all().delete()
|
movie.genres.all().delete()
|
||||||
|
if 'genre' in info:
|
||||||
for i in info['genre']:
|
for i in info['genre']:
|
||||||
debug("add genre", i)
|
debug("add genre", i)
|
||||||
genre = models.Genre.get_or_create(i)
|
genre = models.Genre.get_or_create(i)
|
||||||
|
@ -106,7 +116,7 @@ def loadIMDb(imdbId):
|
||||||
position = 0
|
position = 0
|
||||||
trivia = oxweb.imdb.getMovieTrivia(imdbId)
|
trivia = oxweb.imdb.getMovieTrivia(imdbId)
|
||||||
for i in trivia:
|
for i in trivia:
|
||||||
debug("add trivia", g)
|
debug("add trivia", i)
|
||||||
t = models.Trivia()
|
t = models.Trivia()
|
||||||
t.movie = movie
|
t.movie = movie
|
||||||
t.trivia = i
|
t.trivia = i
|
||||||
|
@ -127,8 +137,16 @@ def loadIMDb(imdbId):
|
||||||
models.Cast.link(movie, person, role, character, position)
|
models.Cast.link(movie, person, role, character, position)
|
||||||
position += 1
|
position += 1
|
||||||
|
|
||||||
#FIXME: connections
|
movie.connections.all().delete()
|
||||||
#m.addMovieConnections(IMDb['connections'])
|
connections = oxweb.imdb.getMovieConnections(imdbId)
|
||||||
|
for relation in connections:
|
||||||
|
for otherId in connections[relation]:
|
||||||
|
try:
|
||||||
|
object = models.Movie.objects.get(imdbId=otherId)
|
||||||
|
debug("add connection", relation, object)
|
||||||
|
models.Connection.get_or_create(movie, relation, object)
|
||||||
|
except models.Movie.DoesNotExist:
|
||||||
|
pass
|
||||||
|
|
||||||
reviews = oxweb.imdb.getMovieExternalReviews(imdbId)
|
reviews = oxweb.imdb.getMovieExternalReviews(imdbId)
|
||||||
for r in reviews:
|
for r in reviews:
|
||||||
|
|
|
@ -1,29 +1,136 @@
|
||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
# vi:si:et:sw=4:sts=4:ts=4
|
# vi:si:et:sw=4:sts=4:ts=4
|
||||||
|
import re
|
||||||
|
from datetime import datetime
|
||||||
|
from urllib2 import unquote
|
||||||
|
|
||||||
from django.contrib.auth.models import User
|
from django.contrib.auth.models import User
|
||||||
from django.core.exceptions import ObjectDoesNotExist
|
from django.core.exceptions import ObjectDoesNotExist
|
||||||
from django.db import models
|
from django.db.models import Q, Manager
|
||||||
from django.db.models import Q
|
|
||||||
|
|
||||||
|
import models
|
||||||
|
|
||||||
class MovieManager(models.Manager):
|
def keyType(key):
|
||||||
|
if key in ('released'):
|
||||||
|
return "date"
|
||||||
|
if key in ('year', 'cast.length'):
|
||||||
|
return "int"
|
||||||
|
if key in ('rating', 'votes'):
|
||||||
|
return "float"
|
||||||
|
return "string"
|
||||||
|
|
||||||
|
class MovieManager(Manager):
|
||||||
def get_query_set(self):
|
def get_query_set(self):
|
||||||
return super(MovieManager, self).get_query_set()
|
return super(MovieManager, self).get_query_set()
|
||||||
|
|
||||||
def find(self, q="", f="all", s="title", a="desc", l="all", o=0, n=100, p=None):
|
def find(self, request):
|
||||||
qs = self.get_query_set()
|
'''
|
||||||
|
construct query set from q value in request,
|
||||||
if q:
|
also checks for lists.
|
||||||
if f == "all":
|
range and order must be applied later
|
||||||
qs = qs.filter(title__icontains=q)
|
'''
|
||||||
|
for i in request.META['QUERY_STRING'].split('&'):
|
||||||
|
if i.startswith('q='):
|
||||||
|
q = i[2:]
|
||||||
|
op = ','
|
||||||
|
if '|' in q:
|
||||||
|
op = '|'
|
||||||
|
conditions = []
|
||||||
|
for e in q.split(op):
|
||||||
|
e = e.split(':')
|
||||||
|
if len(e) == 1: e = ['all'] + e
|
||||||
|
k, v = e
|
||||||
|
exclude = False
|
||||||
|
if v.startswith('!'):
|
||||||
|
v = v[1:]
|
||||||
|
exclude = True
|
||||||
|
if keyType(k) == "string":
|
||||||
|
startswith = v.startswith('^')
|
||||||
|
endswith = v.endswith('$')
|
||||||
|
if startswith and endswith:
|
||||||
|
v = v[1:-1]
|
||||||
|
k = '%s__iexact' % k
|
||||||
|
elif startswith:
|
||||||
|
v = v[1:]
|
||||||
|
k = '%s__istartswith' % k
|
||||||
|
elif v.endswith('$'):
|
||||||
|
v = v[:-1]
|
||||||
|
k = '%s__iendswith' % k
|
||||||
else:
|
else:
|
||||||
field = str("find__%s__icontains"%f)
|
k = '%s__icontains' % k
|
||||||
qs = qs.filter(**{field: q})
|
k = 'find__%s' % k
|
||||||
|
v = unquote(v)
|
||||||
|
if exclude:
|
||||||
|
conditions.append(~Q(**{k:v}))
|
||||||
|
else:
|
||||||
|
conditions.append(Q(**{k:v}))
|
||||||
|
else:
|
||||||
|
def parseDate(d):
|
||||||
|
while len(d) < 3:
|
||||||
|
d.append(1)
|
||||||
|
return datetime(*[int(i) for i in d])
|
||||||
|
#1960-1970
|
||||||
|
match = re.compile("(-?[\d\.]+?)-(-?[\d\.]+$)").findall(v)
|
||||||
|
if match:
|
||||||
|
v1 = match[0][0]
|
||||||
|
v2 = match[0][1]
|
||||||
|
if keyType(k) == "date":
|
||||||
|
v1 = parseDate(v1.split('.'))
|
||||||
|
v2 = parseDate(v2.split('.'))
|
||||||
|
if exclude: #!1960-1970
|
||||||
|
k1 = str('%s__lt' % k)
|
||||||
|
k2 = str('%s__gte' % k)
|
||||||
|
conditions.append(Q(**{k1:v1})|Q(**{k2:v2}))
|
||||||
|
else: #1960-1970
|
||||||
|
k1 = str('%s__gte' % k)
|
||||||
|
k2 = str('%s__lt' % k)
|
||||||
|
conditions.append(Q(**{k1:v1})&Q(**{k2:v2}))
|
||||||
|
else:
|
||||||
|
if keyType(k) == "date":
|
||||||
|
v = parseDate(v.split('.'))
|
||||||
|
k = str('%s' % k)
|
||||||
|
if exclude: #!1960
|
||||||
|
conditions.append(~Q(**{k:v}))
|
||||||
|
else: #1960
|
||||||
|
conditions.append(Q(**{k:v}))
|
||||||
|
|
||||||
order_by = s
|
#join query with operator
|
||||||
if a == "desc":
|
qs = self.get_query_set()
|
||||||
order_by = "-sort__" + order_by
|
if conditions:
|
||||||
qs = qs.order_by(order_by)
|
q = conditions[0]
|
||||||
|
for c in conditions[1:]:
|
||||||
|
if op == '|':
|
||||||
|
q = q | c
|
||||||
|
else:
|
||||||
|
q = q & c
|
||||||
|
qs = qs.filter(q)
|
||||||
|
|
||||||
return qs[o:n]
|
# filter list, works for own or public lists
|
||||||
|
l = request.GET.get('l', 'all')
|
||||||
|
if l != "all":
|
||||||
|
l = l.split(":")
|
||||||
|
only_public = True
|
||||||
|
if not request.user.is_anonymous():
|
||||||
|
if len(l) == 1: l = [request.user.username] + l
|
||||||
|
if request.user.username == l[0]:
|
||||||
|
only_public = False
|
||||||
|
if len(l) == 2:
|
||||||
|
lqs = models.List.objects.filter(name=l[1], user__username=l[0])
|
||||||
|
if only_public:
|
||||||
|
lqs = qls.filter(public=True)
|
||||||
|
if lqs.count() == 1:
|
||||||
|
qs = qs.filter(listitem__list__id=lqs[0].id)
|
||||||
|
return qs
|
||||||
|
|
||||||
|
class ArchiveFileManager(Manager):
|
||||||
|
def get_query_set(self):
|
||||||
|
return super(UserFileManager, self).get_query_set()
|
||||||
|
|
||||||
|
def by_oshash(self, oshash):
|
||||||
|
q = self.get_query_set()
|
||||||
|
q.filter(movie_file__oshash=oshash)
|
||||||
|
if q.count() == 0:
|
||||||
|
raise models.UserFile.DoesNotExist("%s matching oshash %s does not exist." %
|
||||||
|
(models.UserFile._meta.object_name, oshash))
|
||||||
|
return q[0]
|
||||||
|
|
||||||
|
|
|
@ -1,11 +1,14 @@
|
||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
# vi:si:et:sw=4:sts=4:ts=4
|
# vi:si:et:sw=4:sts=4:ts=4
|
||||||
|
import re
|
||||||
import os.path
|
import os.path
|
||||||
from django.db import models
|
from django.db import models
|
||||||
from django.db.models import Q
|
from django.db.models import Q
|
||||||
from django.contrib.auth.models import User
|
from django.contrib.auth.models import User
|
||||||
|
|
||||||
import oxlib
|
import oxlib
|
||||||
|
from oxlib import stripTags
|
||||||
|
from oxlib.normalize import canonicalTitle, canonicalName
|
||||||
|
|
||||||
import utils
|
import utils
|
||||||
import managers
|
import managers
|
||||||
|
@ -61,11 +64,19 @@ class Movie(models.Model):
|
||||||
#FIXME: include role and character
|
#FIXME: include role and character
|
||||||
def cast(self):
|
def cast(self):
|
||||||
cast = []
|
cast = []
|
||||||
for c in Cast.objects.filter(movie=self, role=cast).order_by('position'):
|
for c in Cast.objects.filter(movie=self, role='cast').order_by('position'):
|
||||||
cast.append((c.person.name, c.character))
|
cast.append((c.person.name, c.character))
|
||||||
return tuple(cast)
|
return tuple(cast)
|
||||||
#return self.person.filter(cast__role='cast').order_by('cast__position')
|
#return self.person.filter(cast__role='cast').order_by('cast__position')
|
||||||
|
|
||||||
|
def connections_json(self):
|
||||||
|
connections = {}
|
||||||
|
for connection in self.connections.all():
|
||||||
|
if connection.relation not in connections:
|
||||||
|
connections[connection.relation] = []
|
||||||
|
connections[connection.relation].append(connection.object.movieId)
|
||||||
|
return connections
|
||||||
|
|
||||||
def filtered_reviews(self):
|
def filtered_reviews(self):
|
||||||
whitelist = ReviewWhitelist.objects.all()
|
whitelist = ReviewWhitelist.objects.all()
|
||||||
q = Q(id=-1)
|
q = Q(id=-1)
|
||||||
|
@ -73,22 +84,14 @@ class Movie(models.Model):
|
||||||
q = q | Q(url__contains=w.url)
|
q = q | Q(url__contains=w.url)
|
||||||
return self.reviews.filter(q)
|
return self.reviews.filter(q)
|
||||||
|
|
||||||
risk = models.IntegerField(null=True, blank=True)
|
rights_level = models.IntegerField(default=-1)
|
||||||
rights_level = models.IntegerField(null=True, blank=True)
|
|
||||||
rights_text = models.TextField(blank=True)
|
|
||||||
|
|
||||||
#title_english = models.TextField(blank=True)
|
#FIXME: use data.0xdb.org
|
||||||
#FIXME: join AltTitle
|
|
||||||
|
|
||||||
#FIXME: use mapping
|
|
||||||
tpb_id = models.CharField(max_length=128, blank=True)
|
tpb_id = models.CharField(max_length=128, blank=True)
|
||||||
kg_id = models.CharField(max_length=128, blank=True)
|
kg_id = models.CharField(max_length=128, blank=True)
|
||||||
open_subtitle_id = models.IntegerField(null=True, blank=True)
|
open_subtitle_id = models.IntegerField(null=True, blank=True)
|
||||||
wikipedia_url = models.TextField(blank=True)
|
wikipedia_url = models.TextField(blank=True)
|
||||||
|
|
||||||
#FIXME: join Location
|
|
||||||
#locations = models.TextField(blank=True)
|
|
||||||
|
|
||||||
#Series information
|
#Series information
|
||||||
series_imdb = models.CharField(max_length=7, default='')
|
series_imdb = models.CharField(max_length=7, default='')
|
||||||
series_title = models.TextField(blank=True, default='')
|
series_title = models.TextField(blank=True, default='')
|
||||||
|
@ -107,7 +110,7 @@ class Movie(models.Model):
|
||||||
scene_height = models.IntegerField(null=True, blank=True)
|
scene_height = models.IntegerField(null=True, blank=True)
|
||||||
|
|
||||||
def __unicode__(self):
|
def __unicode__(self):
|
||||||
return "%s (%s)" % (self.title, self.year)
|
return u'%s (%s)' % (self.title, self.year)
|
||||||
|
|
||||||
def save(self, *args, **kwargs):
|
def save(self, *args, **kwargs):
|
||||||
if self.imdbId:
|
if self.imdbId:
|
||||||
|
@ -115,9 +118,10 @@ class Movie(models.Model):
|
||||||
else:
|
else:
|
||||||
mid = self.oxdbId
|
mid = self.oxdbId
|
||||||
self.movieId = mid
|
self.movieId = mid
|
||||||
#FIXME: update sort and find values here
|
|
||||||
|
|
||||||
super(Movie, self).save(*args, **kwargs)
|
super(Movie, self).save(*args, **kwargs)
|
||||||
|
self.updateFind()
|
||||||
|
self.updateSort()
|
||||||
|
|
||||||
_public_fields = {
|
_public_fields = {
|
||||||
'movieId': 'id',
|
'movieId': 'id',
|
||||||
|
@ -138,6 +142,8 @@ class Movie(models.Model):
|
||||||
'episode': 'episode',
|
'episode': 'episode',
|
||||||
'filtered_reviews': 'reviews',
|
'filtered_reviews': 'reviews',
|
||||||
'trivia': 'trivia',
|
'trivia': 'trivia',
|
||||||
|
'alternative_titles': 'alternative_titles',
|
||||||
|
'connections': 'connections_json'
|
||||||
}
|
}
|
||||||
def json(self, fields=None):
|
def json(self, fields=None):
|
||||||
movie = {}
|
movie = {}
|
||||||
|
@ -147,12 +153,25 @@ class Movie(models.Model):
|
||||||
value = getattr(self, key)
|
value = getattr(self, key)
|
||||||
if key in ('directors', 'writers', 'filtered_reviews'):
|
if key in ('directors', 'writers', 'filtered_reviews'):
|
||||||
movie[pub_key] = tuple([v.json() for v in value()])
|
movie[pub_key] = tuple([v.json() for v in value()])
|
||||||
elif key in ('countries', 'keywords', 'genres', 'trivia'):
|
elif key in ('countries', 'keywords', 'genres', 'trivia', 'alternative_titles'):
|
||||||
movie[pub_key] = tuple([v.json() for v in value.all()])
|
movie[pub_key] = tuple([v.json() for v in value.all()])
|
||||||
else:
|
else:
|
||||||
movie[pub_key] = value
|
movie[pub_key] = value
|
||||||
|
for f in fields:
|
||||||
|
if f.endswith('.length') and f[:-7] in ('cast', 'genre', 'trivia'):
|
||||||
|
movie[f] = getattr(self.sort.all()[0], f[:-7])
|
||||||
return movie
|
return movie
|
||||||
|
|
||||||
|
def fields(self):
|
||||||
|
fields = {}
|
||||||
|
for f in self._meta.fields:
|
||||||
|
if f.name in self._public_fields:
|
||||||
|
fields[f.name] = {}
|
||||||
|
fields[f.name]['order'] = 'desc'
|
||||||
|
fields[f.name]['type'] = type(f)
|
||||||
|
return fields
|
||||||
|
fields = classmethod(fields)
|
||||||
|
|
||||||
#Class functions to get Movies by ID, right now movieId, imdbId and oxdbId
|
#Class functions to get Movies by ID, right now movieId, imdbId and oxdbId
|
||||||
#FIXME: this should go into a manager
|
#FIXME: this should go into a manager
|
||||||
def byMovieId(self, movieId):
|
def byMovieId(self, movieId):
|
||||||
|
@ -162,29 +181,124 @@ class Movie(models.Model):
|
||||||
byMovieId = classmethod(byMovieId)
|
byMovieId = classmethod(byMovieId)
|
||||||
|
|
||||||
def byImdbId(self, imdbId):
|
def byImdbId(self, imdbId):
|
||||||
q = self.objects.filter(imdbId=imdbId)
|
return self.objects.get(imdbId=imdbId)
|
||||||
if q.count() == 0:
|
|
||||||
raise self.DoesNotExist("%s matching imdb id %s does not exist." % (self._meta.object_name, imdbId))
|
|
||||||
return q[0]
|
|
||||||
byImdbId = classmethod(byImdbId)
|
byImdbId = classmethod(byImdbId)
|
||||||
|
|
||||||
def byOxdbId(self, oxdbId):
|
def byOxdbId(self, oxdbId):
|
||||||
q = self.objects.filter(oxdbId=oxdbId)
|
return self.objects.get(oxdbId=oxdbId)
|
||||||
if q.count() == 0:
|
|
||||||
raise self.DoesNotExist("%s matching oxdb id %s does not exist." % (self._meta.object_name, oxdbId))
|
|
||||||
return q[0]
|
|
||||||
byOxdbId = classmethod(byOxdbId)
|
byOxdbId = classmethod(byOxdbId)
|
||||||
|
|
||||||
def oxid(self):
|
def oxid(self):
|
||||||
directors = ",".join([d.name for d in self.directors()])
|
directors = ','.join([d.name for d in self.directors()])
|
||||||
return utils.oxid(self.title, directors, self.year, self.series_title, self.episode_title, self.season, self.episode)
|
return utils.oxid(self.title, directors, self.year,
|
||||||
|
self.series_title, self.episode_title, self.season, self.episode)
|
||||||
|
|
||||||
|
def updateFind(self):
|
||||||
|
if self.find.count() == 0:
|
||||||
|
f = MovieFind()
|
||||||
|
f.movie = self
|
||||||
|
else:
|
||||||
|
f = self.find.all()[0]
|
||||||
|
f.title = self.title + ' '.join([t.title for t in self.alternative_titles.all()])
|
||||||
|
f.director = ' '.join([i.name for i in self.directors()])
|
||||||
|
f.country = ' '.join([i.name for i in self.countries.all()])
|
||||||
|
f.year = self.year
|
||||||
|
f.language = ' '.join([i.name for i in self.languages.all()])
|
||||||
|
f.writer = ' '.join([i.name for i in self.writers()])
|
||||||
|
f.producer = ' '.join([i.name for i in self.producers()])
|
||||||
|
f.editor = ' '.join([i.name for i in self.editors()])
|
||||||
|
f.cinematographer = ' '.join([i.name for i in self.cinematographers()])
|
||||||
|
f.cast = ' '.join(['%s %s' % i for i in self.cast()])
|
||||||
|
f.genre = ' '.join([i.name for i in self.genres.all()])
|
||||||
|
f.keywords = ' '.join([i.name for i in self.keywords.all()])
|
||||||
|
f.summary = self.plot + self.plot_outline
|
||||||
|
f.trivia = ' '.join([i.trivia for i in self.trivia.all()])
|
||||||
|
f.location = ' '.join([i.name for i in self.locations.all()])
|
||||||
|
f.filename = self.filename
|
||||||
|
f.all = ' '.join([f.title, f.director, f.country, f.year, f.language,
|
||||||
|
f.writer, f.producer, f.editor, f.cinematographer,
|
||||||
|
f.cast, f.genre, f.keywords, f.summary, f.trivia,
|
||||||
|
f.location, f.filename])
|
||||||
|
f.save()
|
||||||
|
|
||||||
|
def updateSort(self):
|
||||||
|
if self.sort.count() == 0:
|
||||||
|
s = MovieSort()
|
||||||
|
s.movie = self
|
||||||
|
else:
|
||||||
|
s = self.sort.all()[0]
|
||||||
|
|
||||||
|
def sortName(value):
|
||||||
|
sort_value = '~'
|
||||||
|
if value:
|
||||||
|
sort_value = stripTags(value).split(',')
|
||||||
|
sort_value = '; '.join([canonicalName(name.strip()) for name in sort_value])
|
||||||
|
sort_value = sort_value.replace(u'\xc5k', 'A')
|
||||||
|
if not sort_value:
|
||||||
|
sort_value = '~'
|
||||||
|
return sort_value
|
||||||
|
|
||||||
|
#title
|
||||||
|
title = canonicalTitle(self.title)
|
||||||
|
title = re.sub(u'[\'!¿¡,\.;\-"\:\*\[\]]', '', title)
|
||||||
|
title = title.replace(u'Æ', 'Ae')
|
||||||
|
#pad numbered titles
|
||||||
|
numbers = re.compile('^(\d+)').findall(title)
|
||||||
|
if numbers:
|
||||||
|
padded = '%010d' % int(numbers[0])
|
||||||
|
title = re.sub('^(\d+)', padded, title)
|
||||||
|
|
||||||
|
s.title = title.strip()
|
||||||
|
|
||||||
|
directors = ','.join([i.name for i in self.directors()])
|
||||||
|
s.director = sortName(directors)
|
||||||
|
|
||||||
|
s.country = ','.join([i.name for i in self.countries.all()])
|
||||||
|
s.year = self.year
|
||||||
|
|
||||||
|
names = ','.join([i.name for i in self.producers()])
|
||||||
|
s.producer = sortName(names)
|
||||||
|
names = ','.join([i.name for i in self.writers()])
|
||||||
|
s.writer = sortName(names)
|
||||||
|
names = ','.join([i.name for i in self.editors()])
|
||||||
|
s.editor = sortName(names)
|
||||||
|
names = ','.join([i.name for i in self.cinematographers()])
|
||||||
|
s.cinematographer = sortName(names)
|
||||||
|
|
||||||
|
s.country = ','.join([i.name for i in self.languages.all()])
|
||||||
|
s.runtime = self.runtime
|
||||||
|
|
||||||
|
s.keywords = self.keywords.all().count()
|
||||||
|
s.genre = self.genres.all().count()
|
||||||
|
s.cast = len(self.cast())
|
||||||
|
s.summary = len(self.plot.split())
|
||||||
|
s.trivia = self.trivia.all().count()
|
||||||
|
s.connections = self.connections.all().count()
|
||||||
|
s.movieId = self.movieId
|
||||||
|
|
||||||
|
# data from related subtitles
|
||||||
|
s.scenes = 0 #FIXME
|
||||||
|
s.words = 0 #FIXME
|
||||||
|
s.wpm = 0 #FIXME
|
||||||
|
s.risk = 0 #FIXME
|
||||||
|
# data from related files
|
||||||
|
s.duration = 0 #FIXME
|
||||||
|
s.resolution = 0 #FIXME
|
||||||
|
s.aspectratio = 0 #FIXME
|
||||||
|
s.bitrate = 0 #FIXME
|
||||||
|
s.pixels = 0 #FIXME
|
||||||
|
s.filename = 0 #FIXME
|
||||||
|
s.files = 0 #FIXME
|
||||||
|
s.size = 0 #FIXME
|
||||||
|
s.save()
|
||||||
|
|
||||||
'''
|
|
||||||
used to search movies, all search values are in here
|
|
||||||
'''
|
|
||||||
class MovieFind(models.Model):
|
class MovieFind(models.Model):
|
||||||
movie = models.ForeignKey('Movie', related_name='find')
|
"""
|
||||||
|
used to search movies, all search values are in here
|
||||||
|
"""
|
||||||
|
movie = models.ForeignKey('Movie', related_name='find', unique=True)
|
||||||
|
|
||||||
|
all = models.TextField(blank=True)
|
||||||
title = models.CharField(max_length=1000)
|
title = models.CharField(max_length=1000)
|
||||||
director = models.TextField(blank=True)
|
director = models.TextField(blank=True)
|
||||||
country = models.TextField(blank=True)
|
country = models.TextField(blank=True)
|
||||||
|
@ -193,8 +307,8 @@ class MovieFind(models.Model):
|
||||||
writer = models.TextField(blank=True)
|
writer = models.TextField(blank=True)
|
||||||
producer = models.TextField(blank=True)
|
producer = models.TextField(blank=True)
|
||||||
editor = models.TextField(blank=True)
|
editor = models.TextField(blank=True)
|
||||||
cinematographers = models.TextField(blank=True)
|
cinematographer = models.TextField(blank=True)
|
||||||
cast = models.IntegerField(blank=True)
|
cast = models.TextField(blank=True)
|
||||||
#person
|
#person
|
||||||
|
|
||||||
genre = models.TextField(blank=True)
|
genre = models.TextField(blank=True)
|
||||||
|
@ -202,7 +316,6 @@ class MovieFind(models.Model):
|
||||||
summary = models.TextField(blank=True)
|
summary = models.TextField(blank=True)
|
||||||
trivia = models.TextField(blank=True)
|
trivia = models.TextField(blank=True)
|
||||||
locations = models.TextField(blank=True)
|
locations = models.TextField(blank=True)
|
||||||
connections = models.TextField(blank=True)
|
|
||||||
|
|
||||||
#only for own files or as admin?
|
#only for own files or as admin?
|
||||||
filename = models.TextField(blank=True)
|
filename = models.TextField(blank=True)
|
||||||
|
@ -221,11 +334,11 @@ class MovieFind(models.Model):
|
||||||
return tuple(options)
|
return tuple(options)
|
||||||
options = classmethod(options)
|
options = classmethod(options)
|
||||||
|
|
||||||
'''
|
|
||||||
used to sort movies, all sort values are in here
|
|
||||||
'''
|
|
||||||
class MovieSort(models.Model):
|
class MovieSort(models.Model):
|
||||||
movie = models.ForeignKey('Movie', related_name='sort')
|
"""
|
||||||
|
used to sort movies, all sort values are in here
|
||||||
|
"""
|
||||||
|
movie = models.ForeignKey('Movie', related_name='sort', unique=True)
|
||||||
|
|
||||||
title = models.CharField(max_length=1000)
|
title = models.CharField(max_length=1000)
|
||||||
director = models.TextField(blank=True)
|
director = models.TextField(blank=True)
|
||||||
|
@ -235,10 +348,10 @@ class MovieSort(models.Model):
|
||||||
producer = models.TextField(blank=True)
|
producer = models.TextField(blank=True)
|
||||||
writer = models.TextField(blank=True)
|
writer = models.TextField(blank=True)
|
||||||
editor = models.TextField(blank=True)
|
editor = models.TextField(blank=True)
|
||||||
cinematographers = models.TextField(blank=True)
|
cinematographer = models.TextField(blank=True)
|
||||||
|
|
||||||
language = models.TextField(blank=True)
|
language = models.TextField(blank=True)
|
||||||
runtime = models.IntegerField(blank=True)
|
runtime = models.IntegerField(blank=True, null=True)
|
||||||
|
|
||||||
keywords = models.IntegerField(blank=True)
|
keywords = models.IntegerField(blank=True)
|
||||||
genre = models.TextField(blank=True)
|
genre = models.TextField(blank=True)
|
||||||
|
@ -279,8 +392,8 @@ class MovieSort(models.Model):
|
||||||
|
|
||||||
class AlternativeTitle(models.Model):
|
class AlternativeTitle(models.Model):
|
||||||
movie = models.ForeignKey(Movie, related_name='alternative_titles')
|
movie = models.ForeignKey(Movie, related_name='alternative_titles')
|
||||||
type = models.CharField(max_length=128)
|
|
||||||
title = models.TextField()
|
title = models.TextField()
|
||||||
|
type = models.CharField(max_length=1000)
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
ordering = ('title', )
|
ordering = ('title', )
|
||||||
|
@ -288,6 +401,9 @@ class AlternativeTitle(models.Model):
|
||||||
def __unicode__(self):
|
def __unicode__(self):
|
||||||
return self.title
|
return self.title
|
||||||
|
|
||||||
|
def json(self):
|
||||||
|
return (self.title, self.type)
|
||||||
|
|
||||||
def get_or_create(model, name):
|
def get_or_create(model, name):
|
||||||
try:
|
try:
|
||||||
o = model.objects.get(name=name)
|
o = model.objects.get(name=name)
|
||||||
|
@ -408,10 +524,9 @@ class MovieCountry(models.Model):
|
||||||
return link
|
return link
|
||||||
link = classmethod(link)
|
link = classmethod(link)
|
||||||
|
|
||||||
|
|
||||||
class Language(models.Model):
|
class Language(models.Model):
|
||||||
name = models.CharField(max_length=200, unique=True)
|
name = models.CharField(max_length=200, unique=True)
|
||||||
movies = models.ManyToManyField(Movie, related_name='language', through="MovieLanguage")
|
movies = models.ManyToManyField(Movie, related_name='languages', through="MovieLanguage")
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
ordering = ('name', )
|
ordering = ('name', )
|
||||||
|
@ -464,7 +579,6 @@ class Keyword(models.Model):
|
||||||
def json(self):
|
def json(self):
|
||||||
return self.name
|
return self.name
|
||||||
|
|
||||||
|
|
||||||
class Genre(models.Model):
|
class Genre(models.Model):
|
||||||
name = models.CharField(max_length=200, unique=True)
|
name = models.CharField(max_length=200, unique=True)
|
||||||
movies = models.ManyToManyField(Movie, related_name='genres')
|
movies = models.ManyToManyField(Movie, related_name='genres')
|
||||||
|
@ -485,6 +599,14 @@ class Location(models.Model):
|
||||||
movies = models.ManyToManyField(Movie, related_name='locations')
|
movies = models.ManyToManyField(Movie, related_name='locations')
|
||||||
#fixme: geo data
|
#fixme: geo data
|
||||||
|
|
||||||
|
lat_sw = models.FloatField(default=0)
|
||||||
|
lng_sw = models.FloatField(default=0)
|
||||||
|
lat_ne = models.FloatField(default=0)
|
||||||
|
lng_ne = models.FloatField(default=0)
|
||||||
|
lat_center = models.FloatField(default=0)
|
||||||
|
lng_center = models.FloatField(default=0)
|
||||||
|
area = models.FloatField(default=-1)
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
ordering = ('name', )
|
ordering = ('name', )
|
||||||
|
|
||||||
|
@ -508,17 +630,57 @@ class Trivia(models.Model):
|
||||||
return self.trivia
|
return self.trivia
|
||||||
|
|
||||||
def json(self):
|
def json(self):
|
||||||
return self.trivia
|
trivia = self.trivia
|
||||||
|
trivia = oxlib.fixAmpersands(trivia)
|
||||||
|
trivia = re.sub('<a href="(/name/nm.*?)">(.*?)</a>', '<a href="/?f=name&q=\\2">\\2</a>', trivia)
|
||||||
|
trivia = re.sub('<a href="/title/tt(.*?)/">(.*?)</a>', '<a href="/\\1">\\2</a>', trivia)
|
||||||
|
return trivia
|
||||||
|
|
||||||
class MovieFile(models.Model):
|
class Connection(models.Model):
|
||||||
|
subject = models.ForeignKey(Movie, related_name='connections')
|
||||||
|
relation = models.CharField(max_length=512)
|
||||||
|
object = models.ForeignKey(Movie)
|
||||||
|
|
||||||
|
def get_or_create(model, subject, relation, object, reverse=True):
|
||||||
|
q = model.objects.filter(subject=subject, relation=relation, object=object)
|
||||||
|
if q.count() > 0:
|
||||||
|
o = q[0]
|
||||||
|
else:
|
||||||
|
o = model.objects.create(subject=subject, relation=relation, object=object)
|
||||||
|
o.save()
|
||||||
|
if reverse:
|
||||||
|
_map = {
|
||||||
|
'Edited into': 'Edited from',
|
||||||
|
'Features': 'Featured in',
|
||||||
|
'Follows': 'Followed by',
|
||||||
|
'References': 'Referenced in',
|
||||||
|
'Remake of': 'Remade as',
|
||||||
|
'Spin off from': 'Spin off',
|
||||||
|
'Spoofs': 'Spoofed in',
|
||||||
|
'Version of': 'Version of',
|
||||||
|
}
|
||||||
|
if relation in _map.values():
|
||||||
|
for k in _map:
|
||||||
|
if _map[k] == relation:
|
||||||
|
reverse_relation = k
|
||||||
|
else:
|
||||||
|
reverse_relation = _map[relation]
|
||||||
|
o2 = model.get_or_create(object, reverse_relation, subject, reverse=False)
|
||||||
|
return o
|
||||||
|
get_or_create = classmethod(get_or_create)
|
||||||
|
|
||||||
|
def __unicode__(self):
|
||||||
|
return '%s %s %s' % (self.subject, self.relation, self.object)
|
||||||
|
|
||||||
|
class File(models.Model):
|
||||||
created = models.DateTimeField(auto_now_add=True)
|
created = models.DateTimeField(auto_now_add=True)
|
||||||
modified = models.DateTimeField(auto_now=True)
|
modified = models.DateTimeField(auto_now=True)
|
||||||
|
|
||||||
oshash = models.CharField(blank=True, unique=True, max_length=16)
|
oshash = models.CharField(blank=True, unique=True, max_length=16)
|
||||||
sha1hash = models.CharField(blank=True, unique=True, max_length=40)
|
sha1 = models.CharField(blank=True, unique=True, max_length=40)
|
||||||
md5sum = models.CharField(blank=True, unique=True, max_length=32)
|
md5 = models.CharField(blank=True, unique=True, max_length=32)
|
||||||
|
|
||||||
movie = models.ForeignKey('Movie', related_name="files")
|
movie = models.ForeignKey('Movie', related_name="files", default=None)
|
||||||
|
|
||||||
computed_path = models.CharField(blank=True, max_length=2048)
|
computed_path = models.CharField(blank=True, max_length=2048)
|
||||||
size = models.IntegerField(default=-1)
|
size = models.IntegerField(default=-1)
|
||||||
|
@ -540,35 +702,87 @@ class MovieFile(models.Model):
|
||||||
bpp = models.FloatField(default=-1)
|
bpp = models.FloatField(default=-1)
|
||||||
pixels = models.IntegerField(default=0)
|
pixels = models.IntegerField(default=0)
|
||||||
|
|
||||||
part = models.IntegerField(default=1)
|
part = models.IntegerField(default=0)
|
||||||
|
|
||||||
|
def get_or_create(model, oshash):
|
||||||
|
try:
|
||||||
|
f = model.objects.get(oshash=oshash)
|
||||||
|
except model.DoesNotExist:
|
||||||
|
f = model.objects.create(oshash=oshash)
|
||||||
|
f.save()
|
||||||
|
return f
|
||||||
|
get_or_create = classmethod(get_or_create)
|
||||||
|
|
||||||
def __unicode__(self):
|
def __unicode__(self):
|
||||||
return "%s (%s)" % (self.computed_path, self.oshash)
|
return "%s (%s)" % (self.computed_path, self.oshash)
|
||||||
|
|
||||||
class UserFile(models.Model):
|
class Archive(models.Model):
|
||||||
created = models.DateTimeField(auto_now_add=True)
|
created = models.DateTimeField(auto_now_add=True)
|
||||||
modified = models.DateTimeField(auto_now=True)
|
modified = models.DateTimeField(auto_now=True)
|
||||||
user = models.ForeignKey(User)
|
name = models.CharField(max_length=512, unique=True)
|
||||||
movie_file = models.ForeignKey(MovieFile)
|
users = models.ManyToManyField(User, related_name='archives')
|
||||||
|
|
||||||
|
class ArchiveFile(models.Model):
|
||||||
|
created = models.DateTimeField(auto_now_add=True)
|
||||||
|
modified = models.DateTimeField(auto_now=True)
|
||||||
|
archive = models.ForeignKey(Archive)
|
||||||
|
movie_file = models.ForeignKey(File)
|
||||||
path = models.CharField(blank=True, max_length=2048)
|
path = models.CharField(blank=True, max_length=2048)
|
||||||
|
|
||||||
|
objects = managers.ArchiveFileManager()
|
||||||
|
|
||||||
|
def update(self, data):
|
||||||
|
"""
|
||||||
|
only add, do not overwrite keys in movie_file
|
||||||
|
"""
|
||||||
|
for key in ('duration', 'video_codec', 'pixel_format', 'width', 'height',
|
||||||
|
'pixel_aspect_ratio', 'display_aspect_ratio', 'framerate',
|
||||||
|
'audio_codec', 'samplerate', 'channels', 'size', 'sha1', 'md5'):
|
||||||
|
if key in data and not getattr(self.movie_file, key):
|
||||||
|
setattr(self.movie_file, key, data[key])
|
||||||
|
self.path = data.get('path', '')
|
||||||
|
self.movie_file.save()
|
||||||
|
self.save()
|
||||||
|
|
||||||
|
def get_or_create(model, archive, oshash):
|
||||||
|
try:
|
||||||
|
f = model.objects.by_oshash(oshash=oshash)
|
||||||
|
except model.DoesNotExist:
|
||||||
|
f = model.objects.create()
|
||||||
|
f.movie_file = File.get_or_create(oshash)
|
||||||
|
f.archive = archive
|
||||||
|
f.save()
|
||||||
|
return f
|
||||||
|
get_or_create = classmethod(get_or_create)
|
||||||
|
|
||||||
def __unicode__(self):
|
def __unicode__(self):
|
||||||
return "%s (%s)" % (self.path, unicode(self.user))
|
return '%s (%s)' % (self.path, unicode(self.user))
|
||||||
|
|
||||||
class Subtitle(models.Model):
|
class Subtitle(models.Model):
|
||||||
created = models.DateTimeField(auto_now_add=True)
|
created = models.DateTimeField(auto_now_add=True)
|
||||||
modified = models.DateTimeField(auto_now=True)
|
modified = models.DateTimeField(auto_now=True)
|
||||||
user = models.ForeignKey(User)
|
user = models.ForeignKey(User)
|
||||||
|
|
||||||
movie_file = models.ForeignKey(MovieFile)
|
movie_file = models.ForeignKey(File)
|
||||||
language = models.CharField(max_length=16)
|
language = models.CharField(max_length=16)
|
||||||
srt = models.TextField(blank=True)
|
srt = models.TextField(blank=True)
|
||||||
|
|
||||||
|
def get_or_create(model, user, oshash, language):
|
||||||
|
q = model.objects.filter(movie_file__oshash=oshash, language=language, user=user)
|
||||||
|
if q.count() > 0:
|
||||||
|
s = q[0]
|
||||||
|
else:
|
||||||
|
movie_file = models.File.get_or_create(oshash=oshash)
|
||||||
|
s = model.objects.create(user=user, language=language, movie_file=movie_file)
|
||||||
|
s.save()
|
||||||
|
return s
|
||||||
|
get_or_create = classmethod(get_or_create)
|
||||||
|
|
||||||
def __unicode__(self):
|
def __unicode__(self):
|
||||||
return "%s.%s.srt" % (os.path.splitext(self.movie_file.computed_path)[0], self.language)
|
return '%s.%s.srt' % (os.path.splitext(self.movie_file.computed_path)[0], self.language)
|
||||||
|
|
||||||
class Review(models.Model):
|
class Review(models.Model):
|
||||||
movie = models.ForeignKey('Movie', related_name="reviews")
|
movie = models.ForeignKey('Movie', related_name='reviews')
|
||||||
title = models.CharField(blank=True, max_length=2048)
|
title = models.CharField(blank=True, max_length=2048)
|
||||||
url = models.CharField(blank=True, max_length=2048)
|
url = models.CharField(blank=True, max_length=2048)
|
||||||
|
|
||||||
|
@ -602,7 +816,8 @@ class List(models.Model):
|
||||||
created = models.DateTimeField(auto_now_add=True)
|
created = models.DateTimeField(auto_now_add=True)
|
||||||
modified = models.DateTimeField(auto_now=True)
|
modified = models.DateTimeField(auto_now=True)
|
||||||
user = models.ForeignKey(User)
|
user = models.ForeignKey(User)
|
||||||
title = models.CharField(max_length=255, unique=True)
|
name = models.CharField(max_length=255, unique=True)
|
||||||
|
public = models.BooleanField(default=False)
|
||||||
movies = models.ManyToManyField(Movie, related_name='lists', through='ListItem')
|
movies = models.ManyToManyField(Movie, related_name='lists', through='ListItem')
|
||||||
|
|
||||||
def add(self, movie):
|
def add(self, movie):
|
||||||
|
@ -617,7 +832,7 @@ class List(models.Model):
|
||||||
self.ListItem.objects.all().filter(movie=movie, list=self).delete()
|
self.ListItem.objects.all().filter(movie=movie, list=self).delete()
|
||||||
|
|
||||||
def __unicode__(self):
|
def __unicode__(self):
|
||||||
return u"%s (%s)" % (self.title, unicode(self.user))
|
return u'%s (%s)' % (self.title, unicode(self.user))
|
||||||
|
|
||||||
class ListItem(models.Model):
|
class ListItem(models.Model):
|
||||||
created = models.DateTimeField(auto_now_add=True)
|
created = models.DateTimeField(auto_now_add=True)
|
||||||
|
@ -626,5 +841,5 @@ class ListItem(models.Model):
|
||||||
movie = models.ForeignKey(Movie)
|
movie = models.ForeignKey(Movie)
|
||||||
|
|
||||||
def __unicode__(self):
|
def __unicode__(self):
|
||||||
return u"%s in %s" % (unicode(self.movie), unicode(self.list))
|
return u'%s in %s' % (unicode(self.movie), unicode(self.list))
|
||||||
|
|
||||||
|
|
|
@ -24,9 +24,10 @@ urlpatterns = patterns("oxdb.backend.views",
|
||||||
(r'^find', 'find'),
|
(r'^find', 'find'),
|
||||||
(r'^files/find', 'find_files'),
|
(r'^files/find', 'find_files'),
|
||||||
(r'^files/info', 'file_info'),
|
(r'^files/info', 'file_info'),
|
||||||
(r'^files/add', 'add_file'),
|
(r'^files/(?P<archive>.+)/add', 'add_file'),
|
||||||
(r'^files/remove', 'remove_file'),
|
(r'^files/(?P<archive>.+)/remove', 'remove_file'),
|
||||||
(r'^subtitle/get', 'get_subtitle'),
|
(r'^subtitle/get', 'subtitles'),
|
||||||
|
(r'^preferences', 'preferences'),
|
||||||
|
|
||||||
# Example:
|
# Example:
|
||||||
# (r'^oxdata/', include('oxdata.foo.urls')),
|
# (r'^oxdata/', include('oxdata.foo.urls')),
|
||||||
|
|
212
backend/views.py
212
backend/views.py
|
@ -1,19 +1,69 @@
|
||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
# vi:si:et:sw=4:sts=4:ts=4
|
# vi:si:et:sw=4:sts=4:ts=4
|
||||||
import os.path
|
import os.path
|
||||||
|
import re
|
||||||
|
from datetime import datetime
|
||||||
|
from urllib2 import unquote
|
||||||
|
|
||||||
from django.db import models
|
from django.db import models
|
||||||
from django.db.models import Q, Avg, Count
|
from django.db.models import Q, Avg, Count
|
||||||
from django.contrib.auth.models import User
|
from django.contrib.auth.models import User
|
||||||
from django.shortcuts import render_to_response, get_object_or_404, get_list_or_404
|
from django.shortcuts import render_to_response, get_object_or_404, get_list_or_404
|
||||||
from django.template import RequestContext
|
from django.template import RequestContext
|
||||||
from django.core.paginator import Paginator
|
from django.core.paginator import Paginator
|
||||||
|
from django.contrib.auth.decorators import login_required
|
||||||
|
|
||||||
from oxdb.utils.shortcuts import render_to_json_response
|
from oxdb.utils.shortcuts import render_to_json_response
|
||||||
|
|
||||||
import models
|
import models
|
||||||
|
|
||||||
|
|
||||||
'''
|
'''
|
||||||
.length -> _sort
|
|
||||||
|
field.length -> movie.sort.all()[0].field
|
||||||
|
o=0&n=100
|
||||||
|
|
||||||
|
|
||||||
|
a & b | c & d
|
||||||
|
|
||||||
|
query
|
||||||
|
|
||||||
|
|
||||||
|
l=user:name or l=name
|
||||||
|
q=year:1980,hello,country:usa
|
||||||
|
q=year:1980,hello,country:!usa
|
||||||
|
q=title:^the$
|
||||||
|
q=title:^100%24$
|
||||||
|
q=year:<1970,year:>1960
|
||||||
|
q=year:<1960,year:>1950,title:sex
|
||||||
|
|
||||||
|
!1960-1970
|
||||||
|
2009.08.02.22.26.35-2009.08.02.22.26.35
|
||||||
|
|
||||||
|
!^the
|
||||||
|
|
||||||
|
(dddd.dd.dd)-(dddd.dd.dd)
|
||||||
|
|
||||||
|
5-8
|
||||||
|
10000-20000
|
||||||
|
|
||||||
|
<2009-08-02-22-26-35
|
||||||
|
|
||||||
|
>2009-08-02-22-26-35
|
||||||
|
2009-08-02-22-26-35<
|
||||||
|
|
||||||
|
^the the*
|
||||||
|
*foo foo$
|
||||||
|
*foo* foo
|
||||||
|
|
||||||
|
s=director:asc,year:desc default: director:asc,year:desc
|
||||||
|
r=0:100 or r=100 or r=100: default: 0:100
|
||||||
|
k=id,title,director,date,cast.length default: title,director,year,country
|
||||||
|
|
||||||
|
|
||||||
|
id=0133093
|
||||||
|
|
||||||
|
/json/find?l=all&s=date&f=all&q=&a=desc&p=id,title,director,date,cast.length
|
||||||
|
|
||||||
/json/find?o=0&n=100&l=all&s=date&f=all&q=&a=desc&p=id,title,director,date,cast.length
|
/json/find?o=0&n=100&l=all&s=date&f=all&q=&a=desc&p=id,title,director,date,cast.length
|
||||||
{
|
{
|
||||||
|
@ -56,20 +106,32 @@ import models
|
||||||
#auto compleat in find box
|
#auto compleat in find box
|
||||||
|
|
||||||
'''
|
'''
|
||||||
def parse_query(get):
|
|
||||||
|
def order_query(qs, s, prefix='sort__'):
|
||||||
|
order_by = []
|
||||||
|
for e in s.split(','):
|
||||||
|
o = e.split(':')
|
||||||
|
if len(o) == 1: o.append('asc')
|
||||||
|
order = '%s%s' % (prefix, o[0])
|
||||||
|
if o[1] == 'desc':
|
||||||
|
order = '-%s' % order
|
||||||
|
order_by.append(order)
|
||||||
|
if order_by:
|
||||||
|
qs = qs.order_by(*order_by)
|
||||||
|
return qs
|
||||||
|
|
||||||
|
def parse_query(request):
|
||||||
|
get = request.GET
|
||||||
query = {}
|
query = {}
|
||||||
query["o"] = 0
|
query['i'] = 0
|
||||||
query["n"] = 100
|
query['o'] = 100
|
||||||
query["q"] = "The"
|
query['s'] = 'title:asc'
|
||||||
query["f"] = "all"
|
|
||||||
query["s"] = "title"
|
|
||||||
query["a"] = "desc"
|
|
||||||
def parse_dict(s):
|
def parse_dict(s):
|
||||||
d = s.split(",")
|
d = s.split(",")
|
||||||
return [i.strip() for i in d]
|
return [i.strip() for i in d]
|
||||||
_dicts = ['p', ]
|
_dicts = ['k', ]
|
||||||
_ints = ['o', 'n']
|
_ints = ['o', 'n']
|
||||||
for key in ('q', 'f', 's', 'a', 'p', 'g', 'o', 'n'):
|
for key in ('s', 'k', 'g', 'l'):
|
||||||
if key in get:
|
if key in get:
|
||||||
if key in _ints:
|
if key in _ints:
|
||||||
query[key] = int(get[key])
|
query[key] = int(get[key])
|
||||||
|
@ -77,44 +139,57 @@ def parse_query(get):
|
||||||
query[key] = parse_dict(get[key])
|
query[key] = parse_dict(get[key])
|
||||||
else:
|
else:
|
||||||
query[key] = get[key]
|
query[key] = get[key]
|
||||||
print query
|
query['q'] = models.Movie.objects.find(request)
|
||||||
|
if 'r' in get:
|
||||||
|
r = get['r'].split(':')
|
||||||
|
if len(r) == 1: r.append(0)
|
||||||
|
if r[0] == '': r[0] = 0
|
||||||
|
if r[1] == '': r[0] = -1
|
||||||
|
query['i'] = int(r[0])
|
||||||
|
query['o'] = int(r[1])
|
||||||
return query
|
return query
|
||||||
|
|
||||||
def find(request):
|
def find(request):
|
||||||
query = parse_query(request.GET)
|
query = parse_query(request)
|
||||||
response = {}
|
response = {}
|
||||||
if "p" in query:
|
if 'k' in query:
|
||||||
response['movies'] = []
|
response['movies'] = []
|
||||||
|
qs = order_query(query['q'], query['s'])
|
||||||
qs = models.Movie.objects.find(**query)
|
qs = qs[query['i']:query['o']]
|
||||||
p = Paginator(qs, 100)
|
p = Paginator(qs, 100)
|
||||||
for i in p.page_range:
|
for i in p.page_range:
|
||||||
page = p.page(i)
|
page = p.page(i)
|
||||||
for m in page.object_list:
|
for m in page.object_list:
|
||||||
response['movies'].append(m.json(query['p']))
|
response['movies'].append(m.json(query['k']))
|
||||||
elif "g" in query:
|
elif 'g' in query:
|
||||||
|
#FIXME: also filter lists here
|
||||||
response['groups'] = []
|
response['groups'] = []
|
||||||
name = "name"
|
name = 'name'
|
||||||
movies = "movies"
|
movies = 'movies'
|
||||||
if query["g"] == "country":
|
movie_qs = query['q']
|
||||||
qs = models.Country.objects.values("name").annotate(movies=Count('movies'))
|
_objects = {
|
||||||
if query["g"] == "genre":
|
'country': models.Country.objects,
|
||||||
qs = models.Genre.objects.values("name").annotate(movies=Count('movies'))
|
'genre': models.Genre.objects,
|
||||||
if query["g"] == "language":
|
'language': models.Language.objects,
|
||||||
qs = models.Language.objects.values("name").annotate(movies=Count('movies'))
|
'director': models.Person.objects.filter(cast__role='directors'),
|
||||||
if query["g"] == "director":
|
}
|
||||||
qs = models.Person.objects.filter(cast__role="directors").values("name").annotate(movies=Count('movies'))
|
if query['g'] in _objects:
|
||||||
if query["g"] == "year":
|
qs = _objects[query['g']].filter(movies__id__in=movie_qs).values('name').annotate(movies=Count('movies'))
|
||||||
qs = models.Movie.objects.values('year').annotate(movies=Count('id'))
|
elif query['g'] == "year":
|
||||||
name="year"
|
qs = movie_qs.values('year').annotate(movies=Count('id'))
|
||||||
qs = qs[query['o']:query['n']]
|
name='year'
|
||||||
|
qs = order_query(qs, query['s'], '')
|
||||||
|
qs = qs[query['i']:query['o']]
|
||||||
for i in qs:
|
for i in qs:
|
||||||
group = {"name": i[name], "movies": i[movies]}
|
group = {'name': i[name], 'movies': i[movies]}
|
||||||
response['groups'].append(group)
|
response['groups'].append(group)
|
||||||
else:
|
else:
|
||||||
response['movies'] = models.Movie.objects.all().count()
|
#FIXME: also filter lists here
|
||||||
response['files'] = models.MovieFile.objects.all().count()
|
movies = models.Movie.objects.all()
|
||||||
r = models.MovieFile.objects.all().aggregate(Count('size'), Count('pixels'), Count('duration'))
|
files = models.MovieFile.objects.all()
|
||||||
|
response['movies'] = movies.count()
|
||||||
|
response['files'] = files.count()
|
||||||
|
r = files.aggregate(Count('size'), Count('pixels'), Count('duration'))
|
||||||
response['pixels'] = r['pixels__count']
|
response['pixels'] = r['pixels__count']
|
||||||
response['size'] = r['size__count']
|
response['size'] = r['size__count']
|
||||||
response['duration'] = r['duration__count']
|
response['duration'] = r['duration__count']
|
||||||
|
@ -144,6 +219,9 @@ GET info?oshash=a41cde31c581e11d
|
||||||
'''
|
'''
|
||||||
def file_info(request):
|
def file_info(request):
|
||||||
oshash = request.GET['oshash']
|
oshash = request.GET['oshash']
|
||||||
|
f = models.MovieFile.objects.get(oshash=oshash)
|
||||||
|
response = f.json()
|
||||||
|
return render_to_json_response(response)
|
||||||
|
|
||||||
|
|
||||||
'''
|
'''
|
||||||
|
@ -159,9 +237,20 @@ srt =
|
||||||
def subtitles(request):
|
def subtitles(request):
|
||||||
oshash = request.GET['oshash']
|
oshash = request.GET['oshash']
|
||||||
language = request.GET.get('language', None)
|
language = request.GET.get('language', None)
|
||||||
|
if requeset.method == 'POST':
|
||||||
|
user = request.user
|
||||||
|
sub = models.Subtitles.get_or_create(user, oshash, language)
|
||||||
|
sub.srt = request.POST['srt']
|
||||||
|
sub.save()
|
||||||
|
else:
|
||||||
if language:
|
if language:
|
||||||
return srt
|
q = models.Subtitles.objects.filter(movie_file__oshash=oshash, language=language)
|
||||||
return movie.subtitle_languages()
|
if q.count() > 0:
|
||||||
|
return HttpResponse(q[0].srt, content_type='text/x-srt')
|
||||||
|
response = {}
|
||||||
|
l = models.Subtitles.objects.filter(movie_file__oshash=oshash).values('language')
|
||||||
|
response['languages'] = [f['language'] for f in l]
|
||||||
|
return render_to_json_response(response)
|
||||||
|
|
||||||
'''
|
'''
|
||||||
GET list
|
GET list
|
||||||
|
@ -171,9 +260,27 @@ GET list
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
'''
|
'''
|
||||||
|
@login_required
|
||||||
def list_files(request):
|
def list_files(request):
|
||||||
files = {}
|
response['files'] = {}
|
||||||
return dict(files=files)
|
qs = models.UserFile.filter(user=request.user)
|
||||||
|
p = Paginator(qs, 1000)
|
||||||
|
for i in p.page_range:
|
||||||
|
page = p.page(i)
|
||||||
|
for f in page.object_list:
|
||||||
|
response['files'][f.movie_file.oshash] = {'path': f.path, 'size': f.movie_file.size}
|
||||||
|
return render_to_json_response(response)
|
||||||
|
|
||||||
|
def find_files(request):
|
||||||
|
query = parse_query(request)
|
||||||
|
response['files'] = {}
|
||||||
|
qs = models.UserFile.filter(user=request.user).filter(movie_file__movie__id__in=quert['q'])
|
||||||
|
p = Paginator(qs, 1000)
|
||||||
|
for i in p.page_range:
|
||||||
|
page = p.page(i)
|
||||||
|
for f in page.object_list:
|
||||||
|
response['files'][f.movie_file.oshash] = {'path': f.path, 'size': f.movie_file.size}
|
||||||
|
return render_to_json_response(response)
|
||||||
|
|
||||||
'''
|
'''
|
||||||
POST add
|
POST add
|
||||||
|
@ -196,13 +303,36 @@ POST add
|
||||||
"md5":..
|
"md5":..
|
||||||
}
|
}
|
||||||
'''
|
'''
|
||||||
def add_file(request):
|
@login_required
|
||||||
|
def add_file(request, archive):
|
||||||
oshash = request.POST['oshash']
|
oshash = request.POST['oshash']
|
||||||
|
archive = models.Archive.objects.get(name=archive)
|
||||||
|
if archive.users.filter(user=request.user).count() == 1:
|
||||||
|
user_file = models.ArchiveFiles.get_or_create(request.user, oshash)
|
||||||
|
user_file.update(request.POST)
|
||||||
|
response = {'status': 200}
|
||||||
|
else:
|
||||||
|
response = {'status': 404}
|
||||||
|
return render_to_json_response(response)
|
||||||
|
|
||||||
'''
|
'''
|
||||||
POST remove?oshash=
|
POST remove?oshash=
|
||||||
'''
|
'''
|
||||||
def remove_file(request):
|
@login_required
|
||||||
|
def remove_file(request, archive):
|
||||||
oshash = request.POST['oshash']
|
oshash = request.POST['oshash']
|
||||||
|
archive = models.Archive.objects.get(name=archive)
|
||||||
|
models.UserFiles.objects.filter(movie_file__oshash=oshash, user=request.user).delete()
|
||||||
|
response = {'status': 200}
|
||||||
|
return render_to_json_response(response)
|
||||||
|
|
||||||
|
'''
|
||||||
|
POST preferences/get?key=
|
||||||
|
POST preferences/set?key=&value
|
||||||
|
'''
|
||||||
|
@login_required
|
||||||
|
def preferences(request):
|
||||||
|
oshash = request.POST['oshash']
|
||||||
|
return ''
|
||||||
|
|
||||||
|
|
||||||
|
|
34
settings.py
34
settings.py
|
@ -1,4 +1,10 @@
|
||||||
# Django settings for oxdata project.
|
# -*- coding: utf-8 -*-
|
||||||
|
# vi:si:et:sw=4:sts=4:ts=4
|
||||||
|
# Django settings for oxdb project.
|
||||||
|
import os
|
||||||
|
from os.path import join
|
||||||
|
|
||||||
|
PROJECT_PATH = os.path.normpath(os.path.dirname(__file__))
|
||||||
|
|
||||||
DEBUG = True
|
DEBUG = True
|
||||||
TEMPLATE_DEBUG = DEBUG
|
TEMPLATE_DEBUG = DEBUG
|
||||||
|
@ -21,7 +27,7 @@ DATABASE_PORT = '' # Set to empty string for default. Not used with
|
||||||
# although not all choices may be available on all operating systems.
|
# although not all choices may be available on all operating systems.
|
||||||
# If running in a Windows environment this must be set to the same as your
|
# If running in a Windows environment this must be set to the same as your
|
||||||
# system time zone.
|
# system time zone.
|
||||||
TIME_ZONE = 'America/Chicago'
|
TIME_ZONE = 'Europe/Berlin'
|
||||||
|
|
||||||
# Language code for this installation. All choices can be found here:
|
# Language code for this installation. All choices can be found here:
|
||||||
# http://www.i18nguy.com/unicode/language-identifiers.html
|
# http://www.i18nguy.com/unicode/language-identifiers.html
|
||||||
|
@ -35,7 +41,8 @@ USE_I18N = True
|
||||||
|
|
||||||
# Absolute path to the directory that holds media.
|
# Absolute path to the directory that holds media.
|
||||||
# Example: "/home/media/media.lawrence.com/"
|
# Example: "/home/media/media.lawrence.com/"
|
||||||
MEDIA_ROOT = ''
|
MEDIA_ROOT = join(PROJECT_PATH, 'media')
|
||||||
|
STATIC_ROOT = join(PROJECT_PATH, 'static')
|
||||||
|
|
||||||
# URL that handles the media served from MEDIA_ROOT. Make sure to use a
|
# URL that handles the media served from MEDIA_ROOT. Make sure to use a
|
||||||
# trailing slash if there is a path component (optional in other cases).
|
# trailing slash if there is a path component (optional in other cases).
|
||||||
|
@ -45,7 +52,7 @@ MEDIA_URL = ''
|
||||||
# URL prefix for admin media -- CSS, JavaScript and images. Make sure to use a
|
# URL prefix for admin media -- CSS, JavaScript and images. Make sure to use a
|
||||||
# trailing slash.
|
# trailing slash.
|
||||||
# Examples: "http://foo.com/media/", "/media/".
|
# Examples: "http://foo.com/media/", "/media/".
|
||||||
ADMIN_MEDIA_PREFIX = '/media/'
|
ADMIN_MEDIA_PREFIX = '/admin/media/'
|
||||||
|
|
||||||
# Make this unique, and don't share it with anybody.
|
# Make this unique, and don't share it with anybody.
|
||||||
SECRET_KEY = '3fh^twg4!7*xcise#3d5%ty+^-#9+*f0innkjcco+y0dag_nr-'
|
SECRET_KEY = '3fh^twg4!7*xcise#3d5%ty+^-#9+*f0innkjcco+y0dag_nr-'
|
||||||
|
@ -66,9 +73,7 @@ MIDDLEWARE_CLASSES = (
|
||||||
ROOT_URLCONF = 'oxdb.urls'
|
ROOT_URLCONF = 'oxdb.urls'
|
||||||
|
|
||||||
TEMPLATE_DIRS = (
|
TEMPLATE_DIRS = (
|
||||||
# Put strings here, like "/home/html/django_templates" or "C:/www/django/templates".
|
join(PROJECT_PATH, 'templates'),
|
||||||
# Always use forward slashes, even on Windows.
|
|
||||||
# Don't forget to use absolute paths, not relative paths.
|
|
||||||
)
|
)
|
||||||
|
|
||||||
INSTALLED_APPS = (
|
INSTALLED_APPS = (
|
||||||
|
@ -76,5 +81,20 @@ INSTALLED_APPS = (
|
||||||
'django.contrib.contenttypes',
|
'django.contrib.contenttypes',
|
||||||
'django.contrib.sessions',
|
'django.contrib.sessions',
|
||||||
'django.contrib.sites',
|
'django.contrib.sites',
|
||||||
|
'django.contrib.admin',
|
||||||
|
'django.contrib.humanize',
|
||||||
|
|
||||||
'oxdb.backend',
|
'oxdb.backend',
|
||||||
)
|
)
|
||||||
|
|
||||||
|
try:
|
||||||
|
import socket
|
||||||
|
# hostname = socket.gethostname().replace('.','_')
|
||||||
|
# exec "from host_settings.%s import *" % hostname
|
||||||
|
local_settings_module = socket.gethostname().split(".")[0]
|
||||||
|
if local_settings_module:
|
||||||
|
execfile(os.path.join(PROJECT_PATH, "host_settings", "%s.py" % local_settings_module))
|
||||||
|
except ImportError, e:
|
||||||
|
raise e
|
||||||
|
|
||||||
|
|
||||||
|
|
19
urls.py
19
urls.py
|
@ -1,8 +1,11 @@
|
||||||
from django.conf.urls.defaults import *
|
from django.conf.urls.defaults import *
|
||||||
|
|
||||||
|
from django.conf import settings
|
||||||
|
|
||||||
# Uncomment the next two lines to enable the admin:
|
# Uncomment the next two lines to enable the admin:
|
||||||
# from django.contrib import admin
|
from django.contrib import admin
|
||||||
# admin.autodiscover()
|
admin.autodiscover()
|
||||||
|
|
||||||
|
|
||||||
urlpatterns = patterns('',
|
urlpatterns = patterns('',
|
||||||
# Example:
|
# Example:
|
||||||
|
@ -13,5 +16,15 @@ urlpatterns = patterns('',
|
||||||
# (r'^admin/doc/', include('django.contrib.admindocs.urls')),
|
# (r'^admin/doc/', include('django.contrib.admindocs.urls')),
|
||||||
|
|
||||||
# Uncomment the next line to enable the admin:
|
# Uncomment the next line to enable the admin:
|
||||||
# (r'^admin/(.*)', admin.site.root),
|
(r'^admin/(.*)', admin.site.root),
|
||||||
)
|
)
|
||||||
|
|
||||||
|
if settings.DEBUG:
|
||||||
|
urlpatterns += patterns('',
|
||||||
|
(r'^media/(?P<path>.*)$', 'django.views.static.serve',
|
||||||
|
{'document_root': settings.MEDIA_ROOT}),
|
||||||
|
(r'^static/(?P<path>.*)$', 'django.views.static.serve',
|
||||||
|
{'document_root': settings.STATIC_ROOT}),
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue