pull latest version

This commit is contained in:
Ruidy Nemausat 2020-02-25 18:58:32 +01:00
commit 1856c60bca
27 changed files with 566 additions and 456 deletions

26
.gitignore vendored
View file

@ -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*

View file

@ -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);
} }
} }
} }

View file

@ -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();
// // }
} }
} }

View file

@ -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);
}
} }
} }

View file

@ -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);
}
}
}

View file

@ -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();
}
}
}

View file

@ -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);
}
}

View file

@ -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);
}
}

View file

@ -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);
}
}

View file

@ -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);
}
}

View file

@ -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();
}
}

View file

@ -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);
}
}
}

View file

@ -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();
}
}
}

View 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>();
}
}

View 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>();
}
}

View file

@ -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>();

View file

@ -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

View file

@ -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
View 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
View 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"
}

View 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
View file

@ -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*

View file

@ -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>
); );
}; };

View file

@ -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>
); );

View 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
View file

@ -0,0 +1 @@
/// <reference types="react-scripts" />

View file

@ -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"
});