2014-05-04 17:26:43 +00:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
# vi:si:et:sw=4:sts=4:ts=4
|
2014-05-16 08:06:11 +00:00
|
|
|
from __future__ import division
|
|
|
|
|
2014-05-18 23:24:04 +00:00
|
|
|
import os
|
2014-05-17 00:14:15 +00:00
|
|
|
import json
|
2014-05-18 23:24:04 +00:00
|
|
|
|
2014-05-04 17:26:43 +00:00
|
|
|
from oxflask.api import actions
|
|
|
|
from oxflask.shortcuts import returns_json
|
2014-05-18 23:24:04 +00:00
|
|
|
from sqlalchemy.orm import load_only
|
2014-05-04 17:26:43 +00:00
|
|
|
|
|
|
|
import query
|
|
|
|
|
|
|
|
import models
|
|
|
|
import settings
|
|
|
|
import state
|
|
|
|
|
2014-05-14 18:46:31 +00:00
|
|
|
import meta
|
|
|
|
|
2014-05-04 17:26:43 +00:00
|
|
|
import utils
|
|
|
|
|
2014-05-18 23:24:04 +00:00
|
|
|
import logging
|
2014-05-17 14:26:59 +00:00
|
|
|
logger = logging.getLogger('oml.item.api')
|
|
|
|
|
2014-05-04 17:26:43 +00:00
|
|
|
@returns_json
|
|
|
|
def find(request):
|
|
|
|
'''
|
2014-05-18 23:24:04 +00:00
|
|
|
takes {
|
|
|
|
query {
|
|
|
|
conditions [{}]
|
|
|
|
operator string
|
|
|
|
}
|
|
|
|
group string
|
|
|
|
keys [string]
|
|
|
|
sort [{}]
|
|
|
|
range [int, int]
|
|
|
|
}
|
2014-05-04 17:26:43 +00:00
|
|
|
'''
|
|
|
|
response = {}
|
|
|
|
data = json.loads(request.form['data']) if 'data' in request.form else {}
|
|
|
|
q = query.parse(data)
|
|
|
|
if 'group' in q:
|
2014-05-14 09:57:11 +00:00
|
|
|
names = {}
|
|
|
|
groups = {}
|
2014-05-18 23:24:04 +00:00
|
|
|
items = [i.id for i in q['qs'].options(load_only('id'))]
|
2014-05-14 09:57:11 +00:00
|
|
|
qs = models.Find.query.filter_by(key=q['group'])
|
|
|
|
if items:
|
|
|
|
qs = qs.filter(models.Find.item_id.in_(items))
|
2014-05-17 00:14:15 +00:00
|
|
|
for f in qs.values('value', 'findvalue'):
|
|
|
|
value = f[0]
|
|
|
|
findvalue = f[1]
|
|
|
|
if findvalue not in groups:
|
|
|
|
groups[findvalue] = 0
|
|
|
|
groups[findvalue] += 1
|
|
|
|
names[findvalue] = value
|
|
|
|
g = [{'name': names[k], 'items': groups[k]} for k in groups]
|
|
|
|
else:
|
|
|
|
g = []
|
2014-05-14 09:57:11 +00:00
|
|
|
if 'sort' in q:
|
2014-05-04 17:26:43 +00:00
|
|
|
g.sort(key=lambda k: k[q['sort'][0]['key']])
|
|
|
|
if q['sort'][0]['operator'] == '-':
|
|
|
|
g.reverse()
|
|
|
|
if 'positions' in data:
|
|
|
|
response['positions'] = {}
|
|
|
|
ids = [k['name'] for k in g]
|
|
|
|
response['positions'] = utils.get_positions(ids, data['positions'])
|
|
|
|
elif 'range' in data:
|
|
|
|
response['items'] = g[q['range'][0]:q['range'][1]]
|
|
|
|
else:
|
|
|
|
response['items'] = len(g)
|
|
|
|
elif 'position' in data:
|
2014-05-18 23:24:04 +00:00
|
|
|
ids = [i.id for i in q['qs'].options(load_only('id'))]
|
2014-05-04 17:26:43 +00:00
|
|
|
response['position'] = utils.get_positions(ids, [data['qs'][0].id])[0]
|
|
|
|
elif 'positions' in data:
|
2014-05-18 23:24:04 +00:00
|
|
|
ids = [i.id for i in q['qs'].options(load_only('id'))]
|
2014-05-04 17:26:43 +00:00
|
|
|
response['positions'] = utils.get_positions(ids, data['positions'])
|
|
|
|
elif 'keys' in data:
|
|
|
|
response['items'] = []
|
|
|
|
for i in q['qs'][q['range'][0]:q['range'][1]]:
|
|
|
|
j = i.json()
|
|
|
|
response['items'].append({k:j[k] for k in j if not data['keys'] or k in data['keys']})
|
|
|
|
else:
|
2014-05-14 09:57:11 +00:00
|
|
|
response['items'] = q['qs'].count()
|
|
|
|
#from sqlalchemy.sql import func
|
|
|
|
#models.db.session.query(func.sum(models.Item.sort_size).label("size"))
|
|
|
|
#response['size'] = x.scalar()
|
2014-05-18 23:24:04 +00:00
|
|
|
response['size'] = sum([i.sort_size or 0 for i in q['qs'].options(load_only('id', 'sort_size'))])
|
2014-05-04 17:26:43 +00:00
|
|
|
return response
|
|
|
|
actions.register(find)
|
|
|
|
|
|
|
|
@returns_json
|
|
|
|
def get(request):
|
2014-05-18 23:24:04 +00:00
|
|
|
'''
|
|
|
|
takes {
|
|
|
|
id
|
|
|
|
keys
|
|
|
|
}
|
|
|
|
'''
|
2014-05-04 17:26:43 +00:00
|
|
|
response = {}
|
|
|
|
data = json.loads(request.form['data']) if 'data' in request.form else {}
|
|
|
|
item = models.Item.get(data['id'])
|
|
|
|
if item:
|
|
|
|
response = item.json(data['keys'] if 'keys' in data else None)
|
|
|
|
return response
|
|
|
|
actions.register(get)
|
|
|
|
|
|
|
|
@returns_json
|
|
|
|
def edit(request):
|
2014-05-18 23:24:04 +00:00
|
|
|
'''
|
|
|
|
takes {
|
|
|
|
id
|
|
|
|
...
|
|
|
|
}
|
|
|
|
setting identifier or base metadata is possible not both at the same time
|
|
|
|
'''
|
2014-05-04 17:26:43 +00:00
|
|
|
response = {}
|
|
|
|
data = json.loads(request.form['data']) if 'data' in request.form else {}
|
2014-05-18 23:24:04 +00:00
|
|
|
logger.debug('edit %s', data)
|
2014-05-04 17:26:43 +00:00
|
|
|
item = models.Item.get(data['id'])
|
|
|
|
keys = filter(lambda k: k in models.Item.id_keys, data.keys())
|
2014-05-18 23:24:04 +00:00
|
|
|
logger.debug('edit of %s id keys: %s', item, keys)
|
|
|
|
if item and item.json()['mediastate'] == 'available':
|
|
|
|
if keys:
|
|
|
|
key = keys[0]
|
|
|
|
logger.debug('update mainid %s %s', key, data[key])
|
|
|
|
if key in ('isbn10', 'isbn13'):
|
|
|
|
data[key] = utils.normalize_isbn(data[key])
|
|
|
|
item.update_mainid(key, data[key])
|
|
|
|
response = item.json()
|
|
|
|
elif not item.meta.get('mainid'):
|
|
|
|
logger.debug('chustom data %s', data)
|
|
|
|
for key in ('title', 'author', 'date', 'publisher', 'edition'):
|
|
|
|
if key in data:
|
|
|
|
item.meta[key] = data[key]
|
|
|
|
item.update()
|
|
|
|
logger.debug('FIXME: custom metadata not published to changelog!!!')
|
2014-05-04 17:26:43 +00:00
|
|
|
else:
|
2014-05-17 14:26:59 +00:00
|
|
|
logger.info('can only edit available items')
|
2014-05-04 17:26:43 +00:00
|
|
|
return response
|
|
|
|
actions.register(edit, cache=False)
|
|
|
|
|
2014-05-17 11:45:57 +00:00
|
|
|
@returns_json
|
|
|
|
def remove(request):
|
2014-05-18 23:24:04 +00:00
|
|
|
'''
|
|
|
|
takes {
|
|
|
|
id
|
|
|
|
}
|
|
|
|
'''
|
2014-05-17 11:45:57 +00:00
|
|
|
data = json.loads(request.form['data']) if 'data' in request.form else {}
|
2014-05-18 23:24:04 +00:00
|
|
|
logger.debug('remove files %s', data)
|
2014-05-17 11:45:57 +00:00
|
|
|
if 'ids' in data and data['ids']:
|
|
|
|
for i in models.Item.query.filter(models.Item.id.in_(data['ids'])):
|
|
|
|
i.remove_file()
|
|
|
|
return {}
|
|
|
|
actions.register(remove, cache=False)
|
2014-05-04 17:26:43 +00:00
|
|
|
|
|
|
|
@returns_json
|
2014-05-14 18:46:31 +00:00
|
|
|
def findMetadata(request):
|
2014-05-04 17:26:43 +00:00
|
|
|
'''
|
|
|
|
takes {
|
|
|
|
title: string,
|
|
|
|
author: [string],
|
|
|
|
publisher: string,
|
|
|
|
date: string
|
|
|
|
}
|
|
|
|
returns {
|
2014-05-18 23:24:04 +00:00
|
|
|
items: [{
|
|
|
|
key: value
|
|
|
|
}]
|
2014-05-04 17:26:43 +00:00
|
|
|
}
|
2014-05-18 23:24:04 +00:00
|
|
|
key is one of the supported identifiers: isbn10, isbn13...
|
2014-05-04 17:26:43 +00:00
|
|
|
'''
|
|
|
|
response = {}
|
|
|
|
data = json.loads(request.form['data']) if 'data' in request.form else {}
|
2014-05-17 14:26:59 +00:00
|
|
|
logger.debug('findMetadata %s', data)
|
2014-05-14 18:46:31 +00:00
|
|
|
response['items'] = meta.find(**data)
|
|
|
|
return response
|
|
|
|
actions.register(findMetadata)
|
|
|
|
|
|
|
|
@returns_json
|
|
|
|
def getMetadata(request):
|
2014-05-18 23:24:04 +00:00
|
|
|
'''
|
|
|
|
takes {
|
|
|
|
key: value
|
|
|
|
}
|
|
|
|
key can be one of the supported identifiers: isbn10, isbn13, oclc, olid,...
|
|
|
|
'''
|
2014-05-14 18:46:31 +00:00
|
|
|
data = json.loads(request.form['data']) if 'data' in request.form else {}
|
2014-05-17 14:26:59 +00:00
|
|
|
logger.debug('getMetadata %s', data)
|
2014-05-14 18:46:31 +00:00
|
|
|
key, value = data.iteritems().next()
|
|
|
|
if key in ('isbn10', 'isbn13'):
|
|
|
|
value = utils.normalize_isbn(value)
|
|
|
|
response = meta.lookup(key, value)
|
2014-05-18 23:24:04 +00:00
|
|
|
if response:
|
|
|
|
response['mainid'] = key
|
2014-05-04 17:26:43 +00:00
|
|
|
return response
|
2014-05-14 18:46:31 +00:00
|
|
|
actions.register(getMetadata)
|
2014-05-04 17:26:43 +00:00
|
|
|
|
|
|
|
@returns_json
|
|
|
|
def download(request):
|
2014-05-18 23:24:04 +00:00
|
|
|
'''
|
|
|
|
takes {
|
|
|
|
id
|
|
|
|
}
|
|
|
|
'''
|
2014-05-04 17:26:43 +00:00
|
|
|
response = {}
|
|
|
|
data = json.loads(request.form['data']) if 'data' in request.form else {}
|
|
|
|
item = models.Item.get(data['id'])
|
|
|
|
if item:
|
2014-05-12 23:43:27 +00:00
|
|
|
item.queue_download()
|
2014-05-04 17:26:43 +00:00
|
|
|
item.update()
|
|
|
|
response = {'status': 'queued'}
|
|
|
|
return response
|
|
|
|
actions.register(download, cache=False)
|
|
|
|
|
|
|
|
@returns_json
|
2014-05-18 10:44:56 +00:00
|
|
|
def cancelDownloads(request):
|
2014-05-18 23:24:04 +00:00
|
|
|
'''
|
|
|
|
takes {
|
|
|
|
ids
|
|
|
|
}
|
|
|
|
'''
|
2014-05-04 17:26:43 +00:00
|
|
|
response = {}
|
|
|
|
data = json.loads(request.form['data']) if 'data' in request.form else {}
|
2014-05-18 10:44:56 +00:00
|
|
|
ids = data['ids']
|
|
|
|
if ids:
|
|
|
|
for item in models.Item.query.filter(models.Item.id.in_(ids)):
|
|
|
|
item.transferprogress = None
|
|
|
|
item.transferadded = None
|
|
|
|
p = state.user()
|
|
|
|
if p in item.users:
|
|
|
|
item.users.remove(p)
|
|
|
|
for l in item.lists.filter_by(user_id=settings.USER_ID):
|
2014-05-18 10:46:56 +00:00
|
|
|
l.items.remove(item)
|
2014-05-18 10:44:56 +00:00
|
|
|
item.update()
|
2014-05-04 17:26:43 +00:00
|
|
|
response = {'status': 'cancelled'}
|
|
|
|
return response
|
2014-05-18 10:44:56 +00:00
|
|
|
actions.register(cancelDownloads, cache=False)
|
2014-05-04 17:26:43 +00:00
|
|
|
|
|
|
|
@returns_json
|
|
|
|
def scan(request):
|
|
|
|
state.main.add_callback(state.websockets[0].put, json.dumps(['scan', {}]))
|
|
|
|
return {}
|
|
|
|
actions.register(scan, cache=False)
|
|
|
|
|
|
|
|
@returns_json
|
|
|
|
def _import(request):
|
2014-05-18 23:24:04 +00:00
|
|
|
'''
|
|
|
|
takes {
|
|
|
|
path absolute path to import
|
|
|
|
list listename (add new items to this list)
|
|
|
|
mode copy|move
|
|
|
|
}
|
|
|
|
'''
|
2014-05-16 14:30:16 +00:00
|
|
|
data = json.loads(request.form['data']) if 'data' in request.form else {}
|
2014-05-17 14:26:59 +00:00
|
|
|
logger.debug('api.import %s', data)
|
2014-05-16 14:30:16 +00:00
|
|
|
state.main.add_callback(state.websockets[0].put, json.dumps(['import', data]))
|
2014-05-04 17:26:43 +00:00
|
|
|
return {}
|
|
|
|
actions.register(_import, 'import', cache=False)
|
2014-05-18 23:24:04 +00:00
|
|
|
|
|
|
|
@returns_json
|
|
|
|
def cancelImport(request):
|
|
|
|
state.activity['cancel'] = True
|
|
|
|
return {}
|
|
|
|
actions.register(cancelImport, cache=False)
|