Files
Aaru.Server/Aaru.Server.Task/Program.cs

733 lines
28 KiB
C#
Raw Normal View History

2019-11-02 01:40:41 +00:00
// /***************************************************************************
2020-02-29 19:22:23 +00:00
// Aaru Data Preservation Suite
2019-11-02 01:40:41 +00:00
// ----------------------------------------------------------------------------
//
// Filename : Program.cs
// Author(s) : Natalia Portillo <claunia@claunia.com>
//
2020-02-29 19:24:13 +00:00
// Component : Aaru Server Task.
2019-11-02 01:40:41 +00:00
//
// --[ Description ] ----------------------------------------------------------
//
// Runs time consuming server tasks.
//
// --[ 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 <http://www.gnu.org/licenses/>.
//
// ----------------------------------------------------------------------------
2024-04-30 05:11:05 +01:00
// Copyright © 2011-2024 Natalia Portillo
2019-11-02 01:40:41 +00:00
// ****************************************************************************/
using System;
using System.Diagnostics;
using System.IO;
using System.Linq;
using System.Net;
using System.Security.Cryptography;
using System.Text.Json;
using Aaru.CommonTypes.Enums;
2020-02-29 20:00:34 +00:00
using Aaru.Server.Models;
2019-11-02 01:40:41 +00:00
using HtmlAgilityPack;
2019-11-02 21:01:25 +00:00
using Microsoft.EntityFrameworkCore;
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
namespace Aaru.Server.Task;
2023-10-04 00:09:19 +01:00
class Program
2019-11-02 01:40:41 +00:00
{
2021-12-08 17:57:50 +00:00
public static void Main(string[] args)
2019-11-02 01:40:41 +00:00
{
2021-12-08 17:57:50 +00:00
DateTime start, end;
2023-10-04 00:09:19 +01:00
var counter = 0;
2019-11-02 21:01:25 +00:00
2021-12-08 17:57:50 +00:00
start = DateTime.UtcNow;
System.Console.WriteLine("{0}: Connecting to database...", DateTime.UtcNow);
var ctx = new AaruServerContext();
end = DateTime.UtcNow;
System.Console.WriteLine("{0}: Took {1:F2} seconds", end, (end - start).TotalSeconds);
System.Console.WriteLine("{0}: Migrating database to latest version...", DateTime.UtcNow);
start = DateTime.UtcNow;
ctx.Database.Migrate();
end = DateTime.UtcNow;
System.Console.WriteLine("{0}: Took {1:F2} seconds", DateTime.UtcNow, (end - start).TotalSeconds);
WebClient client;
try
{
System.Console.WriteLine("{0}: Retrieving USB IDs from Linux USB...", DateTime.UtcNow);
start = DateTime.UtcNow;
client = new WebClient();
var sr = new StringReader(client.DownloadString("http://www.linux-usb.org/usb.ids"));
2019-11-02 01:40:41 +00:00
end = DateTime.UtcNow;
2019-11-02 21:01:25 +00:00
System.Console.WriteLine("{0}: Took {1:F2} seconds", end, (end - start).TotalSeconds);
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
UsbVendor vendor = null;
2023-10-04 00:09:19 +01:00
var newVendors = 0;
var newProducts = 0;
var modifiedVendors = 0;
var modifiedProducts = 0;
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
start = DateTime.UtcNow;
System.Console.WriteLine("{0}: Adding and updating database entries...", DateTime.UtcNow);
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
do
2019-11-02 01:40:41 +00:00
{
2021-12-08 17:57:50 +00:00
if(counter == 1000)
2019-11-02 01:40:41 +00:00
{
2021-12-08 17:57:50 +00:00
DateTime start2 = DateTime.UtcNow;
System.Console.WriteLine("{0}: Saving changes", start2);
ctx.SaveChanges();
end = DateTime.UtcNow;
System.Console.WriteLine("{0}: Took {1:F2} seconds", end, (end - start2).TotalSeconds);
counter = 0;
}
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
string line = sr.ReadLine();
2019-11-02 23:52:33 +00:00
2021-12-08 17:57:50 +00:00
if(line is null)
break;
2019-11-02 23:52:33 +00:00
2021-12-08 17:57:50 +00:00
if(line.Length == 0 ||
line[0] == '#')
continue;
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
ushort number;
string name;
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
if(line[0] == '\t')
{
2019-11-02 21:39:19 +00:00
try
{
2021-12-08 17:57:50 +00:00
number = Convert.ToUInt16(line.Substring(1, 4), 16);
2019-11-02 21:39:19 +00:00
}
2019-11-02 23:52:33 +00:00
catch(FormatException)
2019-11-02 21:39:19 +00:00
{
continue;
}
2019-11-02 01:40:41 +00:00
2019-11-02 23:52:33 +00:00
if(number == 0)
continue;
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
name = line.Substring(7);
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
UsbProduct product =
ctx.UsbProducts.FirstOrDefault(p => p.ProductId == number && p.Vendor != null &&
p.Vendor.VendorId == vendor.VendorId);
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
if(product is null)
2019-11-02 01:40:41 +00:00
{
2021-12-08 17:57:50 +00:00
product = new UsbProduct(vendor, number, name);
ctx.UsbProducts.Add(product);
2019-11-02 23:52:33 +00:00
2021-12-08 17:57:50 +00:00
System.Console.WriteLine("{0}: Will add product {1} with ID {2:X4} and vendor {3} ({4:X4})",
DateTime.UtcNow, product.Product, product.ProductId,
product.Vendor?.Vendor ?? "null", product.Vendor?.VendorId ?? 0);
2019-11-02 23:52:33 +00:00
2021-12-08 17:57:50 +00:00
newProducts++;
2019-11-02 01:40:41 +00:00
counter++;
}
2021-12-08 17:57:50 +00:00
else if(name != product.Product)
2019-11-02 01:40:41 +00:00
{
2021-12-08 17:57:50 +00:00
System.Console.
2023-10-04 00:09:19 +01:00
WriteLine(
"{0}: Will modify product with ID {1:X4} and vendor {2} ({3:X4}) from \"{4}\" to \"{5}\"",
DateTime.UtcNow, product.ProductId, product.Vendor?.Vendor ?? "null",
product.Vendor?.VendorId ?? 0, product.Product, name);
2021-12-08 17:57:50 +00:00
product.Product = name;
product.ModifiedWhen = DateTime.UtcNow;
modifiedProducts++;
2019-11-02 01:40:41 +00:00
counter++;
}
2021-12-08 17:57:50 +00:00
continue;
}
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
try
{
number = Convert.ToUInt16(line.Substring(0, 4), 16);
}
catch(FormatException)
{
continue;
}
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
if(number == 0)
continue;
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
name = line.Substring(6);
2019-11-02 23:52:33 +00:00
2021-12-08 17:57:50 +00:00
vendor = ctx.UsbVendors.FirstOrDefault(v => v.VendorId == number);
2019-11-02 23:52:33 +00:00
2021-12-08 17:57:50 +00:00
if(vendor is null)
{
vendor = new UsbVendor(number, name);
ctx.UsbVendors.Add(vendor);
2019-11-02 01:40:41 +00:00
2021-12-08 18:26:24 +00:00
System.Console.WriteLine("{0}: Will add vendor {1} with ID {2:X4}", DateTime.UtcNow, vendor.Vendor,
vendor.VendorId);
2019-11-02 23:52:33 +00:00
2021-12-08 17:57:50 +00:00
newVendors++;
counter++;
}
else if(name != vendor.Vendor)
{
System.Console.WriteLine("{0}: Will modify vendor with ID {1:X4} from \"{2}\" to \"{3}\"",
DateTime.UtcNow, vendor.VendorId, vendor.Vendor, name);
2019-11-02 23:52:33 +00:00
2021-12-08 17:57:50 +00:00
vendor.Vendor = name;
vendor.ModifiedWhen = DateTime.UtcNow;
modifiedVendors++;
counter++;
}
} while(true);
2019-11-02 23:52:33 +00:00
2021-12-08 17:57:50 +00:00
end = DateTime.UtcNow;
System.Console.WriteLine("{0}: Took {1:F2} seconds", end, (end - start).TotalSeconds);
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
System.Console.WriteLine("{0}: Saving database changes...", DateTime.UtcNow);
2019-11-02 01:40:41 +00:00
start = DateTime.UtcNow;
2021-12-08 17:57:50 +00:00
ctx.SaveChanges();
end = DateTime.UtcNow;
System.Console.WriteLine("{0}: Took {1:F2} seconds", end, (end - start).TotalSeconds);
2023-10-04 00:09:19 +01:00
System.Console.WriteLine("{0}: {1} vendors added.", DateTime.UtcNow, newVendors);
System.Console.WriteLine("{0}: {1} products added.", DateTime.UtcNow, newProducts);
System.Console.WriteLine("{0}: {1} vendors modified.", DateTime.UtcNow, modifiedVendors);
2021-12-08 17:57:50 +00:00
System.Console.WriteLine("{0}: {1} products modified.", DateTime.UtcNow, modifiedProducts);
2019-11-02 23:52:33 +00:00
2021-12-08 17:57:50 +00:00
System.Console.WriteLine("{0}: Looking up a vendor", DateTime.UtcNow);
start = DateTime.UtcNow;
vendor = ctx.UsbVendors.FirstOrDefault(v => v.VendorId == 0x8086);
if(vendor is null)
System.Console.WriteLine("{0}: Error, could not find vendor.", DateTime.UtcNow);
else
System.Console.WriteLine("{0}: Found {1}.", DateTime.UtcNow, vendor.Vendor);
2019-11-02 23:52:33 +00:00
2019-11-02 01:40:41 +00:00
end = DateTime.UtcNow;
2019-11-02 21:01:25 +00:00
System.Console.WriteLine("{0}: Took {1:F2} seconds", end, (end - start).TotalSeconds);
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
System.Console.WriteLine("{0}: Looking up a product", DateTime.UtcNow);
2019-11-02 01:40:41 +00:00
start = DateTime.UtcNow;
2021-12-08 17:57:50 +00:00
2021-12-08 18:26:24 +00:00
UsbProduct prd = ctx.UsbProducts.FirstOrDefault(p => p.ProductId == 0x0001 && p.Vendor.VendorId == 0x8086);
2021-12-08 17:57:50 +00:00
if(prd is null)
System.Console.WriteLine("{0}: Error, could not find product.", DateTime.UtcNow);
else
System.Console.WriteLine("{0}: Found {1}.", DateTime.UtcNow, prd.Product);
2019-11-02 01:40:41 +00:00
end = DateTime.UtcNow;
2019-11-02 21:01:25 +00:00
System.Console.WriteLine("{0}: Took {1:F2} seconds", end, (end - start).TotalSeconds);
2021-12-08 17:57:50 +00:00
}
catch(Exception ex)
{
#if DEBUG
if(Debugger.IsAttached)
throw;
#endif
System.Console.WriteLine("{0}: Exception {1} filling USB IDs...", DateTime.UtcNow, ex);
}
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
System.Console.WriteLine("{0}: Fixing all devices without modification time...", DateTime.UtcNow);
start = DateTime.UtcNow;
2019-11-02 23:52:33 +00:00
2021-12-08 17:57:50 +00:00
foreach(Device device in ctx.Devices.Where(d => d.ModifiedWhen == null))
device.ModifiedWhen = device.AddedWhen;
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
end = DateTime.UtcNow;
System.Console.WriteLine("{0}: Took {1:F2} seconds", end, (end - start).TotalSeconds);
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
System.Console.WriteLine("{0}: Committing changes...", DateTime.UtcNow);
start = DateTime.UtcNow;
ctx.SaveChanges();
end = DateTime.UtcNow;
System.Console.WriteLine("{0}: Took {1:F2} seconds", end, (end - start).TotalSeconds);
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
try
{
2021-12-08 18:26:24 +00:00
System.Console.WriteLine("{0}: Retrieving CompactDisc read offsets from AccurateRip...", DateTime.UtcNow);
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
start = DateTime.UtcNow;
client = new WebClient();
string html = client.DownloadString("http://www.accuraterip.com/driveoffsets.htm");
end = DateTime.UtcNow;
System.Console.WriteLine("{0}: Took {1:F2} seconds", end, (end - start).TotalSeconds);
// The HTML is too malformed to process easily, so find start of table
html = "<html><body><table><tr>" +
html.Substring(html.IndexOf("<td bgcolor=\"#000000\">", StringComparison.Ordinal));
var doc = new HtmlDocument();
doc.LoadHtml(html);
HtmlNode firstTable = doc.DocumentNode.SelectSingleNode("/html[1]/body[1]/table[1]");
2019-11-02 01:40:41 +00:00
2023-10-04 00:09:19 +01:00
var firstRow = true;
2019-11-02 01:40:41 +00:00
2023-10-04 00:09:19 +01:00
var addedOffsets = 0;
var modifiedOffsets = 0;
2019-11-02 23:52:33 +00:00
2021-12-08 17:57:50 +00:00
System.Console.WriteLine("{0}: Processing offsets...", DateTime.UtcNow);
start = DateTime.UtcNow;
foreach(HtmlNode row in firstTable.Descendants("tr"))
{
HtmlNode[] columns = row.Descendants("td").ToArray();
if(columns.Length != 4)
2019-11-02 01:40:41 +00:00
{
2021-12-08 18:26:24 +00:00
System.Console.WriteLine("{0}: Row does not have correct number of columns...", DateTime.UtcNow);
2021-12-08 17:57:50 +00:00
continue;
}
string column0 = columns[0].InnerText;
string column1 = columns[1].InnerText;
string column2 = columns[2].InnerText;
string column3 = columns[3].InnerText;
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
if(firstRow)
{
if(column0.ToLowerInvariant() != "cd drive")
2019-11-02 01:40:41 +00:00
{
2021-12-08 17:57:50 +00:00
System.Console.WriteLine("{0}: Unexpected header \"{1}\" found...", DateTime.UtcNow,
columns[0].InnerText);
2019-11-02 23:52:33 +00:00
2021-12-08 17:57:50 +00:00
break;
2019-11-02 01:40:41 +00:00
}
2021-12-08 17:57:50 +00:00
if(column1.ToLowerInvariant() != "correction offset")
2019-11-02 01:40:41 +00:00
{
2021-12-08 17:57:50 +00:00
System.Console.WriteLine("{0}: Unexpected header \"{1}\" found...", DateTime.UtcNow,
columns[1].InnerText);
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
break;
}
2019-11-02 23:52:33 +00:00
2021-12-08 17:57:50 +00:00
if(column2.ToLowerInvariant() != "submitted by")
{
System.Console.WriteLine("{0}: Unexpected header \"{1}\" found...", DateTime.UtcNow,
columns[2].InnerText);
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
break;
}
2019-11-02 23:52:33 +00:00
2021-12-08 17:57:50 +00:00
if(column3.ToLowerInvariant() != "percentage agree")
{
System.Console.WriteLine("{0}: Unexpected header \"{1}\" found...", DateTime.UtcNow,
columns[3].InnerText);
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
break;
}
2019-11-02 23:52:33 +00:00
2021-12-08 17:57:50 +00:00
firstRow = false;
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
continue;
}
2019-11-02 23:52:33 +00:00
2021-12-08 17:57:50 +00:00
string manufacturer;
string model;
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
if(column0[0] == '-' &&
column0[1] == ' ')
{
manufacturer = null;
model = column0.Substring(2).Trim();
}
else
{
int cutOffset = column0.IndexOf(" - ", StringComparison.Ordinal);
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
if(cutOffset == -1)
2019-11-02 01:40:41 +00:00
{
manufacturer = null;
2021-12-08 17:57:50 +00:00
model = column0;
2019-11-02 01:40:41 +00:00
}
else
{
2021-12-08 17:57:50 +00:00
manufacturer = column0.Substring(0, cutOffset).Trim();
model = column0.Substring(cutOffset + 3).Trim();
2019-11-02 01:40:41 +00:00
}
2021-12-08 17:57:50 +00:00
}
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
switch(manufacturer)
{
case "Lite-ON":
manufacturer = "JLMS";
2019-11-02 23:52:33 +00:00
2021-12-08 17:57:50 +00:00
break;
case "LG Electronics":
manufacturer = "HL-DT-ST";
2019-11-02 23:52:33 +00:00
2021-12-08 17:57:50 +00:00
break;
case "Panasonic":
manufacturer = "MATSHITA";
2019-11-02 23:52:33 +00:00
2021-12-08 17:57:50 +00:00
break;
}
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
CompactDiscOffset cdOffset =
ctx.CdOffsets.FirstOrDefault(o => o.Manufacturer == manufacturer && o.Model == model);
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
if(column1.ToLowerInvariant() == "[purged]")
{
if(cdOffset != null)
ctx.CdOffsets.Remove(cdOffset);
2019-11-02 23:52:33 +00:00
2021-12-08 17:57:50 +00:00
continue;
}
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
if(!short.TryParse(column1, out short offset))
continue;
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
if(!int.TryParse(column2, out int submissions))
continue;
2019-11-02 23:52:33 +00:00
2021-12-08 17:57:50 +00:00
if(column3[^1] != '%')
continue;
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
column3 = column3.Substring(0, column3.Length - 1);
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
if(!float.TryParse(column3, out float percentage))
continue;
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
percentage /= 100;
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
if(cdOffset is null)
{
cdOffset = new CompactDiscOffset
2019-11-02 01:40:41 +00:00
{
2021-12-08 17:57:50 +00:00
AddedWhen = DateTime.UtcNow,
ModifiedWhen = DateTime.UtcNow,
Agreement = percentage,
Manufacturer = manufacturer,
Model = model,
Offset = offset,
Submissions = submissions
};
ctx.CdOffsets.Add(cdOffset);
addedOffsets++;
}
else
{
if(Math.Abs(cdOffset.Agreement - percentage) > 0)
2019-11-02 01:40:41 +00:00
{
2021-12-08 17:57:50 +00:00
cdOffset.Agreement = percentage;
cdOffset.ModifiedWhen = DateTime.UtcNow;
2019-11-02 01:40:41 +00:00
}
2021-12-08 17:57:50 +00:00
if(cdOffset.Offset != offset)
2019-11-02 01:40:41 +00:00
{
2021-12-08 17:57:50 +00:00
cdOffset.Offset = offset;
cdOffset.ModifiedWhen = DateTime.UtcNow;
}
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
if(cdOffset.Submissions != submissions)
{
cdOffset.Submissions = submissions;
cdOffset.ModifiedWhen = DateTime.UtcNow;
2019-11-02 01:40:41 +00:00
}
2021-12-08 17:57:50 +00:00
if(Math.Abs(cdOffset.Agreement - percentage) > 0 ||
cdOffset.Offset != offset ||
cdOffset.Submissions != submissions)
modifiedOffsets++;
}
2019-11-02 01:40:41 +00:00
2021-12-08 17:57:50 +00:00
foreach(Device device in ctx.Devices.
Where(d => d.Manufacturer == null && d.Model != null &&
d.Model.Trim() == model).
Union(ctx.Devices.Where(d => d.Manufacturer != null &&
d.Manufacturer.Trim() == manufacturer &&
d.Model != null && d.Model == model)))
{
2021-12-08 17:57:50 +00:00
if(device.CdOffset == cdOffset &&
device.ModifiedWhen == cdOffset.ModifiedWhen)
continue;
2021-12-08 17:57:50 +00:00
device.CdOffset = cdOffset;
device.ModifiedWhen = cdOffset.ModifiedWhen;
}
}
2021-12-08 17:57:50 +00:00
end = DateTime.UtcNow;
System.Console.WriteLine("{0}: Took {1:F2} seconds", end, (end - start).TotalSeconds);
if(File.Exists("drive_offsets.json"))
{
var sr = new StreamReader("drive_offsets.json");
CompactDiscOffset[] offsets = JsonSerializer.Deserialize<CompactDiscOffset[]>(sr.ReadToEnd());
if(offsets != null)
{
foreach(CompactDiscOffset offset in offsets)
{
2021-12-08 17:57:50 +00:00
CompactDiscOffset cdOffset =
ctx.CdOffsets.FirstOrDefault(o => o.Manufacturer == offset.Manufacturer &&
o.Model == offset.Model);
if(cdOffset is null)
{
2021-12-08 17:57:50 +00:00
offset.ModifiedWhen = DateTime.UtcNow;
2020-03-01 02:28:07 +00:00
2021-12-08 17:57:50 +00:00
ctx.CdOffsets.Add(offset);
addedOffsets++;
}
else
{
if(Math.Abs(cdOffset.Agreement - offset.Agreement) > 0 ||
offset.Agreement < 0)
2020-03-01 02:28:07 +00:00
{
2021-12-08 17:57:50 +00:00
cdOffset.Agreement = offset.Agreement;
cdOffset.ModifiedWhen = DateTime.UtcNow;
}
2020-03-01 02:28:07 +00:00
2021-12-08 17:57:50 +00:00
if(cdOffset.Offset != offset.Offset)
{
cdOffset.Offset = offset.Offset;
cdOffset.ModifiedWhen = DateTime.UtcNow;
2020-03-01 02:28:07 +00:00
}
2021-12-08 17:57:50 +00:00
if(cdOffset.Submissions != offset.Submissions)
2020-03-01 02:28:07 +00:00
{
2021-12-08 17:57:50 +00:00
cdOffset.Submissions = offset.Submissions;
cdOffset.ModifiedWhen = DateTime.UtcNow;
2020-03-01 02:28:07 +00:00
}
2021-12-08 17:57:50 +00:00
if(Math.Abs(cdOffset.Agreement - offset.Agreement) > 0 ||
cdOffset.Offset != offset.Offset ||
cdOffset.Submissions != offset.Submissions)
modifiedOffsets++;
}
}
}
2021-12-08 17:57:50 +00:00
}
2021-12-08 17:57:50 +00:00
System.Console.WriteLine("{0}: Committing changes...", DateTime.UtcNow);
start = DateTime.UtcNow;
ctx.SaveChanges();
end = DateTime.UtcNow;
System.Console.WriteLine("{0}: Took {1:F2} seconds", end, (end - start).TotalSeconds);
2019-11-02 01:40:41 +00:00
2023-10-04 00:09:19 +01:00
System.Console.WriteLine("{0}: Added {1} offsets", end, addedOffsets);
2021-12-08 17:57:50 +00:00
System.Console.WriteLine("{0}: Modified {1} offsets", end, modifiedOffsets);
}
catch(Exception ex)
{
#if DEBUG
if(Debugger.IsAttached)
throw;
#endif
System.Console.WriteLine("{0}: Exception {1} filling CompactDisc read offsets...", DateTime.UtcNow, ex);
2019-11-02 01:40:41 +00:00
}
if(!Directory.Exists("nes"))
return;
System.Console.WriteLine("{0}: Reading iNES/NES 2.0 headers...", DateTime.UtcNow);
start = DateTime.UtcNow;
2023-10-04 00:09:19 +01:00
var newHeaders = 0;
var updatedHeaders = 0;
counter = 0;
foreach(string file in Directory.GetFiles("nes"))
{
try
{
var fs = new FileStream(file, FileMode.Open, FileAccess.Read);
if(fs.Length <= 16)
continue;
2023-10-04 00:09:19 +01:00
var header = new byte[16];
var data = new byte[fs.Length - 16];
fs.Read(header, 0, 16);
2023-10-04 00:09:19 +01:00
fs.Read(data, 0, data.Length);
bool ines;
bool nes20;
ines = header[0] == 'N' && header[1] == 'E' && header[2] == 'S' && header[3] == 0x1A;
nes20 = ines && (header[7] & 0x0C) == 0x08;
if(!ines)
continue;
counter++;
var info = new NesHeaderInfo
{
NametableMirroring = (header[6] & 0x1) == 0x1,
BatteryPresent = (header[6] & 0x2) == 0x2,
FourScreenMode = (header[6] & 0x8) == 0x8,
Mapper = (ushort)(header[6] >> 4),
ConsoleType = (NesConsoleType)(header[7] & 0x3)
};
info.Mapper += (ushort)(header[7] & 0xF0);
if(nes20)
{
info.Mapper += (ushort)((header[8] & 0xF) << 8);
info.Submapper = (byte)(header[8] >> 4);
info.TimingMode = (NesTimingMode)(header[12] & 0x3);
switch(info.ConsoleType)
{
case NesConsoleType.Vs:
info.VsPpuType = (NesVsPpuType)(header[13] & 0xF);
info.VsHardwareType = (NesVsHardwareType)(header[13] >> 4);
break;
case NesConsoleType.Extended:
info.ExtendedConsoleType = (NesExtendedConsoleType)(header[13] & 0xF);
break;
}
info.DefaultExpansionDevice = (NesDefaultExpansionDevice)header[15];
}
var hasher = SHA256.Create();
byte[] hashBytes = hasher.ComputeHash(data);
2023-10-04 00:09:19 +01:00
var hashChars = new char[64];
2023-10-04 00:09:19 +01:00
for(var i = 0; i < 32; i++)
{
int a = hashBytes[i] >> 4;
int b = hashBytes[i] & 0xF;
hashChars[i * 2] = a > 9 ? (char)(a + 0x57) : (char)(a + 0x30);
2023-10-04 00:09:19 +01:00
hashChars[i * 2 + 1] = b > 9 ? (char)(b + 0x57) : (char)(b + 0x30);
}
info.Sha256 = new string(hashChars);
NesHeaderInfo existing = ctx.NesHeaders.FirstOrDefault(h => h.Sha256 == info.Sha256);
if(existing == null)
{
info.AddedWhen = DateTime.UtcNow;
info.ModifiedWhen = info.AddedWhen;
ctx.NesHeaders.Add(info);
newHeaders++;
continue;
}
2023-10-04 00:09:19 +01:00
var modified = false;
if(existing.NametableMirroring != info.NametableMirroring)
{
existing.NametableMirroring = info.NametableMirroring;
modified = true;
}
if(existing.BatteryPresent != info.BatteryPresent)
{
existing.BatteryPresent = info.BatteryPresent;
modified = true;
}
if(existing.FourScreenMode != info.FourScreenMode)
{
existing.FourScreenMode = info.FourScreenMode;
modified = true;
}
if(existing.Mapper != info.Mapper)
{
existing.Mapper = info.Mapper;
modified = true;
}
if(existing.ConsoleType != info.ConsoleType)
{
existing.ConsoleType = info.ConsoleType;
modified = true;
}
if(existing.Submapper != info.Submapper)
{
existing.Submapper = info.Submapper;
modified = true;
}
if(existing.TimingMode != info.TimingMode)
{
existing.TimingMode = info.TimingMode;
modified = true;
}
if(existing.VsPpuType != info.VsPpuType)
{
existing.VsPpuType = info.VsPpuType;
modified = true;
}
if(existing.VsHardwareType != info.VsHardwareType)
{
existing.VsHardwareType = info.VsHardwareType;
modified = true;
}
if(existing.ExtendedConsoleType != info.ExtendedConsoleType)
{
existing.ExtendedConsoleType = info.ExtendedConsoleType;
modified = true;
}
if(existing.DefaultExpansionDevice != info.DefaultExpansionDevice)
{
existing.DefaultExpansionDevice = info.DefaultExpansionDevice;
modified = true;
}
if(!modified)
continue;
existing.ModifiedWhen = DateTime.UtcNow;
updatedHeaders++;
}
catch(Exception ex)
{
#if DEBUG
if(Debugger.IsAttached)
throw;
#endif
System.Console.WriteLine("{0}: Exception {1} with file {2}...", DateTime.UtcNow, ex, file);
}
}
end = DateTime.UtcNow;
System.Console.WriteLine("{0}: Took {1:F2} seconds", end, (end - start).TotalSeconds);
System.Console.WriteLine("{0}: Processed {1} iNES/NES 2.0 headers...", DateTime.UtcNow, counter);
2023-10-04 00:09:19 +01:00
System.Console.WriteLine("{0}: Added {1} iNES/NES 2.0 headers...", DateTime.UtcNow, newHeaders);
System.Console.WriteLine("{0}: Updated {1} iNES/NES 2.0 headers...", DateTime.UtcNow, updatedHeaders);
System.Console.WriteLine("{0}: Committing changes...", DateTime.UtcNow);
start = DateTime.UtcNow;
ctx.SaveChanges();
end = DateTime.UtcNow;
System.Console.WriteLine("{0}: Took {1:F2} seconds", end, (end - start).TotalSeconds);
2019-11-02 01:40:41 +00:00
}
}