您当前的位置:首页 > 电脑百科 > 程序开发 > 编程百科

实用-API接口设计

时间:2021-06-10 10:19:15  来源:掘金  作者:不求上进的程序员

在实际的业务中,难免会跟第三方系统进行数据的交互与传递,那么如何保证数据在传输过程中的安全呢(防窃取)?除了https的协议之外,能不能加上通用的一套算法以及规范来保证传输的安全性呢?

下面我们就来讨论下常用的一些API设计的安全方法,可能不一定是最好的,有更牛逼的实现方式,但是这篇是我自己的经验分享.

一、token 简介

Token:访问令牌access token, 用于接口中, 用于标识接口调用者的身份、凭证,减少用户名和密码的传输次数。一般情况下客户端(接口调用方)需要先向服务器端申请一个接口调用的账号,服务器会给出一个AppId和一个key, key用于参数签名使用,注意key保存到客户端,需要做一些安全处理,防止泄露。

Token的值一般是UUID,服务端生成Token后需要将token做为key,将一些和token关联的信息作为value保存到缓存服务器中(redis),当一个请求过来后,服务器就去缓存服务器中查询这个Token是否存在,存在则调用接口,不存在返回接口错误,一般通过拦截器或者过滤器来实现,Token分为两种:

API Token(接口令牌): 用于访问不需要用户登录的接口,如登录、注册、一些基本数据的获取等。获取接口令牌需要拿appId、timestamp和sign来换,sign=加密(timestamp+key) USER Token(用户令牌): 用于访问需要用户登录之后的接口,如:获取我的基本信息、保存、修改、删除等操作。获取用户令牌需要拿用户名和密码来换

关于Token的时效性:token可以是一次性的、也可以在一段时间范围内是有效的,具体使用哪种看业务需要。

一般情况下接口最好使用https协议,如果使用http协议,Token机制只是一种减少被黑的可能性,其实只能防君子不能防小人。

一般token、timestamp和sign 三个参数会在接口中会同时作为参数传递,每个参数都有各自的用途。

二、timestamp 简介

timestamp: 时间戳,是客户端调用接口时对应的当前时间戳,时间戳用于防止DoS攻击。

黑客劫持了请求的url去DoS攻击,每次调用接口时接口都会判断服务器当前系统时间和接口中传的的timestamp的差值,如果这个差值超过某个设置的时间(假如5分钟),那么这个请求将被拦截掉,如果在设置的超时时间范围内,是不能阻止DoS攻击的。timestamp机制只能减轻DoS攻击的时间,缩短攻击时间。如果黑客修改了时间戳的值可通过sign签名机制来处理。

DoS

DoS是Denial of Service的简称,即拒绝服务,造成DoS的攻击行为被称为DoS攻击,其目的是使计算机或网络无法提供正常的服务。最常见的DoS攻击有计算机网络带宽攻击和连通性攻击。

DoS攻击是指故意的攻击网络协议实现的缺陷或直接通过野蛮手段残忍地耗尽被攻击对象的资源,目的是让目标计算机或网络无法提供正常的服务或资源访问,使目标系统服务系统停止响应甚至崩溃,而在此攻击中并不包括侵入目标服务器或目标网络设备。这些服务资源包括网络带宽,文件系统空间容量,开放的进程或者允许的连接。这种攻击会导致资源的匮乏,无论计算机的处理速度多快、内存容量多大、网络带宽的速度多快都无法避免这种攻击带来的后果。

Pingflood: 该攻击在短时间内向目的主机发送大量ping包,造成网络堵塞或主机资源耗尽。

Synflood: 该攻击以多个随机的源主机地址向目的主机发送SYN包,而在收到目的主机的SYN ACK后并不回应,这样,目的主机就为这些源主机建立了大量的连接队列,而且由于没有收到ACK一直维护着这些队列,造成了资源的大量消耗而不能向正常请求提供服务。

Smurf:该攻击向一个子网的广播地址发一个带有特定请求(如ICMP回应请求)的包,并且将源地址伪装成想要攻击的主机地址。子网上所有主机都回应广播包请求而向被攻击主机发包,使该主机受到攻击。

Land-based:攻击者将一个包的源地址和目的地址都设置为目标主机的地址,然后将该包通过IP欺骗的方式发送给被攻击主机,这种包可以造成被攻击主机因试图与自己建立连接而陷入死循环,从而很大程度地降低了系统性能。

