Browse Source

open

master
pioneer 2 years ago
commit
df53a1bbdb
  1. 6
      README.md
  2. BIN
      lib/bcprov-jdk15on-1.56.jar
  3. BIN
      lib/finekit-10.0.jar
  4. 24
      plugin.xml
  5. 37
      src/main/java/com/fr/plugin/cmd/LocaleFinder.java
  6. 34
      src/main/java/com/fr/plugin/cmd/PluginMonitor.java
  7. 44
      src/main/java/com/fr/plugin/cmd/config/CmdConfig.java
  8. 244
      src/main/java/com/fr/plugin/cmd/request/OAuthLogin.java
  9. 93
      src/main/java/com/fr/plugin/cmd/utils/AESUtil.java
  10. 5
      src/main/resources/com/fr/plugin/cmd/locale/lang.properties
  11. 5
      src/main/resources/com/fr/plugin/cmd/locale/lang_zh_CN.properties

6
README.md

@ -0,0 +1,6 @@
# open-JSD-10334
JSD-10334 h5单点集成\
免责说明:该源码为第三方爱好者提供,不保证源码和方案的可靠性,也不提供任何形式的源码教学指导和协助!\
仅作为开发者学习参考使用!禁止用于任何商业用途!\
为保护开发者隐私,开发者信息已隐去!若原开发者希望公开自己的信息,可联系【pioneer】处理。

BIN
lib/bcprov-jdk15on-1.56.jar

Binary file not shown.

BIN
lib/finekit-10.0.jar

Binary file not shown.

24
plugin.xml

@ -0,0 +1,24 @@
<?xml version="1.0" encoding="UTF-8"?>
<plugin>
<id>com.fr.plugin.cmd.sso</id>
<name><![CDATA[单点登陆]]></name>
<active>yes</active>
<version>1.2</version>
<env-version>10.0</env-version>
<jartime>2018-07-31</jartime>
<vendor>fr.open</vendor>
<description><![CDATA[单点登陆]]></description>
<change-notes><![CDATA[单点登陆]]></change-notes>
<main-package>com.fr.plugin.cmd</main-package>
<prefer-packages>
<prefer-package>com.fanruan.api</prefer-package>
</prefer-packages>
<lifecycle-monitor class="com.fr.plugin.cmd.PluginMonitor"/>
<extra-core>
<LocaleFinder class="com.fr.plugin.cmd.LocaleFinder"/>
</extra-core>
<extra-decision>
<GlobalRequestFilterProvider class="com.fr.plugin.cmd.request.OAuthLogin"/>
</extra-decision>
<function-recorder class="com.fr.plugin.cmd.LocaleFinder"/>
</plugin>

37
src/main/java/com/fr/plugin/cmd/LocaleFinder.java

@ -0,0 +1,37 @@
/*
* Copyright (C), 2018-2020
* Project: starter
* FileName: LocaleFinder
* Author: xx
* Date: 2020/8/31 22:19
*/
package com.fr.plugin.cmd;
import com.fr.intelli.record.Focus;
import com.fr.intelli.record.Original;
import com.fr.record.analyzer.EnableMetrics;
import com.fr.stable.fun.impl.AbstractLocaleFinder;
import static com.fr.plugin.cmd.config.CmdConfig.PLUGIN_ID;
/**
* <Function Description><br>
* <LocaleFinder>
*
* @author xx
* @since 1.0.0
*/
@EnableMetrics
public class LocaleFinder extends AbstractLocaleFinder {
@Override
@Focus(id = PLUGIN_ID, text = "Plugin-cmd", source = Original.PLUGIN)
public String find() {
return "com/fr/plugin/cmd/locale/lang";
}
@Override
public int currentAPILevel() {
return CURRENT_LEVEL;
}
}

34
src/main/java/com/fr/plugin/cmd/PluginMonitor.java

@ -0,0 +1,34 @@
/*
* Copyright (C), 2018-2021
* Project: starter
* FileName: PluginMonitor
* Author: xx
* Date: 2021/3/30 15:10
*/
package com.fr.plugin.cmd;
import com.fr.plugin.cmd.config.CmdConfig;
import com.fr.plugin.context.PluginContext;
import com.fr.plugin.observer.inner.AbstractPluginLifecycleMonitor;
/**
* <Function Description><br>
* <PluginMonitor>
*
* @author xx
* @since 1.0.0
*/
public class PluginMonitor extends AbstractPluginLifecycleMonitor {
public PluginMonitor() {
}
@Override
public void afterRun(PluginContext pluginContext) {
CmdConfig.getInstance();
}
@Override
public void beforeStop(PluginContext pluginContext) {
}
}

