mirror of
https://github.com/LBPUnion/ProjectLighthouse.git
synced 2025-07-25 14:41:30 +00:00
307 lines
No EOL
13 KiB
C#
307 lines
No EOL
13 KiB
C#
#nullable enable
|
|
using System;
|
|
using System.IO;
|
|
using System.Linq;
|
|
using System.Collections.Generic;
|
|
using System.Threading.Tasks;
|
|
using System.Xml.Serialization;
|
|
using LBPUnion.ProjectLighthouse.Helpers;
|
|
using LBPUnion.ProjectLighthouse.Types;
|
|
using LBPUnion.ProjectLighthouse.Types.Levels;
|
|
using LBPUnion.ProjectLighthouse.Types.Reviews;
|
|
using Microsoft.AspNetCore.Mvc;
|
|
using Microsoft.EntityFrameworkCore;
|
|
using LBPUnion.ProjectLighthouse.Serialization;
|
|
|
|
namespace LBPUnion.ProjectLighthouse.Controllers
|
|
{
|
|
[ApiController]
|
|
[Route("LITTLEBIGPLANETPS3_XML/")]
|
|
[Produces("text/plain")]
|
|
public class ReviewController : ControllerBase
|
|
{
|
|
private readonly Database database;
|
|
|
|
public ReviewController(Database database)
|
|
{
|
|
this.database = database;
|
|
}
|
|
|
|
// LBP1 rating
|
|
[HttpPost("rate/user/{slotId}")]
|
|
public async Task<IActionResult> Rate(int slotId, [FromQuery] int rating)
|
|
{
|
|
User? user = await this.database.UserFromGameRequest(this.Request);
|
|
if (user == null) return this.StatusCode(403, "");
|
|
|
|
Slot? slot = await this.database.Slots.Include(s => s.Creator).Include(s => s.Location).FirstOrDefaultAsync(s => s.SlotId == slotId);
|
|
if (slot == null) return this.StatusCode(403, "");
|
|
|
|
RatedLevel? ratedLevel = await this.database.RatedLevels.FirstOrDefaultAsync(r => r.SlotId == slotId && r.UserId == user.UserId);
|
|
if (ratedLevel == null)
|
|
{
|
|
ratedLevel = new RatedLevel();
|
|
ratedLevel.SlotId = slotId;
|
|
ratedLevel.UserId = user.UserId;
|
|
ratedLevel.Rating = 0;
|
|
this.database.RatedLevels.Add(ratedLevel);
|
|
}
|
|
|
|
ratedLevel.RatingLBP1 = Math.Max(Math.Min(5, rating), 0);
|
|
|
|
await this.database.SaveChangesAsync();
|
|
|
|
return this.Ok();
|
|
}
|
|
|
|
// LBP2 and beyond rating
|
|
[HttpPost("dpadrate/user/{slotId:int}")]
|
|
public async Task<IActionResult> DPadRate(int slotId, [FromQuery] int rating)
|
|
{
|
|
User? user = await this.database.UserFromGameRequest(this.Request);
|
|
if (user == null) return this.StatusCode(403, "");
|
|
|
|
Slot? slot = await this.database.Slots.Include(s => s.Creator).Include(s => s.Location).FirstOrDefaultAsync(s => s.SlotId == slotId);
|
|
if (slot == null) return this.StatusCode(403, "");
|
|
|
|
RatedLevel? ratedLevel = await this.database.RatedLevels.FirstOrDefaultAsync(r => r.SlotId == slotId && r.UserId == user.UserId);
|
|
if (ratedLevel == null)
|
|
{
|
|
ratedLevel = new RatedLevel();
|
|
ratedLevel.SlotId = slotId;
|
|
ratedLevel.UserId = user.UserId;
|
|
ratedLevel.RatingLBP1 = 0;
|
|
this.database.RatedLevels.Add(ratedLevel);
|
|
}
|
|
|
|
ratedLevel.Rating = Math.Max(Math.Min(1, rating), -1);
|
|
|
|
Review? review = await this.database.Reviews.FirstOrDefaultAsync(r => r.SlotId == slotId && r.ReviewerId == user.UserId);
|
|
if (review != null)
|
|
{
|
|
review.Thumb = ratedLevel.Rating;
|
|
}
|
|
|
|
await this.database.SaveChangesAsync();
|
|
|
|
return this.Ok();
|
|
}
|
|
|
|
[HttpPost("postReview/user/{slotId:int}")]
|
|
public async Task<IActionResult> PostReview(int slotId)
|
|
{
|
|
User? user = await this.database.UserFromGameRequest(this.Request);
|
|
if (user == null) return this.StatusCode(403, "");
|
|
|
|
Review? review = await this.database.Reviews.FirstOrDefaultAsync(r => r.SlotId == slotId && r.ReviewerId == user.UserId);
|
|
Review? newReview = await this.GetReviewFromBody();
|
|
if (newReview == null) return this.BadRequest();
|
|
if (review == null)
|
|
{
|
|
review = new();
|
|
review.SlotId = slotId;
|
|
review.ReviewerId = user.UserId;
|
|
review.DeletedBy = DeletedBy.None;
|
|
review.ThumbsUp = 0;
|
|
review.ThumbsDown = 0;
|
|
this.database.Reviews.Add(review);
|
|
}
|
|
review.Thumb = newReview.Thumb;
|
|
review.LabelCollection = newReview.LabelCollection;
|
|
review.Text = newReview.Text;
|
|
review.Deleted = false;
|
|
review.Timestamp = TimeHelper.UnixTimeMilliseconds();
|
|
|
|
// sometimes the game posts/updates a review rating without also calling dpadrate/user/etc (why??)
|
|
RatedLevel? ratedLevel = await this.database.RatedLevels.FirstOrDefaultAsync(r => r.SlotId == slotId && r.UserId == user.UserId);
|
|
if (ratedLevel == null)
|
|
{
|
|
ratedLevel = new RatedLevel();
|
|
ratedLevel.SlotId = slotId;
|
|
ratedLevel.UserId = user.UserId;
|
|
ratedLevel.RatingLBP1 = 0;
|
|
this.database.RatedLevels.Add(ratedLevel);
|
|
}
|
|
|
|
ratedLevel.Rating = newReview.Thumb;
|
|
|
|
await this.database.SaveChangesAsync();
|
|
|
|
return this.Ok();
|
|
}
|
|
|
|
[HttpGet("reviewsFor/user/{slotId:int}")]
|
|
public async Task<IActionResult> ReviewsFor(int slotId, [FromQuery] int pageStart = 1, [FromQuery] int pageSize = 10)
|
|
{
|
|
(User, GameToken)? userAndToken = await this.database.UserAndGameTokenFromRequest(this.Request);
|
|
|
|
if (userAndToken == null) return this.StatusCode(403, "");
|
|
|
|
// ReSharper disable once PossibleInvalidOperationException
|
|
User user = userAndToken.Value.Item1;
|
|
GameToken gameToken = userAndToken.Value.Item2;
|
|
|
|
GameVersion gameVersion = gameToken.GameVersion;
|
|
|
|
Random rand = new();
|
|
|
|
Review? yourReview = await this.database.Reviews.FirstOrDefaultAsync(r => r.ReviewerId == user.UserId && r.SlotId == slotId && r.Slot.GameVersion <= gameVersion);
|
|
|
|
VisitedLevel? visitedLevel = await this.database.VisitedLevels.FirstOrDefaultAsync(v => v.UserId == user.UserId && v.SlotId == slotId && v.Slot.GameVersion <= gameVersion);
|
|
|
|
Slot? slot = await this.database.Slots.FirstOrDefaultAsync(s => s.SlotId == slotId);
|
|
if (slot == null) return this.BadRequest();
|
|
|
|
Boolean canNowReviewLevel = slot.CreatorId != user.UserId && visitedLevel != null && yourReview == null;
|
|
if (canNowReviewLevel)
|
|
{
|
|
RatedLevel? ratedLevel = await this.database.RatedLevels.FirstOrDefaultAsync(r => r.UserId == user.UserId && r.SlotId == slotId && r.Slot.GameVersion <= gameVersion);
|
|
|
|
yourReview = new();
|
|
yourReview.ReviewerId = user.UserId;
|
|
yourReview.Reviewer = user;
|
|
yourReview.Thumb = ratedLevel?.Rating == null ? 0 : ratedLevel.Rating;
|
|
yourReview.Slot = slot;
|
|
yourReview.SlotId = slotId;
|
|
yourReview.Deleted = false;
|
|
yourReview.DeletedBy = DeletedBy.None;
|
|
yourReview.Text = "You haven't reviewed this level yet. Edit this blank review to upload it!";
|
|
yourReview.LabelCollection = "";
|
|
yourReview.Timestamp = TimeHelper.UnixTimeMilliseconds();
|
|
}
|
|
|
|
IQueryable<Review?> reviews = this.database.Reviews.Where(r => r.SlotId == slotId && r.Slot.GameVersion <= gameVersion)
|
|
.Include(r => r.Reviewer)
|
|
.Include(r => r.Slot)
|
|
.OrderByDescending(r => r.ThumbsUp)
|
|
.ThenByDescending(_ => EF.Functions.Random())
|
|
.Skip(pageStart - 1)
|
|
.Take(pageSize);
|
|
|
|
IEnumerable<Review?> prependedReviews;
|
|
if (canNowReviewLevel) // this can only be true if you have not posted a review but have visited the level
|
|
{
|
|
// prepend the fake review to the top of the list to be easily edited
|
|
prependedReviews = reviews.ToList().Prepend(yourReview);
|
|
}
|
|
else
|
|
{
|
|
prependedReviews = reviews.ToList();
|
|
}
|
|
|
|
string inner = Enumerable.Aggregate(prependedReviews, string.Empty, (current, review) =>
|
|
{
|
|
if (review == null) return current;
|
|
return current + review.Serialize();
|
|
});
|
|
|
|
string response = LbpSerializer.TaggedStringElement("reviews", inner, new Dictionary<string, object>
|
|
{
|
|
{
|
|
"hint_start", pageStart + pageSize
|
|
},
|
|
{
|
|
"hint", pageStart // not sure
|
|
},
|
|
});
|
|
return this.Ok(response);
|
|
}
|
|
|
|
[HttpGet("reviewsBy/{username}")]
|
|
public async Task<IActionResult> ReviewsBy(string username, [FromQuery] int pageStart = 1, [FromQuery] int pageSize = 10)
|
|
{
|
|
(User, GameToken)? userAndToken = await this.database.UserAndGameTokenFromRequest(this.Request);
|
|
|
|
if (userAndToken == null) return this.StatusCode(403, "");
|
|
|
|
// ReSharper disable once PossibleInvalidOperationException
|
|
User user = userAndToken.Value.Item1;
|
|
GameToken gameToken = userAndToken.Value.Item2;
|
|
|
|
GameVersion gameVersion = gameToken.GameVersion;
|
|
|
|
IEnumerable<Review> reviews = this.database.Reviews.Where(r => r.Reviewer.Username == username && r.Slot.GameVersion <= gameVersion)
|
|
.Include(r => r.Reviewer)
|
|
.Include(r => r.Slot)
|
|
.OrderByDescending(r => r.Timestamp)
|
|
.Skip(pageStart - 1)
|
|
.Take(pageSize);
|
|
|
|
string inner = Enumerable.Aggregate(reviews, string.Empty, (current, review) =>
|
|
{
|
|
//RatedLevel? ratedLevel = this.database.RatedLevels.FirstOrDefault(r => r.SlotId == review.SlotId && r.UserId == user.UserId);
|
|
//RatedReview? ratedReview = this.database.RatedReviews.FirstOrDefault(r => r.ReviewId == review.ReviewId && r.UserId == user.UserId);
|
|
return current + review.Serialize(/*, ratedReview*/);
|
|
});
|
|
|
|
string response = LbpSerializer.TaggedStringElement("reviews", inner, new Dictionary<string, object>
|
|
{
|
|
{
|
|
"hint_start", pageStart
|
|
},
|
|
{
|
|
"hint", reviews.Last().Timestamp // Seems to be the timestamp of oldest
|
|
},
|
|
});
|
|
|
|
return this.Ok(response);
|
|
}
|
|
|
|
[HttpPost("rateReview/user/{slotId:int}/{username}")]
|
|
public async Task<IActionResult> RateReview(int slotId, string username, [FromQuery] int rating = 0)
|
|
{
|
|
User? user = await this.database.UserFromGameRequest(this.Request);
|
|
if (user == null) return this.StatusCode(403, "");
|
|
|
|
User? reviewer = await this.database.Users.FirstOrDefaultAsync(u => u.Username == username);
|
|
if (reviewer == null) return this.StatusCode(400, "");
|
|
|
|
Review? review = await this.database.Reviews.FirstOrDefaultAsync(r => r.SlotId == slotId && r.ReviewerId == reviewer.UserId);
|
|
if (review == null) return this.StatusCode(400, "");
|
|
|
|
RatedReview? ratedReview = await this.database.RatedReviews.FirstOrDefaultAsync(r => r.ReviewId == review.ReviewId && r.UserId == user.UserId);
|
|
if (ratedReview == null)
|
|
{
|
|
ratedReview = new RatedReview();
|
|
ratedReview.ReviewId = review.ReviewId;
|
|
ratedReview.UserId = user.UserId;
|
|
ratedReview.Thumb = 0;
|
|
this.database.RatedReviews.Add(ratedReview);
|
|
}
|
|
|
|
ratedReview.Thumb = Math.Max(Math.Min(1, rating), -1);
|
|
|
|
await this.database.SaveChangesAsync();
|
|
|
|
return this.Ok();
|
|
}
|
|
|
|
[HttpPost("deleteReview/user/{slotId:int}/{username}")]
|
|
public async Task<IActionResult> DeleteReview(int slotId, string username)
|
|
{
|
|
User? reviewer = await this.database.Users.FirstOrDefaultAsync(u => u.Username == username);
|
|
if (reviewer == null) return this.StatusCode(403, "");
|
|
|
|
Review? review = await this.database.Reviews.FirstOrDefaultAsync(r => r.SlotId == slotId && r.ReviewerId == reviewer.UserId);
|
|
if (review == null) return this.StatusCode(403, "");
|
|
|
|
review.Deleted = true;
|
|
review.DeletedBy = DeletedBy.LevelAuthor;
|
|
|
|
await this.database.SaveChangesAsync();
|
|
return this.Ok();
|
|
}
|
|
|
|
public async Task<Review?> GetReviewFromBody()
|
|
{
|
|
this.Request.Body.Position = 0;
|
|
string bodyString = await new StreamReader(this.Request.Body).ReadToEndAsync();
|
|
|
|
XmlSerializer serializer = new(typeof(Review));
|
|
Review? review = (Review?)serializer.Deserialize(new StringReader(bodyString));
|
|
|
|
return review;
|
|
}
|
|
|
|
}
|
|
} |