212 lines
7.3 KiB
C#
212 lines
7.3 KiB
C#
using System;
|
|
using System.IO;
|
|
using System.Numerics;
|
|
using System.Security.Cryptography;
|
|
using System.Text;
|
|
|
|
namespace ipn_sqlclr
|
|
{
|
|
enum SerialNumberChunks : byte
|
|
{
|
|
Version = 0x01, // 1 byte of data - version
|
|
UserName = 0x02, // 1 + N bytes - length + N bytes of customer's name (without enging \0).
|
|
Email = 0x03, // 1 + N bytes - length + N bytes of customer's email (without ending \0).
|
|
ProductCode = 0x07, // 8 bytes - used for decrypting some parts of exe-file
|
|
UserData = 0x08, // 1 + N bytes - length + N bytes of user data
|
|
MaxBuild = 0x09, // 4 bytes - (year << 16) + (month << 8) + (day)
|
|
End = 0xFF // 4 bytes - checksum: the first four bytes of sha-1 hash from the data before that chunk
|
|
};
|
|
|
|
public static class Rsa
|
|
{
|
|
private const string PublicExpB64 = "AAEAAQ==";
|
|
private const string PrivateExpB64 = "CXHXWx/Z9JqetQWwFpvmD72wrDiqQOXMQs18fhAMjWCfJ/f2r3p2io+iB3gqIuu3LGH3WJ8PQuIzvDMnbwAx+8BbAyYhWhGEbxDdifndjQ2KlDV2Hu8NQgCbc5Wjok0rKwQ+Bxeb2i1+Gu3FsnhRNv9RhSyiwcnH/4Q3+ySE3AFAcAUwuQABePjDKCYOfIyx7RKz5h0sG+v10nkPuuCGPSnh+AXDTBIJFH+yNIjkrfweC9A3dv7URyRJumAMgm/SnDU76rTkFw9vZpupQeMtMtIsZIkeFSngip9KImD5zzbb2vKD63Cg9W/Yvqgvro/d+cR5n6P0t4DzfanNIFRGpFrX8/Q5VjuezDKw/4YbsFYwOhzJPRxglmCEjh8cpfxJ11cUXa/hNBV4c4Dp29D0F+w01OlBnFb1Ck9VXur2qJCsqcWtjsnt/VITsxa1jzr+3C2+uvaI4JSd7yLEnTqSaSsRfWuhDXgjY/YWhmyvMzeQeXBGOXKt2j2lY2Fm0WJx";
|
|
private const string ModulusB64 = "pwUqwaM8IOukyx06Lvi5YNQ70JE7pwg7K+pmM/vCe1CUseHKFM1v1m11geDjVsAt38AnaiFs3JhtTs80ySCIxOSyvMw6Cd52k6N6dn7LAx1mxQLJLhYeMMJYbplMHnMLwYN0+IO58OVbEqRyaJV2ExolnK2EYZL7QRXujGY7/sOoOMF3p6GsWJK6kkBJICIoL9hHWBQMO6/9rmls/+EhaWuP80Vx0+H2OlrQ58K+TJeyE393cvb4QufiEPpCNaB50Klee9QUnsjSW/bTnmGn4Bi5+cowRbawUY73Q5I58fMAXiH9ueDPuNMR9YKDgW9GxunLmYkbuwqIp/v7kw3cfMBM0ihhB0B8UhjyAMAGLzJWX3H/H6Zrz41g9PbPjTAxfsTaCrxoqjaTaO4zk9YsI//VX9Fhivcy913SevBpNandziGfYH/oHW2xDy9AfwkE1wuIBlLj7c/k8U1YmmRAmkoCzlmB7EU4ClNltboh1uARUQ6wW30upppnuYhGkTy7";
|
|
|
|
static BigInteger B2Bi(byte[] b) //reverse & make positive
|
|
{
|
|
Array.Reverse(b);
|
|
var b2 = new byte[b.Length + 1];
|
|
Array.Copy(b, b2, b.Length);
|
|
return new BigInteger(b2);
|
|
}
|
|
|
|
private static readonly BigInteger PublicExp = B2Bi(Convert.FromBase64String(PublicExpB64));
|
|
private static readonly BigInteger PrivateExp = B2Bi(Convert.FromBase64String(PrivateExpB64));
|
|
private static readonly BigInteger Modulus = B2Bi(Convert.FromBase64String(ModulusB64));
|
|
|
|
public static byte[] Encrypt(byte[] paddedData)
|
|
{
|
|
var x = B2Bi(paddedData);
|
|
var y = BigInteger.ModPow(x, PrivateExp, Modulus);
|
|
|
|
byte[] ret = y.ToByteArray();
|
|
Array.Resize(ref ret, paddedData.Length);
|
|
Array.Reverse(ret);
|
|
return ret;
|
|
}
|
|
|
|
public static byte[] Decrypt(byte[] data)
|
|
{
|
|
var x = B2Bi(data);
|
|
var y = BigInteger.ModPow(x, PublicExp, Modulus);
|
|
|
|
byte[] ret = y.ToByteArray();
|
|
Array.Reverse(ret);
|
|
return ret;
|
|
}
|
|
|
|
}
|
|
public static class Keygen
|
|
{
|
|
public static void ParseKey(string key, out int productId, out string customerName, out string eMail, out DateTime maxBuildDt)
|
|
{
|
|
productId = -1;
|
|
customerName = null;
|
|
eMail = null;
|
|
maxBuildDt = new DateTime();
|
|
|
|
var crypted = Convert.FromBase64String(key);
|
|
var data = Rsa.Decrypt(crypted);
|
|
int i;
|
|
for (i = 2; i < data.Length && data[i] != 0; i++) {
|
|
}
|
|
|
|
i++;
|
|
var pos = i;
|
|
while (pos < data.Length)
|
|
{
|
|
var b = data[pos++];
|
|
switch (b)
|
|
{
|
|
case (byte) SerialNumberChunks.Version:
|
|
b = data[pos++];
|
|
if (b < 1 || b > 2)
|
|
throw new InvalidDataException("SerialNumberChunks.Version");
|
|
break;
|
|
case (byte) SerialNumberChunks.UserName:
|
|
b = data[pos++];
|
|
customerName = Encoding.UTF8.GetString(data, pos, b);
|
|
pos += b;
|
|
break;
|
|
case (byte) SerialNumberChunks.Email:
|
|
b = data[pos++];
|
|
eMail = Encoding.UTF8.GetString(data, pos, b);
|
|
pos += b;
|
|
break;
|
|
case (byte)SerialNumberChunks.ProductCode:
|
|
pos += 8;
|
|
break;
|
|
case (byte) SerialNumberChunks.UserData:
|
|
b = data[pos++];
|
|
if (b == 0)
|
|
productId = 0;
|
|
else if(b != 1)
|
|
throw new InvalidDataException("Invalid ProductID");
|
|
else
|
|
productId = data[pos];
|
|
pos += b;
|
|
break;
|
|
case (byte) SerialNumberChunks.MaxBuild:
|
|
maxBuildDt = new DateTime(data[pos + 2] + 256 * data[pos + 3], data[pos + 1],data[pos]);
|
|
pos += 4;
|
|
break;
|
|
case (byte) SerialNumberChunks.End:
|
|
if (pos + 4 > data.Length)
|
|
throw new InvalidDataException("No checksum");
|
|
{
|
|
SHA1 sha = new SHA1Managed();
|
|
sha.Initialize();
|
|
var hash = sha.ComputeHash(data, i, pos - 1 - i);
|
|
for (int j = 0; j < 4; j++)
|
|
{
|
|
if(data[pos + j] == hash[3 - j])
|
|
continue;
|
|
throw new InvalidDataException("Invalid checksum");
|
|
}
|
|
}
|
|
return;
|
|
}
|
|
}
|
|
throw new InvalidDataException("No checksum");
|
|
}
|
|
|
|
public static string GenerateKey(int productId, string customerName, string eMail, DateTime maxBuildDt)
|
|
{
|
|
var data = new MemoryStream();
|
|
data.WriteByte((byte)SerialNumberChunks.Version);
|
|
data.WriteByte(1);
|
|
|
|
data.WriteByte((byte)SerialNumberChunks.UserName);
|
|
var utfCustomer = Encoding.UTF8.GetBytes(customerName);
|
|
if (utfCustomer.Length > 255)
|
|
throw new ArgumentException("Customer name too long", "customerName");
|
|
data.WriteByte((byte)utfCustomer.Length);
|
|
data.Write(utfCustomer, 0, utfCustomer.Length);
|
|
|
|
data.WriteByte((byte)SerialNumberChunks.Email);
|
|
byte[] utfeMail = Encoding.UTF8.GetBytes(eMail);
|
|
if (utfeMail.Length > 255)
|
|
throw new ArgumentException("EMail too long", "eMail");
|
|
data.WriteByte((byte)utfeMail.Length);
|
|
data.Write(utfeMail, 0, utfeMail.Length);
|
|
|
|
data.WriteByte((byte)SerialNumberChunks.ProductCode);
|
|
data.Write(new byte[] { 41, 65, 36, 150, 5, 175, 174, 137 }, 0, 8);
|
|
|
|
data.WriteByte((byte)SerialNumberChunks.UserData);
|
|
data.WriteByte(1);
|
|
data.WriteByte((byte)productId);
|
|
|
|
data.WriteByte((byte)SerialNumberChunks.MaxBuild);
|
|
data.WriteByte((byte)maxBuildDt.Day);
|
|
data.WriteByte((byte)maxBuildDt.Month);
|
|
data.WriteByte((byte)maxBuildDt.Year);
|
|
data.WriteByte((byte)(maxBuildDt.Year >> 8));
|
|
|
|
SHA1 sha = new SHA1Managed();
|
|
sha.Initialize();
|
|
data.Position = 0;
|
|
var hash = sha.ComputeHash(data);
|
|
data.WriteByte((byte)SerialNumberChunks.End);
|
|
data.WriteByte(hash[3]);
|
|
data.WriteByte(hash[2]);
|
|
data.WriteByte(hash[1]);
|
|
data.WriteByte(hash[0]);
|
|
|
|
const int minPadding = 8 + 3;
|
|
const int maxPadding = minPadding + 16;
|
|
const int maxBytes = 3072 / 8;
|
|
if (data.Length + minPadding > maxBytes)
|
|
throw new ApplicationException("Serial number too long");
|
|
|
|
var rnd = new Random();
|
|
var paddingBytes = rnd.Next(minPadding, maxPadding + 1);
|
|
if (data.Length + paddingBytes > maxBytes)
|
|
paddingBytes = maxBytes - (int)data.Length;
|
|
|
|
var paddedData = new byte[maxBytes];
|
|
var nonPaddedData = data.ToArray();
|
|
Array.Copy(nonPaddedData, paddedData, paddingBytes);
|
|
Array.Copy(nonPaddedData, 0, paddedData, paddingBytes, data.Length);
|
|
paddedData[0] = 0;
|
|
paddedData[1] = 2;
|
|
paddedData[paddingBytes - 1] = 0;
|
|
var i = 2;
|
|
for (; i < paddingBytes - 1; i++) {
|
|
byte b = 0;
|
|
while (b == 0) {
|
|
b = (byte)rnd.Next(256);
|
|
}
|
|
paddedData[i] = b;
|
|
}
|
|
i = nonPaddedData.Length + paddingBytes;
|
|
while (i < maxBytes) {
|
|
paddedData[i++] = (byte)rnd.Next(256);
|
|
}
|
|
|
|
var res = Convert.ToBase64String(Rsa.Encrypt(paddedData), Base64FormattingOptions.InsertLineBreaks);
|
|
return res;
|
|
}
|
|
}
|
|
}
|