Ping of Death:根据TCP/IP的规范,一个包的长度最大为65536字节。尽管一个包的长度不能超过65536字节,但是一个包分成的多个片段的叠加却能做到。当一个主机收到了长度大于65536字节的包时,就是受到了Ping of Death攻击,该攻击会造成主机的宕机。

Teardrop:IP数据包在网络传递时,数据包可以分成更小的片段。攻击者可以通过发送两段(或者更多)数据包来实现TearDrop攻击。第一个包的偏移量为0,长度为N,第二个包的偏移量小于N。为了合并这些数据段,TCP/IP堆栈会分配超乎寻常的巨大资源,从而造成系统资源的缺乏甚至机器的重新启动。

PingSweep:使用ICMP Echo轮询多个主机。

三、sign 简介

nonce:随机值,是客户端随机生成的值,作为参数传递过来,随机值的目的是增加sign签名的多变性。随机值一般是数字和字母的组合,6位长度,随机值的组成和长度没有固定规则。

sign: 一般用于参数签名,防止参数被非法篡改,最常见的是修改金额等重要敏感参数, sign的值一般是将所有非空参数按照升续排序然后+token+key+timestamp+nonce(随机数)拼接在一起,然后使用某种加密算法进行加密,作为接口中的一个参数sign来传递,也可以将sign放到请求头中。

接口在网络传输过程中如果被黑客挟持,并修改其中的参数值,然后再继续调用接口,虽然参数的值被修改了,但是因为黑客不知道sign是如何计算出来的,不知道sign都有哪些值构成,不知道以怎样的顺序拼接在一起的,最重要的是不知道签名字符串中的key是什么,所以黑客可以篡改参数的值,但没法修改sign的值,当服务器调用接口前会按照sign的规则重新计算出sign的值然后和接口传递的sign参数的值做比较,如果相等表示参数值没有被篡改,如果不等,表示参数被非法篡改了,就不执行接口了。

四、防止重复提交

对于一些重要的操作需要防止客户端重复提交的(如非幂等性重要操作),具体办法是当请求第一次提交时将sign作为key保存到redis,并设置超时时间,超时时间和Timestamp中设置的差值相同。

当同一个请求第二次访问时会先检测redis是否存在该sign,如果存在则证明重复提交了,接口就不再继续调用了。如果sign在缓存服务器中因过期时间到了,而被删除了,此时当这个url再次请求服务器时,因token的过期时间和sign的过期时间一直,sign过期也意味着token过期,那样同样的url再访问服务器会因token错误会被拦截掉,这就是为什么sign和token的过期时间要保持一致的原因。拒绝重复调用机制确保URL被别人截获了也无法使用(如抓取数据)。

对于哪些接口需要防止重复提交可以自定义个注解来标记。

注意:所有的安全措施都用上的话有时候难免太过复杂,在实际项目中需要根据自身情况作出裁剪,比如可以只使用签名机制就可以保证信息不会被篡改,或者定向提供服务的时候只用Token机制就可以了。如何裁剪,全看项目实际情况和对接口安全性的要求。

五、使用流程

1.接口调用方(客户端)向接口提供方(服务器)申请接口调用账号,申请成功后,接口提供方会给接口调用方一个appId和一个key参数

2.客户端携带参数appId、timestamp、sign去调用服务器端的API token,其中sign=加密(appId + timestamp + key)

3.客户端拿着api_token 去访问不需要登录就能访问的接口

4.当访问用户需要登录的接口时,客户端跳转到登录页面,通过用户名和密码调用登录接口,登录接口会返回一个usertoken, 客户端拿着usertoken 去访问需要登录才能访问的接口

sign的作用是防止参数被篡改,客户端调用服务端时需要传递sign参数,服务器响应客户端时也可以返回一个sign用于客户度校验返回的值是否被非法篡改了。客户端传的sign和服务器端响应的sign算法可能会不同。

六、示例代码

  1. dependency
org.springframework.boot spring-boot-starter-data-redis redis.clients jedis 2.9.0 org.springframework.boot spring-boot-starter-web
  1. RedisConfiguration

