我們一起聊聊如何通過Java實(shí)現(xiàn)類似Nginx代理
proxy-spring-boot-starter
最近遇到一個(gè)問題,在內(nèi)網(wǎng)環(huán)境中部署的項(xiàng)目需要調(diào)用外網(wǎng)完成一些應(yīng)用,一般情況我們可以通過增加一臺機(jī)器,部署到可以訪問外網(wǎng)的服務(wù)器上,然后內(nèi)網(wǎng)直接連接該機(jī)器通過Nginx進(jìn)行代理即可。但是出于安全考慮以及各個(gè)服務(wù)都是由多個(gè)微服務(wù)組成,需要接入SSO實(shí)現(xiàn)認(rèn)證后才能訪問。
實(shí)現(xiàn)過程
- 定義一個(gè)配置文件,后面可以在application.yml中通過配置實(shí)現(xiàn)代理不同網(wǎng)站。
@Getter
@Setter
@ConfigurationProperties(prefix = "proxy")
public class ProxyProperties {
/**
* 需要代理的服務(wù)列表
*/
private Map<String,Server> servers;
/**
*
*/
@Getter@Setter
public static class Server{
private String path;
private String target;
private String name;
}
}
- 引入Spring依賴,在編寫配置時(shí)可以自動提示。
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-autoconfigure-processor</artifactId>
<optional>true</optional>
</dependency>
- 該功能實(shí)現(xiàn)的主要依賴。
<dependency>
<groupId>org.mitre.dsmiley.httpproxy</groupId>
<artifactId>smiley-http-proxy-servlet</artifactId>
<version>2.0</version>
</dependency>
- 增加AutoConfiguration,包含兩個(gè)步驟。
a) 定義一個(gè)ProxyServletConfiguration配置 這里我們基于ImportBeanDefinitionRegistrar接口,動態(tài)讀取代理服務(wù)列表,然后通過ServletRegistrationBean創(chuàng)建Servlet代碼。
public static class ProxyServleImportBeanDefinitionRegistrar implements ImportBeanDefinitionRegistrar, EnvironmentAware {
private ProxyProperties properties;
@Override
public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {
Map<String, ProxyProperties.Server> servers = properties.getServers();
if( !CollectionUtils.isEmpty( servers ) ){
for(Map.Entry<String, ProxyProperties.Server> entry : servers.entrySet()){
ProxyProperties.Server server = entry.getValue();
LOGGER.info("開始注冊服務(wù)代理:{} ( {} => {})", server.getName(), server.getPath(), server.getTarget());
BeanDefinitionBuilder builder = BeanDefinitionBuilder.genericBeanDefinition(ServletRegistrationBean.class);
// builder.addConstructorArgValue(new ProxyServlet() ).addConstructorArgValue(server.getPath());
builder.setFactoryMethodOnBean("getServletRegistrationBean", "proxyServletFactory");
builder.addConstructorArgValue(entry.getKey()).addConstructorArgValue(entry.getValue());
registry.registerBeanDefinition(entry.getKey()+"ServletRegistrationBean", builder.getBeanDefinition());
}
}
}
@Override
public void setEnvironment(Environment environment) {
String prefix = Objects.requireNonNull(AnnotationUtils.getAnnotation(ProxyProperties.class, ConfigurationProperties.class)).prefix();
properties = Binder.get(environment).bind(prefix, ProxyProperties.class).get();
}
}
public static ServletRegistrationBean createServletRegistrationBean(String key, ProxyProperties.Server server){
ServletRegistrationBean servletRegistrationBean = new ServletRegistrationBean(new ProxyServlet(), server.getPath());
servletRegistrationBean.setName(server.getName());
servletRegistrationBean.addInitParameter(ProxyServlet.P_TARGET_URI, server.getTarget());
servletRegistrationBean.addInitParameter(ProxyServlet.P_LOG, String.valueOf(true));
// 自動處理重定向
// servletRegistrationBean.addInitParameter(ProxyServlet.P_HANDLEREDIRECTS, String.valueOf(false));
// // 保持 COOKIES 不變
servletRegistrationBean.addInitParameter(ProxyServlet.P_PRESERVECOOKIES, String.valueOf(true));
// // Set-Cookie 服務(wù)器響應(yīng)標(biāo)頭中保持 cookie 路徑不變
servletRegistrationBean.addInitParameter(ProxyServlet.P_PRESERVECOOKIEPATH, String.valueOf(true));
// // 保持 HOST 參數(shù)不變
// servletRegistrationBean.addInitParameter(ProxyServlet.P_PRESERVEHOST, String.valueOf(true));
return servletRegistrationBean;
}
b) 通過在src/main/resources/META-INF/spring/org.springframework.boot.autoconfigure.AutoConfiguration.imports中引入上面的配置,這樣其他模塊只需要引入該jar則會由Spring自動注入 Configuration。
cn.cycad.proxy.config.ProxyServletConfiguration
使用方法
使用方式非常簡單,在配置文件中添加代理的配置,啟動服務(wù)即可:
- 修改配置文件application.yml
proxy:
servers:
Baidu:
path: /baidu/*
target: 'https://www.baidu.com'
name: 百度
Shici:
path: /shici/*
target: 'https://v1.jinrishici.com'
name: 詩詞
- 啟動服務(wù)
- 示例 http://localhost:8080/shici
可以看到返回的內(nèi)容與https://v1.jinrishici.com相同。
{
"welcome": "歡迎使用古詩詞·一言",
"api-document": "下面為本API可用的所有類型,使用時(shí),在鏈接最后面加上 .svg / .txt / .json / .png 可以獲得不同格式的輸出",
"help": "具體安裝方法請?jiān)L問項(xiàng)目首頁 https://gushi.ci/",
"list": [
{
"全部": "https://v1.jinrishici.com/all"
},
{
"抒情": "https://v1.jinrishici.com/shuqing"
},
{
"四季": "https://v1.jinrishici.com/siji"
},
{
"山水": "https://v1.jinrishici.com/shanshui"
},
{
"天氣": "https://v1.jinrishici.com/tianqi"
}
]
}
優(yōu)缺點(diǎn)
- 通過該方式實(shí)現(xiàn)首先需要有一個(gè)可以訪問外網(wǎng)的服務(wù)器,同時(shí)該服務(wù)器和內(nèi)網(wǎng)環(huán)境互通
- 如果需要添加認(rèn)證模塊,直接引入即可
- 如果代理的網(wǎng)站需要更多的信息才能訪問,則需要進(jìn)一步擴(kuò)展