using System;
|
using System.Collections.Generic;
|
using Top.Api.Util;
|
|
|
namespace Top.Api.Security
|
{
|
/// <summary>
|
/// 加、解密客户端(单例使用,不要初始化多个)
|
/// </summary>
|
public class SecurityClient : SecurityConstants
|
{
|
// 秘钥管理核心类
|
private SecurityCore secretCore;
|
private SecurityCounter securityCounter;
|
|
/// <summary>
|
/// 秘钥管理核心类
|
/// </summary>
|
/// <param name="topClientt"> serverUrl必须是https协议</param>
|
/// <param name="randomNum">伪随机码</param>
|
public SecurityClient(DefaultTopClient topClient, string randomNum): this(topClient, randomNum, false)
|
{
|
}
|
|
/// <summary>
|
/// 秘钥管理核心类
|
/// </summary>
|
/// <param name="topClientt"> serverUrl必须是https协议</param>
|
/// <param name="randomNum">伪随机码</param>
|
/// <param name="streetest">是否全链路压测</param>
|
public SecurityClient(DefaultTopClient topClient, string randomNum, bool streetest)
|
{
|
securityCounter = new SecurityCounter(topClient.appKey);
|
secretCore = new SecurityCore(topClient, randomNum, streetest);
|
}
|
|
/// <summary>
|
/// 初始化秘钥(如果半小时内会调用加、解密方法,建议先初始化秘钥)。所有用户共用秘钥
|
/// </summary>
|
public void InitSecret()
|
{
|
secretCore.GetSecret(null, null);
|
}
|
|
public void SetRandomNum(string randomNum)
|
{
|
this.secretCore.SetRandomNum(randomNum);
|
}
|
|
/// <summary>
|
/// 初始化秘钥(如果半小时内会调用加、解密方法,建议先初始化秘钥)。每个用户单独分配秘钥
|
/// </summary>
|
/// <param name="session"></param>
|
public void InitSecret(string session)
|
{
|
secretCore.GetSecret(session, null);
|
}
|
|
/// <summary>
|
/// 批量解密(所有用户共用秘钥)
|
/// </summary>
|
/// <param name="dataList"></param>
|
/// <param name="type"></param>
|
/// <returns></returns>
|
public IDictionary<string, string> Decrypt(List<string> dataList, string type)
|
{
|
return Decrypt(dataList, type, null);
|
}
|
|
/// <summary>
|
/// 批量解密(每个用户单独分配秘钥)
|
/// </summary>
|
/// <param name="dataList"></param>
|
/// <param name="type"></param>
|
/// <param name="session"></param>
|
/// <returns>key=密文数据,value=明文数据</returns>
|
public IDictionary<string, string> Decrypt(List<string> dataList, string type, string session)
|
{
|
if (dataList == null || dataList.Count == 0)
|
{
|
throw new SecretException("dataList can`t be empty");
|
}
|
IDictionary<string, string> resultMap = new Dictionary<string, string>();
|
foreach (string data in dataList)
|
{
|
if (!resultMap.ContainsKey(data))
|
{
|
string decryptValue = Decrypt(data, type, session);
|
resultMap.Add(data, decryptValue);
|
}
|
}
|
return resultMap;
|
}
|
|
/// <summary>
|
/// 解密(所有用户共用秘钥)
|
/// </summary>
|
/// <param name="data"></param>
|
/// <param name="type"></param>
|
/// <returns></returns>
|
public string Decrypt(string data, string type)
|
{
|
return Decrypt(data, type, null);
|
}
|
|
/// <summary>
|
/// 解密(每个用户单独分配秘钥)
|
/// </summary>
|
/// <param name="data">
|
/// 密文数据 手机号码格式:$手机号码前3位明文$base64(encrypt(phone后8位))$111$
|
/// simple格式:~base64(encrypt(nick))~111~
|
/// </param>
|
/// <param name="type">解密字段类型(例如:simple\phone)</param>
|
/// <param name="session">用户身份,用户级加密必填</param>
|
/// <returns></returns>
|
public string Decrypt(string data, string type, string session)
|
{
|
if (string.IsNullOrEmpty(data) || data.Length < 4)
|
{
|
return data;
|
}
|
|
// 获取分隔符
|
Nullable<char> charValue = null;
|
SecurityBiz.GetSeparatorCharMap().TryGetValue(type, out charValue);
|
|
if (charValue == null)
|
{
|
throw new SecretException("type error");
|
}
|
|
// 校验
|
char separator = charValue.Value;
|
if (!(data[0] == separator && data[data.Length - 1] == separator))
|
{
|
return data;
|
}
|
SecretData secretDataDO = null;
|
if (data[data.Length - 2] == separator)
|
{
|
secretDataDO = SecurityBiz.GetIndexSecretData(data, separator);
|
}
|
else
|
{
|
secretDataDO = SecurityBiz.GetSecretData(data, separator);
|
}
|
|
// 非法密文
|
if (secretDataDO == null)
|
{
|
return data;
|
}
|
|
// 如果密文数据的版本号小于0代表公共秘钥
|
if (secretDataDO.SecretVersion < 0)
|
{
|
secretDataDO.SecretVersion=Math.Abs(secretDataDO.SecretVersion.Value);
|
session = null;
|
}
|
securityCounter.AddDecryptCount(type, session);// 计数器
|
SecretContext secretContextDO = secretCore.GetSecret(session, secretDataDO.SecretVersion);
|
string decryptValue = SecurityUtil.AESDecrypt(secretDataDO.OriginalBase64Value, secretContextDO.Secret);
|
if (PHONE.Equals(type) && !secretDataDO.Search)
|
{
|
// 加上手机号前3位,手机号只加密了后8位
|
return secretDataDO.OriginalValue + decryptValue;
|
}
|
return decryptValue;
|
|
}
|
|
/// <summary>
|
/// 判断list元素是否全部为密文数据
|
/// </summary>
|
/// <param name="dataList"></param>
|
/// <param name="type">加密字段类型(例如:simple\phone)</param>
|
/// <returns></returns>
|
public static bool IsEncryptData(List<string> dataList, string type)
|
{
|
if (dataList == null || dataList.Count == 0)
|
{
|
return false;
|
}
|
bool result = false;
|
foreach (string data in dataList)
|
{
|
result = IsEncryptData(data, type);
|
if (!result)
|
{
|
return false;
|
}
|
}
|
return result;
|
}
|
|
/// <summary>
|
/// 判断list元素是否存在密文数据。只要有一个是密文,则返回true
|
/// </summary>
|
/// <param name="dataList"></param>
|
/// <param name="type">加密字段类型(例如:simple\phone)</param>
|
/// <returns></returns>
|
public static bool IsPartEncryptData(List<String> dataList, String type) {
|
if (dataList == null || dataList.Count == 0)
|
{
|
return false;
|
}
|
bool result = false;
|
foreach (string data in dataList)
|
{
|
result = IsEncryptData(data, type);
|
if (result)
|
{
|
return true;
|
}
|
}
|
return result;
|
}
|
|
/// <summary>
|
/// 判断是否密文数据
|
/// </summary>
|
/// <param name="data"></param>
|
/// <param name="type">加密字段类型(例如:simple\phone)</param>
|
/// <returns></returns>
|
public static bool IsEncryptData(string data, string type)
|
{
|
return SecurityBiz.IsEncryptData(data, type);
|
}
|
|
/// <summary>
|
/// 加密(所有用户共用秘钥)
|
/// </summary>
|
/// <param name="data"></param>
|
/// <param name="type"></param>
|
/// <returns></returns>
|
public string Encrypt(string data, string type)
|
{
|
return Encrypt(data, type, null, null);
|
}
|
|
/// <summary>
|
/// 用老秘钥加密,只在秘钥升级时使用(所有用户共用秘钥)
|
/// </summary>
|
/// <param name="data"></param>
|
/// <param name="type"></param>
|
/// <returns></returns>
|
public string EncryptPrevious(string data, string type)
|
{
|
return Encrypt(data, type, null, -1L);
|
}
|
|
/// <summary>
|
/// 加密(每个用户单独分配秘钥)
|
/// </summary>
|
/// <param name="data"></param>
|
/// <param name="type"></param>
|
/// <param name="session"></param>
|
/// <returns></returns>
|
public string Encrypt(string data, string type, string session)
|
{
|
return Encrypt(data, type, session, null);
|
}
|
|
/// <summary>
|
/// 用老秘钥加密,只在秘钥升级时使用(每个用户单独分配秘钥)
|
/// </summary>
|
/// <param name="data"></param>
|
/// <param name="type"></param>
|
/// <param name="session"></param>
|
/// <returns></returns>
|
public string EncryptPrevious(string data, string type, string session)
|
{
|
return Encrypt(data, type, session, -1L);
|
}
|
|
/// <summary>
|
/// 密文检索(所有用户共用秘钥)
|
/// </summary>
|
/// <param name="data"></param>
|
/// <param name="type"></param>
|
/// <returns></returns>
|
public string Search(string data, string type)
|
{
|
return Search(data, type, null, null);
|
}
|
|
/// <summary>
|
/// 密文检索,在秘钥升级场景下兼容查询(所有用户共用秘钥)
|
/// </summary>
|
/// <param name="data"></param>
|
/// <param name="type"></param>
|
/// <returns></returns>
|
public string SearchPrevious(string data, string type)
|
{
|
return Search(data, type, null, -1L);
|
}
|
|
/// <summary>
|
/// 密文检索(每个用户单独分配秘钥)
|
/// </summary>
|
/// <param name="data"></param>
|
/// <param name="type"></param>
|
/// <param name="session"></param>
|
/// <returns></returns>
|
public string Search(string data, string type, string session)
|
{
|
return Search(data, type, session, null);
|
}
|
|
/// <summary>
|
/// 密文检索,在秘钥升级场景下兼容查询(每个用户单独分配秘钥)
|
/// </summary>
|
/// <param name="data"></param>
|
/// <param name="type"></param>
|
/// <param name="session"></param>
|
/// <returns></returns>
|
public string SearchPrevious(string data, string type, string session)
|
{
|
return Search(data, type, session, -1L);
|
}
|
|
/// <summary>
|
/// 密文检索。 手机号码格式:$base64(H-MAC(phone后4位))$ simple格式:base64(H-MAC(滑窗))
|
/// </summary>
|
/// <param name="data">明文数据</param>
|
/// <param name="type">加密字段类型(例如:simple\phone)</param>
|
/// <param name="session">用户身份,用户级加密必填</param>
|
/// <param name="version">秘钥历史版本</param>
|
/// <returns></returns>
|
private string Search(string data, string type, string session, Nullable<Int64> version)
|
{
|
if (string.IsNullOrEmpty(data))
|
{
|
return data;
|
}
|
|
SecretContext secretContext = secretCore.GetSecret(session, version);
|
if (secretContext == null)
|
{
|
throw new SecretException("secretKey is null");
|
}
|
if (secretContext.Secret == null)
|
{
|
return data;
|
}
|
|
string separator = null;
|
SecurityBiz.GetSeparatorMap().TryGetValue(type, out separator);
|
if (separator == null)
|
{
|
throw new SecretException("type error");
|
}
|
|
// 公共秘钥版本号用负数区分
|
if (session == null)
|
{
|
SecretContext publicSecretContext = new SecretContext();
|
publicSecretContext.Secret = secretContext.Secret;
|
publicSecretContext.SecretVersion = -secretContext.SecretVersion;
|
secretContext = publicSecretContext;
|
}
|
|
securityCounter.AddSearchCount(type, session);// 计数器
|
if (PHONE.Equals(type))
|
{
|
return SecurityBiz.SearchPhoneIndex(data, separator, secretContext);
|
}
|
else
|
{
|
int compressLen = secretCore.GetCompressLen();
|
int slideSize = secretCore.GetSlideSize();
|
return SecurityBiz.SearchNormalIndex(data, compressLen, slideSize, secretContext);
|
}
|
|
}
|
|
/// <summary>
|
/// 加密之后格式。 手机号码格式:$手机号码前3位明文$base64(encrypt(phone后8位))$111$
|
/// simple格式:~base64(encrypt(nick))~111~
|
/// </summary>
|
/// <param name="data">明文数据</param>
|
/// <param name="type">加密字段类型(例如:simple\phone)</param>
|
/// <param name="session">用户身份,用户级加密必填</param>
|
/// <param name="version">秘钥历史版本</param>
|
/// <returns></returns>
|
private string Encrypt(string data, string type, string session, Nullable<Int64> version)
|
{
|
if (string.IsNullOrEmpty(data))
|
{
|
return data;
|
}
|
SecretContext secretContext = secretCore.GetSecret(session, version);
|
if (secretContext == null)
|
{
|
throw new SecretException("secretKey is null");
|
}
|
if (secretContext.Secret == null)
|
{
|
return data;
|
}
|
|
string separator = null;
|
SecurityBiz.GetSeparatorMap().TryGetValue(type, out separator);
|
if (separator == null)
|
{
|
throw new SecretException("type error");
|
}
|
|
// 公共秘钥版本号用负数区分
|
if (session == null)
|
{
|
SecretContext publicSecretContext = new SecretContext();
|
publicSecretContext.Secret = secretContext.Secret;
|
publicSecretContext.SecretVersion = -secretContext.SecretVersion;
|
secretContext = publicSecretContext;
|
}
|
|
securityCounter.AddEncryptCount(type, session);// 计数器
|
bool isEncryptIndex = secretCore.IsIndexEncrypt(type, version);
|
// 支持密文检索
|
if (isEncryptIndex || SEARCH.Equals(type))
|
{
|
if (PHONE.Equals(type))
|
{
|
return SecurityBiz.EncryptPhoneIndex(data, separator, secretContext);
|
}
|
else
|
{
|
int compressLen = secretCore.GetCompressLen();
|
int slideSize = secretCore.GetSlideSize();
|
return SecurityBiz.EncryptNormalIndex(data, compressLen, slideSize, separator, secretContext);
|
}
|
}
|
else
|
{
|
if (PHONE.Equals(type))
|
{
|
return SecurityBiz.EncryptPhone(data, separator, secretContext);
|
}
|
else
|
{
|
return SecurityBiz.EncryptNormal(data, separator, secretContext);
|
}
|
}
|
|
}
|
|
/// <summary>
|
/// 批量加密(所有用户共用秘钥)
|
/// </summary>
|
/// <param name="dataList"></param>
|
/// <param name="type"></param>
|
/// <returns></returns>
|
public IDictionary<string, string> Encrypt(List<string> dataList, string type)
|
{
|
return Encrypt(dataList, type, null);
|
}
|
|
/// <summary>
|
/// 批量加密(每个用户单独分配秘钥)
|
/// </summary>
|
/// <param name="dataList"></param>
|
/// <param name="type"></param>
|
/// <param name="session"></param>
|
/// <returns>key=明文数据,value=密文数据</returns>
|
public IDictionary<string, string> Encrypt(List<string> dataList, string type, string session)
|
{
|
if (dataList == null || dataList.Count == 0)
|
{
|
throw new SecretException("dataList can`t be empty");
|
}
|
IDictionary<string, string> resultMap = new Dictionary<string, string>();
|
foreach (string data in dataList)
|
{
|
if (!resultMap.ContainsKey(data))
|
{
|
string encryptValue = Encrypt(data, type, session, null);
|
resultMap.Add(data, encryptValue);
|
}
|
}
|
return resultMap;
|
}
|
|
/// <summary>
|
/// 生成自定义session,提供给自主账号使用
|
/// </summary>
|
/// <param name="userId"></param>
|
/// <returns></returns>
|
public static string GenerateCustomerSession(long userId)
|
{
|
return UNDERLINE + userId;
|
}
|
|
}
|
}
|