mirror of
https://github.com/rjNemo/ticket_manager
synced 2026-06-06 08:46:39 +00:00
pull latest version
This commit is contained in:
commit
1856c60bca
27 changed files with 566 additions and 456 deletions
26
.gitignore
vendored
26
.gitignore
vendored
|
|
@ -6,5 +6,29 @@ Migrations/
|
||||||
app.db*
|
app.db*
|
||||||
.DS_Store
|
.DS_Store
|
||||||
app.db
|
app.db
|
||||||
|
Data/Interfaces
|
||||||
|
Data/UnitOfWork.cs
|
||||||
|
Data/*Repository.cs
|
||||||
|
|
||||||
|
# client
|
||||||
|
client/src/pages/TestPage.tsx
|
||||||
|
client/react-app-env.d.ts
|
||||||
client/node_modules
|
client/node_modules
|
||||||
client/src/pages/TestPage.tsx
|
client/.pnp
|
||||||
|
client/.pnp.js
|
||||||
|
|
||||||
|
# testing
|
||||||
|
client/coverage
|
||||||
|
|
||||||
|
# production
|
||||||
|
client/build
|
||||||
|
|
||||||
|
# misc
|
||||||
|
client/.DS_Store
|
||||||
|
client/.env.local
|
||||||
|
client/.env.development.local
|
||||||
|
client/.env.test.local
|
||||||
|
client/.env.production.local
|
||||||
|
client/npm-debug.log*
|
||||||
|
client/yarn-debug.log*
|
||||||
|
client/yarn-error.log*
|
||||||
|
|
|
||||||
|
|
@ -1,38 +1,78 @@
|
||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
|
using System.Linq;
|
||||||
using Microsoft.AspNetCore.Mvc;
|
using Microsoft.AspNetCore.Mvc;
|
||||||
using Microsoft.EntityFrameworkCore;
|
using Microsoft.EntityFrameworkCore;
|
||||||
|
using Microsoft.AspNetCore.Authorization;
|
||||||
|
using Microsoft.AspNetCore.Http;
|
||||||
using TicketManager.Data;
|
using TicketManager.Data;
|
||||||
using TicketManager.Models;
|
using TicketManager.Models;
|
||||||
using Microsoft.AspNetCore.Authorization;
|
using TicketManager.DTO;
|
||||||
|
|
||||||
|
|
||||||
namespace TicketManager.Controllers
|
namespace TicketManager.Controllers
|
||||||
{
|
{
|
||||||
// [Authorize]
|
// [Authorize]
|
||||||
|
[Produces("application/json")]
|
||||||
[Route("api/v1/users")]
|
[Route("api/v1/users")]
|
||||||
[ApiController]
|
[ApiController]
|
||||||
public class UsersController : ControllerBase
|
public class UsersController : ControllerBase
|
||||||
{
|
{
|
||||||
private readonly IAppUserRepository _users;
|
private readonly AppDbContext _context;
|
||||||
|
|
||||||
public UsersController(IAppUserRepository users)
|
public UsersController(AppDbContext context)
|
||||||
{
|
{
|
||||||
_users = users;
|
_context = context;
|
||||||
}
|
}
|
||||||
|
|
||||||
// GET: api/Users
|
/// <summary>
|
||||||
|
/// Returns all Users stored in the database.
|
||||||
|
/// </summary>
|
||||||
|
/// <remarks>
|
||||||
|
/// Sample request:
|
||||||
|
///
|
||||||
|
/// GET: api/v1/Users
|
||||||
|
///
|
||||||
|
/// </remarks>
|
||||||
|
/// <response code="200">Returns a list of users</response>
|
||||||
[HttpGet]
|
[HttpGet]
|
||||||
public async Task<IEnumerable<AppUser>> GetUsers()
|
[ProducesResponseType(StatusCodes.Status200OK)]
|
||||||
|
public async Task<IEnumerable<AppUserDTO>> GetUsers()
|
||||||
{
|
{
|
||||||
return await _users.List();
|
return await _context.AppUsers
|
||||||
|
.Include(u => u.Assignments)
|
||||||
|
.ThenInclude(a => a.Project)
|
||||||
|
.Include(u => u.Activities)
|
||||||
|
.AsNoTracking()
|
||||||
|
.Select(u => new AppUserDTO(u))
|
||||||
|
.ToListAsync();
|
||||||
}
|
}
|
||||||
|
|
||||||
// GET: api/Users/5
|
/// <summary>
|
||||||
|
/// Locate a specific User stored in the database by its Id
|
||||||
|
/// </summary>
|
||||||
|
/// <remarks>
|
||||||
|
/// Sample request:
|
||||||
|
///
|
||||||
|
/// GET: api/v1/Users/2
|
||||||
|
///
|
||||||
|
/// </remarks>
|
||||||
|
/// <response code="200">Returns a User object</response>
|
||||||
|
/// <response code="404">If the required User is null</response>
|
||||||
[HttpGet("{id}")]
|
[HttpGet("{id}")]
|
||||||
public async Task<ActionResult<AppUser>> GetUser(Guid id)
|
[ProducesResponseType(StatusCodes.Status200OK)]
|
||||||
|
[ProducesResponseType(StatusCodes.Status404NotFound)]
|
||||||
|
public async Task<ActionResult<AppUserDTO>> GetUser(Guid id)
|
||||||
{
|
{
|
||||||
var user = await _users.GetUser(id);
|
var user = await _context.AppUsers
|
||||||
|
.Include(u => u.Assignments)
|
||||||
|
.ThenInclude(a => a.Project)
|
||||||
|
.Include(u => u.Activities)
|
||||||
|
.AsNoTracking()
|
||||||
|
.Select(u => new AppUserDTO(u))
|
||||||
|
.FirstOrDefaultAsync(u => u.Id == id);
|
||||||
|
|
||||||
if (user == null)
|
if (user == null)
|
||||||
{
|
{
|
||||||
return NotFound();
|
return NotFound();
|
||||||
|
|
@ -40,23 +80,44 @@ namespace TicketManager.Controllers
|
||||||
return user;
|
return user;
|
||||||
}
|
}
|
||||||
|
|
||||||
// PUT: api/Users/5
|
/// <summary>
|
||||||
// To protect from overposting attacks, please enable the specific properties you want to bind to, for
|
/// Updates the specific project with Id.
|
||||||
// more details see https://aka.ms/RazorPagesCRUD.
|
/// </summary>
|
||||||
|
/// <remarks>
|
||||||
|
/// Sample request:
|
||||||
|
///
|
||||||
|
/// PUT: api/v1/Projects/3
|
||||||
|
/// {
|
||||||
|
/// "id": "357727fd-5262-4522-b8a3-38271d43de84",
|
||||||
|
/// "firstName": "Thomas",
|
||||||
|
/// "lastName": "Price",
|
||||||
|
/// "presentation": "New Team?!",
|
||||||
|
/// "email": "tp@mail.com",
|
||||||
|
/// "phone": "0198237645"
|
||||||
|
/// }
|
||||||
|
///
|
||||||
|
/// </remarks>
|
||||||
|
/// <response code="204">Request was succesful but no content is changed</response>
|
||||||
|
/// <response code="404">If the required project is null</response>
|
||||||
[HttpPut("{id}")]
|
[HttpPut("{id}")]
|
||||||
|
[ProducesResponseType(StatusCodes.Status204NoContent)]
|
||||||
|
[ProducesResponseType(StatusCodes.Status404NotFound)]
|
||||||
public async Task<IActionResult> PutUser(Guid id, AppUser user)
|
public async Task<IActionResult> PutUser(Guid id, AppUser user)
|
||||||
{
|
{
|
||||||
if (id != user.Id)
|
if (id != user.Id)
|
||||||
{
|
{
|
||||||
return BadRequest();
|
return BadRequest();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
_context.Entry(user).State = EntityState.Modified;
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
await _users.Update(user);
|
await _context.SaveChangesAsync();
|
||||||
}
|
}
|
||||||
catch (DbUpdateConcurrencyException)
|
catch (DbUpdateConcurrencyException)
|
||||||
{
|
{
|
||||||
if (!_users.Exists(id))
|
if (!UserExists(id))
|
||||||
{
|
{
|
||||||
return NotFound();
|
return NotFound();
|
||||||
}
|
}
|
||||||
|
|
@ -68,49 +129,101 @@ namespace TicketManager.Controllers
|
||||||
return NoContent();
|
return NoContent();
|
||||||
}
|
}
|
||||||
|
|
||||||
// POST: api/Users
|
/// <summary>
|
||||||
// To protect from overposting attacks, please enable the specific properties you want to bind to, for
|
/// Creates a project.
|
||||||
// more details see https://aka.ms/RazorPagesCRUD.
|
/// </summary>
|
||||||
|
/// <remarks>
|
||||||
|
/// Sample request:
|
||||||
|
///
|
||||||
|
/// POST: api/v1/Projects/
|
||||||
|
/// {
|
||||||
|
/// "firstName": "Thomas",
|
||||||
|
/// "lastName": "Price",
|
||||||
|
/// "presentation": "New Team?!",
|
||||||
|
/// "email": "tp@mail.com",
|
||||||
|
/// "phone": "0198237645"
|
||||||
|
/// }
|
||||||
|
///
|
||||||
|
/// </remarks>
|
||||||
|
/// <response code="201">Returns the created project</response>
|
||||||
[HttpPost]
|
[HttpPost]
|
||||||
public async Task<ActionResult<AppUser>> PostUser(AppUser user)
|
[ProducesResponseType(StatusCodes.Status201Created)]
|
||||||
|
[ProducesResponseType(StatusCodes.Status404NotFound)]
|
||||||
|
public async Task<ActionResult<AppUserDTO>> PostUser(AppUser user)
|
||||||
{
|
{
|
||||||
await _users.Add(user);
|
if (!ModelState.IsValid)
|
||||||
return CreatedAtAction("GetUser", new { id = user.Id }, user);
|
{
|
||||||
|
return BadRequest();
|
||||||
|
}
|
||||||
|
|
||||||
|
_context.AppUsers.Add(user);
|
||||||
|
await _context.SaveChangesAsync();
|
||||||
|
|
||||||
|
var dto = new AppUserDTO(user);
|
||||||
|
|
||||||
|
return CreatedAtAction("GetUser", new { id = user.Id }, dto);
|
||||||
}
|
}
|
||||||
|
|
||||||
// DELETE: api/Users/5
|
/// <summary>
|
||||||
|
/// Deletes the project identified by its Id
|
||||||
|
/// </summary>
|
||||||
|
/// <remarks>
|
||||||
|
/// Sample request:
|
||||||
|
///
|
||||||
|
/// DELETE: api/v1/Projects/5
|
||||||
|
///
|
||||||
|
/// </remarks>
|
||||||
|
/// <response code="200">Returns the deleted project</response>
|
||||||
|
[ProducesResponseType(StatusCodes.Status200OK)]
|
||||||
|
[ProducesResponseType(StatusCodes.Status404NotFound)]
|
||||||
[HttpDelete("{id}")]
|
[HttpDelete("{id}")]
|
||||||
public async Task<ActionResult<AppUser>> DeleteUser(Guid id)
|
public async Task<ActionResult<AppUserDTO>> DeleteUser(Guid id)
|
||||||
{
|
{
|
||||||
var user = await _users.GetUser(id);
|
var user = await _context.AppUsers.FindAsync(id);
|
||||||
if (user == null)
|
if (user == null)
|
||||||
{
|
{
|
||||||
return NotFound();
|
return NotFound();
|
||||||
}
|
}
|
||||||
await _users.Delete(user);
|
_context.AppUsers.Remove(user);
|
||||||
return user;
|
await _context.SaveChangesAsync();
|
||||||
|
var dto = new AppUserDTO(user);
|
||||||
|
return Ok(user);
|
||||||
}
|
}
|
||||||
|
|
||||||
[HttpGet("{id}/projects")]
|
[HttpGet("{id}/projects")]
|
||||||
public async Task<ActionResult<IEnumerable<Project>>> GetAppUserProjects(Guid id)
|
public async Task<ActionResult<IEnumerable<ProjectDTO>>> GetAppUserProjects(Guid id)
|
||||||
{
|
{
|
||||||
AppUser user = await _users.GetUser(id);
|
var user = await _context.AppUsers
|
||||||
|
.Include(u => u.Assignments)
|
||||||
|
.ThenInclude(a => a.Project)
|
||||||
|
.AsNoTracking()
|
||||||
|
.FirstOrDefaultAsync(u => u.Id == id);
|
||||||
if (user == null)
|
if (user == null)
|
||||||
{
|
{
|
||||||
return BadRequest();
|
return BadRequest();
|
||||||
}
|
}
|
||||||
return user.GetProjects();
|
return user.GetProjects().Select(p => new ProjectDTO(p)).ToList();
|
||||||
}
|
}
|
||||||
|
|
||||||
[HttpGet("{id}/tickets/")]
|
[HttpGet("{id}/tickets/")]
|
||||||
public async Task<ActionResult<IEnumerable<Ticket>>> GetAppUserTickets(Guid id)
|
public async Task<ActionResult<IEnumerable<TicketDTO>>> GetAppUserTickets(Guid id)
|
||||||
{
|
{
|
||||||
AppUser user = await _users.GetUser(id);
|
var user = await _context.AppUsers
|
||||||
|
.Include(u => u.Assignments)
|
||||||
|
.ThenInclude(a => a.Project)
|
||||||
|
.ThenInclude(p => p.Tickets)
|
||||||
|
.AsNoTracking()
|
||||||
|
.FirstOrDefaultAsync(u => u.Id == id);
|
||||||
if (user == null)
|
if (user == null)
|
||||||
{
|
{
|
||||||
return BadRequest();
|
return BadRequest();
|
||||||
}
|
}
|
||||||
return user.GetTickets();
|
return user.GetTickets().Select(t => new TicketDTO(t)).ToList();
|
||||||
|
}
|
||||||
|
|
||||||
|
private bool UserExists(Guid id)
|
||||||
|
{
|
||||||
|
return _context.AppUsers.Any(e => e.Id == id);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -18,10 +18,11 @@ namespace TicketManager.Controllers
|
||||||
[ApiController]
|
[ApiController]
|
||||||
public class ProjectsController : ControllerBase
|
public class ProjectsController : ControllerBase
|
||||||
{
|
{
|
||||||
private IProjectRepository _projects;
|
private readonly AppDbContext _context;
|
||||||
public ProjectsController(IProjectRepository context)
|
|
||||||
|
public ProjectsController(AppDbContext context)
|
||||||
{
|
{
|
||||||
_projects = context;
|
_context = context;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
|
|
@ -36,9 +37,18 @@ namespace TicketManager.Controllers
|
||||||
/// <response code="200">Returns a list of projects</response>
|
/// <response code="200">Returns a list of projects</response>
|
||||||
[HttpGet]
|
[HttpGet]
|
||||||
[ProducesResponseType(StatusCodes.Status200OK)]
|
[ProducesResponseType(StatusCodes.Status200OK)]
|
||||||
public async Task<IEnumerable<Project>> GetProjects()
|
public async Task<IEnumerable<ProjectDTO>> GetProjects()
|
||||||
{
|
{
|
||||||
return await _projects.List();
|
return await _context.Projects
|
||||||
|
.Include(p => p.Assignments)
|
||||||
|
.ThenInclude(a => a.User)
|
||||||
|
.Include(p => p.Tickets)
|
||||||
|
.Include(p => p.Manager)
|
||||||
|
.Include(p => p.Files)
|
||||||
|
.Include(p => p.Activities)
|
||||||
|
.AsNoTracking()
|
||||||
|
.Select(p => new ProjectDTO(p))
|
||||||
|
.ToListAsync();
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
|
|
@ -58,12 +68,22 @@ namespace TicketManager.Controllers
|
||||||
[ProducesResponseType(StatusCodes.Status404NotFound)]
|
[ProducesResponseType(StatusCodes.Status404NotFound)]
|
||||||
public async Task<ActionResult<ProjectDTO>> GetProject(int id)
|
public async Task<ActionResult<ProjectDTO>> GetProject(int id)
|
||||||
{
|
{
|
||||||
Project project = await _projects.Get(id);
|
var project = await _context.Projects
|
||||||
|
.Include(p => p.Assignments)
|
||||||
|
.ThenInclude(a => a.User)
|
||||||
|
.Include(p => p.Tickets)
|
||||||
|
.Include(p => p.Manager)
|
||||||
|
.Include(p => p.Files)
|
||||||
|
.Include(p => p.Activities)
|
||||||
|
.AsNoTracking()
|
||||||
|
.Select(p => new ProjectDTO(p))
|
||||||
|
.FirstOrDefaultAsync(p => p.Id == id);
|
||||||
|
|
||||||
if (project == null)
|
if (project == null)
|
||||||
{
|
{
|
||||||
return NotFound();
|
return NotFound();
|
||||||
}
|
}
|
||||||
return new ProjectDTO(project);
|
return project;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
|
|
@ -90,15 +110,25 @@ namespace TicketManager.Controllers
|
||||||
[ProducesResponseType(StatusCodes.Status404NotFound)]
|
[ProducesResponseType(StatusCodes.Status404NotFound)]
|
||||||
public async Task<IActionResult> PutProject(int id, Project project)
|
public async Task<IActionResult> PutProject(int id, Project project)
|
||||||
{
|
{
|
||||||
if (id != project.Id) { return BadRequest(); }
|
if (id != project.Id)
|
||||||
|
{
|
||||||
|
return BadRequest();
|
||||||
|
}
|
||||||
|
_context.Entry(project).State = EntityState.Modified;
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
await _projects.Update(project);
|
await _context.SaveChangesAsync();
|
||||||
}
|
}
|
||||||
catch (DbUpdateConcurrencyException)
|
catch (DbUpdateConcurrencyException)
|
||||||
{
|
{
|
||||||
if (!_projects.Exists(id)) { return NotFound(); }
|
if (!ProjectExists(id))
|
||||||
else { throw; }
|
{
|
||||||
|
return NotFound();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
throw;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return NoContent();
|
return NoContent();
|
||||||
}
|
}
|
||||||
|
|
@ -123,11 +153,16 @@ namespace TicketManager.Controllers
|
||||||
[HttpPost]
|
[HttpPost]
|
||||||
[ProducesResponseType(StatusCodes.Status201Created)]
|
[ProducesResponseType(StatusCodes.Status201Created)]
|
||||||
[ProducesResponseType(StatusCodes.Status404NotFound)]
|
[ProducesResponseType(StatusCodes.Status404NotFound)]
|
||||||
public async Task<ActionResult<Project>> PostProject(Project project)
|
public async Task<ActionResult<ProjectDTO>> PostProject(Project project)
|
||||||
{
|
{
|
||||||
if (!ModelState.IsValid) { return BadRequest(); }
|
if (!ModelState.IsValid)
|
||||||
await _projects.Add(project);
|
{
|
||||||
return CreatedAtAction("GetProject", new { id = project.Id }, project);
|
return BadRequest();
|
||||||
|
}
|
||||||
|
_context.Projects.Add(project);
|
||||||
|
await _context.SaveChangesAsync();
|
||||||
|
var dto = new ProjectDTO(project);
|
||||||
|
return CreatedAtAction("GetProject", new { id = project.Id }, dto);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
|
|
@ -145,13 +180,15 @@ namespace TicketManager.Controllers
|
||||||
[HttpDelete("{id}")]
|
[HttpDelete("{id}")]
|
||||||
public async Task<IActionResult> DeleteProject(int id)
|
public async Task<IActionResult> DeleteProject(int id)
|
||||||
{
|
{
|
||||||
var project = await _projects.Get(id);
|
var project = await _context.Projects.FindAsync(id);
|
||||||
if (project == null)
|
if (project == null)
|
||||||
{
|
{
|
||||||
return NotFound();
|
return NotFound();
|
||||||
}
|
}
|
||||||
await _projects.Delete(project);
|
_context.Projects.Remove(project);
|
||||||
return Ok();
|
await _context.SaveChangesAsync();
|
||||||
|
var dto = new ProjectDTO(project);
|
||||||
|
return Ok(dto);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
|
|
@ -169,9 +206,20 @@ namespace TicketManager.Controllers
|
||||||
[HttpGet("{id}/members")]
|
[HttpGet("{id}/members")]
|
||||||
public async Task<ActionResult<List<AppUser>>> GetProjectMembers(int id)
|
public async Task<ActionResult<List<AppUser>>> GetProjectMembers(int id)
|
||||||
{
|
{
|
||||||
var project = await _projects.Get(id);
|
Project project = await _context.Projects
|
||||||
|
.Include(p => p.Assignments)
|
||||||
|
.ThenInclude(a => a.User)
|
||||||
|
.Include(p => p.Tickets)
|
||||||
|
.Include(p => p.Manager)
|
||||||
|
.Include(p => p.Files)
|
||||||
|
.Include(p => p.Activities)
|
||||||
|
.AsNoTracking()
|
||||||
|
.FirstOrDefaultAsync(p => p.Id == id);
|
||||||
|
|
||||||
if (project == null)
|
if (project == null)
|
||||||
{ return NotFound(); }
|
{
|
||||||
|
return NotFound();
|
||||||
|
}
|
||||||
return project.GetMembers();
|
return project.GetMembers();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -198,15 +246,21 @@ namespace TicketManager.Controllers
|
||||||
[HttpPatch("{id}/members")]
|
[HttpPatch("{id}/members")]
|
||||||
public async Task<ActionResult<Project>> SetProjectMembers(int id, List<AppUser> projectMembers)
|
public async Task<ActionResult<Project>> SetProjectMembers(int id, List<AppUser> projectMembers)
|
||||||
{
|
{
|
||||||
Project project = await _projects.Get(id);
|
Project project = await _context.Projects
|
||||||
|
.Include(p => p.Assignments)
|
||||||
|
.FirstOrDefaultAsync(p => p.Id == id);
|
||||||
|
|
||||||
if (project == null)
|
if (project == null)
|
||||||
{
|
{
|
||||||
return NotFound();
|
return NotFound();
|
||||||
}
|
}
|
||||||
|
|
||||||
project.SetMembers(projectMembers);
|
project.SetMembers(projectMembers);
|
||||||
|
_context.Entry(project).State = EntityState.Modified;
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
await _projects.Update(project);
|
|
||||||
|
await _context.SaveChangesAsync();
|
||||||
}
|
}
|
||||||
catch (DbUpdateException /* ex */)
|
catch (DbUpdateException /* ex */)
|
||||||
{
|
{
|
||||||
|
|
@ -218,86 +272,9 @@ namespace TicketManager.Controllers
|
||||||
return NoContent();
|
return NoContent();
|
||||||
}
|
}
|
||||||
|
|
||||||
// // /// <summary>
|
private bool ProjectExists(int id)
|
||||||
// // /// Assign a user to a project.
|
{
|
||||||
// // /// </summary>
|
return _context.Projects.Any(e => e.Id == id);
|
||||||
// // /// <remarks>
|
}
|
||||||
// // /// Sample request:
|
|
||||||
// // ///
|
|
||||||
// // /// POST: api/v1/Projects/addmembers
|
|
||||||
// // /// [{
|
|
||||||
// // /// "id": "357727fd-5262-4522-b8a3-38271d43de84",
|
|
||||||
// // /// "firstName": "Thomas",
|
|
||||||
// // /// "lastName": "Price",
|
|
||||||
// // /// "presentation": "New Team?!",
|
|
||||||
// // /// "email": "tp@mail.com",
|
|
||||||
// // /// "phone": "0198237645"
|
|
||||||
// // /// }]
|
|
||||||
// // ///
|
|
||||||
// // /// </remarks>
|
|
||||||
// // /// <response code="204">Returns the created project</response>
|
|
||||||
// // [ProducesResponseType(StatusCodes.Status204NoContent)]
|
|
||||||
// // [ProducesResponseType(StatusCodes.Status404NotFound)]
|
|
||||||
// // [HttpPut("{id}/addMembers")]
|
|
||||||
// // public async Task<ActionResult<Project>> AddMembersToProject(int id, List<AppUser> usersToAdd)
|
|
||||||
// // {
|
|
||||||
// // if (usersToAdd == null)
|
|
||||||
// // {
|
|
||||||
// // return BadRequest();
|
|
||||||
// // }
|
|
||||||
// // Project project = await GetProjectByIdAsync(id);
|
|
||||||
// // project.AddMembers(usersToAdd);
|
|
||||||
// // try
|
|
||||||
// // {
|
|
||||||
// // await _context.SaveChangesAsync();
|
|
||||||
// // }
|
|
||||||
// // catch (DbUpdateException /* ex */)
|
|
||||||
// // {
|
|
||||||
// // //Log the error (uncomment ex variable name and write a log.)
|
|
||||||
// // ModelState.AddModelError("", "Unable to save changes. " +
|
|
||||||
// // "Try again, and if the problem persists, " +
|
|
||||||
// // "see your system administrator.");
|
|
||||||
// // }
|
|
||||||
// // return NoContent();
|
|
||||||
// // }
|
|
||||||
|
|
||||||
// // /// <summary>
|
|
||||||
// // /// Remove a user to a project.
|
|
||||||
// // /// </summary>
|
|
||||||
// // /// <remarks>
|
|
||||||
// // /// Sample request:
|
|
||||||
// // ///
|
|
||||||
// // /// PUT: api/v1/Projects/removemembers
|
|
||||||
// // /// [{
|
|
||||||
// // /// "id": "357727fd-5262-4522-b8a3-38271d43de84",
|
|
||||||
// // /// "firstName": "Thomas",
|
|
||||||
// // /// "lastName": "Price",
|
|
||||||
// // /// "presentation": "New Team?!",
|
|
||||||
// // /// "email": "tp@mail.com",
|
|
||||||
// // /// "phone": "0198237645"
|
|
||||||
// // /// }]
|
|
||||||
// // ///
|
|
||||||
// // /// </remarks>
|
|
||||||
// // /// <response code="204">Returns the created project</response>
|
|
||||||
// // [ProducesResponseType(StatusCodes.Status204NoContent)]
|
|
||||||
// // [ProducesResponseType(StatusCodes.Status404NotFound)]
|
|
||||||
// // [HttpPut("{id}/removeMembers")]
|
|
||||||
// // public async Task<ActionResult<Project>> RemoveMembersFromProject(int id, List<AppUser> usersToRemove)
|
|
||||||
// // {
|
|
||||||
// // Project project = await GetProjectByIdAsync(id);
|
|
||||||
// // project.RemoveMembers(usersToRemove);
|
|
||||||
// // try
|
|
||||||
// // {
|
|
||||||
// // await _context.SaveChangesAsync();
|
|
||||||
// // }
|
|
||||||
// // catch (DbUpdateException /* ex */)
|
|
||||||
// // {
|
|
||||||
// // //Log the error (uncomment ex variable name and write a log.)
|
|
||||||
// // ModelState.AddModelError("", "Unable to save changes. " +
|
|
||||||
// // "Try again, and if the problem persists, " +
|
|
||||||
// // "see your system administrator.");
|
|
||||||
// // }
|
|
||||||
// // return NoContent();
|
|
||||||
// // }
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -1,9 +1,11 @@
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
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 Microsoft.EntityFrameworkCore;
|
using Microsoft.EntityFrameworkCore;
|
||||||
using TicketManager.Data;
|
using TicketManager.Data;
|
||||||
|
using TicketManager.DTO;
|
||||||
using TicketManager.Models;
|
using TicketManager.Models;
|
||||||
|
|
||||||
namespace TicketManager.Controllers
|
namespace TicketManager.Controllers
|
||||||
|
|
@ -13,25 +15,40 @@ namespace TicketManager.Controllers
|
||||||
[ApiController]
|
[ApiController]
|
||||||
public class TicketsController : ControllerBase
|
public class TicketsController : ControllerBase
|
||||||
{
|
{
|
||||||
private readonly ITicketRepository _tickets;
|
private readonly AppDbContext _context;
|
||||||
|
|
||||||
public TicketsController(ITicketRepository tickets)
|
public TicketsController(AppDbContext context)
|
||||||
{
|
{
|
||||||
_tickets = tickets;
|
_context = context;
|
||||||
}
|
}
|
||||||
|
|
||||||
// GET: api/Tickets
|
// GET: api/Tickets
|
||||||
[HttpGet]
|
[HttpGet]
|
||||||
public async Task<IEnumerable<Ticket>> GetTickets()
|
public async Task<IEnumerable<TicketDTO>> GetTickets()
|
||||||
{
|
{
|
||||||
return await _tickets.List();
|
return await _context.Tickets
|
||||||
|
.Include(t => t.Project)
|
||||||
|
.Include(t => t.Files)
|
||||||
|
.Include(t => t.Activities)
|
||||||
|
.Include(t => t.Notes)
|
||||||
|
.AsNoTracking()
|
||||||
|
.Select(t => new TicketDTO(t))
|
||||||
|
.ToListAsync();
|
||||||
}
|
}
|
||||||
|
|
||||||
// GET: api/Tickets/5
|
// GET: api/Tickets/5
|
||||||
[HttpGet("{id}")]
|
[HttpGet("{id}")]
|
||||||
public async Task<ActionResult<Ticket>> GetTicket(int id)
|
public async Task<ActionResult<TicketDTO>> GetTicket(int id)
|
||||||
{
|
{
|
||||||
var ticket = await _tickets.Get(id);
|
var ticket = await _context.Tickets
|
||||||
|
.Include(t => t.Project)
|
||||||
|
.Include(t => t.Files)
|
||||||
|
.Include(t => t.Activities)
|
||||||
|
.Include(t => t.Notes)
|
||||||
|
.AsNoTracking()
|
||||||
|
.Select(t => new TicketDTO(t))
|
||||||
|
.FirstOrDefaultAsync(t => t.Id == id);
|
||||||
|
|
||||||
if (ticket == null)
|
if (ticket == null)
|
||||||
{
|
{
|
||||||
return NotFound();
|
return NotFound();
|
||||||
|
|
@ -49,13 +66,14 @@ namespace TicketManager.Controllers
|
||||||
{
|
{
|
||||||
return BadRequest();
|
return BadRequest();
|
||||||
}
|
}
|
||||||
|
_context.Entry(ticket).State = EntityState.Modified;
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
await _tickets.Update(ticket);
|
await _context.SaveChangesAsync();
|
||||||
}
|
}
|
||||||
catch (DbUpdateConcurrencyException)
|
catch (DbUpdateConcurrencyException)
|
||||||
{
|
{
|
||||||
if (!_tickets.Exists(id))
|
if (!TicketExists(id))
|
||||||
{
|
{
|
||||||
return NotFound();
|
return NotFound();
|
||||||
}
|
}
|
||||||
|
|
@ -73,36 +91,56 @@ namespace TicketManager.Controllers
|
||||||
[HttpPost]
|
[HttpPost]
|
||||||
public async Task<ActionResult<Ticket>> PostTicket(Ticket ticket)
|
public async Task<ActionResult<Ticket>> PostTicket(Ticket ticket)
|
||||||
{
|
{
|
||||||
await _tickets.Add(ticket);
|
_context.Tickets.Add(ticket);
|
||||||
return CreatedAtAction("GetTicket", new { id = ticket.Id }, ticket);
|
await _context.SaveChangesAsync();
|
||||||
|
|
||||||
|
var dto = new TicketDTO(ticket);
|
||||||
|
return CreatedAtAction("GetTicket", new { id = ticket.Id }, dto);
|
||||||
}
|
}
|
||||||
|
|
||||||
// DELETE: api/Tickets/5
|
// DELETE: api/Tickets/5
|
||||||
[HttpDelete("{id}")]
|
[HttpDelete("{id}")]
|
||||||
public async Task<ActionResult<Ticket>> DeleteTicket(int id)
|
public async Task<ActionResult<TicketDTO>> DeleteTicket(int id)
|
||||||
{
|
{
|
||||||
var ticket = await _tickets.Get(id);
|
var ticket = await _context.Tickets.FindAsync(id);
|
||||||
if (ticket == null)
|
if (ticket == null)
|
||||||
{
|
{
|
||||||
return NotFound();
|
return NotFound();
|
||||||
}
|
}
|
||||||
await _tickets.Delete(ticket);
|
|
||||||
return ticket;
|
_context.Tickets.Remove(ticket);
|
||||||
|
await _context.SaveChangesAsync();
|
||||||
|
|
||||||
|
var dto = new TicketDTO(ticket);
|
||||||
|
return Ok(dto);
|
||||||
}
|
}
|
||||||
|
|
||||||
[HttpGet("{id}/assignees")]
|
[HttpGet("{id}/assignees")]
|
||||||
public async Task<ActionResult<List<AppUser>>> GetTicketAssignees(int id)
|
public async Task<ActionResult<List<AppUserDTO>>> GetTicketAssignees(int id)
|
||||||
{
|
{
|
||||||
Ticket ticket = await _tickets.Get(id);
|
Ticket ticket = await _context.Tickets
|
||||||
return ticket.GetAssignees();
|
.Include(t => t.Project)
|
||||||
|
.ThenInclude(p => p.Assignments)
|
||||||
|
.ThenInclude(a => a.User)
|
||||||
|
.AsNoTracking()
|
||||||
|
.FirstOrDefaultAsync(t => t.Id == id);
|
||||||
|
|
||||||
|
return ticket.GetAssignees().Select(u => new AppUserDTO(u)).ToList();
|
||||||
}
|
}
|
||||||
|
|
||||||
[HttpPut("{id}/closed")]
|
[HttpPut("{id}/closed")]
|
||||||
public async Task<IActionResult> CloseTicket(int id)
|
public async Task<IActionResult> CloseTicket(int id)
|
||||||
{
|
{
|
||||||
Ticket ticket = await _tickets.Get(id);
|
Ticket ticket = await _context.Tickets.FindAsync(id);
|
||||||
ticket.Close();
|
ticket.Close();
|
||||||
|
// _context.Entry(ticket).State = EntityState.Modified;
|
||||||
|
|
||||||
return await PutTicket(ticket.Id, ticket);
|
return await PutTicket(ticket.Id, ticket);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private bool TicketExists(int id)
|
||||||
|
{
|
||||||
|
return _context.Tickets.Any(e => e.Id == id);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -1,37 +0,0 @@
|
||||||
using System.Threading.Tasks;
|
|
||||||
using TicketManager.Models;
|
|
||||||
using System.Linq;
|
|
||||||
using System;
|
|
||||||
using Microsoft.EntityFrameworkCore;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
|
|
||||||
namespace TicketManager.Data
|
|
||||||
{
|
|
||||||
public class AppUserRepository : GenericRepository<AppUser>, IAppUserRepository
|
|
||||||
{
|
|
||||||
private readonly IQueryable<AppUser> _query;
|
|
||||||
public AppUserRepository(AppDbContext context) : base(context)
|
|
||||||
{
|
|
||||||
_query = _dbSet
|
|
||||||
.Include(p => p.Assignments)
|
|
||||||
.ThenInclude(a => a.Project)
|
|
||||||
.ThenInclude(p => p.Tickets)
|
|
||||||
.Include(p => p.Activities);
|
|
||||||
}
|
|
||||||
|
|
||||||
public async Task<AppUser> GetUser(Guid id)
|
|
||||||
{
|
|
||||||
return await _query.FirstOrDefaultAsync(p => p.Id == id);
|
|
||||||
}
|
|
||||||
|
|
||||||
public override async Task<IEnumerable<AppUser>> List()
|
|
||||||
{
|
|
||||||
return await _query.ToListAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool Exists(Guid id)
|
|
||||||
{
|
|
||||||
return _dbSet.Any(e => e.Id == id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
@ -1,55 +0,0 @@
|
||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Linq;
|
|
||||||
using System.Linq.Expressions;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
using Microsoft.EntityFrameworkCore;
|
|
||||||
|
|
||||||
namespace TicketManager.Data
|
|
||||||
{
|
|
||||||
public class GenericRepository<T> : IGenericRepository<T> where T : class
|
|
||||||
{
|
|
||||||
protected readonly AppDbContext _context;
|
|
||||||
protected readonly DbSet<T> _dbSet;
|
|
||||||
public GenericRepository(AppDbContext context)
|
|
||||||
{
|
|
||||||
_context = context;
|
|
||||||
_dbSet = _context.Set<T>();
|
|
||||||
}
|
|
||||||
|
|
||||||
public async Task<int> Add(T entity)
|
|
||||||
{
|
|
||||||
_dbSet.Add(entity);
|
|
||||||
return await _context.SaveChangesAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
public async Task<int> Delete(T entity)
|
|
||||||
{
|
|
||||||
if (_context.Entry(entity).State == EntityState.Detached)
|
|
||||||
{ _dbSet.Attach(entity); }
|
|
||||||
_dbSet.Remove(entity);
|
|
||||||
return await _context.SaveChangesAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
public async Task<IEnumerable<T>> Find(int id, Expression<Func<T, bool>> expr)
|
|
||||||
{
|
|
||||||
return await _dbSet.Where(expr).AsNoTracking().ToListAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
public virtual async Task<T> Get(int id)
|
|
||||||
{
|
|
||||||
return await _dbSet.FindAsync(id);
|
|
||||||
}
|
|
||||||
public virtual async Task<IEnumerable<T>> List()
|
|
||||||
{
|
|
||||||
return await _dbSet.AsNoTracking().ToListAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
public async Task<int> Update(T entity)
|
|
||||||
{
|
|
||||||
_dbSet.Attach(entity);
|
|
||||||
_context.Entry(entity).State = EntityState.Modified;
|
|
||||||
return await _context.SaveChangesAsync();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
@ -1,12 +0,0 @@
|
||||||
using System;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
using TicketManager.Models;
|
|
||||||
|
|
||||||
namespace TicketManager.Data
|
|
||||||
{
|
|
||||||
public interface IAppUserRepository : IGenericRepository<AppUser>
|
|
||||||
{
|
|
||||||
Task<AppUser> GetUser(Guid id);
|
|
||||||
bool Exists(Guid id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
@ -1,20 +0,0 @@
|
||||||
using System;
|
|
||||||
using System.Linq.Expressions;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
|
|
||||||
namespace TicketManager.Data
|
|
||||||
{
|
|
||||||
public interface IGenericRepository<T> where T : class
|
|
||||||
{
|
|
||||||
Task<IEnumerable<T>> List();
|
|
||||||
Task<T> Get(int id);
|
|
||||||
Task<IEnumerable<T>> Find(int id, Expression<Func<T, bool>> expr);
|
|
||||||
|
|
||||||
Task<int> Add(T entity);
|
|
||||||
|
|
||||||
Task<int> Update(T entity);
|
|
||||||
|
|
||||||
Task<int> Delete(T entity);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
@ -1,13 +0,0 @@
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
using TicketManager.Models;
|
|
||||||
|
|
||||||
namespace TicketManager.Data
|
|
||||||
{
|
|
||||||
public interface IProjectRepository : IGenericRepository<Project>
|
|
||||||
{
|
|
||||||
bool Exists(int id);
|
|
||||||
Task<IEnumerable<AppUser>> GetMembers(int id);
|
|
||||||
Task SetMembers(int id, List<AppUser> usersToAdd);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
@ -1,11 +0,0 @@
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
using TicketManager.Models;
|
|
||||||
|
|
||||||
namespace TicketManager.Data
|
|
||||||
{
|
|
||||||
public interface ITicketRepository : IGenericRepository<Ticket>
|
|
||||||
{
|
|
||||||
bool Exists(int id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
@ -1,13 +0,0 @@
|
||||||
using System;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
|
|
||||||
namespace TicketManager.Data
|
|
||||||
{
|
|
||||||
public interface IUnitOfWork : IDisposable
|
|
||||||
{
|
|
||||||
IProjectRepository Projects { get; }
|
|
||||||
IAppUserRepository AppUsers { get; }
|
|
||||||
ITicketRepository Tickets { get; }
|
|
||||||
Task<int> Complete();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
@ -1,40 +0,0 @@
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
using System.Linq;
|
|
||||||
using TicketManager.Models;
|
|
||||||
using Microsoft.EntityFrameworkCore;
|
|
||||||
|
|
||||||
namespace TicketManager.Data
|
|
||||||
{
|
|
||||||
public class TicketRepository : GenericRepository<Ticket>, ITicketRepository
|
|
||||||
{
|
|
||||||
private IQueryable<Ticket> _query;
|
|
||||||
public TicketRepository(AppDbContext context) : base(context)
|
|
||||||
{
|
|
||||||
_query = _dbSet
|
|
||||||
.Include(p => p.Project)
|
|
||||||
.ThenInclude(a => a.Assignments)
|
|
||||||
.ThenInclude(p => p.User)
|
|
||||||
// .Include(p => p.Edits)
|
|
||||||
// .Include(p => p.Notes)
|
|
||||||
// .Include(p => p.Files)
|
|
||||||
// .Include(p => p.Creator)
|
|
||||||
;
|
|
||||||
}
|
|
||||||
|
|
||||||
public override async Task<Ticket> Get(int id)
|
|
||||||
{
|
|
||||||
return await _query.FirstOrDefaultAsync(p => p.Id == id);
|
|
||||||
}
|
|
||||||
|
|
||||||
public override async Task<IEnumerable<Ticket>> List()
|
|
||||||
{
|
|
||||||
return await _query.ToListAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool Exists(int id)
|
|
||||||
{
|
|
||||||
return _dbSet.Any(e => e.Id == id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
@ -1,32 +0,0 @@
|
||||||
using System;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
|
|
||||||
namespace TicketManager.Data
|
|
||||||
{
|
|
||||||
public class UnitOfWork : IUnitOfWork
|
|
||||||
{
|
|
||||||
private readonly AppDbContext _context;
|
|
||||||
public UnitOfWork(AppDbContext context)
|
|
||||||
{
|
|
||||||
_context = context;
|
|
||||||
Projects = new ProjectRepository(_context);
|
|
||||||
Tickets = new TicketRepository(_context);
|
|
||||||
AppUsers = new AppUserRepository(_context);
|
|
||||||
}
|
|
||||||
|
|
||||||
public IProjectRepository Projects { get; private set; }
|
|
||||||
|
|
||||||
public IAppUserRepository AppUsers { get; private set; }
|
|
||||||
|
|
||||||
public ITicketRepository Tickets { get; private set; }
|
|
||||||
|
|
||||||
public async Task<int> Complete()
|
|
||||||
{
|
|
||||||
return await _context.SaveChangesAsync();
|
|
||||||
}
|
|
||||||
public void Dispose()
|
|
||||||
{
|
|
||||||
_context.DisposeAsync();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
53
DataTransfertObjects/AppUserDTO.cs
Normal file
53
DataTransfertObjects/AppUserDTO.cs
Normal file
|
|
@ -0,0 +1,53 @@
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.ComponentModel.DataAnnotations;
|
||||||
|
using TicketManager.Models;
|
||||||
|
|
||||||
|
namespace TicketManager.DTO
|
||||||
|
{
|
||||||
|
public class AppUserDTO
|
||||||
|
{
|
||||||
|
public AppUserDTO(AppUser user)
|
||||||
|
{
|
||||||
|
Id = user.Id;
|
||||||
|
FirstName = user.FirstName;
|
||||||
|
LastName = user.LastName;
|
||||||
|
Presentation = user.Presentation;
|
||||||
|
Email = user.Email;
|
||||||
|
Phone = user.Phone;
|
||||||
|
Created_at = user.Created_at;
|
||||||
|
Picture = user.Picture;
|
||||||
|
Activities = user.Activities;
|
||||||
|
Projects = user.GetProjects();
|
||||||
|
Tickets = user.GetTickets();
|
||||||
|
}
|
||||||
|
|
||||||
|
public Guid Id { get; set; }
|
||||||
|
|
||||||
|
public string FirstName { get; set; }
|
||||||
|
|
||||||
|
public string LastName { get; set; }
|
||||||
|
|
||||||
|
public string FullName => $"{FirstName} {LastName}";
|
||||||
|
|
||||||
|
public string Presentation { get; set; }
|
||||||
|
|
||||||
|
[DataType(DataType.EmailAddress)]
|
||||||
|
public string Email { get; set; }
|
||||||
|
|
||||||
|
[DataType(DataType.PhoneNumber)]
|
||||||
|
public string Phone { get; set; }
|
||||||
|
|
||||||
|
[DataType(DataType.Date)]
|
||||||
|
public DateTime Created_at { get; private set; } = DateTime.Now;
|
||||||
|
|
||||||
|
public string Picture { get; set; }
|
||||||
|
|
||||||
|
public List<Activity> Activities { get; set; } = new List<Activity>();
|
||||||
|
|
||||||
|
public List<Project> Projects { get; set; } = new List<Project>();
|
||||||
|
|
||||||
|
public List<Ticket> Tickets { get; set; } = new List<Ticket>();
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
51
DataTransfertObjects/TicketDTO.cs
Normal file
51
DataTransfertObjects/TicketDTO.cs
Normal file
|
|
@ -0,0 +1,51 @@
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.ComponentModel.DataAnnotations;
|
||||||
|
using TicketManager.Models;
|
||||||
|
|
||||||
|
namespace TicketManager.DTO
|
||||||
|
{
|
||||||
|
public class TicketDTO
|
||||||
|
{
|
||||||
|
public TicketDTO(Ticket ticket)
|
||||||
|
{
|
||||||
|
Id = ticket.Id;
|
||||||
|
Title = ticket.Title;
|
||||||
|
Description = ticket.Description;
|
||||||
|
CreatedAt = ticket.CreatedAt;
|
||||||
|
PlannedEnding = ticket.PlannedEnding;
|
||||||
|
Status = ticket.Status.ToString();
|
||||||
|
Impact = ticket.Impact.ToString();
|
||||||
|
Difficulty = ticket.Difficulty.ToString();
|
||||||
|
Category = ticket.Category.ToString();
|
||||||
|
CreatorId = ticket.CreatorId;
|
||||||
|
Project = ticket.Project;
|
||||||
|
Notes = ticket.Notes;
|
||||||
|
Activities = ticket.Activities;
|
||||||
|
Files = ticket.Files;
|
||||||
|
Users = ticket.GetAssignees();
|
||||||
|
}
|
||||||
|
public int Id { get; set; }
|
||||||
|
|
||||||
|
public string Title { get; set; }
|
||||||
|
|
||||||
|
public string Description { get; set; }
|
||||||
|
|
||||||
|
[DataType(DataType.Date)]
|
||||||
|
public DateTime CreatedAt { get; private set; }
|
||||||
|
|
||||||
|
[DataType(DataType.Date)]
|
||||||
|
public DateTime PlannedEnding { get; set; }
|
||||||
|
|
||||||
|
public string Status { get; set; }
|
||||||
|
public string Impact { get; set; }
|
||||||
|
public string Difficulty { get; set; }
|
||||||
|
public string Category { get; set; }
|
||||||
|
public Guid CreatorId { get; set; }
|
||||||
|
public Project Project { get; set; }
|
||||||
|
public List<Note> Notes { get; set; } = new List<Note>();
|
||||||
|
public List<Activity> Activities { get; set; } = new List<Activity>();
|
||||||
|
public List<File> Files { get; set; } = new List<File>();
|
||||||
|
public List<AppUser> Users { get; set; } = new List<AppUser>();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -36,8 +36,8 @@ namespace TicketManager.Models
|
||||||
[Display(Name = "Member since"), DisplayFormat(DataFormatString = "{0:yyyy-MM-dd}")]
|
[Display(Name = "Member since"), DisplayFormat(DataFormatString = "{0:yyyy-MM-dd}")]
|
||||||
public DateTime Created_at { get; private set; } = DateTime.Now;
|
public DateTime Created_at { get; private set; } = DateTime.Now;
|
||||||
|
|
||||||
// [Display(Name = "Avatar")]
|
[Display(Name = "Avatar")]
|
||||||
// public byte[] Picture { get; set; }
|
public string Picture { get; set; }
|
||||||
|
|
||||||
public List<Assignment> Assignments { get; set; } = new List<Assignment>();
|
public List<Assignment> Assignments { get; set; } = new List<Assignment>();
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -47,3 +47,4 @@
|
||||||
- [ ] check useRef, useReducer, dispatch
|
- [ ] check useRef, useReducer, dispatch
|
||||||
- [ ] error page redirect when offline.
|
- [ ] error page redirect when offline.
|
||||||
- [ ] ticket/files/activities list placeholders when empty
|
- [ ] ticket/files/activities list placeholders when empty
|
||||||
|
- [ ] think about public/private DTO's constructor, getters and setters
|
||||||
|
|
|
||||||
|
|
@ -1 +1,2 @@
|
||||||
curl --insecure https://localhost:5001/api/v1/
|
curl --insecure https://localhost:5001/api/v1/
|
||||||
|
curl --insecure https://localhost:5001/api/v1/projects/1/ | json_pp > Scripts/response.http
|
||||||
5
Scripts/cleanDevDb.sh
Executable file
5
Scripts/cleanDevDb.sh
Executable file
|
|
@ -0,0 +1,5 @@
|
||||||
|
rm -r Migrations
|
||||||
|
rm app.db
|
||||||
|
dotnet ef migrations add Migration1
|
||||||
|
dotnet ef database update
|
||||||
|
dotnet watch run
|
||||||
47
Scripts/response.http
Normal file
47
Scripts/response.http
Normal file
|
|
@ -0,0 +1,47 @@
|
||||||
|
{
|
||||||
|
"activities" : [],
|
||||||
|
"plannedEnding" : "0001-01-01T00:00:00",
|
||||||
|
"id" : 1,
|
||||||
|
"title" : "Secret Project",
|
||||||
|
"createdAt" : "2020-02-24T10:34:18.428046",
|
||||||
|
"users" : [
|
||||||
|
{
|
||||||
|
"firstName" : "Thomas",
|
||||||
|
"phone" : "0198237645",
|
||||||
|
"lastName" : "Price",
|
||||||
|
"created_at" : "2020-02-25T09:42:54.462374",
|
||||||
|
"presentation" : "New Team?!",
|
||||||
|
"email" : "tp@mail.com",
|
||||||
|
"picture" : null,
|
||||||
|
"activities" : [],
|
||||||
|
"id" : "357727fd-5262-4522-b8a3-38271d43de84",
|
||||||
|
"fullName" : "Thomas Price",
|
||||||
|
"assignments" : [
|
||||||
|
{
|
||||||
|
"project" : {
|
||||||
|
"assignments" : [],
|
||||||
|
"createdAt" : "2020-02-24T10:34:18.428046",
|
||||||
|
"title" : "Secret Project",
|
||||||
|
"id" : 1,
|
||||||
|
"plannedEnding" : "2020-02-17T15:51:02.787373",
|
||||||
|
"activities" : [],
|
||||||
|
"description" : "Shhttt Don't tell anyone",
|
||||||
|
"status" : 1,
|
||||||
|
"files" : [],
|
||||||
|
"tickets" : [],
|
||||||
|
"progression" : 0,
|
||||||
|
"manager" : null
|
||||||
|
},
|
||||||
|
"userId" : "357727fd-5262-4522-b8a3-38271d43de84",
|
||||||
|
"projectId" : 1
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"manager" : null,
|
||||||
|
"progression" : 0,
|
||||||
|
"tickets" : [],
|
||||||
|
"files" : [],
|
||||||
|
"status" : "ToDo",
|
||||||
|
"description" : "Shhttt Don't tell anyone"
|
||||||
|
}
|
||||||
6
Scripts/scaffoldControllers.sh
Normal file
6
Scripts/scaffoldControllers.sh
Normal file
|
|
@ -0,0 +1,6 @@
|
||||||
|
rm Controllers/AppUsersController.cs
|
||||||
|
rm Controllers/TicketsController.cs
|
||||||
|
rm Controllers/ProjectsController.cs
|
||||||
|
dotnet aspnet-codegenerator controller -name AppUsersController -async -api -m AppUser -dc AppDbContext -outDir Controllers
|
||||||
|
dotnet aspnet-codegenerator controller -name TicketsController -async -api -m Ticket -dc AppDbContext -outDir Controllers
|
||||||
|
dotnet aspnet-codegenerator controller -name ProjectsController -async -api -m Project -dc AppDbContext -outDir Controllers
|
||||||
25
client/.gitignore
vendored
25
client/.gitignore
vendored
|
|
@ -1,25 +0,0 @@
|
||||||
# See https://help.github.com/articles/ignoring-files/ for more about ignoring files.
|
|
||||||
|
|
||||||
react-app-env.d.ts
|
|
||||||
|
|
||||||
# dependencies
|
|
||||||
/node_modules
|
|
||||||
/.pnp
|
|
||||||
.pnp.js
|
|
||||||
|
|
||||||
# testing
|
|
||||||
/coverage
|
|
||||||
|
|
||||||
# production
|
|
||||||
/build
|
|
||||||
|
|
||||||
# misc
|
|
||||||
.DS_Store
|
|
||||||
.env.local
|
|
||||||
.env.development.local
|
|
||||||
.env.test.local
|
|
||||||
.env.production.local
|
|
||||||
|
|
||||||
npm-debug.log*
|
|
||||||
yarn-debug.log*
|
|
||||||
yarn-error.log*
|
|
||||||
|
|
@ -1,4 +1,4 @@
|
||||||
import React, { FC, useState, CSSProperties } from "react";
|
import React, { FC, CSSProperties } from "react";
|
||||||
|
|
||||||
interface IProps {
|
interface IProps {
|
||||||
handleClose: () => void;
|
handleClose: () => void;
|
||||||
|
|
@ -11,14 +11,6 @@ export const Modal: FC<IProps> = ({ handleClose, show, children }) => {
|
||||||
return (
|
return (
|
||||||
<div className="modal" style={showHideStyle}>
|
<div className="modal" style={showHideStyle}>
|
||||||
<div className="modal-content">{children}</div>
|
<div className="modal-content">{children}</div>
|
||||||
<div className="modal-footer">
|
|
||||||
<button
|
|
||||||
type="submit"
|
|
||||||
className="modal-close waves-effect waves-green btn"
|
|
||||||
>
|
|
||||||
Done
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
|
||||||
|
|
@ -1,11 +1,13 @@
|
||||||
import React, { FC, useState, ChangeEvent, useEffect } from "react";
|
import React, { FC, useState, ChangeEvent, useEffect, FormEvent } from "react";
|
||||||
import { Modal } from "./Modal";
|
import { Modal } from "./Modal";
|
||||||
import { AvatarList } from "./AvatarList";
|
import { AvatarList } from "./AvatarList";
|
||||||
import { User } from "../types/User";
|
import { User } from "../types/User";
|
||||||
import { FilterBar } from "./FilterBar";
|
import { FilterBar } from "./FilterBar";
|
||||||
import { HttpResponse } from "../types/HttpResponse";
|
import { HttpResponse } from "../types/HttpResponse";
|
||||||
import { get } from "../utils/http";
|
import { get, put } from "../utils/http";
|
||||||
import { Constants } from "../utils/Constants";
|
import { Constants } from "../utils/Constants";
|
||||||
|
import { UsersModalEntry } from "./UsersModalEntry";
|
||||||
|
import { useParams } from "react-router-dom";
|
||||||
|
|
||||||
interface IProps {
|
interface IProps {
|
||||||
show: boolean;
|
show: boolean;
|
||||||
|
|
@ -15,12 +17,27 @@ interface IProps {
|
||||||
|
|
||||||
export const UsersModal: FC<IProps> = ({ show, handleClose, users }) => {
|
export const UsersModal: FC<IProps> = ({ show, handleClose, users }) => {
|
||||||
const [filterText, setFilterText] = useState<string>("");
|
const [filterText, setFilterText] = useState<string>("");
|
||||||
|
const [allUsers, setAllUsers] = useState<User[]>([]);
|
||||||
|
const [members, setMembers] = useState<User[]>(users);
|
||||||
|
const { id } = useParams();
|
||||||
|
|
||||||
const handleChange: (e: ChangeEvent<HTMLInputElement>) => void = (
|
const handleChange: (e: ChangeEvent<HTMLInputElement>) => void = (
|
||||||
e: ChangeEvent<HTMLInputElement>
|
e: ChangeEvent<HTMLInputElement>
|
||||||
) => {
|
) => {
|
||||||
setFilterText(e.target.value);
|
setFilterText(e.target.value);
|
||||||
};
|
};
|
||||||
const [allUsers, setAllUsers] = useState();
|
|
||||||
|
const handleSubmit: (event: FormEvent<HTMLFormElement>) => void = async (
|
||||||
|
e: FormEvent
|
||||||
|
) => {
|
||||||
|
e.preventDefault();
|
||||||
|
|
||||||
|
const response: HttpResponse<User[]> = await put<User[]>(
|
||||||
|
`${Constants.projectsURI}/${id}/members`,
|
||||||
|
members
|
||||||
|
);
|
||||||
|
console.log(response);
|
||||||
|
};
|
||||||
|
|
||||||
async function httpGet(): Promise<void> {
|
async function httpGet(): Promise<void> {
|
||||||
try {
|
try {
|
||||||
|
|
@ -28,8 +45,7 @@ export const UsersModal: FC<IProps> = ({ show, handleClose, users }) => {
|
||||||
`${Constants.usersURI}`
|
`${Constants.usersURI}`
|
||||||
);
|
);
|
||||||
if (response.parsedBody !== undefined) {
|
if (response.parsedBody !== undefined) {
|
||||||
setAllUsers(response.parsedBody);
|
setAllUsers((response.parsedBody as unknown) as User[]);
|
||||||
// setIsLoading(false);
|
|
||||||
}
|
}
|
||||||
} catch (ex) {
|
} catch (ex) {
|
||||||
// setHasError(true);
|
// setHasError(true);
|
||||||
|
|
@ -69,34 +85,26 @@ export const UsersModal: FC<IProps> = ({ show, handleClose, users }) => {
|
||||||
handleChange={handleChange}
|
handleChange={handleChange}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
{/* <div className="code">{allUsers}</div> */}
|
|
||||||
<form>
|
<form onSubmit={handleSubmit}>
|
||||||
<ul>
|
<ul>
|
||||||
{users.map((u: User) => (
|
{allUsers.map((u: User) => (
|
||||||
<li key={u.id}>
|
<li key={u.id}>
|
||||||
<div className="row">
|
<UsersModalEntry
|
||||||
<input
|
user={u}
|
||||||
id={u.id}
|
members={members}
|
||||||
type="checkbox"
|
setMembers={setMembers}
|
||||||
name="active"
|
/>
|
||||||
value="true"
|
|
||||||
onChange={() => false}
|
|
||||||
// checked
|
|
||||||
/>
|
|
||||||
<span>
|
|
||||||
{u.fullName}
|
|
||||||
<img
|
|
||||||
className="circle"
|
|
||||||
src={u.picture}
|
|
||||||
width="32vh"
|
|
||||||
height="32vh"
|
|
||||||
alt={u.fullName}
|
|
||||||
/>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</li>
|
</li>
|
||||||
))}
|
))}
|
||||||
</ul>
|
</ul>
|
||||||
|
<div className="modal-footer">
|
||||||
|
<input
|
||||||
|
type="submit"
|
||||||
|
className="modal-close waves-effect waves-green btn"
|
||||||
|
value="Done"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
</form>
|
</form>
|
||||||
</Modal>
|
</Modal>
|
||||||
);
|
);
|
||||||
|
|
|
||||||
38
client/src/components/UsersModalEntry.tsx
Normal file
38
client/src/components/UsersModalEntry.tsx
Normal file
|
|
@ -0,0 +1,38 @@
|
||||||
|
import React, { FC } from "react";
|
||||||
|
import { User } from "../types/User";
|
||||||
|
|
||||||
|
interface IProps {
|
||||||
|
setMembers: React.Dispatch<React.SetStateAction<User[]>>;
|
||||||
|
members: User[];
|
||||||
|
user: User;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const UsersModalEntry: FC<IProps> = ({ user, setMembers, members }) => {
|
||||||
|
return (
|
||||||
|
<div className="row">
|
||||||
|
<label htmlFor={user.id}>
|
||||||
|
<input
|
||||||
|
id={user.id}
|
||||||
|
name={user.fullName}
|
||||||
|
type="checkbox"
|
||||||
|
defaultChecked={members.includes(user)}
|
||||||
|
onChange={() => {
|
||||||
|
!members.includes(user)
|
||||||
|
? setMembers([...members, user])
|
||||||
|
: setMembers(members.filter(p => p !== user));
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<span>
|
||||||
|
{user.fullName}
|
||||||
|
<img
|
||||||
|
className="circle"
|
||||||
|
src={user.picture}
|
||||||
|
width="32vh"
|
||||||
|
height="32vh"
|
||||||
|
alt={user.fullName}
|
||||||
|
/>
|
||||||
|
</span>
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
1
client/src/react-app-env.d.ts
vendored
Normal file
1
client/src/react-app-env.d.ts
vendored
Normal file
|
|
@ -0,0 +1 @@
|
||||||
|
/// <reference types="react-scripts" />
|
||||||
|
|
@ -13,7 +13,7 @@ export async function http<T>(request: RequestInfo): Promise<HttpResponse<T>> {
|
||||||
|
|
||||||
export async function get<T>(
|
export async function get<T>(
|
||||||
path: string,
|
path: string,
|
||||||
args: RequestInit = { method: "get" }
|
args: RequestInit = { method: "get", headers: headers }
|
||||||
): Promise<HttpResponse<T>> {
|
): Promise<HttpResponse<T>> {
|
||||||
return await http<T>(new Request(path, args));
|
return await http<T>(new Request(path, args));
|
||||||
}
|
}
|
||||||
|
|
@ -21,7 +21,11 @@ export async function get<T>(
|
||||||
export async function post<T>(
|
export async function post<T>(
|
||||||
path: string,
|
path: string,
|
||||||
body: any,
|
body: any,
|
||||||
args: RequestInit = { method: "post", body: JSON.stringify(body) }
|
args: RequestInit = {
|
||||||
|
method: "post",
|
||||||
|
headers: headers,
|
||||||
|
body: JSON.stringify(body)
|
||||||
|
}
|
||||||
): Promise<HttpResponse<T>> {
|
): Promise<HttpResponse<T>> {
|
||||||
return await http<T>(new Request(path, args));
|
return await http<T>(new Request(path, args));
|
||||||
}
|
}
|
||||||
|
|
@ -29,7 +33,16 @@ export async function post<T>(
|
||||||
export async function put<T>(
|
export async function put<T>(
|
||||||
path: string,
|
path: string,
|
||||||
body: any,
|
body: any,
|
||||||
args: RequestInit = { method: "put", body: JSON.stringify(body) }
|
args: RequestInit = {
|
||||||
|
method: "put",
|
||||||
|
headers: headers,
|
||||||
|
body: JSON.stringify(body)
|
||||||
|
}
|
||||||
): Promise<HttpResponse<T>> {
|
): Promise<HttpResponse<T>> {
|
||||||
return await http<T>(new Request(path, args));
|
return await http<T>(new Request(path, args));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const headers: Headers = new Headers({
|
||||||
|
Accept: "application/json",
|
||||||
|
"Content-Type": "application/json"
|
||||||
|
});
|
||||||
|
|
|
||||||
Loading…
Reference in a new issue