@Configuration public class RedisConfiguration { @Bean public JedisConnectionFactory jedisConnectionFactory(){ return new JedisConnectionFactory(); }

/**
 * 支持存储对象
 * @return
 */
@Bean
public RedisTemplate<String, String> redisTemplate(){
    RedisTemplate<String, String> redisTemplate = new StringRedisTemplate();
    redisTemplate.setConnectionFactory(jedisConnectionFactory());
    Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class);
    ObjectMapper objectMapper = new ObjectMapper();
    objectMapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
    objectMapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);


    jackson2JsonRedisSerializer.setObjectMapper(objectMapper);
    redisTemplate.setValueSerializer(jackson2JsonRedisSerializer);
    redisTemplate.afterPropertiesSet();


    return redisTemplate;
}
复制代码

}

  1. TokenController

@Slf4j @RestController @RequestMapping("/api/token") public class TokenController {

@Autowired
private RedisTemplate redisTemplate;


/**
 * API Token
 *
 * @param sign
 * @return
 */
@PostMapping("/api_token")
public ApiResponse<AccessToken> apiToken(String appId, @RequestHeader("timestamp") String timestamp, @RequestHeader("sign") String sign) {
    Assert.isTrue(!StringUtils.isEmpty(appId) && !StringUtils.isEmpty(timestamp) && !StringUtils.isEmpty(sign), "参数错误");


    long reqeustInterval = System.currentTimeMillis() - Long.valueOf(timestamp);
    Assert.isTrue(reqeustInterval < 5 * 60 * 1000, "请求过期,请重新请求");


    // 1. 根据appId查询数据库获取appSecret
    AppInfo appInfo = new AppInfo("1", "12345678954556");


    // 2. 校验签名
    String signString = timestamp + appId + appInfo.getKey();
    String signature = MD5Util.encode(signString);
    log.info(signature);
    Assert.isTrue(signature.equals(sign), "签名错误");


    // 3. 如果正确生成一个token保存到redis中,如果错误返回错误信息
    AccessToken accessToken = this.saveToken(0, appInfo, null);


    return ApiResponse.success(accessToken);
}




@NotRepeatSubmit(5000)
@PostMapping("user_token")
public ApiResponse<UserInfo> userToken(String username, String password) {
    // 根据用户名查询密码, 并比较密码(密码可以RSA加密一下)
    UserInfo userInfo = new UserInfo(username, "81255cb0dca1a5f304328a70ac85dcbd", "111111");
    String pwd = password + userInfo.getSalt();
    String passwordMD5 = MD5Util.encode(pwd);
    Assert.isTrue(passwordMD5.equals(userInfo.getPassword()), "密码错误");


    // 2. 保存Token
    AppInfo appInfo = new AppInfo("1", "12345678954556");
    AccessToken accessToken = this.saveToken(1, appInfo, userInfo);
    userInfo.setAccessToken(accessToken);
    return ApiResponse.success(userInfo);
}


private AccessToken saveToken(int tokenType, AppInfo appInfo,  UserInfo userInfo) {
    String token = UUID.randomUUID().toString();


    // token有效期为2小时
    Calendar calendar = Calendar.getInstance();
    calendar.setTime(new Date());
    calendar.add(Calendar.SECOND, 7200);
    Date expireTime = calendar.getTime();


    // 4. 保存token
    ValueOperations<String, TokenInfo> operations = redisTemplate.opsForValue();
    TokenInfo tokenInfo = new TokenInfo();
    tokenInfo.setTokenType(tokenType);
    tokenInfo.setAppInfo(appInfo);


    if (tokenType == 1) {
        tokenInfo.setUserInfo(userInfo);
    }


    operations.set(token, tokenInfo, 7200, TimeUnit.SECONDS);


    AccessToken accessToken = new AccessToken(token, expireTime);


    return accessToken;
}


public static void main(String[] args) {
    long timestamp = System.currentTimeMillis();
    System.out.println(timestamp);
    String signString = timestamp + "1" + "12345678954556";
    String sign = MD5Util.encode(signString);
    System.out.println(sign);


    System.out.println("-------------------");
    signString = "password=123456&username=1&12345678954556" + "ff03e64b-427b-45a7-b78b-47d9e8597d3b1529815393153sdfsdfsfs" + timestamp + "A1scr6";
    sign = MD5Util.encode(signString);
    System.out.println(sign);
}
复制代码

}

  1. WebMvcConfiguration

