2016-04-22 18:58:04 +00:00
|
|
|
import datetime
|
2016-04-21 17:48:47 +00:00
|
|
|
import sqlalchemy
|
2016-04-30 21:17:08 +00:00
|
|
|
from szurubooru import config, db, errors
|
|
|
|
from szurubooru.func import (
|
2016-08-02 10:20:25 +00:00
|
|
|
users, snapshots, scores, comments, tags, util, mime, images, files)
|
2016-04-30 21:17:08 +00:00
|
|
|
|
|
|
|
EMPTY_PIXEL = \
|
|
|
|
b'\x47\x49\x46\x38\x39\x61\x01\x00\x01\x00\x80\x01\x00\x00\x00\x00' \
|
|
|
|
b'\xff\xff\xff\x21\xf9\x04\x01\x00\x00\x01\x00\x2c\x00\x00\x00\x00' \
|
|
|
|
b'\x01\x00\x01\x00\x00\x02\x02\x4c\x01\x00\x3b'
|
2016-04-22 18:58:04 +00:00
|
|
|
|
|
|
|
class PostNotFoundError(errors.NotFoundError): pass
|
|
|
|
class PostAlreadyFeaturedError(errors.ValidationError): pass
|
2016-04-30 21:17:08 +00:00
|
|
|
class PostAlreadyUploadedError(errors.ValidationError): pass
|
2016-06-06 20:25:50 +00:00
|
|
|
class InvalidPostIdError(errors.ValidationError): pass
|
2016-04-30 21:17:08 +00:00
|
|
|
class InvalidPostSafetyError(errors.ValidationError): pass
|
|
|
|
class InvalidPostSourceError(errors.ValidationError): pass
|
|
|
|
class InvalidPostContentError(errors.ValidationError): pass
|
|
|
|
class InvalidPostRelationError(errors.ValidationError): pass
|
|
|
|
class InvalidPostNoteError(errors.ValidationError): pass
|
|
|
|
class InvalidPostFlagError(errors.ValidationError): pass
|
|
|
|
|
|
|
|
SAFETY_MAP = {
|
|
|
|
db.Post.SAFETY_SAFE: 'safe',
|
|
|
|
db.Post.SAFETY_SKETCHY: 'sketchy',
|
|
|
|
db.Post.SAFETY_UNSAFE: 'unsafe',
|
|
|
|
}
|
|
|
|
TYPE_MAP = {
|
|
|
|
db.Post.TYPE_IMAGE: 'image',
|
|
|
|
db.Post.TYPE_ANIMATION: 'animation',
|
|
|
|
db.Post.TYPE_VIDEO: 'video',
|
|
|
|
db.Post.TYPE_FLASH: 'flash',
|
|
|
|
}
|
2016-05-10 09:57:05 +00:00
|
|
|
FLAG_MAP = {
|
|
|
|
db.Post.FLAG_LOOP: 'loop',
|
|
|
|
}
|
2016-04-30 21:17:08 +00:00
|
|
|
|
|
|
|
def get_post_content_url(post):
|
2016-08-14 08:45:00 +00:00
|
|
|
assert post
|
2016-04-30 21:17:08 +00:00
|
|
|
return '%s/posts/%d.%s' % (
|
|
|
|
config.config['data_url'].rstrip('/'),
|
|
|
|
post.post_id,
|
|
|
|
mime.get_extension(post.mime_type) or 'dat')
|
|
|
|
|
|
|
|
def get_post_thumbnail_url(post):
|
2016-08-14 08:45:00 +00:00
|
|
|
assert post
|
2016-04-30 21:17:08 +00:00
|
|
|
return '%s/generated-thumbnails/%d.jpg' % (
|
|
|
|
config.config['data_url'].rstrip('/'),
|
|
|
|
post.post_id)
|
|
|
|
|
|
|
|
def get_post_content_path(post):
|
2016-08-14 08:45:00 +00:00
|
|
|
assert post
|
2016-04-30 21:17:08 +00:00
|
|
|
return 'posts/%d.%s' % (
|
|
|
|
post.post_id, mime.get_extension(post.mime_type) or 'dat')
|
|
|
|
|
|
|
|
def get_post_thumbnail_path(post):
|
2016-08-14 08:45:00 +00:00
|
|
|
assert post
|
2016-04-30 21:17:08 +00:00
|
|
|
return 'generated-thumbnails/%d.jpg' % (post.post_id)
|
|
|
|
|
|
|
|
def get_post_thumbnail_backup_path(post):
|
2016-08-14 08:45:00 +00:00
|
|
|
assert post
|
2016-04-30 21:17:08 +00:00
|
|
|
return 'posts/custom-thumbnails/%d.dat' % (post.post_id)
|
|
|
|
|
|
|
|
def serialize_note(note):
|
2016-08-14 08:45:00 +00:00
|
|
|
assert note
|
2016-04-30 21:17:08 +00:00
|
|
|
return {
|
2016-05-28 09:22:25 +00:00
|
|
|
'polygon': note.polygon,
|
2016-04-30 21:17:08 +00:00
|
|
|
'text': note.text,
|
|
|
|
}
|
2016-04-21 17:48:47 +00:00
|
|
|
|
2016-05-30 21:08:22 +00:00
|
|
|
def serialize_post(post, authenticated_user, options=None):
|
2016-05-30 20:07:44 +00:00
|
|
|
return util.serialize_entity(
|
|
|
|
post,
|
|
|
|
{
|
|
|
|
'id': lambda: post.post_id,
|
2016-08-06 19:16:39 +00:00
|
|
|
'version': lambda: post.version,
|
2016-05-30 20:07:44 +00:00
|
|
|
'creationTime': lambda: post.creation_time,
|
|
|
|
'lastEditTime': lambda: post.last_edit_time,
|
|
|
|
'safety': lambda: SAFETY_MAP[post.safety],
|
|
|
|
'source': lambda: post.source,
|
|
|
|
'type': lambda: TYPE_MAP[post.type],
|
|
|
|
'mimeType': lambda: post.mime_type,
|
|
|
|
'checksum': lambda: post.checksum,
|
|
|
|
'fileSize': lambda: post.file_size,
|
|
|
|
'canvasWidth': lambda: post.canvas_width,
|
|
|
|
'canvasHeight': lambda: post.canvas_height,
|
|
|
|
'contentUrl': lambda: get_post_content_url(post),
|
|
|
|
'thumbnailUrl': lambda: get_post_thumbnail_url(post),
|
|
|
|
'flags': lambda: post.flags,
|
2016-06-06 05:46:19 +00:00
|
|
|
'tags': lambda: [
|
2016-07-30 10:31:04 +00:00
|
|
|
tag.names[0].name for tag in tags.sort_tags(post.tags)],
|
2016-07-03 11:15:41 +00:00
|
|
|
'relations': lambda: sorted(
|
|
|
|
{
|
|
|
|
post['id']:
|
|
|
|
post for post in [
|
2016-07-17 18:58:42 +00:00
|
|
|
serialize_micro_post(rel) for rel in post.relations
|
2016-07-03 11:15:41 +00:00
|
|
|
]
|
|
|
|
}.values(),
|
|
|
|
key=lambda post: post['id']),
|
2016-06-03 18:10:25 +00:00
|
|
|
'user': lambda: users.serialize_micro_user(post.user),
|
2016-05-30 20:07:44 +00:00
|
|
|
'score': lambda: post.score,
|
|
|
|
'ownScore': lambda: scores.get_score(post, authenticated_user),
|
2016-06-06 18:57:12 +00:00
|
|
|
'ownFavorite': lambda: len(
|
|
|
|
[user for user in post.favorited_by \
|
|
|
|
if user.user_id == authenticated_user.user_id]) > 0,
|
2016-05-30 21:23:22 +00:00
|
|
|
'tagCount': lambda: post.tag_count,
|
|
|
|
'favoriteCount': lambda: post.favorite_count,
|
|
|
|
'commentCount': lambda: post.comment_count,
|
|
|
|
'noteCount': lambda: post.note_count,
|
2016-07-03 12:27:36 +00:00
|
|
|
'relationCount': lambda: post.relation_count,
|
2016-05-30 20:07:44 +00:00
|
|
|
'featureCount': lambda: post.feature_count,
|
|
|
|
'lastFeatureTime': lambda: post.last_feature_time,
|
|
|
|
'favoritedBy': lambda: [
|
2016-06-03 18:10:25 +00:00
|
|
|
users.serialize_micro_user(rel.user) \
|
2016-05-30 20:07:44 +00:00
|
|
|
for rel in post.favorited_by],
|
|
|
|
'hasCustomThumbnail':
|
|
|
|
lambda: files.has(get_post_thumbnail_backup_path(post)),
|
|
|
|
'notes': lambda: sorted(
|
|
|
|
[serialize_note(note) for note in post.notes],
|
|
|
|
key=lambda x: x['polygon']),
|
|
|
|
'comments': lambda: [
|
|
|
|
comments.serialize_comment(comment, authenticated_user) \
|
2016-06-11 12:18:33 +00:00
|
|
|
for comment in sorted(
|
|
|
|
post.comments,
|
|
|
|
key=lambda comment: comment.creation_time)],
|
2016-05-30 20:07:44 +00:00
|
|
|
'snapshots': lambda: snapshots.get_serialized_history(post),
|
2016-05-30 21:08:22 +00:00
|
|
|
},
|
|
|
|
options)
|
2016-04-24 07:47:58 +00:00
|
|
|
|
2016-07-03 11:15:41 +00:00
|
|
|
def serialize_micro_post(post):
|
|
|
|
return serialize_post(
|
|
|
|
post,
|
|
|
|
authenticated_user=None,
|
|
|
|
options=['id', 'thumbnailUrl'])
|
|
|
|
|
2016-04-21 17:48:47 +00:00
|
|
|
def get_post_count():
|
|
|
|
return db.session.query(sqlalchemy.func.count(db.Post.post_id)).one()[0]
|
2016-04-22 18:58:04 +00:00
|
|
|
|
2016-04-24 12:24:41 +00:00
|
|
|
def try_get_post_by_id(post_id):
|
2016-06-06 20:25:50 +00:00
|
|
|
try:
|
|
|
|
post_id = int(post_id)
|
|
|
|
except ValueError:
|
|
|
|
raise InvalidPostIdError('Invalid post ID: %r.' % post_id)
|
2016-04-24 07:47:58 +00:00
|
|
|
return db.session \
|
|
|
|
.query(db.Post) \
|
2016-04-22 18:58:04 +00:00
|
|
|
.filter(db.Post.post_id == post_id) \
|
|
|
|
.one_or_none()
|
|
|
|
|
2016-04-24 12:24:41 +00:00
|
|
|
def get_post_by_id(post_id):
|
|
|
|
post = try_get_post_by_id(post_id)
|
|
|
|
if not post:
|
|
|
|
raise PostNotFoundError('Post %r not found.' % post_id)
|
|
|
|
return post
|
|
|
|
|
2016-05-29 10:38:47 +00:00
|
|
|
def try_get_current_post_feature():
|
|
|
|
return db.session \
|
2016-04-22 18:58:04 +00:00
|
|
|
.query(db.PostFeature) \
|
|
|
|
.order_by(db.PostFeature.time.desc()) \
|
|
|
|
.first()
|
2016-05-29 10:38:47 +00:00
|
|
|
|
|
|
|
def try_get_featured_post():
|
|
|
|
post_feature = try_get_current_post_feature()
|
2016-04-22 18:58:04 +00:00
|
|
|
return post_feature.post if post_feature else None
|
|
|
|
|
2016-04-30 21:17:08 +00:00
|
|
|
def create_post(content, tag_names, user):
|
|
|
|
post = db.Post()
|
|
|
|
post.safety = db.Post.SAFETY_SAFE
|
|
|
|
post.user = user
|
2016-07-03 12:46:15 +00:00
|
|
|
post.creation_time = datetime.datetime.utcnow()
|
2016-04-30 21:17:08 +00:00
|
|
|
post.flags = []
|
|
|
|
|
|
|
|
# we'll need post ID
|
|
|
|
post.type = ''
|
|
|
|
post.checksum = ''
|
|
|
|
post.mime_type = ''
|
|
|
|
db.session.add(post)
|
|
|
|
db.session.flush()
|
|
|
|
|
|
|
|
update_post_content(post, content)
|
2016-08-02 10:44:38 +00:00
|
|
|
new_tags = update_post_tags(post, tag_names)
|
|
|
|
return (post, new_tags)
|
2016-04-30 21:17:08 +00:00
|
|
|
|
|
|
|
def update_post_safety(post, safety):
|
2016-08-14 08:45:00 +00:00
|
|
|
assert post
|
2016-04-30 21:17:08 +00:00
|
|
|
safety = util.flip(SAFETY_MAP).get(safety, None)
|
|
|
|
if not safety:
|
|
|
|
raise InvalidPostSafetyError(
|
2016-05-10 09:57:05 +00:00
|
|
|
'Safety can be either of %r.' % list(SAFETY_MAP.values()))
|
2016-04-30 21:17:08 +00:00
|
|
|
post.safety = safety
|
|
|
|
|
|
|
|
def update_post_source(post, source):
|
2016-08-14 08:45:00 +00:00
|
|
|
assert post
|
2016-04-30 21:17:08 +00:00
|
|
|
if util.value_exceeds_column_size(source, db.Post.source):
|
|
|
|
raise InvalidPostSourceError('Source is too long.')
|
|
|
|
post.source = source
|
|
|
|
|
|
|
|
def update_post_content(post, content):
|
2016-08-14 08:45:00 +00:00
|
|
|
assert post
|
2016-04-30 21:17:08 +00:00
|
|
|
if not content:
|
|
|
|
raise InvalidPostContentError('Post content missing.')
|
|
|
|
post.mime_type = mime.get_mime_type(content)
|
|
|
|
if mime.is_flash(post.mime_type):
|
|
|
|
post.type = db.Post.TYPE_FLASH
|
|
|
|
elif mime.is_image(post.mime_type):
|
|
|
|
if mime.is_animated_gif(content):
|
|
|
|
post.type = db.Post.TYPE_ANIMATION
|
|
|
|
else:
|
|
|
|
post.type = db.Post.TYPE_IMAGE
|
|
|
|
elif mime.is_video(post.mime_type):
|
|
|
|
post.type = db.Post.TYPE_VIDEO
|
|
|
|
else:
|
|
|
|
raise InvalidPostContentError('Unhandled file type: %r' % post.mime_type)
|
|
|
|
|
|
|
|
post.checksum = util.get_md5(content)
|
|
|
|
other_post = db.session \
|
|
|
|
.query(db.Post) \
|
|
|
|
.filter(db.Post.checksum == post.checksum) \
|
|
|
|
.filter(db.Post.post_id != post.post_id) \
|
|
|
|
.one_or_none()
|
|
|
|
if other_post:
|
|
|
|
raise PostAlreadyUploadedError(
|
|
|
|
'Post already uploaded (%d)' % other_post.post_id)
|
|
|
|
|
|
|
|
post.file_size = len(content)
|
|
|
|
try:
|
|
|
|
image = images.Image(content)
|
|
|
|
post.canvas_width = image.width
|
|
|
|
post.canvas_height = image.height
|
|
|
|
except errors.ProcessingError:
|
|
|
|
post.canvas_width = None
|
|
|
|
post.canvas_height = None
|
2016-07-30 10:37:39 +00:00
|
|
|
if post.canvas_width <= 0 or post.canvas_height <= 0:
|
|
|
|
post.canvas_width = None
|
|
|
|
post.canvas_height = None
|
2016-04-30 21:17:08 +00:00
|
|
|
files.save(get_post_content_path(post), content)
|
2016-08-02 10:20:25 +00:00
|
|
|
update_post_thumbnail(post, content=None, do_delete=False)
|
2016-04-30 21:17:08 +00:00
|
|
|
|
2016-08-02 10:20:25 +00:00
|
|
|
def update_post_thumbnail(post, content=None, do_delete=True):
|
2016-08-14 08:45:00 +00:00
|
|
|
assert post
|
2016-07-31 21:53:23 +00:00
|
|
|
if not content:
|
2016-04-30 21:17:08 +00:00
|
|
|
content = files.get(get_post_content_path(post))
|
2016-08-02 10:20:25 +00:00
|
|
|
if do_delete:
|
2016-04-30 21:17:08 +00:00
|
|
|
files.delete(get_post_thumbnail_backup_path(post))
|
|
|
|
else:
|
|
|
|
files.save(get_post_thumbnail_backup_path(post), content)
|
2016-05-08 21:26:46 +00:00
|
|
|
generate_post_thumbnail(post)
|
2016-04-30 21:17:08 +00:00
|
|
|
|
2016-05-08 21:26:46 +00:00
|
|
|
def generate_post_thumbnail(post):
|
2016-08-14 08:45:00 +00:00
|
|
|
assert post
|
2016-05-08 21:26:46 +00:00
|
|
|
if files.has(get_post_thumbnail_backup_path(post)):
|
|
|
|
content = files.get(get_post_thumbnail_backup_path(post))
|
|
|
|
else:
|
|
|
|
content = files.get(get_post_content_path(post))
|
2016-04-30 21:17:08 +00:00
|
|
|
try:
|
|
|
|
image = images.Image(content)
|
|
|
|
image.resize_fill(
|
|
|
|
int(config.config['thumbnails']['post_width']),
|
|
|
|
int(config.config['thumbnails']['post_height']))
|
|
|
|
files.save(get_post_thumbnail_path(post), image.to_jpeg())
|
|
|
|
except errors.ProcessingError:
|
|
|
|
files.save(get_post_thumbnail_path(post), EMPTY_PIXEL)
|
|
|
|
|
|
|
|
def update_post_tags(post, tag_names):
|
2016-08-14 08:45:00 +00:00
|
|
|
assert post
|
2016-04-30 21:17:08 +00:00
|
|
|
existing_tags, new_tags = tags.get_or_create_tags_by_names(tag_names)
|
|
|
|
post.tags = existing_tags + new_tags
|
2016-08-02 10:44:38 +00:00
|
|
|
return new_tags
|
2016-04-30 21:17:08 +00:00
|
|
|
|
2016-07-17 18:58:42 +00:00
|
|
|
def update_post_relations(post, new_post_ids):
|
2016-08-14 08:45:00 +00:00
|
|
|
assert post
|
2016-07-17 18:58:42 +00:00
|
|
|
old_posts = post.relations
|
|
|
|
old_post_ids = [p.post_id for p in old_posts]
|
|
|
|
new_posts = db.session \
|
2016-04-30 21:17:08 +00:00
|
|
|
.query(db.Post) \
|
2016-07-17 18:58:42 +00:00
|
|
|
.filter(db.Post.post_id.in_(new_post_ids)) \
|
2016-04-30 21:17:08 +00:00
|
|
|
.all()
|
2016-07-17 18:58:42 +00:00
|
|
|
if len(new_posts) != len(new_post_ids):
|
2016-04-30 21:17:08 +00:00
|
|
|
raise InvalidPostRelationError('One of relations does not exist.')
|
2016-07-17 18:58:42 +00:00
|
|
|
|
|
|
|
relations_to_del = [p for p in old_posts if p.post_id not in new_post_ids]
|
|
|
|
relations_to_add = [p for p in new_posts if p.post_id not in old_post_ids]
|
|
|
|
for relation in relations_to_del:
|
|
|
|
post.relations.remove(relation)
|
|
|
|
relation.relations.remove(post)
|
|
|
|
for relation in relations_to_add:
|
|
|
|
post.relations.append(relation)
|
|
|
|
relation.relations.append(post)
|
2016-04-30 21:17:08 +00:00
|
|
|
|
|
|
|
def update_post_notes(post, notes):
|
2016-08-14 08:45:00 +00:00
|
|
|
assert post
|
2016-04-30 21:17:08 +00:00
|
|
|
post.notes = []
|
|
|
|
for note in notes:
|
|
|
|
for field in ('polygon', 'text'):
|
|
|
|
if field not in note:
|
|
|
|
raise InvalidPostNoteError('Note is missing %r field.' % field)
|
|
|
|
if not note['text']:
|
|
|
|
raise InvalidPostNoteError('A note\'s text cannot be empty.')
|
2016-08-13 21:39:03 +00:00
|
|
|
if not isinstance(note['polygon'], (list, tuple)):
|
|
|
|
raise InvalidPostNoteError(
|
|
|
|
'A note\'s polygon must be a list of points.')
|
2016-04-30 21:17:08 +00:00
|
|
|
if len(note['polygon']) < 3:
|
|
|
|
raise InvalidPostNoteError(
|
|
|
|
'A note\'s polygon must have at least 3 points.')
|
|
|
|
for point in note['polygon']:
|
2016-08-13 21:39:03 +00:00
|
|
|
if not isinstance(point, (list, tuple)):
|
|
|
|
raise InvalidPostNoteError(
|
|
|
|
'A note\'s polygon point must be a list of length 2.')
|
2016-04-30 21:17:08 +00:00
|
|
|
if len(point) != 2:
|
|
|
|
raise InvalidPostNoteError(
|
|
|
|
'A point in note\'s polygon must have two coordinates.')
|
|
|
|
try:
|
|
|
|
pos_x = float(point[0])
|
|
|
|
pos_y = float(point[1])
|
|
|
|
if not 0 <= pos_x <= 1 or not 0 <= pos_y <= 1:
|
|
|
|
raise InvalidPostNoteError(
|
2016-08-04 20:53:45 +00:00
|
|
|
'All points must fit in the image (0..1 range).')
|
2016-04-30 21:17:08 +00:00
|
|
|
except ValueError:
|
|
|
|
raise InvalidPostNoteError(
|
|
|
|
'A point in note\'s polygon must be numeric.')
|
|
|
|
if util.value_exceeds_column_size(note['text'], db.PostNote.text):
|
|
|
|
raise InvalidPostNoteError('Note text is too long.')
|
|
|
|
post.notes.append(
|
2016-08-13 21:39:03 +00:00
|
|
|
db.PostNote(polygon=note['polygon'], text=str(note['text'])))
|
2016-04-30 21:17:08 +00:00
|
|
|
|
|
|
|
def update_post_flags(post, flags):
|
2016-08-14 08:45:00 +00:00
|
|
|
assert post
|
2016-05-10 09:57:05 +00:00
|
|
|
target_flags = []
|
2016-04-30 21:17:08 +00:00
|
|
|
for flag in flags:
|
2016-05-10 09:57:05 +00:00
|
|
|
flag = util.flip(FLAG_MAP).get(flag, None)
|
|
|
|
if not flag:
|
2016-04-30 21:17:08 +00:00
|
|
|
raise InvalidPostFlagError(
|
2016-05-10 09:57:05 +00:00
|
|
|
'Flag must be one of %r.' % list(FLAG_MAP.values()))
|
|
|
|
target_flags.append(flag)
|
|
|
|
post.flags = target_flags
|
2016-04-30 21:17:08 +00:00
|
|
|
|
2016-04-22 18:58:04 +00:00
|
|
|
def feature_post(post, user):
|
2016-08-14 08:45:00 +00:00
|
|
|
assert post
|
2016-04-22 18:58:04 +00:00
|
|
|
post_feature = db.PostFeature()
|
2016-07-03 12:46:15 +00:00
|
|
|
post_feature.time = datetime.datetime.utcnow()
|
2016-04-22 18:58:04 +00:00
|
|
|
post_feature.post = post
|
|
|
|
post_feature.user = user
|
|
|
|
db.session.add(post_feature)
|
2016-08-02 09:44:00 +00:00
|
|
|
|
|
|
|
def delete(post):
|
2016-08-14 08:45:00 +00:00
|
|
|
assert post
|
2016-08-02 09:44:00 +00:00
|
|
|
db.session.delete(post)
|