mirror of
https://github.com/streetwriters/notesnook-sync-server.git
synced 2026-06-10 23:17:46 +02:00
monograph: add new ItemId property that is a simple string
This fixes the issue where an invalid objectid causes error when publishing a monograph
This commit is contained in:
@@ -24,8 +24,10 @@ using System.Text.RegularExpressions;
|
|||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using Microsoft.AspNetCore.Authorization;
|
using Microsoft.AspNetCore.Authorization;
|
||||||
using Microsoft.AspNetCore.Mvc;
|
using Microsoft.AspNetCore.Mvc;
|
||||||
|
using MongoDB.Bson;
|
||||||
using MongoDB.Driver;
|
using MongoDB.Driver;
|
||||||
using Notesnook.API.Models;
|
using Notesnook.API.Models;
|
||||||
|
using Streetwriters.Common;
|
||||||
using Streetwriters.Data.Interfaces;
|
using Streetwriters.Data.Interfaces;
|
||||||
using Streetwriters.Data.Repositories;
|
using Streetwriters.Data.Repositories;
|
||||||
|
|
||||||
@@ -46,53 +48,121 @@ namespace Notesnook.API.Controllers
|
|||||||
unit = unitOfWork;
|
unit = unitOfWork;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static FilterDefinition<Monograph> CreateMonographFilter(string userId, Monograph monograph)
|
||||||
|
{
|
||||||
|
var userIdFilter = Builders<Monograph>.Filter.Eq("UserId", userId);
|
||||||
|
return ObjectId.TryParse(monograph.ItemId, out ObjectId id)
|
||||||
|
? Builders<Monograph>.Filter
|
||||||
|
.And(userIdFilter,
|
||||||
|
Builders<Monograph>.Filter.Or(
|
||||||
|
Builders<Monograph>.Filter.Eq("_id", id), Builders<Monograph>.Filter.Eq("ItemId", monograph.ItemId)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
: Builders<Monograph>.Filter
|
||||||
|
.And(userIdFilter,
|
||||||
|
Builders<Monograph>.Filter.Eq("ItemId", monograph.ItemId)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static FilterDefinition<Monograph> CreateMonographFilter(string itemId)
|
||||||
|
{
|
||||||
|
return ObjectId.TryParse(itemId, out ObjectId id)
|
||||||
|
? Builders<Monograph>.Filter.Or(
|
||||||
|
Builders<Monograph>.Filter.Eq("_id", id),
|
||||||
|
Builders<Monograph>.Filter.Eq("ItemId", itemId))
|
||||||
|
: Builders<Monograph>.Filter.Eq("ItemId", itemId);
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task<Monograph> FindMonographAsync(string userId, Monograph monograph)
|
||||||
|
{
|
||||||
|
var result = await Monographs.Collection.FindAsync(CreateMonographFilter(userId, monograph), new FindOptions<Monograph>
|
||||||
|
{
|
||||||
|
Limit = 1
|
||||||
|
});
|
||||||
|
return await result.FirstOrDefaultAsync();
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task<Monograph> FindMonographAsync(string itemId)
|
||||||
|
{
|
||||||
|
var result = await Monographs.Collection.FindAsync(CreateMonographFilter(itemId), new FindOptions<Monograph>
|
||||||
|
{
|
||||||
|
Limit = 1
|
||||||
|
});
|
||||||
|
return await result.FirstOrDefaultAsync();
|
||||||
|
}
|
||||||
|
|
||||||
[HttpPost]
|
[HttpPost]
|
||||||
public async Task<IActionResult> PublishAsync([FromBody] Monograph monograph)
|
public async Task<IActionResult> PublishAsync([FromBody] Monograph monograph)
|
||||||
{
|
{
|
||||||
var userId = this.User.FindFirstValue("sub");
|
try
|
||||||
if (userId == null) return Unauthorized();
|
|
||||||
|
|
||||||
if (await Monographs.GetAsync(monograph.Id) != null) return base.Conflict("This monograph is already published.");
|
|
||||||
|
|
||||||
if (monograph.EncryptedContent == null)
|
|
||||||
monograph.CompressedContent = monograph.Content.CompressBrotli();
|
|
||||||
monograph.UserId = userId;
|
|
||||||
monograph.DatePublished = DateTimeOffset.UtcNow.ToUnixTimeMilliseconds();
|
|
||||||
|
|
||||||
|
|
||||||
if (monograph.EncryptedContent?.Cipher.Length > MAX_DOC_SIZE || monograph.CompressedContent?.Length > MAX_DOC_SIZE)
|
|
||||||
return base.BadRequest("Monograph is too big. Max allowed size is 15mb.");
|
|
||||||
|
|
||||||
Monographs.Insert(monograph);
|
|
||||||
|
|
||||||
if (!await unit.Commit()) return BadRequest();
|
|
||||||
return Ok(new
|
|
||||||
{
|
{
|
||||||
id = monograph.Id
|
var userId = this.User.FindFirstValue("sub");
|
||||||
});
|
if (userId == null) return Unauthorized();
|
||||||
|
|
||||||
|
if (await FindMonographAsync(userId, monograph) != null) return base.Conflict("This monograph is already published.");
|
||||||
|
|
||||||
|
if (monograph.EncryptedContent == null)
|
||||||
|
monograph.CompressedContent = monograph.Content.CompressBrotli();
|
||||||
|
monograph.UserId = userId;
|
||||||
|
monograph.DatePublished = DateTimeOffset.UtcNow.ToUnixTimeMilliseconds();
|
||||||
|
|
||||||
|
if (monograph.EncryptedContent?.Cipher.Length > MAX_DOC_SIZE || monograph.CompressedContent?.Length > MAX_DOC_SIZE)
|
||||||
|
return base.BadRequest("Monograph is too big. Max allowed size is 15mb.");
|
||||||
|
|
||||||
|
await Monographs.InsertAsync(monograph);
|
||||||
|
|
||||||
|
return Ok(new
|
||||||
|
{
|
||||||
|
id = monograph.ItemId
|
||||||
|
});
|
||||||
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
await Slogger<MonographsController>.Error(nameof(PublishAsync), e.ToString());
|
||||||
|
return BadRequest();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
[HttpPatch]
|
[HttpPatch]
|
||||||
public async Task<IActionResult> UpdateAsync([FromBody] Monograph monograph)
|
public async Task<IActionResult> UpdateAsync([FromBody] Monograph monograph)
|
||||||
{
|
{
|
||||||
if (await Monographs.GetAsync(monograph.Id) == null) return NotFound();
|
try
|
||||||
|
|
||||||
if (monograph.EncryptedContent?.Cipher.Length > MAX_DOC_SIZE || monograph.CompressedContent?.Length > MAX_DOC_SIZE)
|
|
||||||
return base.BadRequest("Monograph is too big. Max allowed size is 15mb.");
|
|
||||||
|
|
||||||
if (monograph.EncryptedContent == null)
|
|
||||||
monograph.CompressedContent = monograph.Content.CompressBrotli();
|
|
||||||
else
|
|
||||||
monograph.Content = null;
|
|
||||||
|
|
||||||
monograph.DatePublished = DateTimeOffset.UtcNow.ToUnixTimeMilliseconds();
|
|
||||||
Monographs.Update(monograph.Id, monograph);
|
|
||||||
|
|
||||||
if (!await unit.Commit()) return BadRequest();
|
|
||||||
return Ok(new
|
|
||||||
{
|
{
|
||||||
id = monograph.Id
|
var userId = this.User.FindFirstValue("sub");
|
||||||
});
|
if (userId == null) return Unauthorized();
|
||||||
|
|
||||||
|
if (await FindMonographAsync(userId, monograph) == null) return NotFound();
|
||||||
|
|
||||||
|
if (monograph.EncryptedContent?.Cipher.Length > MAX_DOC_SIZE || monograph.CompressedContent?.Length > MAX_DOC_SIZE)
|
||||||
|
return base.BadRequest("Monograph is too big. Max allowed size is 15mb.");
|
||||||
|
|
||||||
|
if (monograph.EncryptedContent == null)
|
||||||
|
monograph.CompressedContent = monograph.Content.CompressBrotli();
|
||||||
|
else
|
||||||
|
monograph.Content = null;
|
||||||
|
|
||||||
|
monograph.DatePublished = DateTimeOffset.UtcNow.ToUnixTimeMilliseconds();
|
||||||
|
var result = await Monographs.Collection.UpdateOneAsync(
|
||||||
|
CreateMonographFilter(userId, monograph),
|
||||||
|
Builders<Monograph>.Update
|
||||||
|
.Set(m => m.DatePublished, monograph.DatePublished)
|
||||||
|
.Set(m => m.CompressedContent, monograph.CompressedContent)
|
||||||
|
.Set(m => m.EncryptedContent, monograph.EncryptedContent)
|
||||||
|
.Set(m => m.SelfDestruct, monograph.SelfDestruct)
|
||||||
|
.Set(m => m.Title, monograph.Title)
|
||||||
|
);
|
||||||
|
if (!result.IsAcknowledged) return BadRequest();
|
||||||
|
|
||||||
|
return Ok(new
|
||||||
|
{
|
||||||
|
id = monograph.ItemId
|
||||||
|
});
|
||||||
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
await Slogger<MonographsController>.Error(nameof(UpdateAsync), e.ToString());
|
||||||
|
return BadRequest();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
[HttpGet]
|
[HttpGet]
|
||||||
@@ -103,17 +173,16 @@ namespace Notesnook.API.Controllers
|
|||||||
|
|
||||||
var monographs = (await Monographs.Collection.FindAsync(Builders<Monograph>.Filter.Eq("UserId", userId), new FindOptions<Monograph, ObjectWithId>
|
var monographs = (await Monographs.Collection.FindAsync(Builders<Monograph>.Filter.Eq("UserId", userId), new FindOptions<Monograph, ObjectWithId>
|
||||||
{
|
{
|
||||||
Projection = Builders<Monograph>.Projection.Include("_id"),
|
Projection = Builders<Monograph>.Projection.Include("_id").Include("ItemId"),
|
||||||
})).ToEnumerable();
|
})).ToEnumerable();
|
||||||
return Ok(monographs.Select((m) => m.Id));
|
return Ok(monographs.Select((m) => m.ItemId ?? m.Id));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
[HttpGet("{id}")]
|
[HttpGet("{id}")]
|
||||||
[AllowAnonymous]
|
[AllowAnonymous]
|
||||||
public async Task<IActionResult> GetMonographAsync([FromRoute] string id)
|
public async Task<IActionResult> GetMonographAsync([FromRoute] string id)
|
||||||
{
|
{
|
||||||
var monograph = await Monographs.GetAsync(id);
|
var monograph = await FindMonographAsync(id);
|
||||||
if (monograph == null)
|
if (monograph == null)
|
||||||
{
|
{
|
||||||
return NotFound(new
|
return NotFound(new
|
||||||
@@ -125,6 +194,7 @@ namespace Notesnook.API.Controllers
|
|||||||
|
|
||||||
if (monograph.EncryptedContent == null)
|
if (monograph.EncryptedContent == null)
|
||||||
monograph.Content = monograph.CompressedContent.DecompressBrotli();
|
monograph.Content = monograph.CompressedContent.DecompressBrotli();
|
||||||
|
if (monograph.ItemId == null) monograph.ItemId = monograph.Id;
|
||||||
return Ok(monograph);
|
return Ok(monograph);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -132,7 +202,7 @@ namespace Notesnook.API.Controllers
|
|||||||
[AllowAnonymous]
|
[AllowAnonymous]
|
||||||
public async Task<IActionResult> TrackView([FromRoute] string id)
|
public async Task<IActionResult> TrackView([FromRoute] string id)
|
||||||
{
|
{
|
||||||
var monograph = await Monographs.GetAsync(id);
|
var monograph = await FindMonographAsync(id);
|
||||||
if (monograph == null) return Content(SVG_PIXEL, "image/svg+xml");
|
if (monograph == null) return Content(SVG_PIXEL, "image/svg+xml");
|
||||||
|
|
||||||
if (monograph.SelfDestruct)
|
if (monograph.SelfDestruct)
|
||||||
@@ -144,8 +214,7 @@ namespace Notesnook.API.Controllers
|
|||||||
[HttpDelete("{id}")]
|
[HttpDelete("{id}")]
|
||||||
public async Task<IActionResult> DeleteAsync([FromRoute] string id)
|
public async Task<IActionResult> DeleteAsync([FromRoute] string id)
|
||||||
{
|
{
|
||||||
Monographs.DeleteById(id);
|
await Monographs.Collection.DeleteOneAsync(CreateMonographFilter(id));
|
||||||
if (!await unit.Commit()) return BadRequest();
|
|
||||||
return Ok();
|
return Ok();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -20,27 +20,50 @@ along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|||||||
using System.Text.Json.Serialization;
|
using System.Text.Json.Serialization;
|
||||||
using MongoDB.Bson;
|
using MongoDB.Bson;
|
||||||
using MongoDB.Bson.Serialization.Attributes;
|
using MongoDB.Bson.Serialization.Attributes;
|
||||||
using Notesnook.API.Interfaces;
|
using System.Runtime.Serialization;
|
||||||
|
|
||||||
namespace Notesnook.API.Models
|
namespace Notesnook.API.Models
|
||||||
{
|
{
|
||||||
public class ObjectWithId
|
public class ObjectWithId
|
||||||
{
|
{
|
||||||
[BsonId]
|
[BsonId]
|
||||||
|
[BsonIgnoreIfDefault]
|
||||||
[BsonRepresentation(BsonType.ObjectId)]
|
[BsonRepresentation(BsonType.ObjectId)]
|
||||||
public string Id { get; set; }
|
public string Id
|
||||||
|
{
|
||||||
|
get; set;
|
||||||
|
}
|
||||||
|
|
||||||
|
public string ItemId
|
||||||
|
{
|
||||||
|
get; set;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public class Monograph : IMonograph
|
public class Monograph
|
||||||
{
|
{
|
||||||
public Monograph()
|
public Monograph()
|
||||||
{
|
{
|
||||||
Id = ObjectId.GenerateNewId().ToString();
|
Id = ObjectId.GenerateNewId().ToString();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[DataMember(Name = "id")]
|
||||||
|
[JsonPropertyName("id")]
|
||||||
|
[MessagePack.Key("id")]
|
||||||
|
public string ItemId
|
||||||
|
{
|
||||||
|
get; set;
|
||||||
|
}
|
||||||
|
|
||||||
[BsonId]
|
[BsonId]
|
||||||
|
[BsonIgnoreIfDefault]
|
||||||
[BsonRepresentation(BsonType.ObjectId)]
|
[BsonRepresentation(BsonType.ObjectId)]
|
||||||
public string Id { get; set; }
|
[JsonIgnore]
|
||||||
|
[MessagePack.IgnoreMember]
|
||||||
|
public string Id
|
||||||
|
{
|
||||||
|
get; set;
|
||||||
|
}
|
||||||
|
|
||||||
[JsonPropertyName("title")]
|
[JsonPropertyName("title")]
|
||||||
public string Title { get; set; }
|
public string Title { get; set; }
|
||||||
|
|||||||
Reference in New Issue
Block a user