mirror of
https://github.com/LBPUnion/ProjectLighthouse.git
synced 2025-05-14 22:02:26 +00:00
* Initial commit to support developer slots * Remove hearting story levels, prevent race condition in adding dev slots, and remove LastContactHelper local db object. * Fix photos taken in pod showing wrong level. * Add support for pod and create mode photos * Add time display to photos and added photo display to level page * Add pagination to in game photos * Update in pod description * Fix migration * Adjust wording of photos taken on local slots * Set slot default type to User Fixes old slots being set to developer slots * Apply suggestions * Add player count to developer slots Co-authored-by: Jayden <jvyden@jvyden.xyz>
211 lines
No EOL
8.5 KiB
C#
211 lines
No EOL
8.5 KiB
C#
#nullable enable
|
|
using System.Xml.Serialization;
|
|
using Discord;
|
|
using LBPUnion.ProjectLighthouse.Configuration;
|
|
using LBPUnion.ProjectLighthouse.Helpers;
|
|
using LBPUnion.ProjectLighthouse.Levels;
|
|
using LBPUnion.ProjectLighthouse.Logging;
|
|
using LBPUnion.ProjectLighthouse.PlayerData;
|
|
using LBPUnion.ProjectLighthouse.PlayerData.Profiles;
|
|
using LBPUnion.ProjectLighthouse.Serialization;
|
|
using Microsoft.AspNetCore.Mvc;
|
|
using Microsoft.EntityFrameworkCore;
|
|
|
|
namespace LBPUnion.ProjectLighthouse.Servers.GameServer.Controllers.Resources;
|
|
|
|
[ApiController]
|
|
[Route("LITTLEBIGPLANETPS3_XML/")]
|
|
[Produces("text/xml")]
|
|
public class PhotosController : ControllerBase
|
|
{
|
|
private readonly Database database;
|
|
|
|
public PhotosController(Database database)
|
|
{
|
|
this.database = database;
|
|
}
|
|
|
|
[HttpPost("uploadPhoto")]
|
|
public async Task<IActionResult> UploadPhoto()
|
|
{
|
|
User? user = await this.database.UserFromGameRequest(this.Request);
|
|
if (user == null) return this.StatusCode(403, "");
|
|
|
|
if (user.PhotosByMe >= ServerConfiguration.Instance.UserGeneratedContentLimits.PhotosQuota) return this.BadRequest();
|
|
|
|
this.Request.Body.Position = 0;
|
|
string bodyString = await new StreamReader(this.Request.Body).ReadToEndAsync();
|
|
|
|
XmlSerializer serializer = new(typeof(Photo));
|
|
Photo? photo = (Photo?)serializer.Deserialize(new StringReader(bodyString));
|
|
if (photo == null) return this.BadRequest();
|
|
|
|
SanitizationHelper.SanitizeStringsInClass(photo);
|
|
|
|
foreach (Photo p in this.database.Photos.Where(p => p.CreatorId == user.UserId))
|
|
{
|
|
if (p.LargeHash == photo.LargeHash) return this.Ok(); // photo already uplaoded
|
|
if (p.MediumHash == photo.MediumHash) return this.Ok();
|
|
if (p.SmallHash == photo.SmallHash) return this.Ok();
|
|
if (p.PlanHash == photo.PlanHash) return this.Ok();
|
|
}
|
|
|
|
photo.CreatorId = user.UserId;
|
|
photo.Creator = user;
|
|
|
|
if (photo.XmlLevelInfo != null)
|
|
{
|
|
bool validLevel = false;
|
|
PhotoSlot photoSlot = photo.XmlLevelInfo;
|
|
if (photoSlot.SlotType is SlotType.Pod or SlotType.Local) photoSlot.SlotId = 0;
|
|
switch (photoSlot.SlotType)
|
|
{
|
|
case SlotType.User:
|
|
{
|
|
Slot? slot = await this.database.Slots.FirstOrDefaultAsync(s => s.Type == SlotType.User && s.SlotId == photoSlot.SlotId);
|
|
if (slot != null) validLevel = slot.RootLevel == photoSlot.RootLevel;
|
|
break;
|
|
}
|
|
case SlotType.Pod:
|
|
case SlotType.Local:
|
|
case SlotType.Developer:
|
|
{
|
|
Slot? slot = await this.database.Slots.FirstOrDefaultAsync(s => s.Type == photoSlot.SlotType && s.InternalSlotId == photoSlot.SlotId);
|
|
if (slot != null)
|
|
photoSlot.SlotId = slot.SlotId;
|
|
else
|
|
photoSlot.SlotId = await SlotHelper.GetPlaceholderSlotId(this.database, photoSlot.SlotId, photoSlot.SlotType);
|
|
validLevel = true;
|
|
break;
|
|
}
|
|
default: Logger.Warn($"Invalid photo level type: {photoSlot.SlotType}", LogArea.Photos);
|
|
break;
|
|
}
|
|
|
|
if (validLevel) photo.SlotId = photo.XmlLevelInfo.SlotId;
|
|
}
|
|
|
|
if (photo.Subjects.Count > 4) return this.BadRequest();
|
|
|
|
if (photo.Timestamp > TimeHelper.Timestamp) photo.Timestamp = TimeHelper.Timestamp;
|
|
|
|
foreach (PhotoSubject subject in photo.Subjects)
|
|
{
|
|
subject.User = await this.database.Users.FirstOrDefaultAsync(u => u.Username == subject.Username);
|
|
|
|
if (subject.User == null) continue;
|
|
|
|
subject.UserId = subject.User.UserId;
|
|
Logger.Debug($"Adding PhotoSubject (userid {subject.UserId}) to db", LogArea.Photos);
|
|
|
|
this.database.PhotoSubjects.Add(subject);
|
|
}
|
|
|
|
await this.database.SaveChangesAsync();
|
|
|
|
// Check for duplicate photo subjects
|
|
List<int> subjectUserIds = new(4);
|
|
foreach (PhotoSubject subject in photo.Subjects)
|
|
{
|
|
if (subjectUserIds.Contains(subject.UserId)) return this.BadRequest();
|
|
|
|
subjectUserIds.Add(subject.UserId);
|
|
}
|
|
|
|
photo.PhotoSubjectIds = photo.Subjects.Select(subject => subject.PhotoSubjectId.ToString()).ToArray();
|
|
|
|
// photo.Slot = await this.database.Slots.FirstOrDefaultAsync(s => s.SlotId == photo.SlotId);
|
|
|
|
Logger.Debug($"Adding PhotoSubjectCollection ({photo.PhotoSubjectCollection}) to photo", LogArea.Photos);
|
|
|
|
this.database.Photos.Add(photo);
|
|
|
|
await this.database.SaveChangesAsync();
|
|
|
|
await WebhookHelper.SendWebhook
|
|
(
|
|
new EmbedBuilder
|
|
{
|
|
Title = "New photo uploaded!",
|
|
Description = $"{user.Username} uploaded a new photo.",
|
|
ImageUrl = $"{ServerConfiguration.Instance.ExternalUrl}/gameAssets/{photo.LargeHash}",
|
|
Color = WebhookHelper.UnionColor,
|
|
}
|
|
);
|
|
|
|
return this.Ok();
|
|
}
|
|
|
|
[HttpGet("photos/{slotType}/{id:int}")]
|
|
public async Task<IActionResult> SlotPhotos([FromQuery] int pageStart, [FromQuery] int pageSize, string slotType, int id)
|
|
{
|
|
User? user = await this.database.UserFromGameRequest(this.Request);
|
|
if (user == null) return this.StatusCode(403, "");
|
|
|
|
if (SlotHelper.IsTypeInvalid(slotType)) return this.BadRequest();
|
|
|
|
if (slotType == "developer") id = await SlotHelper.GetPlaceholderSlotId(this.database, id, SlotType.Developer);
|
|
|
|
List<Photo> photos = await this.database.Photos.Include(p => p.Creator)
|
|
.Where(p => p.SlotId == id)
|
|
.OrderByDescending(s => s.Timestamp)
|
|
.Skip(pageStart - 1)
|
|
.Take(Math.Min(pageSize, 30))
|
|
.ToListAsync();
|
|
string response = photos.Aggregate(string.Empty, (s, photo) => s + photo.Serialize(id, SlotHelper.ParseType(slotType)));
|
|
return this.Ok(LbpSerializer.StringElement("photos", response));
|
|
}
|
|
|
|
[HttpGet("photos/by")]
|
|
public async Task<IActionResult> UserPhotosBy([FromQuery] string user, [FromQuery] int pageStart, [FromQuery] int pageSize)
|
|
{
|
|
User? userFromQuery = await this.database.Users.FirstOrDefaultAsync(u => u.Username == user);
|
|
// ReSharper disable once ConditionIsAlwaysTrueOrFalse
|
|
if (userFromQuery == null) return this.NotFound();
|
|
|
|
List<Photo> photos = await this.database.Photos.Include
|
|
(p => p.Creator)
|
|
.Where(p => p.CreatorId == userFromQuery.UserId)
|
|
.OrderByDescending(s => s.Timestamp)
|
|
.Skip(pageStart - 1)
|
|
.Take(Math.Min(pageSize, 30))
|
|
.ToListAsync();
|
|
string response = photos.Aggregate(string.Empty, (s, photo) => s + photo.Serialize());
|
|
return this.Ok(LbpSerializer.StringElement("photos", response));
|
|
}
|
|
|
|
[HttpGet("photos/with")]
|
|
public async Task<IActionResult> UserPhotosWith([FromQuery] string user, [FromQuery] int pageStart, [FromQuery] int pageSize)
|
|
{
|
|
User? userFromQuery = await this.database.Users.FirstOrDefaultAsync(u => u.Username == user);
|
|
// ReSharper disable once ConditionIsAlwaysTrueOrFalse
|
|
if (userFromQuery == null) return this.NotFound();
|
|
|
|
List<Photo> photos = new();
|
|
foreach (Photo photo in this.database.Photos.Include
|
|
(p => p.Creator)) photos.AddRange(photo.Subjects.Where(subject => subject.User.UserId == userFromQuery.UserId).Select(_ => photo));
|
|
|
|
string response = photos.OrderByDescending
|
|
(s => s.Timestamp)
|
|
.Skip(pageStart - 1)
|
|
.Take(Math.Min(pageSize, 30))
|
|
.Aggregate(string.Empty, (s, photo) => s + photo.Serialize());
|
|
|
|
return this.Ok(LbpSerializer.StringElement("photos", response));
|
|
}
|
|
|
|
[HttpPost("deletePhoto/{id:int}")]
|
|
public async Task<IActionResult> DeletePhoto(int id)
|
|
{
|
|
User? user = await this.database.UserFromGameRequest(this.Request);
|
|
if (user == null) return this.StatusCode(403, "");
|
|
|
|
Photo? photo = await this.database.Photos.FirstOrDefaultAsync(p => p.PhotoId == id);
|
|
if (photo == null) return this.NotFound();
|
|
if (photo.CreatorId != user.UserId) return this.StatusCode(401, "");
|
|
|
|
this.database.Photos.Remove(photo);
|
|
await this.database.SaveChangesAsync();
|
|
return this.Ok();
|
|
}
|
|
} |