admin管理员组

文章数量:1341758

c#软件注册

    • 整体思路
    • 生成机器码
    • 服务端生成密钥
    • 客户端根据密钥注册软件
    • 在打开软件的时候,判断是否已经注册/过期

整体思路

服务端
根据客户端的唯一机器码,和一些扩展字段(有效期、角色等),加密生成密钥。
客户端:
根据服务端生成的密钥进行注册,注册成功后将密钥存储在注册表;
软件启动时判断注册表是否存在对应的键值,取出密钥用进行解密,判断机器码是否一致、是否过期。

生成机器码

根据硬件信息生成唯一的机器码(要进行加密),有些同品牌的电脑生成的机器码可能会相同,实际使用中可以把mac网络地址加上。

/// <summary>
       /// MD5哈希加密
       /// </summary>
       /// <param name="scr">原始string数据</param>
       /// <returns>加密后的数据</returns>
       private static string MD5(string scr)
       {
           MD5 md5 = new MD5CryptoServiceProvider();
           byte[] palindata = Encoding.Default.GetBytes(scr);//将要加密的字符串转换为字节数组
           byte[] encryptdata = md5.ComputeHash(palindata);//将字符串加密后也转换为字符数组
           return GetHexString(encryptdata);//将加密后的字节数组转换为加密字符串
       }
          /// <summary>
       /// 获取机器码
       /// </summary>
       /// <returns></returns>
       public static string GetMachineCode()
       {
           string MachineCode = string.Empty;
           if (string.IsNullOrEmpty(MachineCode))
           {
               string omsg = " CPU >> " + CpuId() + " BIOS >> " +
                       BiosId() + " BASE >> " + BaseId() + " VIDEO >> " + VideoId() + " MAC >> " + MacId();
               MachineCode = MD5(omsg);
           }
           return MachineCode;
       }
       /// <summary>
       /// byte[]转换成十六进制
       /// </summary>
       /// <param name="bt"></param>
       /// <returns></returns>
       private static string GetHexString(byte[] bt)
       {
           string s = string.Empty;
           for (int i = 0; i < bt.Length; i++)
           {
               byte b = bt[i];
               int n, n1, n2;
               n = b;
               n1 = n & 15;
               n2 = n >> 4 & 15;
               if (n2 > 9)
                   s += ((char)(n2 - 10 + 'A')).ToString();
               else
                   s += n2.ToString();
               if (n1 > 9)
                   s += ((char)(n1 - 10 + 'A')).ToString();
               else
                   s += n1.ToString();
               if (i + 1 != bt.Length && (i + 1) % 2 == 0) s += "-";
           }
           return s;
       }

       public static string CpuId()
       {
           //Uses first CPU identifier available in order of preference
           //Don't get all identifiers, as it is very time consuming
           string retVal = identifier("Win32_Processor", "UniqueId");
           if (retVal == "") //If no UniqueID, use ProcessorID
           {
               retVal = identifier("Win32_Processor", "ProcessorId");
               if (retVal == "") //If no ProcessorId, use Name
               {
                   retVal = identifier("Win32_Processor", "Name");
                   if (retVal == "") //If no Name, use Manufacturer
                   {
                       retVal = identifier("Win32_Processor", "Manufacturer");
                   }
                   //Add clock speed for extra security
                   retVal += identifier("Win32_Processor", "MaxClockSpeed");
               }
           }
           return retVal;
       }
       //BIOS Identifier
       public static string BiosId()
       {
           return identifier("Win32_BIOS", "Manufacturer")
           + identifier("Win32_BIOS", "SMBIOSBIOSVersion")
           + identifier("Win32_BIOS", "IdentificationCode")
           + identifier("Win32_BIOS", "SerialNumber")
           + identifier("Win32_BIOS", "ReleaseDate")
           + identifier("Win32_BIOS", "Version");
       }
       //Main physical hard drive ID
       public static string DiskId()
       {
           return identifier("Win32_DiskDrive", "Model")
           + identifier("Win32_DiskDrive", "Manufacturer")
           + identifier("Win32_DiskDrive", "Signature")
           + identifier("Win32_DiskDrive", "TotalHeads");
       }
       //Motherboard ID
       public static string BaseId()
       {
           return identifier("Win32_BaseBoard", "Model")
           + identifier("Win32_BaseBoard", "Manufacturer")
           + identifier("Win32_BaseBoard", "Name")
           + identifier("Win32_BaseBoard", "SerialNumber");
       }
       //Primary video controller ID
       public static string VideoId()
       {
           return identifier("Win32_VideoController", "DriverVersion")
           + identifier("Win32_VideoController", "Name");
       }
       //First enabled network card ID
       public static string MacId()
       {
           return identifier("Win32_NetworkAdapterConfiguration",
               "MACAddress", "IPEnabled");
       }
       //Return a hardware identifier
       private static string identifier(string wmiClass, string wmiProperty, string wmiMustBeTrue)
       {
           string result = "";
           ManagementClass mc = new ManagementClass(wmiClass);
           ManagementObjectCollection moc = mc.GetInstances();
           foreach (ManagementObject mo in moc)
           {
               if (mo[wmiMustBeTrue].ToString() == "True")
               {
                   //Only get the first one
                   if (result == "")
                   {
                       try
                       {
                           result = mo[wmiProperty].ToString();
                           break;
                       }
                       catch
                       {
                       }
                   }
               }
           }
           return result;
       }
       //Return a hardware identifier
       private static string identifier(string wmiClass, string wmiProperty)
       {
           string result = "";
           ManagementClass mc = new ManagementClass(wmiClass);
           ManagementObjectCollection moc = mc.GetInstances();
           foreach (ManagementObject mo in moc)
           {
               //Only get the first one
               if (result == "")
               {
                   try
                   {
                       result = mo[wmiProperty]?.ToString();
                       break;
                   }
                   catch
                   {
                   }
               }
           }
           return result;
       }

