// /*************************************************************************** // The Disc Image Chef // ---------------------------------------------------------------------------- // // Filename : UploadReportController.cs // Author(s) : Natalia Portillo // // Component : DiscImageChef Server. // // --[ Description ] ---------------------------------------------------------- // // Handles report uploads. // // --[ License ] -------------------------------------------------------------- // // This library is free software; you can redistribute it and/or modify // it under the terms of the GNU Lesser General Public License as // published by the Free Software Foundation; either version 2.1 of the // License, or (at your option) any later version. // // This library 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 // Lesser General Public License for more details. // // You should have received a copy of the GNU Lesser General Public // License along with this library; if not, see . // // ---------------------------------------------------------------------------- // Copyright © 2011-2019 Natalia Portillo // ****************************************************************************/ using System; using System.Diagnostics; using System.IO; using System.Net; using System.Text; using System.Threading.Tasks; using System.Xml.Serialization; using Cinchoo.PGP; using DiscImageChef.CommonTypes.Metadata; using DiscImageChef.Server.Models; using MailKit.Net.Smtp; using Microsoft.AspNetCore.Hosting; using Microsoft.AspNetCore.Mvc; using MimeKit; using Newtonsoft.Json; namespace DiscImageChef.Server.Controllers { public class UploadReportController : Controller { private readonly IWebHostEnvironment _environment; private readonly DicServerContext ctx; public UploadReportController(IWebHostEnvironment environment, DicServerContext _ctx) { _environment = environment; ctx = _ctx; } /// /// Receives a report from DiscImageChef.Core, verifies it's in the correct format and stores it on the server /// /// HTTP response [Route("api/uploadreport")] [HttpPost] public async Task UploadReport() { var response = new ContentResult {StatusCode = (int) HttpStatusCode.OK, ContentType = "text/plain"}; try { var newReport = new DeviceReport(); var request = HttpContext.Request; var xs = new XmlSerializer(newReport.GetType()); newReport = (DeviceReport) xs.Deserialize( new StringReader(await new StreamReader(request.Body).ReadToEndAsync())); if (newReport == null) { response.Content = "notstats"; return response; } var reportV2 = new DeviceReportV2(newReport); var jsonSw = new StringWriter(); jsonSw.Write(JsonConvert.SerializeObject(reportV2, Formatting.Indented, new JsonSerializerSettings { NullValueHandling = NullValueHandling.Ignore })); var reportV2String = jsonSw.ToString(); jsonSw.Close(); ctx.Reports.Add(new UploadedReport(reportV2)); ctx.SaveChanges(); var pgpIn = new MemoryStream(Encoding.UTF8.GetBytes(reportV2String)); var pgpOut = new MemoryStream(); var pgp = new ChoPGPEncryptDecrypt(); pgp.Encrypt(pgpIn, pgpOut, Path.Combine(_environment.ContentRootPath ?? throw new InvalidOperationException(), "public.asc")); pgpOut.Position = 0; reportV2String = Encoding.UTF8.GetString(pgpOut.ToArray()); var message = new MimeMessage { Subject = "New device report (old version)", Body = new TextPart("plain") {Text = reportV2String} }; message.From.Add(new MailboxAddress("DiscImageChef", "dic@claunia.com")); message.To.Add(new MailboxAddress("Natalia Portillo", "claunia@claunia.com")); using (var client = new SmtpClient()) { client.Connect("mail.claunia.com", 25, false); client.Send(message); client.Disconnect(true); } response.Content = "ok"; return response; } // ReSharper disable once RedundantCatchClause catch { #if DEBUG if (Debugger.IsAttached) throw; #endif response.Content = "error"; return response; } } /// /// Receives a report from DiscImageChef.Core, verifies it's in the correct format and stores it on the server /// /// HTTP response [Route("api/uploadreportv2")] [HttpPost] public async Task UploadReportV2() { var response = new ContentResult {StatusCode = (int) HttpStatusCode.OK, ContentType = "text/plain"}; try { var request = HttpContext.Request; var sr = new StreamReader(request.Body); var reportJson = await sr.ReadToEndAsync(); var newReport = JsonConvert.DeserializeObject(reportJson); if (newReport == null) { response.Content = "notstats"; return response; } ctx.Reports.Add(new UploadedReport(newReport)); ctx.SaveChanges(); var pgpIn = new MemoryStream(Encoding.UTF8.GetBytes(reportJson)); var pgpOut = new MemoryStream(); var pgp = new ChoPGPEncryptDecrypt(); pgp.Encrypt(pgpIn, pgpOut, Path.Combine(_environment.ContentRootPath ?? throw new InvalidOperationException(), "public.asc")); pgpOut.Position = 0; reportJson = Encoding.UTF8.GetString(pgpOut.ToArray()); var message = new MimeMessage { Subject = "New device report", Body = new TextPart("plain") {Text = reportJson} }; message.From.Add(new MailboxAddress("DiscImageChef", "dic@claunia.com")); message.To.Add(new MailboxAddress("Natalia Portillo", "claunia@claunia.com")); using (var client = new SmtpClient()) { client.Connect("mail.claunia.com", 25, false); client.Send(message); client.Disconnect(true); } response.Content = "ok"; return response; } // ReSharper disable once RedundantCatchClause catch { #if DEBUG if (Debugger.IsAttached) throw; #endif response.Content = "error"; return response; } } } }