gallery.accords-library.com/server/szurubooru/func/posts.py

584 lines
18 KiB
Python
Raw Normal View History

2016-04-22 18:58:04 +00:00
import datetime
import sqlalchemy
2016-04-30 21:17:08 +00:00
from szurubooru import config, db, errors
from szurubooru.func import (
users, scores, comments, tags, util, mime, images, files, image_hash)
2016-04-30 21:17:08 +00:00
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
class PostAlreadyUploadedError(errors.ValidationError):
def __init__(self, other_post):
super().__init__(
'Post already uploaded (%d)' % other_post.post_id,
{
'otherPostUrl': get_post_content_url(other_post),
'otherPostId': other_post.post_id,
})
class InvalidPostIdError(errors.ValidationError):
pass
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
2016-04-30 21:17:08 +00:00
class PostLookalike(image_hash.Lookalike):
def __init__(self, score, distance, post):
super().__init__(score, distance, post.post_id)
self.post = post
2016-04-30 21:17:08 +00:00
SAFETY_MAP = {
db.Post.SAFETY_SAFE: 'safe',
db.Post.SAFETY_SKETCHY: 'sketchy',
db.Post.SAFETY_UNSAFE: 'unsafe',
}
2016-04-30 21:17:08 +00:00
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
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')
2016-04-30 21:17:08 +00:00
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)
2016-04-30 21:17:08 +00:00
def get_post_content_path(post):
2016-08-14 08:45:00 +00:00
assert post
assert post.post_id
2016-04-30 21:17:08 +00:00
return 'posts/%d.%s' % (
post.post_id, mime.get_extension(post.mime_type) or 'dat')
2016-04-30 21:17:08 +00:00
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)
2016-04-30 21:17:08 +00:00
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)
2016-04-30 21:17:08 +00:00
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,
}
def serialize_post(post, auth_user, options=None):
return util.serialize_entity(
post,
{
'id': lambda: post.post_id,
2016-08-06 19:16:39 +00:00
'version': lambda: post.version,
'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,
'tags': lambda: [
tag.names[0].name for tag in tags.sort_tags(post.tags)],
'relations': lambda: sorted(
{
post['id']:
post for post in [
serialize_micro_post(rel, auth_user)
for rel in post.relations]
}.values(),
key=lambda post: post['id']),
'user': lambda: users.serialize_micro_user(post.user, auth_user),
'score': lambda: post.score,
'ownScore': lambda: scores.get_score(post, auth_user),
'ownFavorite': lambda: len([
user for user in post.favorited_by
if user.user_id == auth_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,
'featureCount': lambda: post.feature_count,
'lastFeatureTime': lambda: post.last_feature_time,
'favoritedBy': lambda: [
users.serialize_micro_user(rel.user, auth_user)
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, auth_user)
for comment in sorted(
post.comments,
key=lambda comment: comment.creation_time)],
},
options)
def serialize_micro_post(post, auth_user):
return serialize_post(
post,
auth_user=auth_user,
options=['id', 'thumbnailUrl'])
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
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)
return db.session \
.query(db.Post) \
2016-04-22 18:58:04 +00:00
.filter(db.Post.post_id == post_id) \
.one_or_none()
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
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()
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 = []
post.type = ''
post.checksum = ''
post.mime_type = ''
db.session.add(post)
update_post_content(post, content)
new_tags = update_post_tags(post, tag_names)
return (post, new_tags)
2016-04-30 21:17:08 +00:00
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
2016-04-30 21:17:08 +00:00
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
@sqlalchemy.events.event.listens_for(db.Post, 'after_insert')
def _after_post_insert(_mapper, _connection, post):
_sync_post_content(post)
@sqlalchemy.events.event.listens_for(db.Post, 'after_update')
def _after_post_update(_mapper, _connection, post):
_sync_post_content(post)
@sqlalchemy.events.event.listens_for(db.Post, 'before_delete')
def _before_post_delete(_mapper, _connection, post):
image_hash.delete_image(post.post_id)
def _sync_post_content(post):
regenerate_thumb = False
if hasattr(post, '__content'):
content = getattr(post, '__content')
files.save(get_post_content_path(post), content)
delattr(post, '__content')
regenerate_thumb = True
if post.type in (db.Post.TYPE_IMAGE, db.Post.TYPE_ANIMATION):
image_hash.delete_image(post.post_id)
image_hash.add_image(post.post_id, content)
if hasattr(post, '__thumbnail'):
if getattr(post, '__thumbnail'):
files.save(
get_post_thumbnail_backup_path(post),
getattr(post, '__thumbnail'))
else:
files.delete(get_post_thumbnail_backup_path(post))
delattr(post, '__thumbnail')
regenerate_thumb = True
if regenerate_thumb:
generate_post_thumbnail(post)
2016-04-30 21:17:08 +00:00
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)
2016-04-30 21:17:08 +00:00
post.checksum = util.get_sha1(content)
2016-04-30 21:17:08 +00:00
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 \
and other_post.post_id \
and other_post.post_id != post.post_id:
raise PostAlreadyUploadedError(other_post)
2016-04-30 21:17:08 +00:00
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
if (post.canvas_width is not None and post.canvas_width <= 0) \
or (post.canvas_height is not None and post.canvas_height <= 0):
post.canvas_width = None
post.canvas_height = None
setattr(post, '__content', content)
2016-04-30 21:17:08 +00:00
def update_post_thumbnail(post, content=None):
2016-08-14 08:45:00 +00:00
assert post
setattr(post, '__thumbnail', content)
2016-04-30 21:17:08 +00:00
def generate_post_thumbnail(post):
2016-08-14 08:45:00 +00:00
assert post
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)
2016-04-30 21:17:08 +00:00
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
return new_tags
2016-04-30 21:17:08 +00:00
def update_post_relations(post, new_post_ids):
2016-08-14 08:45:00 +00:00
assert post
try:
new_post_ids = [int(id) for id in new_post_ids]
except ValueError:
raise InvalidPostRelationError(
'A relation must be numeric post ID.')
old_posts = post.relations
old_post_ids = [int(p.post_id) for p in old_posts]
if new_post_ids:
new_posts = db.session \
.query(db.Post) \
.filter(db.Post.post_id.in_(new_post_ids)) \
.all()
else:
new_posts = []
if len(new_posts) != len(new_post_ids):
2016-04-30 21:17:08 +00:00
raise InvalidPostRelationError('One of relations does not exist.')
if post.post_id in new_post_ids:
raise InvalidPostRelationError('Post cannot relate to itself.')
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
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.')
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']:
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(
'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(
db.PostNote(polygon=note['polygon'], text=str(note['text'])))
2016-04-30 21:17:08 +00:00
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)
def delete(post):
2016-08-14 08:45:00 +00:00
assert post
db.session.delete(post)
2016-10-21 19:48:08 +00:00
def merge_posts(source_post, target_post, replace_content):
2016-10-21 19:48:08 +00:00
assert source_post
assert target_post
if source_post.post_id == target_post.post_id:
raise InvalidPostRelationError('Cannot merge post with itself.')
def merge_tables(table, anti_dup_func, source_post_id, target_post_id):
2016-10-22 15:57:25 +00:00
alias1 = table
alias2 = sqlalchemy.orm.util.aliased(table)
update_stmt = (sqlalchemy.sql.expression.update(alias1)
.where(alias1.post_id == source_post_id))
2016-10-21 19:48:08 +00:00
if anti_dup_func is not None:
update_stmt = (update_stmt
.where(~sqlalchemy.exists()
2016-10-22 15:57:25 +00:00
.where(anti_dup_func(alias1, alias2))
.where(alias2.post_id == target_post_id)))
2016-10-21 19:48:08 +00:00
2016-10-22 15:57:25 +00:00
update_stmt = update_stmt.values(post_id=target_post_id)
2016-10-21 19:48:08 +00:00
db.session.execute(update_stmt)
def merge_tags(source_post_id, target_post_id):
merge_tables(
db.PostTag,
lambda alias1, alias2: alias1.tag_id == alias2.tag_id,
source_post_id,
target_post_id)
def merge_scores(source_post_id, target_post_id):
merge_tables(
db.PostScore,
lambda alias1, alias2: alias1.user_id == alias2.user_id,
source_post_id,
target_post_id)
def merge_favorites(source_post_id, target_post_id):
merge_tables(
db.PostFavorite,
lambda alias1, alias2: alias1.user_id == alias2.user_id,
source_post_id,
target_post_id)
def merge_comments(source_post_id, target_post_id):
merge_tables(db.Comment, None, source_post_id, target_post_id)
def merge_relations(source_post_id, target_post_id):
2016-10-22 15:57:25 +00:00
alias1 = db.PostRelation
alias2 = sqlalchemy.orm.util.aliased(db.PostRelation)
update_stmt = (sqlalchemy.sql.expression.update(alias1)
.where(alias1.parent_id == source_post_id)
.where(alias1.child_id != target_post_id)
2016-10-21 19:48:08 +00:00
.where(~sqlalchemy.exists()
2016-10-22 15:57:25 +00:00
.where(alias2.child_id == alias1.child_id)
.where(alias2.parent_id == target_post_id))
2016-10-21 19:48:08 +00:00
.values(parent_id=target_post_id))
db.session.execute(update_stmt)
2016-10-22 15:57:25 +00:00
update_stmt = (sqlalchemy.sql.expression.update(alias1)
.where(alias1.child_id == source_post_id)
.where(alias1.parent_id != target_post_id)
2016-10-21 19:48:08 +00:00
.where(~sqlalchemy.exists()
2016-10-22 15:57:25 +00:00
.where(alias2.parent_id == alias1.parent_id)
.where(alias2.child_id == target_post_id))
2016-10-21 19:48:08 +00:00
.values(child_id=target_post_id))
db.session.execute(update_stmt)
merge_tags(source_post.post_id, target_post.post_id)
merge_comments(source_post.post_id, target_post.post_id)
merge_scores(source_post.post_id, target_post.post_id)
merge_favorites(source_post.post_id, target_post.post_id)
merge_relations(source_post.post_id, target_post.post_id)
delete(source_post)
db.session.flush()
if replace_content:
content = files.get(get_post_content_path(source_post))
update_post_content(target_post, content)
def search_by_image_exact(image_content):
checksum = util.get_sha1(image_content)
return db.session \
.query(db.Post) \
.filter(db.Post.checksum == checksum) \
.one_or_none()
def search_by_image(image_content):
for result in image_hash.search_by_image(image_content):
yield PostLookalike(
score=result.score,
distance=result.distance,
post=get_post_by_id(result.path))
def populate_reverse_search():
excluded_post_ids = image_hash.get_all_paths()
post_ids_to_hash = (db.session
.query(db.Post.post_id)
.filter(
(db.Post.type == db.Post.TYPE_IMAGE) |
(db.Post.type == db.Post.TYPE_ANIMATION))
.filter(~db.Post.post_id.in_(excluded_post_ids))
.order_by(db.Post.post_id.asc())
.all())
for post_ids_chunk in util.chunks(post_ids_to_hash, 100):
posts_chunk = (db.session
.query(db.Post)
.filter(db.Post.post_id.in_(post_ids_chunk))
.all())
for post in posts_chunk:
content_path = get_post_content_path(post)
if files.has(content_path):
image_hash.add_image(post.post_id, files.get(content_path))