Nessuna descrizione

views.py 11KB

    # -*- coding: utf-8 -*- import os import shortuuid from curtail_uuid import CurtailUUID from django.conf import settings from django.core.files.storage import default_storage from django.db import transaction from django.shortcuts import render from django_q.tasks import async from logit import logit from rest_framework import viewsets from TimeConvert import TimeConvert as tc from account.models import LensmanInfo, UserInfo from group.models import GroupInfo, GroupPhotoInfo, GroupUserInfo from photo.models import PhotosInfo from photo.serializers import PhotosInfoSerializer from utils.error.errno_utils import LensmanStatusCode, PhotoStatusCode from utils.error.response_utils import response from utils.ip_utils import ip_addr from utils.redis.rgroup import get_group_info, set_group_info, set_group_users_info from utils.redis.rkeys import (GROUP_LAST_PHOTO_PK, GROUP_USERS_DELETED_SET, GROUP_USERS_PASSED_SET, GROUP_USERS_QUIT_SET, GROUP_USERS_REFUSED_SET, UUID_LIST) from utils.redis.ruuid import generate_uuids, update_uuids from utils.thumbnail_utils import make_thumbnail from utils.watermark_utils import watermark_wrap r = settings.REDIS_CACHE @logit def uuid_init(request): """ 生成唯一标识 :param request: :return: """ num = int(request.GET.get('num', 1000)) # 生成 UUID generate_uuids(num) return response(200, 'UUID Refresh Success', u'UUID 更新成功') # curl -X POST -F user_id=xxxxxxx -F num=100 http://api.pai.ai/uuid @logit @transaction.atomic def uuid(request): """ 获取唯一标识 :param request: :return: """ lensman_id = request.POST.get('user_id', '') num = int(request.POST.get('num', 100)) # 从 Redis 中 Pop 中指定数量的 UUID uuids, succeed, left = r.multi_pop(UUID_LIST, num) # 异步更新 UUID 数据库中状态 if uuids: async(update_uuids, lensman_id, uuids) # 当可用 UUID 数量少于 500 时, 异步创建 if left < 500: async(generate_uuids) return response(200, 'Get UUID Success', u'获取唯一标识成功', uuids) # [How to do a PUT request with curl?](http://stackoverflow.com/questions/13782198/how-to-do-a-put-request-with-curl) # Unfortunately, the -T is no substitute for -X PUT if you want to specify parameters with -d or -F. # -T sends the content of a file via PUT. To achieve the GET after a redirect, add the parameter --location # # -F, --form <name=content> # (HTTP) This lets curl emulate a filled-in form in which a user has pressed the submit button. This causes curl to POST data # using the Content-Type multipart/form-data according to RFC 2388. This enables uploading of binary files etc. To force the # 'content' part to be a file, prefix the file name with an @ sign. To just get the content part from a file, prefix the file # name with the symbol <. The difference between @ and < is then that @ makes a file get attached in the post as a file upload, # while the < makes a text field and just get the contents for that text field from a file. # # curl -X POST -F user_id=xxxxxxx -F session_id=xxxxxxx -F photo_id=xxxxxxx -F photo=@xxxxxxx.jpg http://api.pai.ai/photos/upload @logit def upload_photo(request): """ 上传图片 :param request: :return: """ lensman_id = request.POST.get('user_id', '') session_id = request.POST.get('session_id', '') photo_id = request.POST.get('photo_id', '') photo = request.FILES.get('photo', '') if not (lensman_id and session_id and photo): return response(PhotoStatusCode.PARAMS_ERROR) try: LensmanInfo.objects.get(lensman_id=lensman_id) except LensmanInfo.DoesNotExist: return response(LensmanStatusCode.LENSMAN_NOT_FOUND) # photo_id = curtailUUID(PhotosInfo, 'photo_id') _, extension = os.path.splitext(photo.name) extension = extension or 'jpeg' m_photo_path = 'photo/{uuid}{extension}'.format(uuid=shortuuid.uuid(), extension=extension) if default_storage.exists(m_photo_path): default_storage.delete(m_photo_path) default_storage.save(m_photo_path, photo) p_photo_path = 'photo/{uuid}{extension}'.format(uuid=shortuuid.uuid(), extension=extension) watermark_wrap( os.path.join(settings.MEDIA_ROOT, m_photo_path).replace('\\', '/'), settings.WATERMARK_LOGO, os.path.join(settings.MEDIA_ROOT, p_photo_path).replace('\\', '/') ) photo, created = PhotosInfo.objects.get_or_create( lensman_id=lensman_id, session_id=session_id, photo_id=photo_id, p_photo_path=p_photo_path, m_photo_path=m_photo_path, ) return response(200, 'Photo Upload Success', u'照片上传成功', photo.data) @logit def session_join_api(request): """ Session 详情 :param request: :return: """ user_id = request.POST.get('user_id', '') session_id = request.POST.get('session_id', '') nickname = request.POST.get('nickname', '') current_id = -1 # 判断 user_id 是否存在,如果不存在,则直接分配帐户 user, user_created = UserInfo.objects.get_or_create(user_id=user_id, defaults={ 'user_id': CurtailUUID.uuid(UserInfo, 'user_id'), 'user_status': UserInfo.ASSIGN, 'assign_ip': ip_addr(request), 'assign_at': tc.utc_datetime(), }) user_id = user.user_id # 判断通过 session_id 创建的群组是否存在,如果不存在,则直接创建 group, group_created = GroupInfo.objects.get_or_create(session_id=session_id, group_from=GroupInfo.SESSION_GROUP, defaults={ 'group_id': CurtailUUID.uuid(GroupInfo, 'group_id'), 'admin_id': user_id, 'group_name': user.final_nickname, 'group_default_avatar': 0, }) group_id = group.group_id # Redis 群组数据缓存 group_info = set_group_info(group) if group_created else get_group_info(group_id) # 判断 group_id/user_id 的群组用户是否存在,如果不存在,则直接创建 group_user, group_user_created = GroupUserInfo.objects.get_or_create(group_id=group_id, user_id=user_id, defaults={ # 'current_id': int(r.get(GROUP_LAST_PHOTO_PK % group_id) or -1), 'current_id': current_id, # 通过扫描 session_id 二维码进群的用户,默认可以查看该群组所有照片 'nickname': nickname or user.final_nickname, 'avatar': user.avatar, 'admin': group_created, 'user_status': GroupUserInfo.PASSED, 'passed_at': tc.utc_datetime(), }) if not group_user_created: group_user.current_id = current_id group_user.user_status = GroupUserInfo.PASSED group_user.save() # Redis 群组用户数据缓存 group_users = set_group_users_info(group) # Redis 群组通过集合缓存 r.srem(GROUP_USERS_REFUSED_SET % group_id, user_id) r.srem(GROUP_USERS_DELETED_SET % group_id, user_id) r.srem(GROUP_USERS_QUIT_SET % group_id, user_id) r.sadd(GROUP_USERS_PASSED_SET % group_id, user_id) # 获取 Session 照片 photos = PhotosInfo.objects.filter(session_id=session_id) # if group_created: for photo in photos: photo_path = photo.p_photo_path photo_thumbnail_path = photo_path.replace('.', '_thumbnail.') photo_thumbnail2_path = photo_path.replace('.', '_thumbnail2.') # 群组照片缩略图生成 # 双列: 540, 40-50K photo_w, photo_h, photo_thumbnail_w, photo_thumbnail_h = make_thumbnail( os.path.join(settings.MEDIA_ROOT, photo_path).replace('\\', '/'), os.path.join(settings.MEDIA_ROOT, photo_thumbnail_path).replace('\\', '/'), settings.THUMBNAIL_MAX_WIDTH ) # 单列: 1080, xx-100K photo_w, photo_h, photo_thumbnail2_w, photo_thumbnail2_h = make_thumbnail( os.path.join(settings.MEDIA_ROOT, photo_path).replace('\\', '/'), os.path.join(settings.MEDIA_ROOT, photo_thumbnail2_path).replace('\\', '/'), settings.THUMBNAIL_MAX_WIDTH2 ) # 群组照片记录创建 group_photo = GroupPhotoInfo.objects.create( group_id=group_id, user_id=user_id, nickname=user.final_nickname, avatar=user.avatar, photo_path=photo_path, photo_w=photo_w, photo_h=photo_h, photo_thumbnail_path=photo_thumbnail_path, photo_thumbnail_w=photo_thumbnail_w, photo_thumbnail_h=photo_thumbnail_h, photo_thumbnail2_path=photo_thumbnail2_path, photo_thumbnail2_w=photo_thumbnail2_w, photo_thumbnail2_h=photo_thumbnail2_h, photo_from=GroupPhotoInfo.SESSION_GROUP, session_id=photo.session_id, lensman_id=photo.lensman_id, lensman_photo_id=photo.photo_id, ) # 设置群组最后一张照片PK r.set(GROUP_LAST_PHOTO_PK % group_id, group_photo.pk) # 获取从 current_id 到 now 的群组照片列表 group_photos = GroupPhotoInfo.objects.filter( group_id=group_id, status=True, pk__gt=group_user.current_id, ).order_by( '-pk' ) latest_photo = group_photos.first() return response(200, 'Apply Success', u'申请成功', { 'current_id': latest_photo and latest_photo.pk or current_id, 'photos': [photo.photo_info(user_id) for photo in group_photos], 'group_id': group_id, 'group': group_info, 'user_id': user_id, 'users': group_users, }) @logit def session_detail(request, session_id): photos = PhotosInfo.objects.filter(session_id=session_id) return render(request, 'photo/session_detail.html', {'photos': photos}) @logit def photo_standard(request, photo_id): photo = PhotosInfo.objects.get(photo_id=photo_id) return render(request, 'photo/photo_detail.html', {'photo_url': photo.p_photo_url}) @logit def photo_standard_api(request, photo_id): photo = PhotosInfo.objects.get(photo_id=photo_id) return response(200, 'Get Photo Detail Success', u'获取照片详情成功', { 'photo': photo.detail, }) @logit def photo_medium(request, photo_id): photo = PhotosInfo.objects.get(photo_id=photo_id) return render(request, 'photo/photo_detail.html', {'photo_url': photo.m_photo_url}) @logit def photo_large(request, photo_id): photo = PhotosInfo.objects.get(photo_id=photo_id) return render(request, 'photo/photo_detail.html', {'photo_url': photo.l_photo_url}) @logit def photo_raw(request, photo_id): photo = PhotosInfo.objects.get(photo_id=photo_id) return render(request, 'photo/photo_detail.html', {'photo_url': photo.r_photo_url}) class PhotoInfoViewSet(viewsets.ModelViewSet): queryset = PhotosInfo.objects.all().order_by('-pk') serializer_class = PhotosInfoSerializer