44
src/main/java/com/fr/plugin/cmd/config/CmdConfig.java

@ -0,0 +1,44 @@
/*
* Copyright (C), 2018-2021
* Project: starter
* FileName: IbahConfig
* Author: xx
* Date: 2021/3/30 9:38
*/
package com.fr.plugin.cmd.config;
import com.fr.config.*;
import com.fr.config.holder.Conf;
import com.fr.config.holder.factory.Holders;
/**
* <Function Description><br>
* <CmdConfig>
*
* @author xx
* @since 1.0.0
*/
@Visualization(category = "Plugin-cmd_Group")
public class CmdConfig extends DefaultConfiguration {
public static final String PLUGIN_ID = "com.fr.plugin.cmd.sso";
public static final String MOBILE_INTERFACE = "https://xx/MobileInterface.aspx?Action=Authorization";
private static volatile CmdConfig config = null;
@Identifier(value = "uriBase", name = "Plugin-cmd_Config_UriBase", description = "Plugin-cmd_Config_UriBase_Description", status = Status.SHOW)
private final Conf<String> uriBase = Holders.simple(MOBILE_INTERFACE);
public static CmdConfig getInstance() {
if (config == null) {
config = ConfigContext.getConfigInstance(CmdConfig.class);
}
return config;
}
public String getUriBase() {
return uriBase.get();
}
public void setUriBase(String uriBase) {
this.uriBase.set(uriBase);
}
}

244
src/main/java/com/fr/plugin/cmd/request/OAuthLogin.java