服务端生成密钥

这里我们先创建一个实体Model,后面需要把对象序列化,所以要加上Serializable。

[Serializable]
   public class LicenseModel
   {
       //客户机器唯一识别码,由客户端生成
       public string CustomMachineCode { get; set; }
       //过期时间expire
       public DateTime ExpireTime { get; set; }
   }
这里只有机器码和有效期两个属性,可以根据自己项目需求扩展。

下面就是生成密钥的关键代码:

/// <summary>
       /// 服务端生成License文本  可授权给客户
       /// </summary>
       /// <param name="lic">LicenseModel对象,由客户提供机器码,并由商业提供期限和权限角色</param>
       /// <returns></returns>
       public static string CreateLicenseString(LicenseModel lic)
       {
           byte[] licByte = SerializeHelper.SerializeToBinary(lic);
           return EncodeHelper.AES(Convert.ToBase64String(licByte), LicenseManage.aeskey);
       }
        const string AesIV_128 = "liumao设计";//定义默认加密密钥 16个字节 Unicode编码为8个英文或汉字
       /// <summary>
       /// 按指定键值进行AES加密
       /// </summary>
       /// <param name="plainText">要解密字符</param>
       /// <param name="strKey">加密时使用的键值 Unicode编码 必须大于或等于8个英文或汉字</param>
       /// <returns></returns>
       public static string AES(string strContent, string strKey)
       {
           if (string.IsNullOrEmpty(strContent)) return string.Empty;
           if (strKey.Length > 8) strKey = strKey.Substring(0, 8);

           using (AesCryptoServiceProvider aesAlg = new AesCryptoServiceProvider())
           {
               aesAlg.Key = Encoding.Unicode.GetBytes(strKey);
               aesAlg.IV = Encoding.Unicode.GetBytes(AesIV_128);
               ICryptoTransform encryptor = aesAlg.CreateEncryptor(aesAlg.Key, aesAlg.IV);
               using (MemoryStream msEncrypt = new MemoryStream())
               {
                   using (CryptoStream csEncrypt = new CryptoStream(msEncrypt, encryptor, CryptoStreamMode.Write))
                   {
                       using (StreamWriter swEncrypt = new StreamWriter(csEncrypt))
                       {
                           swEncrypt.Write(strContent);
                       }
                       return Convert.ToBase64String(msEncrypt.ToArray()); //返回Base64密文方便传输
                   }
               }
           }
       }
       /// <summary>
       /// 按指定键值进行AES解密
       /// </summary>
       /// <param name="strContent">要解密字符</param>
       /// <param name="strKey">加密时使用的键值 Unicode编码 必须大于或等于8个英文或汉字</param>
       /// <returns></returns>
       public static string AESDecrypt(string strContent, string strKey)
       {
           if (string.IsNullOrEmpty(strContent)) return string.Empty;
           if (strKey.Length > 8) strKey = strKey.Substring(0, 8);
           //与加密时Base64对应
           byte[] byEnc;
           try
           {
               byEnc = Convert.FromBase64String(strContent);
               //解密
               using (AesCryptoServiceProvider aesAlg = new AesCryptoServiceProvider())
               {
                   aesAlg.Key = Encoding.Unicode.GetBytes(strKey);
                   aesAlg.IV = Encoding.Unicode.GetBytes(AesIV_128);

                   // Create a decryptor to perform the stream transform.
                   ICryptoTransform decryptor = aesAlg.CreateDecryptor(aesAlg.Key, aesAlg.IV);

                   // Create the streams used for decryption.
                   using (MemoryStream msDecrypt = new MemoryStream(byEnc))
                   {
                       using (CryptoStream csDecrypt = new CryptoStream(msDecrypt, decryptor, CryptoStreamMode.Read))
                       {
                           using (StreamReader srDecrypt = new StreamReader(csDecrypt))
                           {
                               return srDecrypt.ReadToEnd();
                           }
                       }
                   }
               }
           }
           catch
           {
               return string.Empty;
           }
       }

