Обновление не работает для отношений «многие ко многим». Entity Framework Core 2.2 - PullRequest
0 голосов
/ 29 января 2020

У меня много-много отношений между играми и категориями. Я использую Sql Сервер . Одна игра может иметь несколько категорий. Операция создания работает, но обновление не работает. Получение ошибки:

InnerException: Нарушение ограничения PRIMARY KEY 'PK_GameCategories'. Невозможно вставить дубликат ключа в объект «dbo.GameCategories». Значение ключа-дубликата равно (1003, 1). \ R \ nОпределение завершено.

public class Game
{
    [Key, DatabaseGenerated(DatabaseGeneratedOption.Identity)]
    public int Id { get; set; }

    [Display(Name = "Game Name")]
    public string Name { get; set; }

    [Display(Name = "Game Categories")]
    public ICollection<GameCategory> GameCategories { get; set; }
}

public class Category
{
    [Key, DatabaseGenerated(DatabaseGeneratedOption.Identity)]
    public int Id { get; set; }

    [Display(Name = "Game Category")]
    public string Name { get; set; }

    [Display(Name = "Game Categories")]
    public ICollection<GameCategory> GameCategories { get; set; }
}

public class GameCategory
{
    public int GameId { get; set; }
    public Game Game { get; set; }
    public int CategoryId { get; set; }
    public Category Category { get; set; }
}

// --------------------------
GameController/Edit Action [Working Perfectly]
// -------------------------
[HttpGet]
[Route("edit-game/{id}")]
public async Task<IActionResult> EditGame(int id)
{
      try
      {
                if (id == null) return NotFound();
                var gameInDb = await _db.Games.FirstOrDefaultAsync(g => g.Id == id);
                if (gameInDb == null) return NotFound();

                var model = new GameChangeViewModel()
                {
                    Title = $"Edit {gameInDb.Name}",
                    Name = gameInDb.Name,
                    // Displaying list of categories
                    GameCategoryList = _db.Categories
                        .Select(c => new SelectListItem() { Text = c.Name, Value = c.Id.ToString()}).ToList(),
                    // loading selected values
                    GameCategoryId = _db.GameCategories.Where(c=>c.GameId == gameInDb.Id).Select(s=>s.CategoryId).ToList()
                };

                return View(model);
            }
            catch (Exception e)
            {
                Console.WriteLine(e);
                return NotFound();
            }
        }

Приведенный ниже код не работает и вызывает указанное выше исключение.

[HttpPost]
        [Route("edit-game-post")]
        [ValidateAntiForgeryToken]
        public async Task<IActionResult> EditGamePost(GameChangeViewModel model)
        {
            try
            {
                if (!ModelState.IsValid) return View(nameof(EditGame), model);

                var gameInDb = await _db.Games.FirstOrDefaultAsync(g => g.Id == model.Id);
                if (gameInDb == null)
                {
                    ModelState.AddModelError("", "No Data");
                    return View(nameof(EditGame), model);
                }

                _db.GameCategories.RemoveRange(gameInDb.GameCategories);
                _db.SaveChanges();

                gameInDb.Name = model.Name;
                var category = _db.Categories.Where(c => model.GameCategoryId.Contains(c.Id)).ToList();
                foreach (Category cat in category)
                {
                    gameInDb.GameCategories.Add(new GameCategory()
                    {
                        Category = cat, Game = gameInDb
                    });
                }

                await _db.SaveChangesAsync();
                return RedirectToAction(nameof(Index));

            }
            catch (Exception e)
            {
                Console.WriteLine(e);
                ModelState.AddModelError("", "Exception Happen");
                return View(nameof(EditGame), model);
            }
        }


К сожалению, я не нашел подходящих операций CRUD для отношений «многие ко многим». Спасибо

1 Ответ

2 голосов
/ 30 января 2020

Попробуйте создать новый идентификатор столбца в качестве первичного ключа в таблице GameCategory.

...