@ -0,0 +1,244 @@
/*
* Copyright (C), 2018-2021
* Project: starter
* FileName: OAuthLogin
* Author: xx
* Date: 2021/3/30 22:09
*/
package com.fr.plugin.cmd.request;
import com.fanruan.api.decision.login.LoginKit;
import com.fanruan.api.decision.user.UserKit;
import com.fanruan.api.i18n.I18nKit;
import com.fanruan.api.log.LogKit;
import com.fanruan.api.net.NetworkKit;
import com.fanruan.api.net.http.HttpKit;
import com.fanruan.api.util.StringKit;
import com.fr.decision.fun.impl.AbstractGlobalRequestFilterProvider;
import com.fr.decision.webservice.utils.DecisionServiceConstants;
import com.fr.decision.webservice.v10.login.LoginService;
import com.fr.general.ComparatorUtils;
import com.fr.json.JSONObject;
import com.fr.plugin.cmd.config.CmdConfig;
import com.fr.plugin.cmd.utils.AESUtil;
import com.fr.plugin.context.PluginContexts;
import com.fr.stable.fun.Authorize;
import com.fr.third.org.apache.http.NameValuePair;
import com.fr.third.org.apache.http.client.utils.URIBuilder;
import com.fr.third.org.apache.http.entity.StringEntity;
import com.fr.web.utils.WebUtils;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.net.URI;
import java.net.URISyntaxException;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import static com.fr.plugin.cmd.config.CmdConfig.PLUGIN_ID;
/**
* <Function Description><br>
* <OAuthLogin>
*
* @author xx
* @since 1.0.0
*/
@Authorize(callSignKey = PLUGIN_ID)
public class OAuthLogin extends AbstractGlobalRequestFilterProvider {
public static final String REFRESH_AUTH = "/refreshMobileAuth";
private CmdConfig config;
/**
* 过滤器名称
*
* @return
*/
@Override
public String filterName() {
return "cmdFilter";
}
/**
* 过滤规则
*
* @return
*/
@Override
public String[] urlPatterns() {
return new String[]{"/decision", "/decision/url/mobile", "/decision/view/form", "/decision/view/report", "/decision/v10/entry/access/*"};
}
/**
* 过滤器初始化
*
* @param filterConfig
*/
@Override
public void init(FilterConfig filterConfig) {
this.config = CmdConfig.getInstance();
super.init(filterConfig);
}
/**
* 过滤器处理
*
* @param request
* @param response
* @param filterChain
*/
@Override
public void doFilter(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain) {
try {
if (operation(request, response)) {
filterChain.doFilter(request, response);
}
} catch (Exception e) {
LogKit.error(e.getMessage(), e);
}
}
/**
* 用户验证登陆操作
*
* @param req
* @param res
* @throws Exception
*/
private boolean operation(HttpServletRequest req, HttpServletResponse res) throws Exception {
if (LoginService.getInstance().isLogged(req)) {
return true;
}
String mobileToken = decodeStr(NetworkKit.getHTTPRequestParameter(req, "Token"));
String mobileUserName = NetworkKit.getHTTPRequestParameter(req, "UserAccount");
LogKit.info("cmd-OAuthLogin-operation-mobileToken:{}, mobileUserName:{}", mobileToken, mobileUserName);
if (StringKit.isBlank(mobileToken) || StringKit.isBlank(mobileUserName)) {
return true;
}
String username = getUsername(mobileToken, mobileUserName);
if (StringKit.isEmpty(username) || !UserKit.existUsername(username)) {
return true;
}
if (!PluginContexts.currentContext().isAvailable()) {
LogKit.error(I18nKit.getLocText("Plugin-cmd_Licence_Expired"));
return true;
}
String tokenFR = LoginKit.login(req, res, username);
req.setAttribute(DecisionServiceConstants.FINE_AUTH_TOKEN_NAME, tokenFR);
String refreshMobileUrl = resolveURL(req.getRequestURL().toString(), WebUtils.createServletURL(req) + REFRESH_AUTH);
refreshMobileAuth(mobileToken, mobileUserName, refreshMobileUrl);
res.sendRedirect(removeToken(req));
return false;
}
private String resolveURL(String url, String path) throws Exception {
URI uri = new URI(url);
return uri.resolve(path).toString();
}
/**
* 处理加号和等号特殊字符
*
* @param str
* @return
*/
private String decodeStr(String str) {
if (StringKit.isBlank(str)) {
return str;
}
return str.replaceAll(" ", "+");
}
/**
* 通过凭证获得username
*
* @param mobileToken
* @param mobileUserName
* @return
*/
private String getUsername(String mobileToken, String mobileUserName) throws Exception {
String username = StringKit.EMPTY;
if (isValidToken(mobileToken, mobileUserName)) {
username = UserKit.existUsername(mobileUserName) ? mobileUserName : UserKit.getUserNamesFromEmail(mobileUserName).get(0);
}
return username;
}
/**
* 验证移动端token
*
* @param token
* @param userAccount
* @return
*/
private boolean authToken(String token, String userAccount) throws Exception {
String msgDecrypt = AESUtil.decrypt(token, AESUtil.sKey);
assert msgDecrypt != null;
String tmpUserName = msgDecrypt.substring(0, msgDecrypt.indexOf("&"));
return ComparatorUtils.equals(tmpUserName, userAccount);
}
/**
* 验证移动端token
*
* @param token
* @param userAccount
* @return
*/
private boolean isValidToken(String token, String userAccount) throws Exception {
Map<String, String> headers = new HashMap<>();
headers.put("Content-Type", "application/json");
JSONObject params = new JSONObject();
params.put("UserAccount", userAccount);
params.put("Token", token);
LogKit.info("cmd-OAuthLogin-isValidToken-params:{}", params.encode());
StringEntity stringEntity = new StringEntity(params.encode(), "UTF-8");
String res = HttpKit.executeAndParse(com.fanruan.api.net.http.rs.HttpRequest.custom()
.url(this.config.getUriBase()).post(stringEntity).headers(headers).build());
LogKit.info("cmd-OAuthLogin-isValidToken-res:{}", res);
if (StringKit.isBlank(res)) {
return false;
}
String resultCode = new JSONObject(res).getString("Code");
return ComparatorUtils.equals(resultCode, "200");
}
/**
* 刷新移动端权限
*
* @param token
* @param userAccount
* @return
*/
private boolean refreshMobileAuth(String token, String userAccount, String refreshMobileUrl) throws Exception {
Map<String, String> headers = new HashMap<>();
headers.put("Content-Type", "application/json");
JSONObject params = new JSONObject();
params.put("username", userAccount);
params.put("token", token);
LogKit.info("cmd-OAuthLogin-refreshMobileAuth-params:{}, refreshMobileUrl:{}", params.encode(), refreshMobileUrl);
StringEntity stringEntity = new StringEntity(params.encode(), "UTF-8");
String res = HttpKit.executeAndParse(com.fanruan.api.net.http.rs.HttpRequest.custom()
.url(refreshMobileUrl).post(stringEntity).headers(headers).build());
LogKit.info("cmd-OAuthLogin-refreshMobileAuth-res:{}", res);
if (StringKit.isBlank(res)) {
return false;
}
String resultCode = new JSONObject(res).getString("code");
return ComparatorUtils.equals(resultCode, "200");
}
private String removeToken(HttpServletRequest request) throws URISyntaxException {
URIBuilder uriBuilder = new URIBuilder(WebUtils.getOriginalURL(request).replaceAll(" ", "%2B"));
List<NameValuePair> params = uriBuilder.getQueryParams();
params.removeIf(pair -> ComparatorUtils.equals(pair.getName(), "token"));
uriBuilder.clearParameters();
if (!params.isEmpty()) {
uriBuilder.setParameters(params);
}
return uriBuilder.build().toString();
}
}

