optimized

This commit is contained in:
Tsanie Lily 2023-05-26 13:39:58 +08:00
parent 098f64451a
commit e04ac0f541
4 changed files with 185 additions and 36 deletions

View File

@ -250,6 +250,28 @@ public abstract partial class BaseController : ControllerBase
var path = Path.Combine(directory, file.Path);
await System.IO.File.WriteAllBytesAsync(path, file.Content, token);
}
/// <summary>
/// 删除花草下的文件
/// </summary>
/// <param name="uid">用户唯一 id</param>
/// <param name="fid">花草唯一 id</param>
/// <param name="path">文件路径</param>
/// <returns>返回是否已删除</returns>
protected bool DeleteFile(int uid, int fid, string path)
{
var directory = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "uploads", uid.ToString(), fid.ToString());
if (Directory.Exists(directory))
{
path = Path.Combine(directory, path);
if (System.IO.File.Exists(path))
{
System.IO.File.Delete(path);
return true;
}
}
return false;
}
}
/// <summary>

View File

@ -71,6 +71,16 @@ partial class BaseController
return database.Database.ExecuteSql($"UPDATE \"users\" SET \"avatar\" = NULL WHERE \"uid\" = {uid}");
}
/// <summary>
/// 添加事件项
/// </summary>
/// <param name="item">事件对象</param>
/// <returns></returns>
protected int AddRecordItem(RecordItem item)
{
return database.Database.ExecuteSql($"INSERT INTO \"records\"(\"uid\",\"fid\",\"eid\",\"date\",\"byuid\",\"byname\",\"memo\") VALUES({item.OwnerId},{item.FlowerId},{item.EventId},{item.DateUnixTime},{item.ByUserId},{item.ByUserName},{item.Memo})");
}
/// <summary>
/// 添加照片项
/// </summary>

View File

