構建微服務時,常見的問題是爲系統的客戶端應用程序提供惟一的網關。java
事實上,您的服務被拆分爲小型微服務應用程序,這些應用程序應該對用戶不可見,不然可能會致使大量的開發/維護工做。還有一些狀況,整個生態系統網絡流量可能會經過一個可能影響羣集性能的點。git
爲了解決這個問題,Netflix(微服務的一個主要採用者)建立並開源了它的Zuul,Zuul是Netflix的基於JVM的路由器和服務器端負載均衡器。後來Spring在Pivotal下已經在其Spring Cloud中對其進行了調整,使咱們可以經過簡單的步驟輕鬆有效地使用zuul。github
Zuul是一種邊緣服務,它支持對多個服務請求的代理。它爲您的生態系統提供統一的「前門」,容許任何瀏覽器,移動應用程序或其餘用戶界面使用來自多個主機的服務。您能夠將Zuul與其餘Netflix堆棧組件(如Hystrix)集成以實現容錯,使用Eureka進行服務發現,或者使用它來管理整個系統中的路由規則,過濾器和負載平衡。web
最重要的是,Spring框架經過Spring boot/cloud很好地適應了全部這些組件。spring
路由是微服務架構不可或缺的一部分。例如,/
能夠映射到您的Web應用程序,/api/users
映射到用戶服務並/api/shop
映射到商店服務。數據庫
Netflix使用Zuul進行如下操做:後端
認證api
洞察瀏覽器
壓力測試安全
金絲雀測試
動態路由
服務遷移
負載脫落
安全
靜態響應處理
主動/主動流量管理
Zuul的規則引擎容許規則和過濾器基本上以任何JVM語言編寫,內置支持Java和Groovy。
Zuul主要有四種類型的過濾器,使咱們可以在任何特定事務的請求處理的不一樣時間線中攔截流量。咱們能夠爲特定的url模式添加任意數量的過濾器。
前置過濾器 - 在路由請求以前調用。
後置過濾器 - 在路由請求後調用。
路由過濾器 - 用於路由請求。
錯誤過濾器 - 在處理請求時發生錯誤時調用。
使用不一樣的過濾器在Zuul內部請求處理流程
關鍵詞 | 備註 |
---|---|
類型Type | 定義在路由過程當中,過濾器被應用的階段 |
執行順序Execution Order | 在同一個Type中,定義過濾器執行的順序 |
條件Criteria | 過濾器被執行必須知足的條件 |
動做Action | 若是條件知足,過濾器中將被執行的動做 |
PRE
在請求被路由到源服務器前要執行的過濾器
適用業務場景:
認證
選路由
請求日誌
ROUTING
處理將請求發送到源服務器的過濾器
POST
在響應從源服務器返回時要被執行的過濾器
對響應增長HTTP 頭
收集統計和度量
將響應以流的方式發送回客戶端
ERROR
上述階段中出現錯誤要執行的過濾器
public class PreFilter extends ZuulFilter {
// 過濾器類型
如今讓咱們經過使用Zuul建立一個簡單而有意義的生態系統來嘗試一下。咱們將建立下面的組件來演示整個事物:
學生微服務 - 基於spring boot啓動的微服務,它只是暴露單個URL以啓用一些搜索功能。爲簡單起見,咱們將返回硬編碼值,但在現實世界中,咱們可讓此服務鏈接數據庫以獲取數據。
Zuul網關服務
它基於spring boot啓動,它將基本上攔截學生服務的全部流量並應用一系列請求過濾器而後路由到底層服務,並在響應服務時再次,它將應用一些響應過濾。因爲它是一個網關,咱們可使用過濾器有效地採起許多有趣和有用的操做。
網關服務的一些共同責任是 -
在網關層應用微服務身份驗證和安全性以保護實際服務
咱們能夠經過使一些日誌記錄在邊緣獲取有意義的數據和統計數據來實現微服務洞察和監控進入生態系統的全部流量,從而爲咱們提供準確的生產視圖。
動態路由能夠根據須要將請求路由到不一樣的後端羣集。
咱們能夠經過逐漸增長到新集羣的流量來進行運行時壓力測試,以便在許多狀況下衡量性能,例如集羣有新的H / W和網絡設置,或者部署了新版本的生產代碼。
咱們能夠進行動態負載,即爲每種類型的請求分配容量,並刪除超出限制的請求。
咱們能夠應用靜態響應處理,即直接在邊緣構建一些響應,而不是將它們轉發到內部集羣進行處理。
技術棧和運行環境
Java 1.8和IntelliJIDEA做爲開發環境
Spring cloud Zuul做爲網關代理提供商
Spring boot做爲應用程序框架
Spring Rest用於將微服務暴露爲REST
Maven做爲構建工具
按照如下步驟開發學生微服務,稍後將經過zuul代理訪問的幾個REST端點。稍後咱們將研究zuul部分,如今讓咱們先建立學生服務。
建立一個Spring boot項目從spring初始化網站,依賴於Web。
將項目解壓縮並導入到IDEA中。在此步驟中,使用命令執行maven構建,mvn clean install
以便正確下載全部maven依賴項。
咱們如今只需向此服務添加一些REST端點,以便稍後測試網關。爲此,咱們須要經過添加註釋添加一個REST控制器@RestController
。爲簡單起見,咱們將添加一個模型類Student
。
完成全部更改後,該類將以下所示。
package com.example.springboostudentservice;
import java.util.Date;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
spring
這裏咱們按屬性給這個服務命名,spring.application.name=student
咱們也定義了默認端口server.port=8090
。咱們須要覆蓋默認端口,由於咱們將在localhost中運行不一樣微服務的多個實例。
最後使用命令執行maven構建,mvn clean install
並經過運行命令將此項目做爲spring boot應用程序啓動java -jar target\spring-boot-zuulgatway-student-service-0.0.1-SNAPSHOT.jar
。如今,一旦服務器啓動,轉到瀏覽器並測試端點是否正常工做。
http://localhost:8090/echoStudentName/james
http://localhost:8090/getStudentDetails/james
建立過程和學生微服務同樣,可是因爲服務之間的功能和差別性,咱們須要對接口進行簡單的修改
咱們如今只需向此服務添加一些REST端點,爲此,咱們須要經過添加註釋添加一個REST控制器@RestController
。爲簡單起見,咱們將添加一個模型類School
。
完成全部更改後,該類將以下所示。
@SpringBootApplication
@RestController
public class SpringBootZuulgatewaySchoolServiceApplication {
@RequestMapping(value = "/echoSchoolName/{name}")
public String echoSchoolName(@PathVariable(name = "name") String name)
{
return "hello <strong style=\"color: green;\">" + name + " </strong> Responsed on : " + new Date();
}
@RequestMapping(value = "/getSchoolDetails/{name}")
public School getSchoolDetails(@PathVariable(name = "name") String name)
{
return new School(name, "China", "ZheJiang");
}
public static void main(String[] args) {
SpringApplication.run(SpringBootZuulgatewaySchoolServiceApplication.class, args);
}
}
class School
{
String name;
String address;
String cls;
public School(String name, String address, String cls) {
super();
this.name = name;
this.address = address;
this.cls = cls;
}
public String getName() {
return name;
}
public String getAddress() {
return address;
}
public String getCls() {
return cls;
}
}
如今打開application.properties
文件並添加這些條目。
spring:
application:
name: school
server:
port: 8100
這裏咱們按屬性給這個服務命名,spring.application.name=school咱們也定義了默認端口
server.port=8100`。咱們須要覆蓋默認端口,由於咱們將在localhost中運行不一樣微服務的多個實例。
最後使用命令執行maven構建,mvn clean install
並經過運行命令將此項目做爲spring boot應用程序啓動java -jar target\spring-boot-zuulgatway-school-service-0.0.1-SNAPSHOT.jar
。如今,一旦服務器啓動,轉到瀏覽器並測試端點是否正常工做。
http://localhost:8100/echoSchoolName/學軍中學
http://localhost:8100/getSchoolDetails/學軍中學
如今咱們將使用Zuul建立實際的網關服務。
這將是一個基於Spring boot的微服務,但它有一個特殊的功能。它將使用zuul建立一個表明學生服務的API網關。稍後咱們能夠添加任意數量的微服務,如學生服務,學校服務並可以建立一個強大的微服務生態系統。
從spring初始化網站建立一個具備Zuul
依賴關係的Spring boot項目。
將項目做爲現有maven項目解壓縮並導入IDEA。在此步驟中,使用命令執行maven構建,mvn clean install
以便正確下載全部maven依賴項。
如今@EnableZuulProxy
在src
文件夾中的Spring啓動應用程序類中添加註釋。使用此批註,此工件將像Zuul服務代理同樣運行,並將啓用API網關層的全部功能,如前所述。而後咱們將添加一些過濾器和路由配置。
import com.example.zuuldemo.filters.ErrorFilter;
import com.example.zuuldemo.filters.PostFilter;
import com.example.zuuldemo.filters.PreFilter;
import com.example.zuuldemo.filters.RouteFilter;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.netflix.zuul.EnableZuulProxy;
import org.springframework.context.annotation.Bean;
打開application.yml並在下面添加條目
zuul:
routes:
student:
url: http://localhost:8090
school:
url: http://localhost:8100
server:
port: 8080
這裏zuul.routes.student.url
將路由全部流量以請求/student
到實際的學生服務服務器。zuul.routes.school.url
將路由全部流量以請求/school
到實際的學校服務服務器 server.port
- 須要覆蓋默認端口,由於咱們將在localhost中運行不一樣微服務的多個實例。
正如咱們已經描述了zuul組件,咱們將添加一些過濾器,Zuul支持4種類型的過濾器,即pre
,post
,route
和error
。在這裏,咱們將建立每種類型的過濾器。
要編寫過濾器,咱們基本上須要執行如下步驟:
須要擴展 com.netflix.zuul.ZuulFilter
須要重寫filterType
,filterOrder
,shouldFilter
和run
方法。這裏的filterType
方法只能返回四個String中的任何一個 - pre/post/route/error
。下降此值後,過濾器將像特定過濾器同樣運行。
run
method是根據咱們的要求放置濾波器邏輯的地方。
此外,咱們能夠根據須要添加任意數量的任何特定過濾器,這種狀況filterOrder
將用於肯定該過濾器執行階段該文件管理器的順序。
前置過濾器代碼 - 咱們將添加如下預過濾器。目前,過濾器除了println
用於測試目的以外什麼都不作。但實際上那些功能足以完成前面提到的許多重要方面。
package com.example.springbootzuulgateway.filters;
import javax.servlet.http.HttpServletRequest;
import com.netflix.zuul.ZuulFilter;
import com.netflix.zuul.context.RequestContext;
public class PreFilter extends ZuulFilter {
@Override
public String filterType() {
return "pre";
}
@Override
public int filterOrder() {
return 1;
}
@Override
public boolean shouldFilter() {
return true;
}
@Override
public Object run() {
RequestContext ctx = RequestContext.getCurrentContext();
HttpServletRequest request = ctx.getRequest();
System.out.println("Request Method : " + request.getMethod() + " Request URL : " + request.getRequestURL().toString());
return null;
}
}
後置過濾器
package com.example.springbootzuulgateway.filters; import com.netflix.zuul.ZuulFilter; public class PostFilter extends ZuulFilter { @Override public String filterType() { return "post"; } @Override public int filterOrder() { return 1; } @Override public boolean shouldFilter() { return true; } @Override public Object run() { System.out.println("Inside Response Filter"); return null; } }
路由過濾器
package com.example.springbootzuulgateway.filters; import com.netflix.zuul.ZuulFilter; public class RouteFilter extends ZuulFilter { @Override public String filterType() { return "route"; } @Override public int filterOrder() { return 1; } @Override public boolean shouldFilter() { return true; } @Override public Object run() { System.out.println("Inside Route Filter"); return null; } }
錯誤過濾器
package com.example.springbootzuulgateway.filters; import com.netflix.zuul.ZuulFilter; public class ErrorFilter extends ZuulFilter { @Override public String filterType() { return "error"; } @Override public int filterOrder() { return 1; } @Override public boolean shouldFilter() { return true; } @Override public Object run() { System.out.println("Inside Route Filter"); return null; } }
建立要自動註冊和啓用的這些過濾器的bean定義。
@Bean
public PreFilter preFilter() {
return new PreFilter();
}
@Bean
public PostFilter postFilter() {
return new PostFilter();
}
@Bean
public ErrorFilter errorFilter() {
return new ErrorFilter();
}
@Bean
public RouteFilter routeFilter() {
return new RouteFilter();
}
使用命令執行maven構建,mvn clean install
並經過運行命令將此項目做爲spring boot應用程序啓動java -jar target\spring-boot-zuulgateway-0.0.1-SNAPSHOT.jar
。
如今,一旦服務器啓動,轉到瀏覽器並經過訪問學生服務名稱和學校服務來測試端點是否正常工做,即/student
和/school
。
http://localhost:8080/student/echoStudentName/james
http://localhost:8080/school/echoSchoolName/學軍學校
這就是netflix zuul過濾器示例。我建議你本身作,經過代理添加一些更多的底層服務和路由請求,應用不一樣類型的過濾器並在過濾器中添加真正的邏輯。
連接: https://pan.baidu.com/s/1zpUBTCDNVHO4s8TAIOxKVA 提取碼: 8v7w
請在評論部分將您的問題提交給我。
快樂學習!!