@Configuration public class WebMvcConfiguration extends WebMvcConfigurationSupport {

private static final String[] excludePathPatterns  = {"/api/token/api_token"};


@Autowired
private TokenInterceptor tokenInterceptor;


@Override
public void addInterceptors(InterceptorRegistry registry) {
    super.addInterceptors(registry);
    registry.addInterceptor(tokenInterceptor)
            .addPathPatterns("/api/**")
            .excludePathPatterns(excludePathPatterns);
}
复制代码

} 5. TokenInterceptor @Component public class TokenInterceptor extends HandlerInterceptorAdapter {

@Autowired
private RedisTemplate redisTemplate;


/**
 *
 * @param request
 * @param response
 * @param handler 访问的目标方法
 * @return
 * @throws Exception
 */
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
    String token = request.getHeader("token");
    String timestamp = request.getHeader("timestamp");
    // 随机字符串
    String nonce = request.getHeader("nonce");
    String sign = request.getHeader("sign");
    Assert.isTrue(!StringUtils.isEmpty(token) && !StringUtils.isEmpty(timestamp) && !StringUtils.isEmpty(sign), "参数错误");


    // 获取超时时间
    NotRepeatSubmit notRepeatSubmit = ApiUtil.getNotRepeatSubmit(handler);
    long expireTime = notRepeatSubmit == null ? 5 * 60 * 1000 : notRepeatSubmit.value();


    // 2. 请求时间间隔
    long reqeustInterval = System.currentTimeMillis() - Long.valueOf(timestamp);
    Assert.isTrue(reqeustInterval < expireTime, "请求超时,请重新请求");


    // 3. 校验Token是否存在
    ValueOperations<String, TokenInfo> tokenRedis = redisTemplate.opsForValue();
    TokenInfo tokenInfo = tokenRedis.get(token);
    Assert.notNull(tokenInfo, "token错误");


    // 4. 校验签名(将所有的参数加进来,防止别人篡改参数) 所有参数看参数名升续排序拼接成url
    // 请求参数 + token + timestamp + nonce
    String signString = ApiUtil.concatSignString(request) + tokenInfo.getAppInfo().getKey() + token + timestamp + nonce;
    String signature = MD5Util.encode(signString);
    boolean flag = signature.equals(sign);
    Assert.isTrue(flag, "签名错误");


    // 5. 拒绝重复调用(第一次访问时存储,过期时间和请求超时时间保持一致), 只有标注不允许重复提交注解的才会校验
    if (notRepeatSubmit != null) {
        ValueOperations<String, Integer> signRedis = redisTemplate.opsForValue();
        boolean exists = redisTemplate.hasKey(sign);
        Assert.isTrue(!exists, "请勿重复提交");
        signRedis.set(sign, 0, expireTime, TimeUnit.MILLISECONDS);
    }


    return super.preHandle(request, response, handler);
}
复制代码

}

  1. MD5Util ----MD5工具类,加密生成数字签名

public class MD5Util {

private static final String hexDigits[] = { "0", "1", "2", "3", "4", "5",
        "6", "7", "8", "9", "a", "b", "c", "d", "e", "f" };


private static String byteArrayToHexString(byte b[]) {
    StringBuffer resultSb = new StringBuffer();
    for (int i = 0; i < b.length; i++)
        resultSb.append(byteToHexString(b[i]));


    return resultSb.toString();
}


private static String byteToHexString(byte b) {
    int n = b;
    if (n < 0)
        n += 256;
    int d1 = n / 16;
    int d2 = n % 16;
    return hexDigits[d1] + hexDigits[d2];
}


public static String encode(String origin) {
    return encode(origin, "UTF-8");
}
public static String encode(String origin, String charsetname) {
    String resultString = null;
    try {
        resultString = new String(origin);
        MessageDigest md = MessageDigest.getInstance("MD5");
        if (charsetname == null || "".equals(charsetname))
            resultString = byteArrayToHexString(md.digest(resultString
                    .getBytes()));
        else
            resultString = byteArrayToHexString(md.digest(resultString
                    .getBytes(charsetname)));
    } catch (Exception exception) {
    }
    return resultString;
}
复制代码

}

  1. @NotRepeatSubmit -----自定义注解,防止重复提交。

/**

  • 禁止重复提交

/ @Target(ElementType.METHOD) @Retention(RetentionPolicy.RUNTIME) public @interface NotRepeatSubmit { /* 过期时间,单位毫秒 **/ long value() default 5000; }

  1. AccessToken