@ -3,6 +3,7 @@ using Blahblah.FlowerStory.Server.Data.Model;
using Microsoft.AspNetCore.Mvc;
using Microsoft.EntityFrameworkCore;
using System.ComponentModel.DataAnnotations;
using System.IO;
namespace Blahblah.FlowerStory.Server.Controller;
@ -293,7 +294,6 @@ public class EventApiController : BaseController
var record = database.Records.SingleOrDefault(r => r.Id == update.Id && r.OwnerId == user.Id);
if (record == null)
{
SaveDatabase();
return NotFound(update.Id);
}
record.FlowerId = update.FlowerId;
@ -360,10 +360,11 @@ public class EventApiController : BaseController
return NotFound();
}
SaveDatabase();
var record = database.Records.SingleOrDefault(r => r.Id == id && r.OwnerId == user.Id);
if (record == null)
{
SaveDatabase();
return NotFound(id);
}
if (photo.Length > 0)
@ -371,10 +372,13 @@ public class EventApiController : BaseController
var file = WrapFormFile(photo);
if (file == null)
{
SaveDatabase();
return BadRequest();
}
try
{
await ExecuteTransaction(async token =>
{
var p = new PhotoItem
{
FlowerId = record.FlowerId,
@ -384,20 +388,17 @@ public class EventApiController : BaseController
Path = file.Path,
DateUploadUnixTime = user.ActiveDateUnixTime ?? DateTimeOffset.UtcNow.ToUnixTimeMilliseconds()
};
database.Photos.Add(p);
AddPhotoItem(p);
try
{
await WriteToFile(user.Id, record.FlowerId, file);
await WriteToFile(user.Id, record.FlowerId, file, token);
});
}
catch (Exception ex)
{
SaveDatabase();
return Problem(ex.ToString(), "api/event/add_photo");
// TODO: Logger
}
}
SaveDatabase();
return NoContent();
}
@ -451,14 +452,12 @@ public class EventApiController : BaseController
if (photos == null || photos.Length == 0)
{
SaveDatabase();
return BadRequest();
}
var record = database.Records.SingleOrDefault(r => r.Id == id && r.OwnerId == user.Id);
if (record == null)
{
SaveDatabase();
return NotFound(id);
}
@ -503,6 +502,125 @@ public class EventApiController : BaseController
return NoContent();
}
/// <summary>
/// 移除事件关联照片
/// </summary>
/// <remarks>
/// 请求示例:
///
/// DELETE /api/event/remove_photo
/// Authorization: authorization id
///
/// 参数:
///
/// id: int
///
/// </remarks>
/// <param name="id">图片唯一 id</param>
/// <returns>移除成功则返回 HTTP 204</returns>
/// <response code="204">移除成功</response>
/// <response code="401">未找到登录会话或已过期或图片所有者不符</response>
/// <response code="403">用户已禁用</response>
/// <response code="404">未找到关联用户或者照片</response>
[Route("remove_photo", Name = "removeEventPhoto")]
[ProducesResponseType(StatusCodes.Status204NoContent)]
[ProducesResponseType(StatusCodes.Status401Unauthorized)]
[ProducesResponseType(StatusCodes.Status403Forbidden)]
[ProducesResponseType(StatusCodes.Status404NotFound)]
[HttpDelete]
public ActionResult<int> RemoveEventPhoto([FromQuery][Required] int id)
{
var (result, user) = CheckPermission();
if (result != null)
{
return result;
}
if (user == null)
{
return NotFound();
}
var photo = database.Photos.Where(p => p.Id == id).Include(p => p.Record).SingleOrDefault();
if (photo == null)
{
return NotFound();
}
if (photo.Record != null && photo.Record.OwnerId != user.Id)
{
return Unauthorized();
}
database.Photos.Remove(photo);
SaveDatabase();
if (photo.Record != null)
{
DeleteFile(user.Id, photo.Record.FlowerId, photo.Path);
}
return NoContent();
}
/// <summary>
/// 批量移除事件关联的照片
/// </summary>
/// <remarks>
/// 请求示例:
///
/// POST /api/event/remove_photos
/// Authorization: authorization id
/// [
/// 2, 4, 5, 11
/// ]
///
/// </remarks>
/// <param name="ids">要移除的事件关联图片唯一 id 的数组</param>
/// <returns>会话有效则返回操作影响的数据库行数</returns>
/// <response code="200">返回操作影响的数据库行数</response>
/// <response code="401">未找到登录会话或已过期</response>
/// <response code="403">用户已禁用</response>
/// <response code="404">未找到关联用户</response>
[Route("remove_photos", Name = "removeEventPhotos")]
[ProducesResponseType(StatusCodes.Status200OK)]
[ProducesResponseType(StatusCodes.Status401Unauthorized)]
[ProducesResponseType(StatusCodes.Status403Forbidden)]
[ProducesResponseType(StatusCodes.Status404NotFound)]
[HttpPost]
[Consumes("application/json")]
public ActionResult<int> RemoveEventPhotos([FromBody] int[] ids)
{
var (result, user) = CheckPermission();
if (result != null)
{
return result;
}
if (user == null)
{
return NotFound();
}
if (database.Photos.Any(p => ids.Contains(p.Id) && database.Records.Any(r => r.Id == p.RecordId && r.OwnerId != user.Id)))
{
return Unauthorized();
}
var photos = database.Photos.Where(p => ids.Contains(p.Id)).Include(p => p.Record).ToList();
var count = database.Photos.Where(p => ids.Contains(p.Id)).ExecuteDelete();
SaveDatabase();
foreach (var photo in photos)
{
if (photo.Record != null)
{
DeleteFile(user.Id, photo.Record.FlowerId, photo.Path);
}
}
return Ok(count);
}
/// <summary>
/// 获取事件关联的照片列表
/// </summary>

View File

@ -316,7 +316,6 @@ public class FlowerApiController : BaseController
var flower = database.Flowers.SingleOrDefault(f => f.Id == update.Id && f.OwnerId == user.Id);
if (flower == null)
{
SaveDatabase();
return NotFound(update.Id);
}
flower.CategoryId = update.CategoryId;
@ -378,7 +377,6 @@ public class FlowerApiController : BaseController
var flower = database.Flowers.SingleOrDefault(f => f.Id == id && f.OwnerId == user.Id);
if (flower == null)
{
SaveDatabase();
return NotFound(id);
}
if (photo.Length > 0)
@ -386,7 +384,6 @@ public class FlowerApiController : BaseController
var file = WrapFormFile(photo);
if (file == null)
{
SaveDatabase();
return BadRequest();
}
@ -406,30 +403,32 @@ public class FlowerApiController : BaseController
};
database.Records.Add(record);
}
SaveDatabase();
try
{
await ExecuteTransaction(async token =>
{
var cover = new PhotoItem
{
FlowerId = id,
Record = record,
RecordId = record.Id,
FileType = file.FileType,
FileName = file.Filename,
Path = file.Path,
DateUploadUnixTime = now
};
database.Photos.Add(cover);
AddPhotoItem(cover);
try
{
await WriteToFile(user.Id, id, file);
await WriteToFile(user.Id, id, file, token);
});
}
catch (Exception ex)
{
SaveDatabase();
return Problem(ex.ToString(), "api/flower/add_cover");
// TODO: Logger
}
}
SaveDatabase();
return NoContent();
}