客户端根据密钥注册软件

注册成功后将密钥存储在注册表

/// <summary>
      /// 客户端验证License,存储
      /// </summary>
      /// <param name="lic">服务端授权给客户的License密文</param>
      /// <returns></returns>
      public static bool VerifyLicense(string lic)
      {
          if (string.IsNullOrEmpty(lic)) return false;
          try
          {
              string strlic = EncodeHelper.AESDecrypt(lic, aeskey);
              byte[] licbyte = Convert.FromBase64String(strlic);
              LicenseModel lm = SerializeHelper.DeserializeWithBinary<LicenseModel>(licbyte);
              //简单验证机器码、期限。具体角色权限限制需要在业务系统中实现。
              if (VerifyLicense(lm))
              {
                  ApplicationLicense = lm;
                  return true;
              }

          }
          catch (Exception ex)
          {
              throw ex;
          }
          //否则直接返回原始试用版
          return false;
      }
      /// <summary>
      /// 简单验证licensemode对象是否合法,不存储
      /// </summary>
      /// <param name="licmod"></param>
      /// <returns></returns>
      public static bool VerifyLicense(LicenseModel licmod)
      {
          //简单验证机器码、role、期限。具体角色权限限制需要在业务系统中实现。
          bool isHaveRight = false;
          if (licmod.CustomMachineCode == ComputerCode.GetMachineCode())
          {
              if (licmod.ExpireTime > DateTime.Now) 
                  isHaveRight = true;
          }
          //写入注册表
          LicencesRegister.WriteRegedit(regeditkey, LicencesRegister.CreateLicenseString(licmod));
          return isHaveRight;
      }
/// <summary>
      /// 将二进制数据反序列化为指定类型对象
      /// </summary>
      /// <typeparam name="T"></typeparam>
      /// <param name="data"></param>
      /// <returns></returns>
      public static T DeserializeWithBinary<T>(byte[] data)
      {
          return (T)DeserializeWithBinary(data);
      }
