2025-11-13 18:22:44 +00:00
|
|
|
/*******************************************************************************
|
|
|
|
|
// MARECHAI: Master repository of computing history artifacts information
|
|
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
|
//
|
|
|
|
|
// Author(s) : Natalia Portillo <claunia@claunia.com>
|
|
|
|
|
//
|
|
|
|
|
// --[ License ] -----------------------------------------------------------
|
|
|
|
|
//
|
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
|
// it under the terms of the GNU General Public License as
|
|
|
|
|
// published by the Free Software Foundation, either version 3 of the
|
|
|
|
|
// License, or (at your option) any later version.
|
|
|
|
|
//
|
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
|
//
|
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
//
|
|
|
|
|
// ---------------------------------------------------------------------------
|
2025-11-14 05:08:14 +00:00
|
|
|
// Copyright © 2003-2026 Natalia Portillo
|
2025-11-13 18:22:44 +00:00
|
|
|
*******************************************************************************/
|
|
|
|
|
|
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Security.Claims;
|
|
|
|
|
using System.Threading.Tasks;
|
2025-11-14 16:31:35 +00:00
|
|
|
using Marechai.Data;
|
2025-11-13 18:22:44 +00:00
|
|
|
using Marechai.Data.Dtos;
|
|
|
|
|
using Marechai.Database.Models;
|
|
|
|
|
using Microsoft.AspNetCore.Authorization;
|
|
|
|
|
using Microsoft.AspNetCore.Http;
|
|
|
|
|
using Microsoft.AspNetCore.Mvc;
|
|
|
|
|
using Microsoft.EntityFrameworkCore;
|
|
|
|
|
|
|
|
|
|
namespace Marechai.Server.Controllers;
|
|
|
|
|
|
|
|
|
|
[Route("/machines")]
|
|
|
|
|
[ApiController]
|
2025-11-15 02:38:47 +00:00
|
|
|
public class MachinesController(MarechaiContext context) : ControllerBase
|
2025-11-13 18:22:44 +00:00
|
|
|
{
|
|
|
|
|
[HttpGet]
|
|
|
|
|
[AllowAnonymous]
|
|
|
|
|
[ProducesResponseType(StatusCodes.Status200OK)]
|
|
|
|
|
[ProducesResponseType(StatusCodes.Status400BadRequest)]
|
2025-11-13 18:27:00 +00:00
|
|
|
public Task<List<MachineDto>> GetAsync() => context.Machines.OrderBy(m => m.Company.Name)
|
|
|
|
|
.ThenBy(m => m.Name)
|
|
|
|
|
.ThenBy(m => m.Family.Name)
|
|
|
|
|
.Select(m => new MachineDto
|
|
|
|
|
{
|
|
|
|
|
Id = m.Id,
|
|
|
|
|
Company = m.Company.Name,
|
|
|
|
|
Name = m.Name,
|
|
|
|
|
Model = m.Model,
|
|
|
|
|
Introduced = m.Introduced,
|
|
|
|
|
Type = m.Type,
|
|
|
|
|
Family = m.Family.Name
|
|
|
|
|
})
|
|
|
|
|
.ToListAsync();
|
2025-11-13 18:22:44 +00:00
|
|
|
|
2025-11-13 22:20:03 +00:00
|
|
|
[HttpGet("{id:int}")]
|
2025-11-13 18:22:44 +00:00
|
|
|
[AllowAnonymous]
|
|
|
|
|
[ProducesResponseType(StatusCodes.Status200OK)]
|
|
|
|
|
[ProducesResponseType(StatusCodes.Status400BadRequest)]
|
2025-11-13 18:27:00 +00:00
|
|
|
public Task<MachineDto> GetAsync(int id) => context.Machines.Where(m => m.Id == id)
|
|
|
|
|
.Select(m => new MachineDto
|
|
|
|
|
{
|
|
|
|
|
Id = m.Id,
|
|
|
|
|
Company = m.Company.Name,
|
|
|
|
|
CompanyId = m.CompanyId,
|
|
|
|
|
Name = m.Name,
|
|
|
|
|
Model = m.Model,
|
|
|
|
|
Introduced = m.Introduced,
|
|
|
|
|
Type = m.Type,
|
|
|
|
|
FamilyId = m.FamilyId
|
|
|
|
|
})
|
|
|
|
|
.FirstOrDefaultAsync();
|
2025-11-13 18:22:44 +00:00
|
|
|
|
2025-11-13 21:31:49 +00:00
|
|
|
[HttpPut("{id:int}")]
|
2025-11-13 18:22:44 +00:00
|
|
|
[Authorize(Roles = "Admin,UberAdmin")]
|
|
|
|
|
[ProducesResponseType(StatusCodes.Status200OK)]
|
|
|
|
|
[ProducesResponseType(StatusCodes.Status400BadRequest)]
|
2025-11-13 19:10:08 +00:00
|
|
|
[ProducesResponseType(StatusCodes.Status404NotFound)]
|
2025-11-13 19:28:21 +00:00
|
|
|
[ProducesResponseType(StatusCodes.Status401Unauthorized)]
|
2025-11-13 21:31:49 +00:00
|
|
|
public async Task<ActionResult> UpdateAsync(int id, [FromBody] MachineDto dto)
|
2025-11-13 18:22:44 +00:00
|
|
|
{
|
|
|
|
|
string userId = User.FindFirstValue(ClaimTypes.Sid);
|
2025-11-13 18:27:00 +00:00
|
|
|
|
2025-11-13 18:52:45 +00:00
|
|
|
if(userId is null) return Unauthorized();
|
2025-11-13 21:31:49 +00:00
|
|
|
Machine model = await context.Machines.FindAsync(id);
|
2025-11-13 18:22:44 +00:00
|
|
|
|
2025-11-13 18:52:45 +00:00
|
|
|
if(model is null) return NotFound();
|
2025-11-13 18:22:44 +00:00
|
|
|
|
|
|
|
|
model.CompanyId = dto.CompanyId;
|
|
|
|
|
model.Name = dto.Name;
|
|
|
|
|
model.Model = dto.Model;
|
|
|
|
|
model.Introduced = dto.Introduced;
|
|
|
|
|
model.Type = dto.Type;
|
|
|
|
|
model.FamilyId = dto.FamilyId;
|
|
|
|
|
|
|
|
|
|
var news = new News
|
|
|
|
|
{
|
|
|
|
|
AddedId = model.Id,
|
|
|
|
|
Date = DateTime.UtcNow
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
switch(model.Type)
|
|
|
|
|
{
|
|
|
|
|
case MachineType.Computer:
|
|
|
|
|
news.Type = NewsType.UpdatedComputerInDb;
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
case MachineType.Console:
|
|
|
|
|
news.Type = NewsType.UpdatedConsoleInDb;
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
news = null;
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(news != null) await context.News.AddAsync(news);
|
|
|
|
|
|
|
|
|
|
await context.SaveChangesWithUserAsync(userId);
|
2025-11-13 18:52:45 +00:00
|
|
|
|
|
|
|
|
return Ok();
|
2025-11-13 18:22:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[HttpPost]
|
|
|
|
|
[Authorize(Roles = "Admin,UberAdmin")]
|
|
|
|
|
[ProducesResponseType(StatusCodes.Status200OK)]
|
|
|
|
|
[ProducesResponseType(StatusCodes.Status400BadRequest)]
|
2025-11-13 19:28:21 +00:00
|
|
|
[ProducesResponseType(StatusCodes.Status401Unauthorized)]
|
2025-11-13 21:22:49 +00:00
|
|
|
public async Task<ActionResult<long>> CreateAsync([FromBody] MachineDto dto)
|
2025-11-13 18:22:44 +00:00
|
|
|
{
|
|
|
|
|
string userId = User.FindFirstValue(ClaimTypes.Sid);
|
2025-11-13 18:27:00 +00:00
|
|
|
|
2025-11-13 18:52:45 +00:00
|
|
|
if(userId is null) return Unauthorized();
|
2025-11-13 18:27:00 +00:00
|
|
|
|
2025-11-13 18:22:44 +00:00
|
|
|
var model = new Machine
|
|
|
|
|
{
|
|
|
|
|
CompanyId = dto.CompanyId,
|
|
|
|
|
Name = dto.Name,
|
|
|
|
|
Model = dto.Model,
|
|
|
|
|
Introduced = dto.Introduced,
|
|
|
|
|
Type = dto.Type,
|
|
|
|
|
FamilyId = dto.FamilyId
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
await context.Machines.AddAsync(model);
|
|
|
|
|
await context.SaveChangesWithUserAsync(userId);
|
|
|
|
|
|
|
|
|
|
var news = new News
|
|
|
|
|
{
|
|
|
|
|
AddedId = model.Id,
|
|
|
|
|
Date = DateTime.UtcNow
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
switch(model.Type)
|
|
|
|
|
{
|
|
|
|
|
case MachineType.Computer:
|
|
|
|
|
news.Type = NewsType.NewComputerInDb;
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
case MachineType.Console:
|
|
|
|
|
news.Type = NewsType.NewConsoleInDb;
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
news = null;
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(news != null)
|
|
|
|
|
{
|
|
|
|
|
await context.News.AddAsync(news);
|
|
|
|
|
await context.SaveChangesWithUserAsync(userId);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return model.Id;
|
|
|
|
|
}
|
|
|
|
|
|
2025-11-13 22:20:03 +00:00
|
|
|
[HttpGet("{id:int}/full")]
|
2025-11-13 18:22:44 +00:00
|
|
|
[AllowAnonymous]
|
|
|
|
|
[ProducesResponseType(StatusCodes.Status200OK)]
|
|
|
|
|
[ProducesResponseType(StatusCodes.Status400BadRequest)]
|
|
|
|
|
public async Task<MachineDto> GetMachine(int id)
|
|
|
|
|
{
|
|
|
|
|
Machine machine = await context.Machines.FindAsync(id);
|
|
|
|
|
|
|
|
|
|
if(machine is null) return null;
|
|
|
|
|
|
|
|
|
|
var model = new MachineDto
|
|
|
|
|
{
|
|
|
|
|
Introduced = machine.Introduced,
|
|
|
|
|
Name = machine.Name,
|
|
|
|
|
CompanyId = machine.CompanyId,
|
|
|
|
|
Model = machine.Model,
|
|
|
|
|
Type = machine.Type
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
Company company = await context.Companies.FindAsync(model.CompanyId);
|
|
|
|
|
|
|
|
|
|
if(company != null)
|
|
|
|
|
{
|
|
|
|
|
model.Company = company.Name;
|
|
|
|
|
|
|
|
|
|
IQueryable<CompanyLogo> logos = context.CompanyLogos.Where(l => l.CompanyId == company.Id);
|
|
|
|
|
|
|
|
|
|
if(model.Introduced.HasValue)
|
|
|
|
|
model.CompanyLogo = (await logos.FirstOrDefaultAsync(l => l.Year >= model.Introduced.Value.Year))?.Guid;
|
|
|
|
|
|
|
|
|
|
if(model.CompanyLogo is null && logos.Any()) model.CompanyLogo = (await logos.FirstAsync())?.Guid;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
MachineFamily family = await context.MachineFamilies.FindAsync(machine.FamilyId);
|
|
|
|
|
|
|
|
|
|
if(family != null)
|
|
|
|
|
{
|
|
|
|
|
model.FamilyName = family.Name;
|
|
|
|
|
model.FamilyId = family.Id;
|
|
|
|
|
}
|
|
|
|
|
|
2025-11-15 02:38:47 +00:00
|
|
|
model.Gpus = await context.GpusByMachine.Where(g => g.MachineId == machine.Id)
|
|
|
|
|
.Select(g => g.Gpu)
|
|
|
|
|
.OrderBy(g => g.Company.Name)
|
|
|
|
|
.ThenBy(g => g.Name)
|
|
|
|
|
.Select(g => new GpuDto
|
|
|
|
|
{
|
|
|
|
|
Id = g.Id,
|
|
|
|
|
Name = g.Name,
|
|
|
|
|
Company = g.Company.Name,
|
|
|
|
|
CompanyId = g.Company.Id,
|
|
|
|
|
ModelCode = g.ModelCode,
|
|
|
|
|
Introduced = g.Introduced,
|
|
|
|
|
Package = g.Package,
|
|
|
|
|
Process = g.Process,
|
|
|
|
|
ProcessNm = g.ProcessNm,
|
|
|
|
|
DieSize = g.DieSize,
|
|
|
|
|
Transistors = g.Transistors
|
|
|
|
|
})
|
|
|
|
|
.ToListAsync();
|
2025-11-13 18:22:44 +00:00
|
|
|
|
|
|
|
|
model.Memory = await context.MemoryByMachine.Where(m => m.MachineId == machine.Id)
|
2025-11-13 18:27:00 +00:00
|
|
|
.Select(m => new MemoryDto
|
|
|
|
|
{
|
|
|
|
|
Type = m.Type,
|
|
|
|
|
Usage = m.Usage,
|
|
|
|
|
Size = m.Size,
|
|
|
|
|
Speed = m.Speed
|
|
|
|
|
})
|
|
|
|
|
.ToListAsync();
|
2025-11-13 18:22:44 +00:00
|
|
|
|
2025-11-15 02:38:47 +00:00
|
|
|
model.Processors = await context.ProcessorsByMachine.Where(p => p.MachineId == machine.Id)
|
|
|
|
|
.Select(p => new ProcessorDto
|
|
|
|
|
{
|
|
|
|
|
Name = p.Processor.Name,
|
|
|
|
|
CompanyName = p.Processor.Company.Name,
|
|
|
|
|
CompanyId = p.Processor.Company.Id,
|
|
|
|
|
ModelCode = p.Processor.ModelCode,
|
|
|
|
|
Introduced = p.Processor.Introduced,
|
|
|
|
|
Speed = p.Speed,
|
|
|
|
|
Package = p.Processor.Package,
|
|
|
|
|
Gprs = p.Processor.Gprs,
|
|
|
|
|
GprSize = p.Processor.GprSize,
|
|
|
|
|
Fprs = p.Processor.Fprs,
|
|
|
|
|
FprSize = p.Processor.FprSize,
|
|
|
|
|
Cores = p.Processor.Cores,
|
|
|
|
|
ThreadsPerCore = p.Processor.ThreadsPerCore,
|
|
|
|
|
Process = p.Processor.Process,
|
|
|
|
|
ProcessNm = p.Processor.ProcessNm,
|
|
|
|
|
DieSize = p.Processor.DieSize,
|
|
|
|
|
Transistors = p.Processor.Transistors,
|
|
|
|
|
DataBus = p.Processor.DataBus,
|
|
|
|
|
AddrBus = p.Processor.AddrBus,
|
|
|
|
|
SimdRegisters = p.Processor.SimdRegisters,
|
|
|
|
|
SimdSize = p.Processor.SimdSize,
|
|
|
|
|
L1Instruction = p.Processor.L1Instruction,
|
|
|
|
|
L1Data = p.Processor.L1Data,
|
|
|
|
|
L2 = p.Processor.L2,
|
|
|
|
|
L3 = p.Processor.L3,
|
|
|
|
|
InstructionSet = p.Processor.InstructionSet.Name,
|
|
|
|
|
Id = p.Processor.Id,
|
|
|
|
|
InstructionSetExtensions = p.Processor.InstructionSetExtensions
|
|
|
|
|
.Select(e => e.Extension.Extension)
|
|
|
|
|
.ToList()
|
|
|
|
|
})
|
|
|
|
|
.ToListAsync();
|
|
|
|
|
|
|
|
|
|
model.SoundSynthesizers = await context.SoundByMachine.Where(s => s.MachineId == machine.Id)
|
|
|
|
|
.Select(s => s.SoundSynth)
|
|
|
|
|
.OrderBy(s => s.Company.Name)
|
|
|
|
|
.ThenBy(s => s.Name)
|
|
|
|
|
.ThenBy(s => s.ModelCode)
|
|
|
|
|
.Select(s => new SoundSynthDto
|
|
|
|
|
{
|
|
|
|
|
Id = s.Id,
|
|
|
|
|
Name = s.Name,
|
|
|
|
|
CompanyId = s.Company.Id,
|
|
|
|
|
CompanyName = s.Company.Name,
|
|
|
|
|
ModelCode = s.ModelCode,
|
|
|
|
|
Introduced = s.Introduced,
|
|
|
|
|
Voices = s.Voices,
|
|
|
|
|
Frequency = s.Frequency,
|
|
|
|
|
Depth = s.Depth,
|
|
|
|
|
SquareWave = s.SquareWave,
|
|
|
|
|
WhiteNoise = s.WhiteNoise,
|
|
|
|
|
Type = s.Type
|
|
|
|
|
})
|
|
|
|
|
.ToListAsync();
|
2025-11-13 18:22:44 +00:00
|
|
|
|
|
|
|
|
model.Storage = await context.StorageByMachine.Where(s => s.MachineId == machine.Id)
|
2025-11-13 18:27:00 +00:00
|
|
|
.Select(s => new StorageDto
|
|
|
|
|
{
|
|
|
|
|
Type = s.Type,
|
|
|
|
|
Interface = s.Interface,
|
|
|
|
|
Capacity = s.Capacity
|
|
|
|
|
})
|
|
|
|
|
.ToListAsync();
|
2025-11-13 18:22:44 +00:00
|
|
|
|
|
|
|
|
return model;
|
|
|
|
|
}
|
|
|
|
|
|
2025-11-13 22:20:03 +00:00
|
|
|
[HttpDelete("{id:int}")]
|
2025-11-13 18:22:44 +00:00
|
|
|
[Authorize(Roles = "Admin,UberAdmin")]
|
|
|
|
|
[ProducesResponseType(StatusCodes.Status200OK)]
|
|
|
|
|
[ProducesResponseType(StatusCodes.Status400BadRequest)]
|
2025-11-13 19:10:08 +00:00
|
|
|
[ProducesResponseType(StatusCodes.Status404NotFound)]
|
2025-11-13 19:28:21 +00:00
|
|
|
[ProducesResponseType(StatusCodes.Status401Unauthorized)]
|
2025-11-13 18:52:45 +00:00
|
|
|
public async Task<ActionResult> DeleteAsync(int id)
|
2025-11-13 18:22:44 +00:00
|
|
|
{
|
|
|
|
|
string userId = User.FindFirstValue(ClaimTypes.Sid);
|
2025-11-13 18:27:00 +00:00
|
|
|
|
2025-11-13 18:52:45 +00:00
|
|
|
if(userId is null) return Unauthorized();
|
2025-11-13 18:22:44 +00:00
|
|
|
Machine item = await context.Machines.FindAsync(id);
|
|
|
|
|
|
2025-11-13 18:52:45 +00:00
|
|
|
if(item is null) return NotFound();
|
2025-11-13 18:22:44 +00:00
|
|
|
|
|
|
|
|
context.Machines.Remove(item);
|
|
|
|
|
|
|
|
|
|
await context.SaveChangesWithUserAsync(userId);
|
2025-11-13 18:52:45 +00:00
|
|
|
|
|
|
|
|
return Ok();
|
2025-11-13 18:22:44 +00:00
|
|
|
}
|
2025-11-13 18:27:00 +00:00
|
|
|
}
|