@Data @AllArgsConstructor public class AccessToken { /** token */ private String token;

/** 失效时间 */
private Date expireTime;
复制代码

}

  1. AppInfo

@Data @NoArgsConstructor @AllArgsConstructor public class AppInfo { /** App id / private String appId; /* API 秘钥 */ private String key; }

  1. TokenInfo

@Data public class TokenInfo { /** token类型: api:0 、user:1 */ private Integer tokenType;

/** App 信息 */
private AppInfo appInfo;


/** 用户其他数据 */
private UserInfo userInfo;
复制代码

}

  1. UserInfo

@Data public class UserInfo { /** 用户名 / private String username; /* 手机/ private String mobile; /* 邮箱 / private String email; /* 密码 / private String password; /* 盐 */ private String salt;

private AccessToken accessToken;


public UserInfo(String username, String password, String salt) {
    this.username = username;
    this.password = password;
    this.salt = salt;
}
复制代码

}

  1. ApiCodeEnum

/**

  • 错误码code可以使用纯数字,使用不同区间标识一类错误,也可以使用纯字符,也可以使用前缀+编号
  •  
  • 错误码:ERR + 编号
  •  
  • 可以使用日志级别的前缀作为错误类型区分 Info(I) Error(E) Warning(W)
  •  
  • 或者以业务模块 + 错误号
  •  
  • TODO 错误码设计
  •  
  • Alipay 用了两个code,两个msg(docs.open.alipay.com/api_1/alipa…)

*/ public enum ApiCodeEnum { SUCCESS("10000", "success"), UNKNOW_ERROR("ERR0001","未知错误"), PARAMETER_ERROR("ERR0002","参数错误"), TOKEN_EXPIRE("ERR0003","认证过期"), REQUEST_TIMEOUT("ERR0004","请求超时"), SIGN_ERROR("ERR0005","签名错误"), REPEAT_SUBMIT("ERR0006","请不要频繁操作"), ;

/** 代码 */
private String code;


/** 结果 */
private String msg;


ApiCodeEnum(String code, String msg) {
    this.code = code;
    this.msg = msg;
}


public String getCode() {
    return code;
}


public String getMsg() {
    return msg;
}
复制代码

}

  1. ApiResult

@Data @NoArgsConstructor @AllArgsConstructor public class ApiResult {

/** 代码 */
private String code;


/** 结果 */
private String msg;
复制代码

}

  1. ApiUtil -------这个参考支付宝加密的算法写的.我直接Copy过来了。

public class ApiUtil {

/**
 * 按参数名升续拼接参数
 * @param request
 * @return
 */
public static String concatSignString(HttpServletRequest request) {
    Map<String, String> paramterMap = new HashMap<>();
    request.getParameterMap().forEach((key, value) -> paramterMap.put(key, value[0]));
    // 按照key升续排序,然后拼接参数
    Set<String> keySet = paramterMap.keySet();
    String[] keyArray = keySet.toArray(new String[keySet.size()]);
    Arrays.sort(keyArray);
    StringBuilder sb = new StringBuilder();
    for (String k : keyArray) {
        // 或略掉的字段
        if (k.equals("sign")) {
            continue;
        }
        if (paramterMap.get(k).trim().length() > 0) {
            // 参数值为空,则不参与签名
            sb.append(k).append("=").append(paramterMap.get(k).trim()).append("&");
        }
    }


    return sb.toString();
}


public static String concatSignString(Map<String, String> map) {
    Map<String, String> paramterMap = new HashMap<>();
    map.forEach((key, value) -> paramterMap.put(key, value));
    // 按照key升续排序,然后拼接参数
    Set<String> keySet = paramterMap.keySet();
    String[] keyArray = keySet.toArray(new String[keySet.size()]);
    Arrays.sort(keyArray);
    StringBuilder sb = new StringBuilder();
    for (String k : keyArray) {
        if (paramterMap.get(k).trim().length() > 0) {
            // 参数值为空,则不参与签名
            sb.append(k).append("=").append(paramterMap.get(k).trim()).append("&");
        }
    }
    return sb.toString();
}


/**
 * 获取方法上的@NotRepeatSubmit注解
 * @param handler
 * @return
 */
public static NotRepeatSubmit getNotRepeatSubmit(Object handler) {
    if (handler instanceof HandlerMethod) {
        HandlerMethod handlerMethod = (HandlerMethod) handler;
        Method method = handlerMethod.getMethod();
        NotRepeatSubmit annotation = method.getAnnotation(NotRepeatSubmit.class);


        return annotation;
    }


    return null;
}
复制代码

}

  1. ApiResponse