/// <summary>
      /// 向你的软件注册表菜单下添加键值
      /// </summary>
      /// <param name="keyname">键名</param>
      /// <param name="keyvalue">值</param>
      public static void WriteRegedit(string keyname, string keyvalue)
      {
          RegistryKey software = Registry.LocalMachine.OpenSubKey("SOFTWARE", RegistryKeyPermissionCheck.ReadWriteSubTree, System.Security.AccessControl.RegistryRights.FullControl);

          RegistryKey aimdir;
          if (!IsYourSoftkeyExit()) //不存在则创建
          {
              aimdir = software.CreateSubKey(YourSoftName);
          }
          else //存在则open
          {
              aimdir = software.OpenSubKey(YourSoftName, true);
          }
          aimdir.SetValue(keyname, keyvalue, RegistryValueKind.String);
          aimdir.Close();
      }

在打开软件的时候,判断是否已经注册/过期

从注册表取出密钥,没有的话表示没有注册,有密钥的话就验证密钥是否合规或者是否过期

public void IsRegister()
     {
         ComputerCode.GetMachineCode();
         try
         {
             bool active = false;
             var licenseModel = LicenseManage.GetLicense();
             if (licenseModel != null)
             {
                 active = LicenseManage.VerifyLicense(licenseModel);
             }
             if (active)
                 this.title.Text = "软件已激活";
             else
                 this.title.Text = "激活失败";
         }
         catch (Exception ex)
         {
             return;
         }
     }
/// <summary>
     /// 客户端获取本地的license  根据自己设计的存储介质,可以是从文件中取、也可以是注册表或远程服务器上取。
     /// </summary>
     /// <returns></returns>
     public static LicenseModel GetLicense()
     {
         if (ApplicationLicense != null) return ApplicationLicense;

         try
         {
             //如果以前装过,则从注册表取值 这里可以改成从数据库、文件、或服务端
             //未取到键则建一个
             if (!LicencesRegister.IsRegeditExit(regeditkey))
             {
                 //第一次使用默认是试用版7天
                 LicenseModel license = new LicenseModel()
                 {
                     CustomMachineCode = ComputerCode.GetMachineCode(),
                     ExpireTime = DateTime.Now.AddDays(7)
                 };
                 LicencesRegister.WriteRegedit(regeditkey, LicencesRegister.CreateLicenseString(license));
                 ApplicationLicense = license;
             }
             else
             {
                 string licFromReg = LicencesRegister.GetRegistData(regeditkey);
                 try
                 {
                     string strlic = EncodeHelper.AESDecrypt(licFromReg, aeskey);
                     byte[] licbyte = Convert.FromBase64String(strlic);
                     LicenseModel lm = SerializeHelper.DeserializeWithBinary<LicenseModel>(licbyte);
                     //取到的值还原license并返回
                     ApplicationLicense = lm;
                 }
                 catch (Exception ex1)
                 {
                     //_log.Error(ex1);
                     //如果从注册表中取到的值发现被篡改,则直接试用版到期,不给使用。
                     LicenseModel licenseErr = new LicenseModel()
                     {
                         CustomMachineCode = ComputerCode.GetMachineCode(),
                         ExpireTime = DateTime.Now
                     };
                 }
             }
         }
         catch (Exception ex)
         {
             throw ex;
         }
         return ApplicationLicense;
     }
/// <summary>
     /// 简单验证licensemode对象是否合法,不存储
     /// </summary>
     /// <param name="licmod"></param>
     /// <returns></returns>
     public static bool VerifyLicense(LicenseModel licmod)
     {
         //简单验证机器码、role、期限。具体角色权限限制需要在业务系统中实现。
         bool isHaveRight = false;
         if (licmod.CustomMachineCode == ComputerCode.GetMachineCode())
         {
             if (licmod.ExpireTime > DateTime.Now) 
                 isHaveRight = true;
         }
         //写入注册表
         LicencesRegister.WriteRegedit(regeditkey, LicencesRegister.CreateLicenseString(licmod));
         return isHaveRight;
     }

想要完整代码可以联系我哦!
如需转载请注明出处,尊重劳动成果!

本文标签: 软件