93
src/main/java/com/fr/plugin/cmd/utils/AESUtil.java

@ -0,0 +1,93 @@
package com.fr.plugin.cmd.utils;
import com.fanruan.api.log.LogKit;
import com.fr.third.org.apache.commons.codec.binary.Base64;
import org.bouncycastle.jce.provider.BouncyCastleProvider;
import javax.crypto.Cipher;
import javax.crypto.spec.IvParameterSpec;
import javax.crypto.spec.SecretKeySpec;
import java.security.Security;
public class AESUtil {
/**
* 密钥算法
*/
private static final String KEY_ALGORITHM = "AES";
/**
* 加密/解密算法 / 工作模式 / 填充方式
* Java 6支持PKCS5Padding填充方式
* Bouncy Castle支持PKCS7Padding填充方式
*/
private static final String CIPHER_ALGORITHM = "AES/CBC/PKCS7Padding";
/**
* 偏移量只有CBC模式才需要
*/
private final static String ivParameter = "xxx";
/**
* AES要求密钥长度为128位或192位或256位java默认限制AES密钥长度最多128位
*/
public static String sKey = "xxx";
/**
* 编码格式
*/
public static final String ENCODING = "utf-8";
static {
//如果是PKCS7Padding填充方式,则必须加上下面这行
Security.addProvider(new BouncyCastleProvider());
}
/**
* AES加密
*
* @param source 源字符串
* @param key 密钥
* @return 加密后的密文
* @throws Exception
*/
public static String encrypt(String source, String key) {
try {
byte[] sourceBytes = source.getBytes(ENCODING);
byte[] keyBytes = key.getBytes(ENCODING);
Cipher cipher = Cipher.getInstance(CIPHER_ALGORITHM, "BC");
IvParameterSpec iv = new IvParameterSpec(ivParameter.getBytes(ENCODING));
cipher.init(Cipher.ENCRYPT_MODE, new SecretKeySpec(keyBytes, KEY_ALGORITHM), iv);
byte[] decrypted = cipher.doFinal(sourceBytes);
return Base64.encodeBase64String(decrypted);
} catch (Exception e) {
e.printStackTrace();
}
return null;
}
/**
* AES解密
*
* @param encryptStr 加密后的密文
* @param key 密钥
* @return 源字符串
* @throws Exception
*/
public static String decrypt(String encryptStr, String key) {
try {
byte[] sourceBytes = Base64.decodeBase64(encryptStr);
byte[] keyBytes = key.getBytes(ENCODING);
Cipher cipher = Cipher.getInstance(CIPHER_ALGORITHM, "BC");
IvParameterSpec iv = new IvParameterSpec(ivParameter.getBytes(ENCODING));
cipher.init(Cipher.DECRYPT_MODE, new SecretKeySpec(keyBytes, KEY_ALGORITHM), iv);
byte[] decoded = cipher.doFinal(sourceBytes);
return new String(decoded, ENCODING);
} catch (Exception e) {
LogKit.error(e.getMessage(), e);
}
return null;
}
}

5
src/main/resources/com/fr/plugin/cmd/locale/lang.properties

@ -0,0 +1,5 @@
Plugin-cmd=Sso Plugin
Plugin-cmd_Group=Sso Plugin
Plugin-cmd_Config_UriBase=Uri Base
Plugin-cmd_Config_UriBase_Description=Uri Base
Plugin-cmd_Licence_Expired=Sso Plugin Licence Expired

5
src/main/resources/com/fr/plugin/cmd/locale/lang_zh_CN.properties

@ -0,0 +1,5 @@
Plugin-cmd=\u5355\u70B9\u767B\u9646\u63D2\u4EF6
Plugin-cmd_Group=\u5355\u70B9\u767B\u9646\u63D2\u4EF6
Plugin-cmd_Config_UriBase=Token\u63A5\u53E3\u5730\u5740
Plugin-cmd_Config_UriBase_Description=Token\u63A5\u53E3\u5730\u5740
Plugin-cmd_Licence_Expired=\u5355\u70B9\u767B\u9646\u63D2\u4EF6\u8BB8\u53EF\u8FC7\u671F
Loading…
Cancel
Save