@Data @Slf4j public class ApiResponse { /** 结果 */ private ApiResult result;

/** 数据 */
private T data;


/** 签名 */
private String sign;




public static <T> ApiResponse success(T data) {
    return response(ApiCodeEnum.SUCCESS.getCode(), ApiCodeEnum.SUCCESS.getMsg(), data);
}


public static ApiResponse error(String code, String msg) {
    return response(code, msg, null);
}


public static <T> ApiResponse response(String code, String msg, T data) {
    ApiResult result = new ApiResult(code, msg);
    ApiResponse response = new ApiResponse();
    response.setResult(result);
    response.setData(data);


    String sign = signData(data);
    response.setSign(sign);


    return response;
}


private static <T> String signData(T data) {
    // TODO 查询key
    String key = "12345678954556";
    Map<String, String> responseMap = null;
    try {
        responseMap = getFields(data);
    } catch (IllegalAccessException e) {
        return null;
    }
    String urlComponent = ApiUtil.concatSignString(responseMap);
    String signature = urlComponent + "key=" + key;
    String sign = MD5Util.encode(signature);


    return sign;
}


/**
 * @param data 反射的对象,获取对象的字段名和值
 * @throws IllegalArgumentException
 * @throws IllegalAccessException
 */
public static Map<String, String> getFields(Object data) throws IllegalAccessException, IllegalArgumentException {
    if (data == null) return null;
    Map<String, String> map = new HashMap<>();
    Field[] fields = data.getClass().getDeclaredFields();
    for (int i = 0; i < fields.length; i++) {
        Field field = fields[i];
        field.setAccessible(true);


        String name = field.getName();
        Object value = field.get(data);
        if (field.get(data) != null) {
            map.put(name, value.toString());
        }
    }


    return map;
}
复制代码

}

七、ThreadLocal

ThreadLocal是线程内的全局上下文。就是在单个线程中,方法之间共享的内存,每个方法都可以从该上下文中获取值和修改值。

实际案例:

在调用api时都会传一个token参数,通常会写一个拦截器来校验token是否合法,我们可以通过token找到对应的用户信息(User),如果token合法,然后将用户信息存储到ThreadLocal中,这样无论是在controller、service、dao的哪一层都能访问到该用户的信息。作用类似于Web中的request作用域。

传统方式我们要在方法中访问某个变量,可以通过传参的形式往方法中传参,如果多个方法都要使用那么每个方法都要传参;如果使用ThreadLocal所有方法就不需要传该参数了,每个方法都可以通过ThreadLocal来访问该值。

ThreadLocalUtil.set("key", value); 保存值 T value = ThreadLocalUtil.get("key"); 获取值

ThreadLocalUtil

public class ThreadLocalUtil { private static final ThreadLocal<Map<String, Object>> threadLocal = new ThreadLocal() { @Override protected Map<String, Object> initialValue() { return new HashMap<>(4); } };

public static Map<String, Object> getThreadLocal(){
    return threadLocal.get();
}


public static <T> T get(String key) {
    Map map = (Map)threadLocal.get();
    return (T)map.get(key);
}


public static <T> T get(String key,T defaultValue) {
    Map map = (Map)threadLocal.get();
    return (T)map.get(key) == null ? defaultValue : (T)map.get(key);
}


public static void set(String key, Object value) {
    Map map = (Map)threadLocal.get();
    map.put(key, value);
}


public static void set(Map<String, Object> keyValueMap) {
    Map map = (Map)threadLocal.get();
    map.putAll(keyValueMap);
}


public static void remove() {
    threadLocal.remove();
}


public static <T> Map<String,T> fetchVarsByPrefix(String prefix) {
    Map<String,T> vars = new HashMap<>();
    if( prefix == null ){
        return vars;
    }
    Map map = (Map)threadLocal.get();
    Set<Map.Entry> set = map.entrySet();


    for( Map.Entry entry : set){
        Object key = entry.getKey();
        if( key instanceof String ){
            if( ((String) key).startsWith(prefix) ){
                vars.put((String)key,(T)entry.getValue());
            }
        }
    }
    return vars;
}


public static <T> T remove(String key) {
    Map map = (Map)threadLocal.get();
    return (T)map.remove(key);
}


public static void clear(String prefix) {
    if( prefix == null ){
        return;
    }
    Map map = (Map)threadLocal.get();
    Set<Map.Entry> set = map.entrySet();
    List<String> removeKeys = new ArrayList<>();


    for( Map.Entry entry : set ){
        Object key = entry.getKey();
        if( key instanceof String ){
            if( ((String) key).startsWith(prefix) ){
                removeKeys.add((String)key);
            }
        }
    }
    for( String key : removeKeys ){
        map.remove(key);
    }
}
复制代码

}

