springboot实现轻量级网关

spring boot实现超轻量级网关(反向代理、转发) 

在我们的rest服务中,需要暴露一个中间件的接口给用户,但是需要经过rest服务的认证,这是典型的网关使用场景。可以引入网关组件来搞定,但是引入zuul等中间件会增加系统复杂性,这里实现一个超轻量级的网关,只实现请求转发,认证等由rest服务的spring security来搞定。

如何进行请求转发呢? 熟悉网络请求的同学应该很清楚,请求无非就是请求方式、HTTP header,以及请求body,我们将这些信息取出来,透传给转发的url即可。

举例:

/graphdb/** 转发到 Graph_Server/**

获取转发目的地址:

private String createRedictUrl(HttpServletRequest request, String routeUrl, String prefix) {
        String queryString = request.getQueryString();
        return routeUrl + request.getRequestURI().replace(prefix, "") +
                (queryString != null ? "?" + queryString : "");
    }

解析请求头和内容

然后从request中提取出header、body等内容,构造一个RequestEntity,后续可以用RestTemplate来请求。

private RequestEntity createRequestEntity(HttpServletRequest request, String url) throws URISyntaxException, IOException {
        String method = request.getMethod();
        HttpMethod httpMethod = HttpMethod.resolve(method);
        MultiValueMap<String, String> headers = parseRequestHeader(request);
        byte[] body = parseRequestBody(request);
        return new RequestEntity<>(body, headers, httpMethod, new URI(url));
    }


    private byte[] parseRequestBody(HttpServletRequest request) throws IOException {
        InputStream inputStream = request.getInputStream();
        return StreamUtils.copyToByteArray(inputStream);
    }

    private MultiValueMap<String, String> parseRequestHeader(HttpServletRequest request) {
        HttpHeaders headers = new HttpHeaders();
        List<String> headerNames = Collections.list(request.getHeaderNames());
        for (String headerName : headerNames) {
            List<String> headerValues = Collections.list(request.getHeaders(headerName));
            for (String headerValue : headerValues) {
                headers.add(headerName, headerValue);
            }
        }
        return headers;
    }

透明转发

最后用RestTemplate来实现请求:

 private ResponseEntity<String> route(RequestEntity requestEntity) {
        RestTemplate restTemplate = new RestTemplate();
        return restTemplate.exchange(requestEntity, String.class);
    }

全部代码

以下是轻量级转发全部代码:

import org.springframework.http.*;
import org.springframework.stereotype.Service;
import org.springframework.util.MultiValueMap;
import org.springframework.util.StreamUtils;
import org.springframework.web.client.RestTemplate;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.InputStream;
import java.net.URI;
import java.net.URISyntaxException;
import java.util.Collections;
import java.util.List;

@Service
public class RoutingDelegate {


    public ResponseEntity<String> redirect(HttpServletRequest request, HttpServletResponse response,String routeUrl, String prefix) {
        try {
            // build up the redirect URL
            String redirectUrl = createRedictUrl(request,routeUrl, prefix);
            RequestEntity requestEntity = createRequestEntity(request, redirectUrl);
            return route(requestEntity);
        } catch (Exception e) {
            return new ResponseEntity("REDIRECT ERROR", HttpStatus.INTERNAL_SERVER_ERROR);
        }
    }

    private String createRedictUrl(HttpServletRequest request, String routeUrl, String prefix) {
        String queryString = request.getQueryString();
        return routeUrl + request.getRequestURI().replace(prefix, "") +
                (queryString != null ? "?" + queryString : "");
    }


    private RequestEntity createRequestEntity(HttpServletRequest request, String url) throws URISyntaxException, IOException {
        String method = request.getMethod();
        HttpMethod httpMethod = HttpMethod.resolve(method);
        MultiValueMap<String, String> headers = parseRequestHeader(request);
        byte[] body = parseRequestBody(request);
        return new RequestEntity<>(body, headers, httpMethod, new URI(url));
    }
	
    private ResponseEntity<String> route(RequestEntity requestEntity) {
        RestTemplate restTemplate = new RestTemplate();
        return restTemplate.exchange(requestEntity, String.class);
    }


    private byte[] parseRequestBody(HttpServletRequest request) throws IOException {
        InputStream inputStream = request.getInputStream();
        return StreamUtils.copyToByteArray(inputStream);
    }

    private MultiValueMap<String, String> parseRequestHeader(HttpServletRequest request) {
        HttpHeaders headers = new HttpHeaders();
        List<String> headerNames = Collections.list(request.getHeaderNames());
        for (String headerName : headerNames) {
            List<String> headerValues = Collections.list(request.getHeaders(headerName));
            for (String headerValue : headerValues) {
                headers.add(headerName, headerValue);
            }
        }
        return headers;
    }
}

Spring 集成

Spring Controller,RequestMapping里把GET \ POST\PUT\DELETE 支持的请求带上,就能实现转发了。

@RestController
@RequestMapping(GraphDBController.DELEGATE_PREFIX)
@Api(value = "GraphDB", tags = {
        "graphdb-Api"
})
public class GraphDBController {

    @Autowired
    GraphProperties graphProperties;

    public final static String DELEGATE_PREFIX = "/graphdb";

    @Autowired
    private RoutingDelegate routingDelegate;

    @RequestMapping(value = "/**", method = {RequestMethod.GET, RequestMethod.POST, RequestMethod.PUT, RequestMethod.DELETE}, produces = MediaType.TEXT_PLAIN_VALUE)
    public ResponseEntity catchAll(HttpServletRequest request, HttpServletResponse response) {
        return routingDelegate.redirect(request, response, graphProperties.getGraphServer(), DELEGATE_PREFIX);
    }
}

分类: java编程

标签: java

好文要顶 关注我 收藏该文 微信分享

JadePeng
粉丝 - 353 关注 - 22

+加关注

3

0

« 上一篇: hugegraph 支持sparql 与cypher
» 下一篇: hugegraph 存取数据解析

posted @ 2020-11-18 10:11  JadePeng  阅读(13349)  评论(4)  编辑  收藏  举报

会员力量,点亮园子希望

刷新页面返回顶部

登录后才能查看或发表评论,立即 登录 或者 逛逛 博客园首页

【推荐】在 ASP.NET Core 中运行 WebForms 业务代码
【推荐】园子周边第二季:更大的鼠标垫,没有logo的鼠标垫
【推荐】阿里云云市场联合博客园推出开发者商店,欢迎关注
【推荐】会员力量,点亮园子希望,期待您升级成为园子会员

编辑推荐:
· 日常 Bug 排查 - 连接突然全部关闭
· 「动画进阶」极具创意的鼠标交互动画
· WPF 稳定的全屏化窗口方法
· 云服务器遭到黑客入侵植入木马病毒排查过程
· HTTPS 是如何进行安全传输的 ?

阅读排行:
· 一个不知名的开源项目可以带来多少收入
· 厉害了!12秒将百万数据通过EasyExcel导入MySQL数据库中
· 日常Bug排查-连接突然全部关闭
· Visual Studio中的四款代码格式化工具
· C#.Net筑基-模式匹配汇总

公告

欢迎关注作者微信公众号, 一起交流软件开发
 

欢迎关注作者微信公众号

关注作者同名微信公众号

昵称: JadePeng
园龄: 15年8个月
粉丝: 353
关注: 22

+加关注

<2024年5月>
2829301234
567891011
12131415161718
19202122232425
2627282930311
2345678

搜索

常用链接

最新随笔

我的标签

积分与排名

  • 积分 - 440046
  • 排名 - 1627

随笔分类 (217)

随笔档案 (222)

阅读排行榜

评论排行榜

推荐排行榜

最新评论

  • 9
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,我可以为您提供一些基本思路,但是具体实现需要您自己进行开发和测试。 首先,您需要了解MQTT协议的基本概念和用法。MQTT是一种轻量级的消息协议,适用于物联网设备之间的通信。它基于发布-订阅模型,支持多个客户端同时订阅同一个主题,实现消息的传递。 接下来,您可以使用Eclipse Paho项目提供的Java客户端库来实现MQTT网关。这个库提供了一些类和接口,可以帮助您轻松地连接到MQTT代理,发布和订阅主题,处理消息等等。 然后,您可以使用Spring Boot框架来搭建一个Web应用程序,作为MQTT网关的控制中心。这个应用程序可以提供HTTP接口,允许用户动态地配置MQTT代理连接信息,主题订阅和消息处理规则等等。 最后,您需要编写一些代码来实现MQTT网关的核心逻辑。这个逻辑包括: 1. 连接到MQTT代理,并注册一个唯一的客户端ID。 2. 订阅用户指定的主题,处理收到的消息,并根据用户指定的规则进行转发或过滤。 3. 将来自外部系统的消息转换为MQTT消息,并发布到指定的主题中。 4. 处理MQTT代理发送回来的错误信息和断开连接的事件。 需要注意的是,MQTT网关需要保持长时间的连接,处理大量的消息,并确保消息的可靠性和安全性。因此,您需要仔细考虑性能、可靠性和安全性方面的问题,并进行充分的测试和调试。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值