2017-06-03 01:10:46 +01:00
|
|
|
|
// /***************************************************************************
|
|
|
|
|
|
// The Disc Image Chef
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
|
|
//
|
|
|
|
|
|
// Filename : UploadStatsController.cs
|
2017-12-19 03:50:57 +00:00
|
|
|
|
// Author(s) : Natalia Portillo <claunia@claunia.com>
|
2017-06-03 01:10:46 +01:00
|
|
|
|
//
|
2017-12-19 03:50:57 +00:00
|
|
|
|
// Component : DiscImageChef Server.
|
2017-06-03 01:10:46 +01:00
|
|
|
|
//
|
|
|
|
|
|
// --[ Description ] ----------------------------------------------------------
|
|
|
|
|
|
//
|
2017-12-19 03:50:57 +00:00
|
|
|
|
// Handles statistics uploads.
|
2017-06-03 01:10:46 +01:00
|
|
|
|
//
|
|
|
|
|
|
// --[ License ] --------------------------------------------------------------
|
|
|
|
|
|
//
|
2017-12-19 03:50:57 +00:00
|
|
|
|
// 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
|
2017-06-03 01:10:46 +01:00
|
|
|
|
// License, or (at your option) any later version.
|
|
|
|
|
|
//
|
2017-12-19 03:50:57 +00:00
|
|
|
|
// 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.
|
2017-06-03 01:10:46 +01:00
|
|
|
|
//
|
2017-12-19 03:50:57 +00:00
|
|
|
|
// You should have received a copy of the GNU Lesser General Public
|
|
|
|
|
|
// License along with this library; if not, see <http://www.gnu.org/licenses/>.
|
2017-06-03 01:10:46 +01:00
|
|
|
|
//
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
2017-12-19 03:50:57 +00:00
|
|
|
|
// Copyright © 2011-2018 Natalia Portillo
|
2017-06-03 01:10:46 +01:00
|
|
|
|
// ****************************************************************************/
|
2017-12-19 03:50:57 +00:00
|
|
|
|
|
2017-06-03 01:10:46 +01:00
|
|
|
|
using System;
|
2017-12-21 14:30:38 +00:00
|
|
|
|
using System.Collections.Generic;
|
2017-06-03 01:19:47 +01:00
|
|
|
|
using System.IO;
|
2017-06-05 17:46:27 +01:00
|
|
|
|
using System.Linq;
|
2017-12-21 14:30:38 +00:00
|
|
|
|
using System.Net;
|
2017-06-03 01:19:47 +01:00
|
|
|
|
using System.Net.Http;
|
2017-12-21 14:30:38 +00:00
|
|
|
|
using System.Text;
|
|
|
|
|
|
using System.Threading;
|
2017-06-03 01:19:47 +01:00
|
|
|
|
using System.Web;
|
2017-12-21 14:30:38 +00:00
|
|
|
|
using System.Web.Hosting;
|
2017-06-03 01:10:46 +01:00
|
|
|
|
using System.Web.Http;
|
2017-06-03 01:19:47 +01:00
|
|
|
|
using System.Xml.Serialization;
|
|
|
|
|
|
using DiscImageChef.Metadata;
|
2017-06-03 01:10:46 +01:00
|
|
|
|
|
2017-06-03 01:19:47 +01:00
|
|
|
|
namespace DiscImageChef.Server.Controllers
|
2017-06-03 01:10:46 +01:00
|
|
|
|
{
|
2017-06-03 01:19:47 +01:00
|
|
|
|
public class UploadStatsController : ApiController
|
2017-06-03 01:10:46 +01:00
|
|
|
|
{
|
2017-12-23 02:57:47 +00:00
|
|
|
|
/// <summary>
|
2017-12-24 03:23:06 +00:00
|
|
|
|
/// Receives statistics from DiscImageChef.Core, processes them and adds them to a server-side global statistics XML
|
2017-12-23 02:57:47 +00:00
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <returns>HTTP response</returns>
|
2017-06-03 01:19:47 +01:00
|
|
|
|
[Route("api/uploadstats")]
|
2017-06-03 01:10:46 +01:00
|
|
|
|
[HttpPost]
|
2017-06-03 01:19:47 +01:00
|
|
|
|
public HttpResponseMessage UploadStats()
|
2017-06-03 01:10:46 +01:00
|
|
|
|
{
|
2017-12-22 18:17:36 +00:00
|
|
|
|
HttpResponseMessage response = new HttpResponseMessage {StatusCode = HttpStatusCode.OK};
|
2017-06-03 01:19:47 +01:00
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
|
{
|
|
|
|
|
|
Stats newStats = new Stats();
|
|
|
|
|
|
HttpRequest request = HttpContext.Current.Request;
|
|
|
|
|
|
|
|
|
|
|
|
XmlSerializer xs = new XmlSerializer(newStats.GetType());
|
|
|
|
|
|
newStats = (Stats)xs.Deserialize(request.InputStream);
|
|
|
|
|
|
|
|
|
|
|
|
if(newStats == null)
|
|
|
|
|
|
{
|
2017-12-21 14:30:38 +00:00
|
|
|
|
response.Content = new StringContent("notstats", Encoding.UTF8, "text/plain");
|
2017-06-03 01:19:47 +01:00
|
|
|
|
return response;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2017-12-19 20:33:03 +00:00
|
|
|
|
FileStream fs =
|
2017-12-21 14:41:38 +00:00
|
|
|
|
WaitForFile(Path.Combine(HostingEnvironment.MapPath("~") ?? throw new InvalidOperationException(), "Statistics", "Statistics.xml"),
|
2017-12-19 20:33:03 +00:00
|
|
|
|
FileMode.Open, FileAccess.ReadWrite, FileShare.None);
|
2017-06-03 01:19:47 +01:00
|
|
|
|
|
|
|
|
|
|
if(fs == null)
|
|
|
|
|
|
{
|
2017-12-21 14:30:38 +00:00
|
|
|
|
response.Content = new StringContent("retry", Encoding.UTF8, "text/plain");
|
2017-06-03 01:19:47 +01:00
|
|
|
|
return response;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Stats oldStats = new Stats();
|
|
|
|
|
|
xs = new XmlSerializer(oldStats.GetType());
|
|
|
|
|
|
oldStats = (Stats)xs.Deserialize(fs);
|
|
|
|
|
|
|
|
|
|
|
|
if(newStats.Commands != null)
|
2017-12-19 20:33:03 +00:00
|
|
|
|
if(oldStats.Commands == null) oldStats.Commands = newStats.Commands;
|
2017-06-03 01:19:47 +01:00
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
oldStats.Commands.Analyze += newStats.Commands.Analyze;
|
|
|
|
|
|
oldStats.Commands.Benchmark += newStats.Commands.Benchmark;
|
|
|
|
|
|
oldStats.Commands.Checksum += newStats.Commands.Checksum;
|
|
|
|
|
|
oldStats.Commands.Compare += newStats.Commands.Compare;
|
|
|
|
|
|
oldStats.Commands.CreateSidecar += newStats.Commands.CreateSidecar;
|
|
|
|
|
|
oldStats.Commands.Decode += newStats.Commands.Decode;
|
|
|
|
|
|
oldStats.Commands.DeviceInfo += newStats.Commands.DeviceInfo;
|
|
|
|
|
|
oldStats.Commands.DeviceReport += newStats.Commands.DeviceReport;
|
|
|
|
|
|
oldStats.Commands.DumpMedia += newStats.Commands.DumpMedia;
|
|
|
|
|
|
oldStats.Commands.Entropy += newStats.Commands.Entropy;
|
|
|
|
|
|
oldStats.Commands.Formats += newStats.Commands.Formats;
|
|
|
|
|
|
oldStats.Commands.MediaInfo += newStats.Commands.MediaInfo;
|
|
|
|
|
|
oldStats.Commands.MediaScan += newStats.Commands.MediaScan;
|
|
|
|
|
|
oldStats.Commands.PrintHex += newStats.Commands.PrintHex;
|
|
|
|
|
|
oldStats.Commands.Verify += newStats.Commands.Verify;
|
2017-06-04 23:07:31 +01:00
|
|
|
|
oldStats.Commands.Ls += newStats.Commands.Ls;
|
|
|
|
|
|
oldStats.Commands.ExtractFiles += newStats.Commands.ExtractFiles;
|
2017-08-21 21:04:44 +01:00
|
|
|
|
oldStats.Commands.ListDevices += newStats.Commands.ListDevices;
|
2017-10-12 22:41:31 +01:00
|
|
|
|
oldStats.Commands.ListEncodings += newStats.Commands.ListEncodings;
|
2017-12-29 00:46:03 +00:00
|
|
|
|
oldStats.Commands.ConvertImage += newStats.Commands.ConvertImage;
|
2018-01-28 16:05:54 +00:00
|
|
|
|
oldStats.Commands.ImageInfo += newStats.Commands.ImageInfo;
|
2017-06-03 01:19:47 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if(newStats.OperatingSystems != null)
|
2017-12-19 20:33:03 +00:00
|
|
|
|
if(oldStats.OperatingSystems == null) oldStats.OperatingSystems = newStats.OperatingSystems;
|
2017-06-03 01:19:47 +01:00
|
|
|
|
else
|
2017-09-11 02:05:07 +01:00
|
|
|
|
foreach(OsStats newNvs in newStats.OperatingSystems)
|
2017-06-03 01:19:47 +01:00
|
|
|
|
{
|
2017-09-11 02:05:07 +01:00
|
|
|
|
OsStats removeNvs = null;
|
|
|
|
|
|
OsStats addNvs = null;
|
2017-06-03 01:19:47 +01:00
|
|
|
|
|
2017-12-24 03:23:06 +00:00
|
|
|
|
foreach(OsStats oldNvs in
|
|
|
|
|
|
oldStats.OperatingSystems.Where(oldNvs => oldNvs.name == newNvs.name &&
|
|
|
|
|
|
oldNvs.version == newNvs.version))
|
|
|
|
|
|
{
|
2017-12-21 07:08:26 +00:00
|
|
|
|
addNvs = new OsStats
|
2017-06-03 01:19:47 +01:00
|
|
|
|
{
|
2017-12-21 07:08:26 +00:00
|
|
|
|
name = oldNvs.name,
|
|
|
|
|
|
Value = oldNvs.Value + newNvs.Value,
|
|
|
|
|
|
version = oldNvs.version
|
|
|
|
|
|
};
|
|
|
|
|
|
removeNvs = oldNvs;
|
|
|
|
|
|
break;
|
|
|
|
|
|
}
|
2017-06-03 01:19:47 +01:00
|
|
|
|
|
2017-12-21 17:34:47 +00:00
|
|
|
|
if(removeNvs != null)
|
2017-06-03 01:19:47 +01:00
|
|
|
|
{
|
|
|
|
|
|
oldStats.OperatingSystems.Remove(removeNvs);
|
|
|
|
|
|
oldStats.OperatingSystems.Add(addNvs);
|
|
|
|
|
|
}
|
2017-12-19 20:33:03 +00:00
|
|
|
|
else oldStats.OperatingSystems.Add(newNvs);
|
2017-06-03 01:19:47 +01:00
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
if(oldStats.OperatingSystems == null)
|
2017-12-24 03:23:06 +00:00
|
|
|
|
oldStats.OperatingSystems = new List<OsStats> {new OsStats {name = "Linux", Value = 1}};
|
2017-06-03 01:19:47 +01:00
|
|
|
|
else
|
|
|
|
|
|
{
|
2017-09-11 02:05:07 +01:00
|
|
|
|
OsStats removeNvs = null;
|
|
|
|
|
|
OsStats addNvs = null;
|
2017-06-03 01:19:47 +01:00
|
|
|
|
|
2017-12-24 03:23:06 +00:00
|
|
|
|
foreach(OsStats oldNvs in oldStats.OperatingSystems.Where(oldNvs => oldNvs.name == "Linux"))
|
|
|
|
|
|
{
|
2017-12-21 07:08:26 +00:00
|
|
|
|
addNvs = new OsStats
|
2017-06-03 01:19:47 +01:00
|
|
|
|
{
|
2017-12-21 07:08:26 +00:00
|
|
|
|
name = oldNvs.name,
|
|
|
|
|
|
Value = oldNvs.Value + 1,
|
|
|
|
|
|
version = oldNvs.version
|
|
|
|
|
|
};
|
|
|
|
|
|
removeNvs = oldNvs;
|
|
|
|
|
|
break;
|
|
|
|
|
|
}
|
2017-06-03 01:19:47 +01:00
|
|
|
|
|
2017-12-21 17:34:47 +00:00
|
|
|
|
if(removeNvs != null)
|
2017-06-03 01:19:47 +01:00
|
|
|
|
{
|
|
|
|
|
|
oldStats.OperatingSystems.Remove(removeNvs);
|
|
|
|
|
|
oldStats.OperatingSystems.Add(addNvs);
|
|
|
|
|
|
}
|
2017-12-19 20:33:03 +00:00
|
|
|
|
else oldStats.OperatingSystems.Add(new OsStats {name = "Linux", Value = 1});
|
2017-06-03 01:19:47 +01:00
|
|
|
|
}
|
2017-06-04 22:49:20 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if(newStats.Versions != null)
|
2017-12-19 20:33:03 +00:00
|
|
|
|
if(oldStats.Versions == null) oldStats.Versions = newStats.Versions;
|
2017-06-04 22:49:20 +01:00
|
|
|
|
else
|
|
|
|
|
|
foreach(NameValueStats newNvs in newStats.Versions)
|
|
|
|
|
|
{
|
|
|
|
|
|
NameValueStats removeNvs = null;
|
|
|
|
|
|
NameValueStats addNvs = null;
|
|
|
|
|
|
|
2017-12-24 03:23:06 +00:00
|
|
|
|
foreach(NameValueStats oldNvs in
|
|
|
|
|
|
oldStats.Versions.Where(oldNvs => oldNvs.name == newNvs.name))
|
|
|
|
|
|
{
|
|
|
|
|
|
addNvs = new NameValueStats {name = oldNvs.name, Value = oldNvs.Value + newNvs.Value};
|
2017-12-21 07:08:26 +00:00
|
|
|
|
removeNvs = oldNvs;
|
|
|
|
|
|
break;
|
|
|
|
|
|
}
|
2017-06-04 22:49:20 +01:00
|
|
|
|
|
2017-12-21 17:34:47 +00:00
|
|
|
|
if(removeNvs != null)
|
2017-06-04 22:49:20 +01:00
|
|
|
|
{
|
|
|
|
|
|
oldStats.Versions.Remove(removeNvs);
|
|
|
|
|
|
oldStats.Versions.Add(addNvs);
|
|
|
|
|
|
}
|
2017-12-19 20:33:03 +00:00
|
|
|
|
else oldStats.Versions.Add(newNvs);
|
2017-06-04 22:49:20 +01:00
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
if(oldStats.Versions == null)
|
2017-12-19 20:33:03 +00:00
|
|
|
|
oldStats.Versions =
|
2017-12-24 03:23:06 +00:00
|
|
|
|
new List<NameValueStats> {new NameValueStats {name = "previous", Value = 1}};
|
2017-06-04 22:49:20 +01:00
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
NameValueStats removeNvs = null;
|
|
|
|
|
|
NameValueStats addNvs = null;
|
|
|
|
|
|
|
2017-12-24 03:23:06 +00:00
|
|
|
|
foreach(NameValueStats oldNvs in oldStats.Versions.Where(oldNvs => oldNvs.name == "previous"))
|
|
|
|
|
|
{
|
2017-12-21 07:08:26 +00:00
|
|
|
|
addNvs = new NameValueStats {name = oldNvs.name, Value = oldNvs.Value + 1};
|
|
|
|
|
|
removeNvs = oldNvs;
|
|
|
|
|
|
break;
|
|
|
|
|
|
}
|
2017-06-04 22:49:20 +01:00
|
|
|
|
|
2017-12-21 17:34:47 +00:00
|
|
|
|
if(removeNvs != null)
|
2017-06-04 22:49:20 +01:00
|
|
|
|
{
|
|
|
|
|
|
oldStats.Versions.Remove(removeNvs);
|
|
|
|
|
|
oldStats.Versions.Add(addNvs);
|
|
|
|
|
|
}
|
2017-12-19 20:33:03 +00:00
|
|
|
|
else oldStats.Versions.Add(new NameValueStats {name = "previous", Value = 1});
|
2017-06-04 22:49:20 +01:00
|
|
|
|
}
|
2017-06-03 01:19:47 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if(newStats.Filesystems != null)
|
2017-12-19 20:33:03 +00:00
|
|
|
|
if(oldStats.Filesystems == null) oldStats.Filesystems = newStats.Filesystems;
|
2017-06-03 01:19:47 +01:00
|
|
|
|
else
|
|
|
|
|
|
foreach(NameValueStats newNvs in newStats.Filesystems)
|
|
|
|
|
|
{
|
|
|
|
|
|
NameValueStats removeNvs = null;
|
|
|
|
|
|
NameValueStats addNvs = null;
|
|
|
|
|
|
|
2017-12-24 03:23:06 +00:00
|
|
|
|
foreach(NameValueStats oldNvs in
|
|
|
|
|
|
oldStats.Filesystems.Where(oldNvs => oldNvs.name == newNvs.name))
|
|
|
|
|
|
{
|
|
|
|
|
|
addNvs = new NameValueStats {name = oldNvs.name, Value = oldNvs.Value + newNvs.Value};
|
2017-12-21 07:08:26 +00:00
|
|
|
|
removeNvs = oldNvs;
|
|
|
|
|
|
break;
|
|
|
|
|
|
}
|
2017-06-03 01:19:47 +01:00
|
|
|
|
|
2017-12-21 17:34:47 +00:00
|
|
|
|
if(removeNvs != null)
|
2017-06-03 01:19:47 +01:00
|
|
|
|
{
|
|
|
|
|
|
oldStats.Filesystems.Remove(removeNvs);
|
|
|
|
|
|
oldStats.Filesystems.Add(addNvs);
|
|
|
|
|
|
}
|
2017-12-19 20:33:03 +00:00
|
|
|
|
else oldStats.Filesystems.Add(newNvs);
|
2017-06-03 01:19:47 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if(newStats.Partitions != null)
|
2017-12-19 20:33:03 +00:00
|
|
|
|
if(oldStats.Partitions == null) oldStats.Partitions = newStats.Partitions;
|
2017-06-03 01:19:47 +01:00
|
|
|
|
else
|
|
|
|
|
|
foreach(NameValueStats newNvs in newStats.Partitions)
|
|
|
|
|
|
{
|
|
|
|
|
|
NameValueStats removeNvs = null;
|
|
|
|
|
|
NameValueStats addNvs = null;
|
|
|
|
|
|
|
2017-12-24 03:23:06 +00:00
|
|
|
|
foreach(NameValueStats oldNvs in
|
|
|
|
|
|
oldStats.Partitions.Where(oldNvs => oldNvs.name == newNvs.name))
|
|
|
|
|
|
{
|
|
|
|
|
|
addNvs = new NameValueStats {name = oldNvs.name, Value = oldNvs.Value + newNvs.Value};
|
2017-12-21 07:08:26 +00:00
|
|
|
|
removeNvs = oldNvs;
|
|
|
|
|
|
break;
|
|
|
|
|
|
}
|
2017-06-03 01:19:47 +01:00
|
|
|
|
|
2017-12-21 17:34:47 +00:00
|
|
|
|
if(removeNvs != null)
|
2017-06-03 01:19:47 +01:00
|
|
|
|
{
|
|
|
|
|
|
oldStats.Partitions.Remove(removeNvs);
|
|
|
|
|
|
oldStats.Partitions.Add(addNvs);
|
|
|
|
|
|
}
|
2017-12-19 20:33:03 +00:00
|
|
|
|
else oldStats.Partitions.Add(newNvs);
|
2017-06-03 01:19:47 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if(newStats.MediaImages != null)
|
2017-12-19 20:33:03 +00:00
|
|
|
|
if(oldStats.MediaImages == null) oldStats.MediaImages = newStats.MediaImages;
|
2017-06-03 01:19:47 +01:00
|
|
|
|
else
|
|
|
|
|
|
foreach(NameValueStats newNvs in newStats.MediaImages)
|
|
|
|
|
|
{
|
|
|
|
|
|
NameValueStats removeNvs = null;
|
|
|
|
|
|
NameValueStats addNvs = null;
|
|
|
|
|
|
|
2017-12-24 03:23:06 +00:00
|
|
|
|
foreach(NameValueStats oldNvs in
|
|
|
|
|
|
oldStats.MediaImages.Where(oldNvs => oldNvs.name == newNvs.name))
|
|
|
|
|
|
{
|
|
|
|
|
|
addNvs = new NameValueStats {name = oldNvs.name, Value = oldNvs.Value + newNvs.Value};
|
2017-12-21 07:08:26 +00:00
|
|
|
|
removeNvs = oldNvs;
|
|
|
|
|
|
break;
|
|
|
|
|
|
}
|
2017-06-03 01:19:47 +01:00
|
|
|
|
|
2017-12-21 17:34:47 +00:00
|
|
|
|
if(removeNvs != null)
|
2017-06-03 01:19:47 +01:00
|
|
|
|
{
|
|
|
|
|
|
oldStats.MediaImages.Remove(removeNvs);
|
|
|
|
|
|
oldStats.MediaImages.Add(addNvs);
|
|
|
|
|
|
}
|
2017-12-19 20:33:03 +00:00
|
|
|
|
else oldStats.MediaImages.Add(newNvs);
|
2017-06-03 01:19:47 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if(newStats.Filters != null)
|
2017-12-19 20:33:03 +00:00
|
|
|
|
if(oldStats.Filters == null) oldStats.Filters = newStats.Filters;
|
2017-06-03 01:19:47 +01:00
|
|
|
|
else
|
|
|
|
|
|
foreach(NameValueStats newNvs in newStats.Filters)
|
|
|
|
|
|
{
|
|
|
|
|
|
NameValueStats removeNvs = null;
|
|
|
|
|
|
NameValueStats addNvs = null;
|
|
|
|
|
|
|
2017-12-24 03:23:06 +00:00
|
|
|
|
foreach(NameValueStats oldNvs in
|
|
|
|
|
|
oldStats.Filters.Where(oldNvs => oldNvs.name == newNvs.name))
|
|
|
|
|
|
{
|
|
|
|
|
|
addNvs = new NameValueStats {name = oldNvs.name, Value = oldNvs.Value + newNvs.Value};
|
2017-12-21 07:08:26 +00:00
|
|
|
|
removeNvs = oldNvs;
|
|
|
|
|
|
break;
|
|
|
|
|
|
}
|
2017-06-03 01:19:47 +01:00
|
|
|
|
|
2017-12-21 17:34:47 +00:00
|
|
|
|
if(removeNvs != null)
|
2017-06-03 01:19:47 +01:00
|
|
|
|
{
|
|
|
|
|
|
oldStats.Filters.Remove(removeNvs);
|
|
|
|
|
|
oldStats.Filters.Add(addNvs);
|
|
|
|
|
|
}
|
2017-12-19 20:33:03 +00:00
|
|
|
|
else oldStats.Filters.Add(newNvs);
|
2017-06-03 01:19:47 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if(newStats.Devices != null)
|
2017-12-19 20:33:03 +00:00
|
|
|
|
if(oldStats.Devices == null) oldStats.Devices = newStats.Devices;
|
2017-06-03 01:19:47 +01:00
|
|
|
|
else
|
2017-12-21 07:08:26 +00:00
|
|
|
|
foreach(DeviceStats newDev in from newDev in newStats.Devices
|
|
|
|
|
|
let found =
|
|
|
|
|
|
oldStats.Devices.Any(oldDev =>
|
|
|
|
|
|
oldDev.Manufacturer ==
|
|
|
|
|
|
newDev.Manufacturer &&
|
|
|
|
|
|
oldDev.Model == newDev.Model &&
|
|
|
|
|
|
oldDev.Revision == newDev.Revision &&
|
|
|
|
|
|
oldDev.Bus == newDev.Bus)
|
|
|
|
|
|
where !found
|
2017-12-24 03:23:06 +00:00
|
|
|
|
select newDev) oldStats.Devices.Add(newDev);
|
2017-06-03 01:19:47 +01:00
|
|
|
|
|
|
|
|
|
|
if(newStats.Medias != null)
|
2017-12-19 20:33:03 +00:00
|
|
|
|
if(oldStats.Medias == null) oldStats.Medias = newStats.Medias;
|
2017-06-03 01:19:47 +01:00
|
|
|
|
else
|
|
|
|
|
|
foreach(MediaStats newMstat in newStats.Medias)
|
|
|
|
|
|
{
|
|
|
|
|
|
MediaStats removeMstat = null;
|
|
|
|
|
|
MediaStats addMstat = null;
|
|
|
|
|
|
|
2017-12-24 03:23:06 +00:00
|
|
|
|
foreach(MediaStats oldMstat in
|
|
|
|
|
|
oldStats.Medias.Where(oldMstat => oldMstat.real == newMstat.real &&
|
|
|
|
|
|
oldMstat.type == newMstat.type))
|
|
|
|
|
|
{
|
2017-12-21 07:08:26 +00:00
|
|
|
|
addMstat = new MediaStats
|
2017-06-03 01:19:47 +01:00
|
|
|
|
{
|
2017-12-21 07:08:26 +00:00
|
|
|
|
real = oldMstat.real,
|
|
|
|
|
|
type = oldMstat.type,
|
|
|
|
|
|
Value = oldMstat.Value + newMstat.Value
|
|
|
|
|
|
};
|
|
|
|
|
|
removeMstat = oldMstat;
|
|
|
|
|
|
break;
|
|
|
|
|
|
}
|
2017-06-03 01:19:47 +01:00
|
|
|
|
|
|
|
|
|
|
if(removeMstat != null && addMstat != null)
|
|
|
|
|
|
{
|
|
|
|
|
|
oldStats.Medias.Remove(removeMstat);
|
|
|
|
|
|
oldStats.Medias.Add(addMstat);
|
|
|
|
|
|
}
|
2017-12-19 20:33:03 +00:00
|
|
|
|
else oldStats.Medias.Add(newMstat);
|
2017-06-03 01:19:47 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if(newStats.MediaScan != null)
|
2017-12-19 20:33:03 +00:00
|
|
|
|
if(oldStats.MediaScan == null) oldStats.MediaScan = newStats.MediaScan;
|
2017-06-03 01:19:47 +01:00
|
|
|
|
else
|
|
|
|
|
|
{
|
2017-12-19 20:33:03 +00:00
|
|
|
|
if(oldStats.MediaScan.Sectors == null) oldStats.MediaScan.Sectors = newStats.MediaScan.Sectors;
|
2017-06-03 01:19:47 +01:00
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
oldStats.MediaScan.Sectors.Correct = newStats.MediaScan.Sectors.Correct;
|
|
|
|
|
|
oldStats.MediaScan.Sectors.Error = newStats.MediaScan.Sectors.Error;
|
|
|
|
|
|
oldStats.MediaScan.Sectors.Total = newStats.MediaScan.Sectors.Total;
|
|
|
|
|
|
oldStats.MediaScan.Sectors.Unverifiable = newStats.MediaScan.Sectors.Unverifiable;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2017-12-19 20:33:03 +00:00
|
|
|
|
if(oldStats.MediaScan.Times == null) oldStats.MediaScan.Times = newStats.MediaScan.Times;
|
2017-06-03 01:19:47 +01:00
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
oldStats.MediaScan.Times.LessThan10ms = newStats.MediaScan.Times.LessThan10ms;
|
|
|
|
|
|
oldStats.MediaScan.Times.LessThan150ms = newStats.MediaScan.Times.LessThan150ms;
|
|
|
|
|
|
oldStats.MediaScan.Times.LessThan3ms = newStats.MediaScan.Times.LessThan3ms;
|
|
|
|
|
|
oldStats.MediaScan.Times.LessThan500ms = newStats.MediaScan.Times.LessThan500ms;
|
|
|
|
|
|
oldStats.MediaScan.Times.LessThan50ms = newStats.MediaScan.Times.LessThan50ms;
|
|
|
|
|
|
oldStats.MediaScan.Times.MoreThan500ms = newStats.MediaScan.Times.MoreThan500ms;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if(newStats.Verify != null)
|
2017-12-19 20:33:03 +00:00
|
|
|
|
if(oldStats.Verify == null) oldStats.Verify = newStats.Verify;
|
2017-06-03 01:19:47 +01:00
|
|
|
|
else
|
|
|
|
|
|
{
|
2017-12-19 20:33:03 +00:00
|
|
|
|
if(oldStats.Verify.Sectors == null) oldStats.Verify.Sectors = newStats.Verify.Sectors;
|
2017-06-03 01:19:47 +01:00
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
oldStats.Verify.Sectors.Correct = newStats.Verify.Sectors.Correct;
|
|
|
|
|
|
oldStats.Verify.Sectors.Error = newStats.Verify.Sectors.Error;
|
|
|
|
|
|
oldStats.Verify.Sectors.Total = newStats.Verify.Sectors.Total;
|
|
|
|
|
|
oldStats.Verify.Sectors.Unverifiable = newStats.Verify.Sectors.Unverifiable;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if(oldStats.Verify.MediaImages == null)
|
|
|
|
|
|
oldStats.Verify.MediaImages = newStats.Verify.MediaImages;
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
oldStats.Verify.MediaImages.Correct = newStats.Verify.MediaImages.Correct;
|
|
|
|
|
|
oldStats.Verify.MediaImages.Failed = newStats.Verify.MediaImages.Failed;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2017-06-05 17:46:27 +01:00
|
|
|
|
if(oldStats.Devices != null)
|
2017-12-19 20:33:03 +00:00
|
|
|
|
oldStats.Devices = oldStats.Devices.OrderBy(device => device.Manufacturer)
|
|
|
|
|
|
.ThenBy(device => device.Model).ThenBy(device => device.Revision)
|
|
|
|
|
|
.ThenBy(device => device.Bus).ToList();
|
2017-06-05 17:46:27 +01:00
|
|
|
|
|
2017-06-03 01:19:47 +01:00
|
|
|
|
Random rng = new Random();
|
2017-12-21 17:58:51 +00:00
|
|
|
|
string filename = $"BackupStats_{DateTime.UtcNow:yyyyMMddHHmmssfff}_{rng.Next()}.xml";
|
2017-12-24 03:23:06 +00:00
|
|
|
|
while(File.Exists(Path.Combine(HostingEnvironment.MapPath("~"), "Statistics", filename)))
|
2017-12-21 17:58:51 +00:00
|
|
|
|
filename = $"BackupStats_{DateTime.UtcNow:yyyyMMddHHmmssfff}_{rng.Next()}.xml";
|
2017-06-03 01:19:47 +01:00
|
|
|
|
|
2017-12-19 20:33:03 +00:00
|
|
|
|
FileStream backup =
|
2017-12-24 03:23:06 +00:00
|
|
|
|
new FileStream(Path.Combine(HostingEnvironment.MapPath("~"), "Statistics", filename),
|
2017-12-19 20:33:03 +00:00
|
|
|
|
FileMode.CreateNew, FileAccess.ReadWrite, FileShare.None);
|
2017-06-03 01:19:47 +01:00
|
|
|
|
fs.Seek(0, SeekOrigin.Begin);
|
|
|
|
|
|
fs.CopyTo(backup);
|
|
|
|
|
|
backup.Close();
|
|
|
|
|
|
fs.Seek(0, SeekOrigin.Begin);
|
|
|
|
|
|
xs = new XmlSerializer(oldStats.GetType());
|
|
|
|
|
|
xs.Serialize(fs, oldStats);
|
|
|
|
|
|
fs.SetLength(fs.Position);
|
|
|
|
|
|
fs.Close();
|
|
|
|
|
|
|
2017-12-21 14:30:38 +00:00
|
|
|
|
response.Content = new StringContent("ok", Encoding.UTF8, "text/plain");
|
2017-06-03 01:19:47 +01:00
|
|
|
|
return response;
|
|
|
|
|
|
}
|
|
|
|
|
|
catch(Exception ex)
|
|
|
|
|
|
{
|
|
|
|
|
|
#if DEBUG
|
2018-04-11 08:41:12 +01:00
|
|
|
|
System.Console.WriteLine("{0} {1}", ex.Message, ex.InnerException);
|
2017-06-03 01:19:47 +01:00
|
|
|
|
throw;
|
|
|
|
|
|
#else
|
|
|
|
|
|
response.Content = new StringContent("error", System.Text.Encoding.UTF8, "text/plain");
|
|
|
|
|
|
return response;
|
|
|
|
|
|
#endif
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
FileStream WaitForFile(string fullPath, FileMode mode, FileAccess access, FileShare share)
|
|
|
|
|
|
{
|
|
|
|
|
|
for(int numTries = 0; numTries < 100; numTries++)
|
|
|
|
|
|
{
|
|
|
|
|
|
FileStream fs = null;
|
|
|
|
|
|
try
|
|
|
|
|
|
{
|
|
|
|
|
|
fs = new FileStream(fullPath, mode, access, share);
|
|
|
|
|
|
return fs;
|
|
|
|
|
|
}
|
|
|
|
|
|
catch(IOException)
|
|
|
|
|
|
{
|
2017-12-20 23:07:46 +00:00
|
|
|
|
if(fs != null) fs.Dispose();
|
2017-12-21 14:30:38 +00:00
|
|
|
|
Thread.Sleep(50);
|
2017-06-03 01:19:47 +01:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return null;
|
2017-06-03 01:10:46 +01:00
|
|
|
|
}
|
|
|
|
|
|
}
|
2017-12-19 20:33:03 +00:00
|
|
|
|
}
|