| 
									
										
										
										
											2025-08-14 11:34:31 +00:00
										 |  |  |  | package handlers | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | import ( | 
					
						
							|  |  |  |  | 	"encoding/json" | 
					
						
							| 
									
										
										
										
											2025-08-17 11:58:43 +00:00
										 |  |  |  | 	"fmt" | 
					
						
							|  |  |  |  | 	"io" | 
					
						
							| 
									
										
										
										
											2025-08-14 11:34:31 +00:00
										 |  |  |  | 	"net/http" | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	"github.com/gorilla/mux" | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-28 21:25:21 +03:00
										 |  |  |  | 	"neomovies-api/pkg/config" | 
					
						
							| 
									
										
										
										
											2025-08-14 11:34:31 +00:00
										 |  |  |  | 	"neomovies-api/pkg/middleware" | 
					
						
							|  |  |  |  | 	"neomovies-api/pkg/models" | 
					
						
							|  |  |  |  | 	"neomovies-api/pkg/services" | 
					
						
							|  |  |  |  | ) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | type FavoritesHandler struct { | 
					
						
							|  |  |  |  | 	favoritesService *services.FavoritesService | 
					
						
							| 
									
										
										
										
											2025-08-17 11:58:43 +00:00
										 |  |  |  | 	config           *config.Config | 
					
						
							| 
									
										
										
										
											2025-08-14 11:34:31 +00:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-17 11:58:43 +00:00
										 |  |  |  | func NewFavoritesHandler(favoritesService *services.FavoritesService, cfg *config.Config) *FavoritesHandler { | 
					
						
							| 
									
										
										
										
											2025-08-14 11:34:31 +00:00
										 |  |  |  | 	return &FavoritesHandler{ | 
					
						
							|  |  |  |  | 		favoritesService: favoritesService, | 
					
						
							| 
									
										
										
										
											2025-08-28 21:25:21 +03:00
										 |  |  |  | 		config:           cfg, | 
					
						
							| 
									
										
										
										
											2025-08-14 11:34:31 +00:00
										 |  |  |  | 	} | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | func (h *FavoritesHandler) GetFavorites(w http.ResponseWriter, r *http.Request) { | 
					
						
							|  |  |  |  | 	userID, ok := middleware.GetUserIDFromContext(r.Context()) | 
					
						
							|  |  |  |  | 	if !ok { | 
					
						
							|  |  |  |  | 		http.Error(w, "User ID not found in context", http.StatusUnauthorized) | 
					
						
							|  |  |  |  | 		return | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	favorites, err := h.favoritesService.GetFavorites(userID) | 
					
						
							|  |  |  |  | 	if err != nil { | 
					
						
							|  |  |  |  | 		http.Error(w, "Failed to get favorites: "+err.Error(), http.StatusInternalServerError) | 
					
						
							|  |  |  |  | 		return | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	w.Header().Set("Content-Type", "application/json") | 
					
						
							|  |  |  |  | 	json.NewEncoder(w).Encode(models.APIResponse{ | 
					
						
							|  |  |  |  | 		Success: true, | 
					
						
							|  |  |  |  | 		Data:    favorites, | 
					
						
							|  |  |  |  | 		Message: "Favorites retrieved successfully", | 
					
						
							|  |  |  |  | 	}) | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | func (h *FavoritesHandler) AddToFavorites(w http.ResponseWriter, r *http.Request) { | 
					
						
							|  |  |  |  | 	userID, ok := middleware.GetUserIDFromContext(r.Context()) | 
					
						
							|  |  |  |  | 	if !ok { | 
					
						
							|  |  |  |  | 		http.Error(w, "User ID not found in context", http.StatusUnauthorized) | 
					
						
							|  |  |  |  | 		return | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	vars := mux.Vars(r) | 
					
						
							|  |  |  |  | 	mediaID := vars["id"] | 
					
						
							|  |  |  |  | 	mediaType := r.URL.Query().Get("type") | 
					
						
							| 
									
										
										
										
											2025-08-28 21:25:21 +03:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-14 11:34:31 +00:00
										 |  |  |  | 	if mediaID == "" { | 
					
						
							|  |  |  |  | 		http.Error(w, "Media ID is required", http.StatusBadRequest) | 
					
						
							|  |  |  |  | 		return | 
					
						
							|  |  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2025-08-28 21:25:21 +03:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-14 11:34:31 +00:00
										 |  |  |  | 	if mediaType == "" { | 
					
						
							|  |  |  |  | 		mediaType = "movie" // По умолчанию фильм для обратной совместимости | 
					
						
							|  |  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2025-08-28 21:25:21 +03:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-14 11:34:31 +00:00
										 |  |  |  | 	if mediaType != "movie" && mediaType != "tv" { | 
					
						
							|  |  |  |  | 		http.Error(w, "Media type must be 'movie' or 'tv'", http.StatusBadRequest) | 
					
						
							|  |  |  |  | 		return | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-17 11:58:43 +00:00
										 |  |  |  | 	// Получаем информацию о медиа на русском языке | 
					
						
							|  |  |  |  | 	mediaInfo, err := h.fetchMediaInfoRussian(mediaID, mediaType) | 
					
						
							|  |  |  |  | 	if err != nil { | 
					
						
							|  |  |  |  | 		http.Error(w, "Failed to fetch media information: "+err.Error(), http.StatusInternalServerError) | 
					
						
							|  |  |  |  | 		return | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	err = h.favoritesService.AddToFavoritesWithInfo(userID, mediaID, mediaType, mediaInfo) | 
					
						
							| 
									
										
										
										
											2025-08-14 11:34:31 +00:00
										 |  |  |  | 	if err != nil { | 
					
						
							|  |  |  |  | 		http.Error(w, "Failed to add to favorites: "+err.Error(), http.StatusInternalServerError) | 
					
						
							|  |  |  |  | 		return | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	w.Header().Set("Content-Type", "application/json") | 
					
						
							|  |  |  |  | 	json.NewEncoder(w).Encode(models.APIResponse{ | 
					
						
							|  |  |  |  | 		Success: true, | 
					
						
							|  |  |  |  | 		Message: "Added to favorites successfully", | 
					
						
							|  |  |  |  | 	}) | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | func (h *FavoritesHandler) RemoveFromFavorites(w http.ResponseWriter, r *http.Request) { | 
					
						
							|  |  |  |  | 	userID, ok := middleware.GetUserIDFromContext(r.Context()) | 
					
						
							|  |  |  |  | 	if !ok { | 
					
						
							|  |  |  |  | 		http.Error(w, "User ID not found in context", http.StatusUnauthorized) | 
					
						
							|  |  |  |  | 		return | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	vars := mux.Vars(r) | 
					
						
							|  |  |  |  | 	mediaID := vars["id"] | 
					
						
							|  |  |  |  | 	mediaType := r.URL.Query().Get("type") | 
					
						
							| 
									
										
										
										
											2025-08-28 21:25:21 +03:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-14 11:34:31 +00:00
										 |  |  |  | 	if mediaID == "" { | 
					
						
							|  |  |  |  | 		http.Error(w, "Media ID is required", http.StatusBadRequest) | 
					
						
							|  |  |  |  | 		return | 
					
						
							|  |  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2025-08-28 21:25:21 +03:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-14 11:34:31 +00:00
										 |  |  |  | 	if mediaType == "" { | 
					
						
							|  |  |  |  | 		mediaType = "movie" // По умолчанию фильм для обратной совместимости | 
					
						
							|  |  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2025-08-28 21:25:21 +03:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-14 11:34:31 +00:00
										 |  |  |  | 	if mediaType != "movie" && mediaType != "tv" { | 
					
						
							|  |  |  |  | 		http.Error(w, "Media type must be 'movie' or 'tv'", http.StatusBadRequest) | 
					
						
							|  |  |  |  | 		return | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	err := h.favoritesService.RemoveFromFavorites(userID, mediaID, mediaType) | 
					
						
							|  |  |  |  | 	if err != nil { | 
					
						
							|  |  |  |  | 		http.Error(w, "Failed to remove from favorites: "+err.Error(), http.StatusInternalServerError) | 
					
						
							|  |  |  |  | 		return | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	w.Header().Set("Content-Type", "application/json") | 
					
						
							|  |  |  |  | 	json.NewEncoder(w).Encode(models.APIResponse{ | 
					
						
							|  |  |  |  | 		Success: true, | 
					
						
							|  |  |  |  | 		Message: "Removed from favorites successfully", | 
					
						
							|  |  |  |  | 	}) | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | func (h *FavoritesHandler) CheckIsFavorite(w http.ResponseWriter, r *http.Request) { | 
					
						
							|  |  |  |  | 	userID, ok := middleware.GetUserIDFromContext(r.Context()) | 
					
						
							|  |  |  |  | 	if !ok { | 
					
						
							|  |  |  |  | 		http.Error(w, "User ID not found in context", http.StatusUnauthorized) | 
					
						
							|  |  |  |  | 		return | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	vars := mux.Vars(r) | 
					
						
							|  |  |  |  | 	mediaID := vars["id"] | 
					
						
							|  |  |  |  | 	mediaType := r.URL.Query().Get("type") | 
					
						
							| 
									
										
										
										
											2025-08-28 21:25:21 +03:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-14 11:34:31 +00:00
										 |  |  |  | 	if mediaID == "" { | 
					
						
							|  |  |  |  | 		http.Error(w, "Media ID is required", http.StatusBadRequest) | 
					
						
							|  |  |  |  | 		return | 
					
						
							|  |  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2025-08-28 21:25:21 +03:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-14 11:34:31 +00:00
										 |  |  |  | 	if mediaType == "" { | 
					
						
							|  |  |  |  | 		mediaType = "movie" // По умолчанию фильм для обратной совместимости | 
					
						
							|  |  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2025-08-28 21:25:21 +03:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-14 11:34:31 +00:00
										 |  |  |  | 	if mediaType != "movie" && mediaType != "tv" { | 
					
						
							|  |  |  |  | 		http.Error(w, "Media type must be 'movie' or 'tv'", http.StatusBadRequest) | 
					
						
							|  |  |  |  | 		return | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	isFavorite, err := h.favoritesService.IsFavorite(userID, mediaID, mediaType) | 
					
						
							|  |  |  |  | 	if err != nil { | 
					
						
							|  |  |  |  | 		http.Error(w, "Failed to check favorite status: "+err.Error(), http.StatusInternalServerError) | 
					
						
							|  |  |  |  | 		return | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	w.Header().Set("Content-Type", "application/json") | 
					
						
							|  |  |  |  | 	json.NewEncoder(w).Encode(models.APIResponse{ | 
					
						
							|  |  |  |  | 		Success: true, | 
					
						
							|  |  |  |  | 		Data:    map[string]bool{"isFavorite": isFavorite}, | 
					
						
							|  |  |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2025-08-17 11:58:43 +00:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | // fetchMediaInfoRussian получает информацию о медиа на русском языке из TMDB | 
					
						
							| 
									
										
										
										
											2025-08-28 21:25:21 +03:00
										 |  |  |  | func (h *FavoritesHandler) fetchMediaInfoRussian(mediaID, mediaType string) (*models.MediaInfo, error) { | 
					
						
							| 
									
										
										
										
											2025-08-17 11:58:43 +00:00
										 |  |  |  | 	var url string | 
					
						
							|  |  |  |  | 	if mediaType == "movie" { | 
					
						
							| 
									
										
										
										
											2025-08-28 21:25:21 +03:00
										 |  |  |  | 		url = fmt.Sprintf("https://api.themoviedb.org/3/movie/%s?api_key=%s&language=ru-RU", mediaID, h.config.TMDBAccessToken) | 
					
						
							| 
									
										
										
										
											2025-08-17 11:58:43 +00:00
										 |  |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2025-08-28 21:25:21 +03:00
										 |  |  |  | 		url = fmt.Sprintf("https://api.themoviedb.org/3/tv/%s?api_key=%s&language=ru-RU", mediaID, h.config.TMDBAccessToken) | 
					
						
							| 
									
										
										
										
											2025-08-17 11:58:43 +00:00
										 |  |  |  | 	} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	resp, err := http.Get(url) | 
					
						
							|  |  |  |  | 	if err != nil { | 
					
						
							|  |  |  |  | 		return nil, fmt.Errorf("failed to fetch from TMDB: %w", err) | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  | 	defer resp.Body.Close() | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	if resp.StatusCode != http.StatusOK { | 
					
						
							|  |  |  |  | 		return nil, fmt.Errorf("TMDB API error: status %d", resp.StatusCode) | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	body, err := io.ReadAll(resp.Body) | 
					
						
							|  |  |  |  | 	if err != nil { | 
					
						
							|  |  |  |  | 		return nil, fmt.Errorf("failed to read response body: %w", err) | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	var tmdbResponse map[string]interface{} | 
					
						
							|  |  |  |  | 	if err := json.Unmarshal(body, &tmdbResponse); err != nil { | 
					
						
							|  |  |  |  | 		return nil, fmt.Errorf("failed to parse TMDB response: %w", err) | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-28 21:25:21 +03:00
										 |  |  |  | 	mediaInfo := &models.MediaInfo{ | 
					
						
							| 
									
										
										
										
											2025-08-17 11:58:43 +00:00
										 |  |  |  | 		ID:        mediaID, | 
					
						
							|  |  |  |  | 		MediaType: mediaType, | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	// Заполняем информацию в зависимости от типа медиа | 
					
						
							|  |  |  |  | 	if mediaType == "movie" { | 
					
						
							|  |  |  |  | 		if title, ok := tmdbResponse["title"].(string); ok { | 
					
						
							|  |  |  |  | 			mediaInfo.Title = title | 
					
						
							|  |  |  |  | 		} | 
					
						
							|  |  |  |  | 		if originalTitle, ok := tmdbResponse["original_title"].(string); ok { | 
					
						
							|  |  |  |  | 			mediaInfo.OriginalTitle = originalTitle | 
					
						
							|  |  |  |  | 		} | 
					
						
							|  |  |  |  | 		if releaseDate, ok := tmdbResponse["release_date"].(string); ok { | 
					
						
							|  |  |  |  | 			mediaInfo.ReleaseDate = releaseDate | 
					
						
							|  |  |  |  | 		} | 
					
						
							|  |  |  |  | 	} else { | 
					
						
							|  |  |  |  | 		if name, ok := tmdbResponse["name"].(string); ok { | 
					
						
							|  |  |  |  | 			mediaInfo.Title = name | 
					
						
							|  |  |  |  | 		} | 
					
						
							|  |  |  |  | 		if originalName, ok := tmdbResponse["original_name"].(string); ok { | 
					
						
							|  |  |  |  | 			mediaInfo.OriginalTitle = originalName | 
					
						
							|  |  |  |  | 		} | 
					
						
							|  |  |  |  | 		if firstAirDate, ok := tmdbResponse["first_air_date"].(string); ok { | 
					
						
							|  |  |  |  | 			mediaInfo.FirstAirDate = firstAirDate | 
					
						
							|  |  |  |  | 		} | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	// Общие поля | 
					
						
							|  |  |  |  | 	if overview, ok := tmdbResponse["overview"].(string); ok { | 
					
						
							|  |  |  |  | 		mediaInfo.Overview = overview | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  | 	if posterPath, ok := tmdbResponse["poster_path"].(string); ok { | 
					
						
							|  |  |  |  | 		mediaInfo.PosterPath = posterPath | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  | 	if backdropPath, ok := tmdbResponse["backdrop_path"].(string); ok { | 
					
						
							|  |  |  |  | 		mediaInfo.BackdropPath = backdropPath | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  | 	if voteAverage, ok := tmdbResponse["vote_average"].(float64); ok { | 
					
						
							|  |  |  |  | 		mediaInfo.VoteAverage = voteAverage | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  | 	if voteCount, ok := tmdbResponse["vote_count"].(float64); ok { | 
					
						
							|  |  |  |  | 		mediaInfo.VoteCount = int(voteCount) | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  | 	if popularity, ok := tmdbResponse["popularity"].(float64); ok { | 
					
						
							|  |  |  |  | 		mediaInfo.Popularity = popularity | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	// Жанры | 
					
						
							|  |  |  |  | 	if genres, ok := tmdbResponse["genres"].([]interface{}); ok { | 
					
						
							|  |  |  |  | 		for _, genre := range genres { | 
					
						
							|  |  |  |  | 			if genreMap, ok := genre.(map[string]interface{}); ok { | 
					
						
							|  |  |  |  | 				if genreID, ok := genreMap["id"].(float64); ok { | 
					
						
							|  |  |  |  | 					mediaInfo.GenreIDs = append(mediaInfo.GenreIDs, int(genreID)) | 
					
						
							|  |  |  |  | 				} | 
					
						
							|  |  |  |  | 			} | 
					
						
							|  |  |  |  | 		} | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	return mediaInfo, nil | 
					
						
							| 
									
										
										
										
											2025-08-28 21:25:21 +03:00
										 |  |  |  | } |