总结:这个是目前第三方数据接口交互过程中常用的一些参数与使用示例,希望对大家有点帮助。

当然如果为了保证更加的安全,可以加上RSA,RSA2,AES等等加密方式,保证了数据的更加的安全,但是唯一的缺点是加密与解密比较耗费CPU的资源。

一般是RSA+AES结合!原理:AES加密参数,rsa加密AES的密钥,后端通过RSA解密得到AES密钥,然后通过AES解密,得到前端传的参数

没有绝对安全,只有相对安全

此文转载:cnblogs.com/jurendage/p/12653865.html


作者:不求上进的程序员
链接:https://juejin.cn/post/6970195224807096351
来源:掘金
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。


Tags:API接口   点击:()  评论:()
声明:本站部分内容及图片来自互联网,转载是出于传递更多信息之目的,内容观点仅代表作者本人,如有任何标注错误或版权侵犯请与我们联系(Email:2595517585@qq.com),我们将及时更正、删除,谢谢。
▌相关推荐
前言前后端分离开发模式中,api文档是最好的沟通方式。今天就来说一说如何整合Swagger生成一套漂亮、美观、实用的接口文档。 源码传送门: https://gitee.com/huoqstudy/xiliu-...【详细内容】
2021-09-08  Tags: API接口  点击:(65)  评论:(0)  加入收藏
注:商业级功能效果演示,非开源,无源码。研发基础在之前AJAX请求数据加密效果之上,更进一步,对返回数据加密。之前是单纯用于登录场景。更广泛的场景是所有此类AJAX WEB API接口。...【详细内容】
2021-09-03  Tags: API接口  点击:(73)  评论:(0)  加入收藏
在实际的业务中,难免会跟第三方系统进行数据的交互与传递,那么如何保证数据在传输过程中的安全呢(防窃取)?除了https的协议之外,能不能加上通用的一套算法以及规范来保证传输的安...【详细内容】
2021-06-10  Tags: API接口  点击:(136)  评论:(0)  加入收藏
前几天我发文就已经给各位朋友提醒过了,你们的api早就被盗了。只是没有到那些人收割的时候而已。这不我在我们公司的官方群里就看见用户分享他朋友用其他的一些量化交易机...【详细内容】
2021-06-08  Tags: API接口  点击:(129)  评论:(0)  加入收藏
今天介绍我正在用的一款高效敏捷开发工具magic-api,顺便分享一点工作中使用它的心得缘起先说一下我为什么会使用这个工具?最近新启动一个项目,业务并不算复杂,那种典型的管理系...【详细内容】
2021-06-01  Tags: API接口  点击:(194)  评论:(0)  加入收藏
一、axios的封装在vue项目中,和后台交互获取数据这块,我们通常使用的是axios库,它是基于promise的http库,可运行在浏览器端和node.js中。他有很多优秀的特性,例如拦截请求和响应...【详细内容】
2020-10-12  Tags: API接口  点击:(256)  评论:(0)  加入收藏
API译者:DevOps亮哥如今,API已在软件、Web和移动应用程序开发领域无处不在,从企业内部到面向公众的应用以及与合作伙伴进行系统集成。通过使用API,开发人员可以创建满足各种客户...【详细内容】
2020-08-01  Tags: API接口  点击:(95)  评论:(0)  加入收藏
在6月中旬时,拼多多为了保障数据安全和运营信息安全,依据《拼多多开放平台开发者协议》等相关协议和规则,发布了对商家在使用API接口的时候开始收费的公告。 所谓API,百度百科...【详细内容】
2020-07-09  Tags: API接口  点击:(885)  评论:(0)  加入收藏
有没有遇到这样子的接口,放到互联网上面去,谁都可以调用,谁都可以访问,完全就是公开的,这样子的接口,如果只是普通的数据,其实可以考虑,只是可以考虑,但是,一般情况下,我们是不允许这样...【详细内容】
2020-06-20  Tags: API接口  点击:(85)  评论:(0)  加入收藏
滑动验证是网站反爬虫、反作弊的升级,滑动验证也是机器学习在反爬虫、反作弊领域的应用; 本项目也是一个简单的全栈项目,使用tornado做的后端、Bootstrap4做的前端;核心的识别...【详细内容】
2020-06-04  Tags: API接口  点击:(33)  评论:(0)  加入收藏
▌简易百科推荐
摘 要 (OF作品展示)OF之前介绍了用python实现数据可视化、数据分析及一些小项目,但基本都是后端的知识。想要做一个好看的可视化大屏,我们还要学一些前端的知识(vue),网上有很多比...【详细内容】
2021-12-27  项目与数据管理    Tags:Vue   点击:(1)  评论:(0)  加入收藏
程序是如何被执行的&emsp;&emsp;程序是如何被执行的?许多开发者可能也没法回答这个问题,大多数人更注重的是如何编写程序,却不会太注意编写好的程序是如何被运行,这并不是一个好...【详细内容】
2021-12-23  IT学习日记    Tags:程序   点击:(9)  评论:(0)  加入收藏
阅读收获✔️1. 了解单点登录实现原理✔️2. 掌握快速使用xxl-sso接入单点登录功能一、早期的多系统登录解决方案 单系统登录解决方案的核心是cookie,cookie携带会话id在浏览器...【详细内容】
2021-12-23  程序yuan    Tags:单点登录(   点击:(8)  评论:(0)  加入收藏
下载Eclipse RCP IDE如果你电脑上还没有安装Eclipse,那么请到这里下载对应版本的软件进行安装。具体的安装步骤就不在这赘述了。创建第一个标准Eclipse RCP应用(总共分为六步)1...【详细内容】
2021-12-22  阿福ChrisYuan    Tags:RCP应用   点击:(7)  评论:(0)  加入收藏
今天想简单聊一聊 Token 的 Value Capture,就是币的价值问题。首先说明啊,这个话题包含的内容非常之光,Token 的经济学设计也可以包含诸多问题,所以几乎不可能把这个问题说的清...【详细内容】
2021-12-21  唐少华TSH    Tags:Token   点击:(9)  评论:(0)  加入收藏
实现效果:假如有10条数据,分组展示,默认在当前页面展示4个,点击换一批,从第5个开始继续展示,到最后一组,再重新返回到第一组 data() { return { qList: [], //处理后...【详细内容】
2021-12-17  Mason程    Tags:VUE   点击:(14)  评论:(0)  加入收藏
什么是性能调优?(what) 为什么需要性能调优?(why) 什么时候需要性能调优?(when) 什么地方需要性能调优?(where) 什么时候来进行性能调优?(who) 怎么样进行性能调优?(How) 硬件配...【详细内容】
2021-12-16  软件测试小p    Tags:性能调优   点击:(19)  评论:(0)  加入收藏
Tasker 是一款适用于 Android 设备的高级自动化应用,它可以通过脚本让重复性的操作自动运行,提高效率。 不知道从哪里听说的抖音 app 会导致 OLED 屏幕烧屏。于是就现学现卖,自...【详细内容】
2021-12-15  ITBang    Tags:抖音防烧屏   点击:(23)  评论:(0)  加入收藏
11 月 23 日,Rust Moderation Team(审核团队)在 GitHub 上发布了辞职公告,即刻生效。根据公告,审核团队集体辞职是为了抗议 Rust 核心团队(Core team)在执行社区行为准则和标准上...【详细内容】
2021-12-15  InfoQ    Tags:Rust   点击:(24)  评论:(0)  加入收藏
一个项目的大部分API,测试用例在参数和参数值等信息会有很多相似的地方。我们可以复制API,复制用例来快速生成,然后做细微调整既可以满足我们的测试需求1.复制API:在菜单发布单...【详细内容】
2021-12-14  AutoMeter    Tags:AutoMeter   点击:(20)  评论:(0)  加入收藏
最新更新
栏目热门
栏目头条