Files
rogaining_srv/rog/views_apis/api_photos.py

208 lines
7.9 KiB
Python
Raw Normal View History

2025-08-20 19:15:19 +09:00
# 既存のインポート部分に追加
from rest_framework.decorators import api_view
from rest_framework.response import Response
from rest_framework import status
from rog.models import NewEvent2, Entry, GpsCheckin, Team
2025-08-20 19:15:19 +09:00
import logging
from django.db.models import F, Q
from django.conf import settings
import os
from urllib.parse import urljoin, quote
from django.http import JsonResponse
from django.views.decorators.csrf import csrf_exempt
from django.views.decorators.http import require_http_methods
import boto3
from botocore.exceptions import ClientError
2025-08-20 19:15:19 +09:00
logger = logging.getLogger(__name__)
def generate_image_url(image_address, event_code, zekken_number):
"""
画像アドレスからS3 URLまたは適切なURLを生成
"""
if not image_address:
return None
# 既にHTTP URLの場合はそのまま返す
if image_address.startswith('http'):
return image_address
# simulation_image.jpgなどのテスト画像の場合はS3にないのでスキップ
if image_address in ['simulation_image.jpg', 'test_image']:
return f"/media/{image_address}"
# S3パスを構築してURLを生成
s3_key = f"{event_code}/{zekken_number}/{image_address}"
try:
# S3 URLを生成
s3_url = f"https://{settings.AWS_STORAGE_BUCKET_NAME}.s3.{settings.AWS_S3_REGION_NAME}.amazonaws.com/{quote(s3_key)}"
return s3_url
except Exception as e:
# S3設定に問題がある場合はmediaパスを返す
return f"/media/{image_address}"
2025-08-20 19:15:19 +09:00
"""
解説
この実装では以下の処理を行っています
1.2つのエンドポイントを提供しています
- /get_photo_list - 認証なしで写真とレポートURLを取得
- /get_photo_list_prod - パスワード認証付きで同じ情報を取得
2.共通のロジックは get_team_photos 関数に集約し以下の情報を取得します
- チームの基本情報名前ゼッケン番号クラス名
- チェックポイント通過時の写真時間順サービスチェック情報含む
- スタート写真とゴール写真あれば
- チームレポートのURL存在する場合
- スコアボードのURL存在する場合
- チームのスコアゴール済みの場合
3.レポートとスコアボードのファイルパスを実際に確認し存在する場合のみURLを提供します
4.写真の表示順はスタートチェックポイント時間順ゴールとなっておりチェックポイントについてはそれぞれ番号撮影時間サービスチェック状態などの情報も含めています
この実装によりチームは自分たちの競技中の写真やレポートを簡単に確認できます
本番環境_prod版ではパスワード認証によりセキュリティを確保しています
"""
@api_view(['GET'])
def get_photo_list(request):
"""
チームの写真とレポートURLを取得認証なし版
パラメータ:
- zekken: ゼッケン番号
- event: イベントコード
"""
logger.info("get_photo_list called")
# リクエストからパラメータを取得
zekken_number = request.query_params.get('zekken')
event_code = request.query_params.get('event')
logger.debug(f"Parameters: zekken={zekken_number}, event={event_code}")
# パラメータ検証
if not all([zekken_number, event_code]):
logger.warning("Missing required parameters")
return Response({
"status": "ERROR",
"message": "ゼッケン番号とイベントコードが必要です"
}, status=status.HTTP_400_BAD_REQUEST)
return get_team_photos(zekken_number, event_code)
@api_view(['GET'])
def get_photo_list_prod(request):
"""
チームの写真とレポートURLを取得認証あり版
パラメータ:
- zekken: ゼッケン番号
- pw: パスワード
- event: イベントコード
"""
logger.info("get_photo_list_prod called")
# リクエストからパラメータを取得
zekken_number = request.query_params.get('zekken')
password = request.query_params.get('pw')
event_code = request.query_params.get('event')
logger.debug(f"Parameters: zekken={zekken_number}, event={event_code}, has_password={bool(password)}")
# パラメータ検証
if not all([zekken_number, password, event_code]):
logger.warning("Missing required parameters")
return Response({
"status": "ERROR",
"message": "ゼッケン番号、パスワード、イベントコードが必要です"
}, status=status.HTTP_400_BAD_REQUEST)
try:
# イベントの存在確認
event = NewEvent2.objects.filter(event_name=event_code).first()
if not event:
logger.warning(f"Event not found: {event_code}")
return Response({
"status": "ERROR",
"message": "指定されたイベントが見つかりません"
}, status=status.HTTP_404_NOT_FOUND)
# チームの存在確認とパスワード検証
entry = Entry.objects.filter(
event=event,
zekken_number=zekken_number
).first()
if not entry:
logger.warning(f"Team with zekken number {zekken_number} not found in event: {event_code}")
return Response({
"status": "ERROR",
"message": "指定されたゼッケン番号のチームが見つかりません"
}, status=status.HTTP_404_NOT_FOUND)
# パスワード検証
if not hasattr(entry, 'password') or entry.password != password:
logger.warning(f"Invalid password for team: {entry.team.team_name if entry.team else 'Unknown'}")
2025-08-20 19:15:19 +09:00
return Response({
"status": "ERROR",
"message": "パスワードが一致しません"
}, status=status.HTTP_401_UNAUTHORIZED)
return get_team_photos(zekken_number, event_code)
except Exception as e:
logger.error(f"Error in get_photo_list_prod: {str(e)}")
return Response({
"status": "ERROR",
"message": "サーバーエラーが発生しました"
}, status=status.HTTP_500_INTERNAL_SERVER_ERROR)
def get_team_photos(request):
2025-08-20 19:15:19 +09:00
"""
チーム別の写真データを取得するAPI
2025-08-20 19:15:19 +09:00
"""
zekken = request.GET.get('zekken')
event = request.GET.get('event')
if not zekken or not event:
return JsonResponse({
'error': 'zekken and event parameters are required'
}, status=400)
2025-08-20 19:15:19 +09:00
try:
# GpsCheckinからチームの画像データを取得
gps_checkins = GpsCheckin.objects.filter(
zekken_number=zekken,
event_code=event
).exclude(
image_address__isnull=True
).exclude(
image_address=''
).order_by('create_at')
2025-08-20 19:15:19 +09:00
photos = []
for gps in gps_checkins:
# image_addressを処理してS3 URLまたは既存URLを生成
image_url = generate_image_url(gps.image_address, event, zekken)
2025-08-20 19:15:19 +09:00
photos.append({
'id': gps.id,
'image_url': image_url,
'created_at': gps.create_at.strftime('%Y-%m-%d %H:%M:%S') if gps.create_at else None,
'point_name': gps.checkpoint_id,
'latitude': float(gps.lattitude) if gps.lattitude else None,
'longitude': float(gps.longitude) if gps.longitude else None,
})
return JsonResponse({
'photos': photos,
'count': len(photos)
})
2025-08-20 19:15:19 +09:00
except Exception as e:
return JsonResponse({
'error': f'Error retrieving photos: {str(e)}'
}, status=500)