Zeeshaun 9b646f501c
All checks were successful
Deploy Promiscuity Auth API / deploy (push) Successful in 47s
Deploy Promiscuity Character API / deploy (push) Successful in 57s
Deploy Promiscuity Locations API / deploy (push) Successful in 45s
k8s smoke test / test (push) Successful in 7s
Adding loggin to troubleshoot location generation issue
2026-03-13 13:59:04 -05:00

134 lines
5.1 KiB
C#

using CharacterApi.Models;
using MongoDB.Bson;
using MongoDB.Driver;
namespace CharacterApi.Services;
public class CharacterStore
{
private readonly IMongoCollection<Character> _col;
private readonly IMongoCollection<VisibleLocation> _locations;
public sealed record VisibleLocationResult(List<VisibleLocation> Locations, int GeneratedCount);
public CharacterStore(IConfiguration cfg)
{
var cs = cfg["MongoDB:ConnectionString"] ?? "mongodb://127.0.0.1:27017";
var dbName = cfg["MongoDB:DatabaseName"] ?? "GameDb";
var client = new MongoClient(cs);
var db = client.GetDatabase(dbName);
_col = db.GetCollection<Character>("Characters");
_locations = db.GetCollection<VisibleLocation>("Locations");
var ownerIndex = Builders<Character>.IndexKeys.Ascending(c => c.OwnerUserId);
_col.Indexes.CreateOne(new CreateIndexModel<Character>(ownerIndex));
}
public Task CreateAsync(Character character) => _col.InsertOneAsync(character);
public Task<List<Character>> GetForOwnerAsync(string ownerUserId) =>
_col.Find(c => c.OwnerUserId == ownerUserId).ToListAsync();
public async Task<Character?> GetForOwnerByIdAsync(string id, string ownerUserId, bool allowAnyOwner)
{
var filter = Builders<Character>.Filter.Eq(c => c.Id, id);
if (!allowAnyOwner)
{
filter = Builders<Character>.Filter.And(
filter,
Builders<Character>.Filter.Eq(c => c.OwnerUserId, ownerUserId)
);
}
return await _col.Find(filter).FirstOrDefaultAsync();
}
public Task<List<VisibleLocation>> GetVisibleLocationsAsync(Character character)
{
return GetVisibleLocationsInternalAsync(character, ensureGenerated: false);
}
public async Task<VisibleLocationResult> GetOrCreateVisibleLocationsAsync(Character character)
{
var generatedCount = await EnsureVisibleLocationsExistAsync(character);
var locations = await GetVisibleLocationsInternalAsync(character, ensureGenerated: true);
return new VisibleLocationResult(locations, generatedCount);
}
private Task<List<VisibleLocation>> GetVisibleLocationsInternalAsync(Character character, bool ensureGenerated)
{
var radius = character.VisionRadius > 0 ? character.VisionRadius : 3;
var minX = character.Coord.X - radius;
var maxX = character.Coord.X + radius;
var minY = character.Coord.Y - radius;
var maxY = character.Coord.Y + radius;
var filter = Builders<VisibleLocation>.Filter.And(
Builders<VisibleLocation>.Filter.Gte(l => l.Coord.X, minX),
Builders<VisibleLocation>.Filter.Lte(l => l.Coord.X, maxX),
Builders<VisibleLocation>.Filter.Gte(l => l.Coord.Y, minY),
Builders<VisibleLocation>.Filter.Lte(l => l.Coord.Y, maxY)
);
return _locations.Find(filter).ToListAsync();
}
private async Task<int> EnsureVisibleLocationsExistAsync(Character character)
{
var radius = character.VisionRadius > 0 ? character.VisionRadius : 3;
var generatedCount = 0;
for (var x = character.Coord.X - radius; x <= character.Coord.X + radius; x++)
{
for (var y = character.Coord.Y - radius; y <= character.Coord.Y + radius; y++)
{
var filter = Builders<VisibleLocation>.Filter.And(
Builders<VisibleLocation>.Filter.Eq(l => l.Coord.X, x),
Builders<VisibleLocation>.Filter.Eq(l => l.Coord.Y, y)
);
var update = Builders<VisibleLocation>.Update
.SetOnInsert(l => l.Name, DefaultLocationName(x, y))
.SetOnInsert(l => l.Coord, new LocationCoord { X = x, Y = y })
.SetOnInsert(l => l.Id, ObjectId.GenerateNewId().ToString())
.SetOnInsert("createdUtc", DateTime.UtcNow);
try
{
var result = await _locations.UpdateOneAsync(filter, update, new UpdateOptions { IsUpsert = true });
if (result.UpsertedId is not null)
generatedCount += 1;
}
catch (MongoWriteException ex) when (ex.WriteError.Category == ServerErrorCategory.DuplicateKey)
{
// Another request or service instance created it first.
}
}
}
return generatedCount;
}
private static string DefaultLocationName(int x, int y)
{
if (x == 0 && y == 0)
return "Origin";
return $"Location {x},{y}";
}
public async Task<bool> DeleteForOwnerAsync(string id, string ownerUserId, bool allowAnyOwner)
{
var filter = Builders<Character>.Filter.Eq(c => c.Id, id);
if (!allowAnyOwner)
{
filter = Builders<Character>.Filter.And(
filter,
Builders<Character>.Filter.Eq(c => c.OwnerUserId, ownerUserId)
);
}
var result = await _col.DeleteOneAsync(filter);
return result.DeletedCount > 0;
}
}