国产xxxx99真实实拍_久久不雅视频_高清韩国a级特黄毛片_嗯老师别我我受不了了小说

資訊專欄INFORMATION COLUMN

讓ERP的服務(wù)更開放! ——用微服務(wù)架構(gòu)搭建的一套基于EBS的API服務(wù)系統(tǒng)

JouyPub / 2744人閱讀

摘要:每個服務(wù)運行在其獨立的進程中,服務(wù)與服務(wù)間采用輕量級的通信機制互相溝通通常是基于的。在微服務(wù)架構(gòu)下,故障會被隔離在單個服務(wù)中。

1. 源碼下載地址

源碼鏈接: https://github.com/samt007/xy...

這是用Spring Cloud微服務(wù)架構(gòu)搭建的一套基于EBS的API服務(wù)系統(tǒng)

如對本文有任何的疑問,請聯(lián)系我:samt007@qq.com

2. Introduction介紹

這是一篇傳統(tǒng)ERP系統(tǒng)和基于Java的微服務(wù)架構(gòu)有效結(jié)合的技術(shù)文檔。

傳統(tǒng)ERP關(guān)注的是企業(yè)內(nèi)部的信息化管理。當ERP系統(tǒng)能將其服務(wù)發(fā)布出去之后(結(jié)合微服務(wù)架構(gòu)),就可以很好實現(xiàn)與第三方系統(tǒng)的無縫對接,同時也可以實現(xiàn)擴展ERP本身的功能。
目標是:讓ERP的服務(wù)更開放!
2.1 它有什么用?

簡單來說,就是:

相當于做一個中間服務(wù)平臺,把ERP的功能做成Web Service與其它系統(tǒng)集成。

下面具體說明它的作用。

1. 如果沒有它...

如果沒有一個統(tǒng)一的API對接平臺,那么ERP和第三方系統(tǒng)做對接,那會是下圖所示的結(jié)構(gòu):

從上圖可以看出,各個第三方系統(tǒng)分別和ERP做對接,無論是DBLINK還是通過自己的Web Service, 都是雜亂的,沒能統(tǒng)一管理的。簡單來說就是各自為政,為對接而需要做的事情都是零碎的。

當?shù)谌较到y(tǒng)越來越多的時候,那對于日常的運維將會是一個災(zāi)難的問題。舉個例子,就一個簡單的運維:密碼修改,需要調(diào)整的地方就會很多,也很容易遺漏。

特別指出的是,接口功能的復(fù)用方面也是個難題。假設(shè)一個查詢庫存的接口,CRM系統(tǒng)和在線下單系統(tǒng)都可以用的,需要開發(fā)2次。

2. 自從有了它...

有了這套統(tǒng)一的API系統(tǒng)之后,ERP系統(tǒng)和別的系統(tǒng)之間的對接就變成了這個結(jié)構(gòu):

所以,有了它,相當于ERP的API都可以通過這服務(wù)平臺給開發(fā)出去,基本上所有的接口可以完成的業(yè)務(wù),都可以通過這套服務(wù)平臺來完成。

可以實現(xiàn):

對外服務(wù)的統(tǒng)一

API服務(wù)之間可以實現(xiàn)互相調(diào)用,并且實現(xiàn)服務(wù)取數(shù)和處理的邏輯的統(tǒng)一

代碼的統(tǒng)一,提高開發(fā)效率。特別是comm代碼的部分。

提高與第三方系統(tǒng)對接的穩(wěn)定性:只需要關(guān)注該微服務(wù)系統(tǒng)的運行穩(wěn)定性即可。

3. 有哪些實例?

舉個例子:

1) 成品進出條碼管理系統(tǒng):

大概這個需求:
成品入庫的時候,直接可以用條碼槍掃條碼或者二維碼就可以入庫;銷售出庫的時候,也可以通過刷成品的條碼直接進行出庫。JIT管理。

通過這個系統(tǒng)的實現(xiàn)邏輯是:
通過EBS的用戶名和密碼可以登入條碼槍上的APP系統(tǒng)。然后,刷條碼的時候,通過該Web Service可以產(chǎn)生對應(yīng)的事務(wù)處理!例如完工入庫,處理物料搬運單等。

下面是該系統(tǒng)的一些截圖

注意:該功能后臺API由該微服務(wù)提供,前臺是安卓的APP

2) 與各個第三方系統(tǒng)的集成:

每個企業(yè)內(nèi)部都有各種第三方系統(tǒng),這些系統(tǒng)或多或少都需要和EBS進行集成。傳統(tǒng)的集成方法是通過DBLINK。

但是有這套Web Service系統(tǒng)之后,就可以統(tǒng)一通過該Web Service作為中介,和EBS進行數(shù)據(jù)的集成交互!

2.2 什么是微服務(wù)?

關(guān)于它的解析,網(wǎng)上資料很多。
這里引用某位大神的總結(jié)(引用:http://blog.51cto.com/ityoukn...),解析得比較到位:

微服務(wù)的概念源于2014年3月Martin Fowler所寫的一篇文章“Microservices”。

微服務(wù)架構(gòu)是一種架構(gòu)模式,它提倡將單一應(yīng)用程序劃分成一組小的服務(wù),服務(wù)之間互相協(xié)調(diào)、互相配合,為用戶提供最終價值。每個服務(wù)運行在其獨立的進程中,服務(wù)與服務(wù)間采用輕量級的通信機制互相溝通(通常是基于HTTP的RESTful
API)。每個服務(wù)都圍繞著具體業(yè)務(wù)進行構(gòu)建,并且能夠被獨立地部署到生產(chǎn)環(huán)境、類生產(chǎn)環(huán)境等。另外,應(yīng)盡量避免統(tǒng)一的、集中式的服務(wù)管理機制,對具體的一個服務(wù)而言,應(yīng)根據(jù)業(yè)務(wù)上下文,選擇合適的語言、工具對其進行構(gòu)建。

微服務(wù)是一種架構(gòu)風格,一個大型復(fù)雜軟件應(yīng)用由一個或多個微服務(wù)組成。系統(tǒng)中的各個微服務(wù)可被獨立部署,各個微服務(wù)之間是松耦合的。每個微服務(wù)僅關(guān)注于完成一件任務(wù)并很好地完成該任務(wù)。在所有情況下,每個任務(wù)代表著一個小的業(yè)務(wù)能力。

微服務(wù)架構(gòu)優(yōu)勢

復(fù)雜度可控:在將應(yīng)用分解的同時,規(guī)避了原本復(fù)雜度無止境的積累。每一個微服務(wù)專注于單一功能,并通過定義良好的接口清晰表述服務(wù)邊界。由于體積小、復(fù)雜度低,每個微服務(wù)可由一個小規(guī)模開發(fā)團隊完全掌控,易于保持高可維護性和開發(fā)效率。

獨立部署:由于微服務(wù)具備獨立的運行進程,所以每個微服務(wù)也可以獨立部署。當某個微服務(wù)發(fā)生變更時無需編譯、部署整個應(yīng)用。由微服務(wù)組成的應(yīng)用相當于具備一系列可并行的發(fā)布流程,使得發(fā)布更加高效,同時降低對生產(chǎn)環(huán)境所造成的風險,最終縮短應(yīng)用交付周期。

技術(shù)選型靈活:微服務(wù)架構(gòu)下,技術(shù)選型是去中心化的。每個團隊可以根據(jù)自身服務(wù)的需求和行業(yè)發(fā)展的現(xiàn)狀,自由選擇最適合的技術(shù)棧。由于每個微服務(wù)相對簡單,故需要對技術(shù)棧進行升級時所面臨的風險就較低,甚至完全重構(gòu)一個微服務(wù)也是可行的。

容錯:當某一組建發(fā)生故障時,在單一進程的傳統(tǒng)架構(gòu)下,故障很有可能在進程內(nèi)擴散,形成應(yīng)用全局性的不可用。在微服務(wù)架構(gòu)下,故障會被隔離在單個服務(wù)中。若設(shè)計良好,其他服務(wù)可通過重試、平穩(wěn)退化等機制實現(xiàn)應(yīng)用層面的容錯。

擴展:單塊架構(gòu)應(yīng)用也可以實現(xiàn)橫向擴展,就是將整個應(yīng)用完整的復(fù)制到不同的節(jié)點。當應(yīng)用的不同組件在擴展需求上存在差異時,微服務(wù)架構(gòu)便體現(xiàn)出其靈活性,因為每個服務(wù)可以根據(jù)實際需求獨立進行擴展。

2.3 ERP API微服務(wù)系統(tǒng)架構(gòu)說明 2.3.1 系統(tǒng)開發(fā)邏輯說明

從上面的解析得知:微服務(wù)是一種技術(shù)架構(gòu),將一個龐大的服務(wù)體系拆分為若干個子服務(wù)執(zhí)行。
問題來了,應(yīng)該如何拆分呢?就是服務(wù)的拆分原則是什么。

這個問題就像是一個大表如何進行分區(qū)一樣,其實我覺得是具體問題具體分析。
由于我開發(fā)的是基于EBS的微服務(wù)系統(tǒng),正常來說,比較合理的劃分規(guī)則應(yīng)該是以EBS的模塊來分。

相當于每個模塊都劃分為一個多帶帶的微服務(wù)。例如FND模塊,INV模塊,WIP模塊等等。

有時候,為了某個目的,可能有些功能是定制的,需要提取幾個模塊的數(shù)據(jù)來用,而且被別的模塊重用的概率很低。所以,實際上也可以以定制的功能來劃分微服務(wù)。

目前來說,該系統(tǒng)包括2個子服務(wù):

xygerp-ald服務(wù):ald模塊

這個是整個微服務(wù)API的核心ald模塊。這個模塊的主要功能是驗證用戶的登錄,為所有的api模塊提供統(tǒng)一的token認證。相當于ebs的FND模塊。

xygerp-albc服務(wù):albc子模塊

這個項目是屬于微服務(wù)的API模塊之一:條碼管理系統(tǒng)提供數(shù)據(jù)以及數(shù)據(jù)處理的API。
主要是為條形碼傳輸系統(tǒng)用。

當然,未來可以添加若干個服務(wù)。微服務(wù)架構(gòu)的優(yōu)勢是有很好的橫向擴展能力!

2.3.2 微服務(wù)系統(tǒng)架構(gòu)圖

該系統(tǒng)的架構(gòu)圖如下所示。

注意:

1.Spring Cloud模塊中,實際上Spring Security并不是多帶帶的一個模塊,而是融入到每一個業(yè)務(wù)微服務(wù)模塊中! 每個微服務(wù)都必須要有token認證才允許訪問API,它非常重要! 所以我將它給列到Spring Cloud模塊中。

2.圖中有些模塊目前還沒有實現(xiàn)。
目前實現(xiàn)了架構(gòu)整體,包括以下的服務(wù)(下一個章節(jié)會具體說明每個模塊的用途):

xygerp-ald

xygerp-albc

xygerp-server-eureka

xygerp-server-zuul

注意: 以后會按需添加別的模塊。
3 系統(tǒng)開發(fā)流程

接下來是一步一步來開發(fā)一套這個基于Spring Cloud的微服務(wù)系統(tǒng)。

3.1 必須掌握的基礎(chǔ)開發(fā)技術(shù)知識點

開發(fā)系統(tǒng)都必須要打好基礎(chǔ)。所以,這里列出了開發(fā)基于Spring Cloud的微服務(wù)系統(tǒng)需要掌握的開發(fā)技術(shù)。

下面我不會具體解說每一個開發(fā)技術(shù)如何學習,因為這并不是本文的重點。工欲善其事必先利其器,基礎(chǔ)還是必須要打好。
1)Java語言

必須要熟悉java,否則基本不用看文檔了。先打好基礎(chǔ)吧!

2)Maven項目管理工具

系統(tǒng)開發(fā)的項目都是以maven做項目管理的,所以必須要先安裝并掌握maven工具。

3)Oracle數(shù)據(jù)庫+PLSQL+SQL語言

數(shù)據(jù)庫端的開發(fā)技術(shù)。這里選用Oracle數(shù)據(jù)庫,因為EBS就是基于Oracle數(shù)據(jù)庫的ERP系統(tǒng)。

3.2 需要熟悉的java框架

Java技術(shù)發(fā)展到現(xiàn)在,已經(jīng)出現(xiàn)了許多非常優(yōu)秀的開源框架,我們可以借助這些框架來快速開發(fā)系統(tǒng)。

3.2.1 Spring框架技術(shù)棧(全家桶)

Spring是目前開源的主流的技術(shù)包。
該系統(tǒng)主要用到的技術(shù)棧是:Spring boot,Spring Security以及Spring Cloud。

1.Spring Boot

系統(tǒng)基于SpringBoot快速開發(fā)。選擇目前熱度很高的SpringBoot,最大限度地降低配置復(fù)雜度,把大量的精力投入到業(yè)務(wù)開發(fā)中來。

2.Spring MVC

利用Spring MVC框架處理所有的url請求,簡單易用。

3.Spring Security

Spring security是一個強大的和高度可定制的身份驗證和訪問控制框架。它是確保基于Spring的應(yīng)用程序的標準。
這里主要是用Spring Security框架處理Token機制。

4.Spring Cloud

Spring Cloud是一系列框架的有序集合。
它利用Spring Boot的開發(fā)便利性巧妙地簡化了分布式系統(tǒng)基礎(chǔ)設(shè)施的開發(fā),如服務(wù)發(fā)現(xiàn)注冊、配置中心、消息總線、負載均衡、斷路器、數(shù)據(jù)監(jiān)控等,都可以用Spring Boot的開發(fā)風格做到一鍵啟動和部署。
注意:本系統(tǒng)目前使用Spring Cloud的2個模塊

請求統(tǒng)一通過API網(wǎng)關(guān)(Zuul)來訪問內(nèi)部服務(wù).

網(wǎng)關(guān)接收到請求后,從注冊中心(Eureka)獲取可用服務(wù)

3.2.2 MyBatis

ORM框架選用MyBatis。

主要是考慮到它能夠很好支持SQL語句:MyBatis是支持普通SQL查詢,存儲過程和高級映射的優(yōu)秀持久層框架。
另外,還用到了MyBatis的一些提高開發(fā)效率的插件,特別是通用Mapper和PageHelper分頁插件!

3.2.3 Alibaba druid

DRUID是阿里巴巴開源平臺上一個數(shù)據(jù)庫連接池實現(xiàn)。它結(jié)合了C3P0、DBCP、PROXOOL等DB池的優(yōu)點,同時加入了日志監(jiān)控,可以很好的監(jiān)控DB池連接和SQL的執(zhí)行情況。

3.2.4 Swagger

前端和后端的唯一聯(lián)系,變成了API接口。

API文檔變成了前后端開發(fā)人員聯(lián)系的紐帶,變得越來越重要,swagger就是一款讓你更好的書寫API文檔的框架。

3.3 需要準備的軟件工具 3.3.1Redis

目前用Redis的主要作用是存取token,以配合實現(xiàn)Spring Security完成api訪問的安全機制。

以后可以考慮做緩存或者消息隊列等高級功能。
3.3.2Docker

基于Docker的容器化部署。

由于使用了微服務(wù)架構(gòu)后,我們的系統(tǒng)將會由很多子系統(tǒng)構(gòu)成。
為了達到多個系統(tǒng)之間環(huán)境隔離的目的,我們可以將它們部署在多臺服務(wù)器上,可這樣的成本會比較高,而且每臺服務(wù)器的性能可能都沒有充分利用起來。

所以我們很自然地想到了虛擬機,在同一臺服務(wù)器上運行多個虛擬機,從而實現(xiàn)環(huán)境的隔離,每個虛擬機上運行獨立的服務(wù)。

然而虛擬機的隔離成本依舊很高,因為它需要占用服務(wù)器較多的硬件資源和軟件資源。
所以,在微服務(wù)結(jié)構(gòu)下,要實現(xiàn)服務(wù)環(huán)境的隔離,Docker是最佳選擇。它比虛擬機更加輕量級,占用資源較少,而且能夠?qū)崿F(xiàn)快速部署。

備注:后面有專題說明這個工具如何安裝使用。由于篇幅原因,本文檔暫時不講解容器化部署。
3.3.3 Jenkins

Jenkins自動化構(gòu)建工具。

當我們采用了微服務(wù)架構(gòu)后,我們會發(fā)現(xiàn)這樣一個問題。整個系統(tǒng)由許許多多的服務(wù)構(gòu)成,這些服務(wù)都需要運行在多帶帶的容器中,那么每次發(fā)布的復(fù)雜度將非常高。

首先你要搞清楚這些服務(wù)之間的依賴關(guān)系、啟動的先后順序,然后再將多個子系統(tǒng)挨個編譯、打包、發(fā)布。這些操作技術(shù)難度低,卻又容易出錯。

那么有什么工具能夠幫助我們解決這些問題呢?答案就是——Jenkins。

它是一款自動化構(gòu)建的工具,簡單的來說,就是我們只需要在它的界面上按一個按鈕,就可以實現(xiàn)上述一系列復(fù)雜的過程。

備注:后面有專題說明這個工具如何安裝使用。由于篇幅原因,本文檔暫時不講解自動化構(gòu)建。
3.4 具體開發(fā)流程

現(xiàn)在開始手把手來搭建一套這樣子的系統(tǒng)。

3.4.1 創(chuàng)建Maven項目的組織結(jié)構(gòu)

先創(chuàng)建一個微服務(wù)系統(tǒng)的父級項目:xygerp-api

再在這個項目下面分別創(chuàng)建下面幾個子項目:

項目名稱 說明
xygerp-ald ald模塊,端口:8180。這個是整個微服務(wù)API的核心ald模塊。這個模塊的主要功能是驗證用戶的登錄,為所有的api模塊提供統(tǒng)一的token認證。相當于ebs的FND模塊。
xygerp-albc albc子模塊,端口:8181。這個項目是屬于微服務(wù)的API模塊之一:條碼管理系統(tǒng)提供數(shù)據(jù)以及數(shù)據(jù)處理的API。主要是為條形碼傳輸系統(tǒng)用。
xygerp-comm comm模塊這個項目是所有API項目的核心依賴項目。說白了就是將API微服務(wù)架構(gòu)的所有項目的公用代碼可以抽取在這里。
xygerp-basic-support 核心基礎(chǔ)支撐模塊這個項目是所有API項目的基礎(chǔ)數(shù)據(jù)支撐項目。這里統(tǒng)一歸集了所有的Entity!因為對于Entity來說,應(yīng)該是整個微服務(wù)都公用的。
xygerp-server-eureka Spring Cloud的服務(wù)與發(fā)現(xiàn)的服務(wù)中心。端口:8101。這個模塊是Spring cloud的最核心的模塊了,用來處理各個微服務(wù)之間的服務(wù)調(diào)用的。
xygerp-server-zuul Spring Cloud服務(wù)網(wǎng)關(guān)。端口:8102。在Spring Cloud架構(gòu)體系內(nèi)的所有微服務(wù)都通過Zuul來對外提供統(tǒng)一的訪問入口,所有需要和微服務(wù)架構(gòu)內(nèi)部服務(wù)進行通訊的請求都走統(tǒng)一網(wǎng)關(guān)。
它們的目錄結(jié)果是這樣子的:

注意:
關(guān)于xygerp-basic-support:核心基礎(chǔ)支撐模塊

可能您會有疑問:為什么不將entity歸并在它所屬的模塊?其實是這樣的,我主要是考慮到服務(wù)之間的互相調(diào)用的問題。

微服務(wù)雖然客觀上是一個多帶帶的服務(wù),但是,實際上大部分的功能肯定是互相調(diào)用的。舉個例子,銷售訂單模塊調(diào)用庫存模塊的功能查詢個庫存是很正常的業(yè)務(wù)吧?

如果entity不共用的話,相當于銷售模塊得到的庫存模塊的結(jié)果無法歸集為bean來處理,這樣子對于后臺的處理會帶來極大的不便!

3.4.2 構(gòu)建模塊的依賴關(guān)系

接著需要通過pom文件來指定它們之間的依賴關(guān)系,依賴關(guān)系如下圖所示。

1. 業(yè)務(wù)服務(wù)部分:


注意,上面的4個項目是有依賴關(guān)系的。
所以,xygerp-ald和xygerp-albc部署方式 改為war部署 (主要是為了利用jenkins進行自動化部署)。

而xygerp-comm和xygerp-basic-support只是為各個微服務(wù)提供基礎(chǔ)代碼支撐,所以是jar部署即可。

此外,為了簡化各個模塊的配置,我們將所有模塊的通用依賴放在Project的pom文件中,然后讓所有模塊作為Project的子模塊。
這樣子模塊就可以從父模塊中繼承所有的依賴,而不需要自己再配置了。

在父pom中指定子模塊

modules標簽指定了當前模塊的子模塊是誰,但是僅在父模塊的pom文件中指定子模塊還不夠,還需要在子模塊的pom文件中指定父模塊是誰。

    
        xygerp-comm          
        xygerp-basic-support    
        xygerp-server-eureka   
        xygerp-server-zuul      
        xygerp-ald             
        xygerp-albc           
    

需要在子模塊中指定父模塊


        xygerp
        com.xygerp
        1.0-SNAPSHOT
        ../pom.xml

備注:具體代碼直接看源碼吧。這里只是提及了一些重點設(shè)置而已。

所以,到此為止,模塊的依賴關(guān)系配置完畢!但要注意模塊打包的順序。

由于所有模塊都依賴于xygerp-comm模塊和xygerp-basic-support模塊,因此在構(gòu)建模塊時,首先需要編譯、打包、安裝xygerp-comm模塊和xygerp-basic-support模塊,將它打包進本地倉庫中,這樣上層模塊才能引用到。當該模塊安裝完畢后,再構(gòu)建上層模塊。
否則在構(gòu)建上層模塊的時候會出現(xiàn)找不到xygerp-comm模塊中類庫的問題。

Tips: 其實,如果是在父級目錄直接用mvn package整體打包的話,那打包構(gòu)建的順序在父pom中是直接指定了!
2. 微服務(wù)架構(gòu)服務(wù)治理部分

xygerp-server-eureka:Spring Cloud服務(wù)注冊和發(fā)現(xiàn)。就是處理服務(wù)之間的治理。

xygerp-server-zuul:Spring Cloud的統(tǒng)一API網(wǎng)關(guān)服務(wù)。

Tips: 這2個項目是為了實現(xiàn)微服務(wù)架構(gòu)而用到的核心服務(wù)。所以,它們是相對獨立的。不需要依賴父pom。
3.4.3用mvn編譯命令打包代碼

上面的項目都建立好之后,再添加所有項目都需要用到的依賴(具體代碼可以參考我的源碼)。

都沒問題的話,就可以用mvn命令進行打包項目了:

mvn clean install -Dmaven.test.skip=true -P dev

這里簡單解析一下指令: 

mvn:Maven的統(tǒng)一指令。

clean install:表示要構(gòu)建該項目。

-Dmaven.test.skip=true:表示構(gòu)建的時候要跳過測試模塊。

-P dev:表示構(gòu)建的時候,啟用 dev 的Spring boot參數(shù)運行系統(tǒng)。

如果一切都OK,那正常的結(jié)果如下:

3.5 本地電腦測試系統(tǒng)

代碼搞定了,接下來需要考慮的事情應(yīng)該是如何測試。
畢竟所有的系統(tǒng)都必須要經(jīng)過測試,特別是這種配置多,涉及范圍廣的系統(tǒng)。

這個就不得不說一下Spring boot的優(yōu)勢了。Spring boot的打包應(yīng)用默認內(nèi)置了tomcat服務(wù)。
換句話說,只需要java命令執(zhí)行一下Spring boot打包的target結(jié)果,就可以啟動一個tomcat服務(wù)啦!真挺方便測試的!
3.5.1 啟動本地系統(tǒng)的服務(wù)

假設(shè)我的xygerp-api項目在這里:D:JSP_MyEclipsexygerp-api

然后分別打開4個cmd命令窗口,執(zhí)行:

D:JSP_MyEclipsexygerp-apixygerp-server-eureka	arget>java -jar xygerp-server-eureka-1.0-SNAPSHOT.war
D:JSP_MyEclipsexygerp-apixygerp-server-zuul	arget>java -jar xygerp-server-zuul-1.0-SNAPSHOT.war
D:JSP_MyEclipsexygerp-apixygerp-ald	arget>java -jar xygerp-ald-1.0-SNAPSHOT.war
D:JSP_MyEclipsexygerp-apixygerp-albc	arget>java -jar xygerp-albc-1.0-SNAPSHOT.war

如下圖:

3.5.2 本地測試API服務(wù)系統(tǒng)

本地測試環(huán)境的服務(wù)啟動起來了,接著就是進行具體的數(shù)據(jù)測試。

首先測試Eureka的服務(wù)注冊以及發(fā)現(xiàn),確認服務(wù)是否都已經(jīng)注冊到系統(tǒng)中:

然后,用swagger測試用戶登錄的功能:
http://127.0.0.1:8102/xygerp/ald/swagger-ui.html
目前是測試是否可以正常產(chǎn)生token。


說明已經(jīng)登錄成功,并且產(chǎn)生了本次訪問的token!

將token記錄下來,接著測試。
繼續(xù)測試一個查詢的功能:
http://127.0.0.1:8102/xygerp/albc/swagger-ui.html

注意,這里用了Spring Security框架,所以的API請求頭都必須攜帶token信息。否則請求會返回401。

如果測試OK,那說明基本上系統(tǒng)已經(jīng)成功搭建好了。
下一步就是如何在測試環(huán)境或者正式環(huán)境部署它,以及如何一鍵構(gòu)建項目的問題了。

簡單來說,系統(tǒng)的部署是用 docker工具 ,一鍵部署項目用的是 Jenkins工具。后面將會用專題來說明這2個工具的使用。
3.6 該API微服務(wù)系統(tǒng)實現(xiàn)的功能難點 3.6.1 解決數(shù)據(jù)庫Session的環(huán)境變量問題,特別是語言環(huán)境和用戶環(huán)境。
關(guān)于這個問題,目前我用的辦法可能不一定是最優(yōu)的,如果有別的兄臺有更好的解決辦法,請留言給我,十分感謝!

問題來源:

熟悉EBS開發(fā)的兄臺都應(yīng)該知道,登錄ERP之后,我們每次打開Form,系統(tǒng)就會申請一個新的數(shù)據(jù)庫Session,這時候,EBS系統(tǒng)會 自動幫我們初始化該Session的環(huán)境變量 :例如基本的語言環(huán)境,用戶環(huán)境,業(yè)務(wù)實體等等。

這時候,我們在包里面可以直接用FND_GLOBAL.USER_ID之類的函數(shù)就可以非常方便獲取環(huán)境變量的信息。

但是,在Java Web開發(fā)里面就不一樣了!

在Java訪問數(shù)據(jù)庫的理念中,Session的申請是一個極耗資源的動作!所以,大部分連接數(shù)據(jù)庫的Java軟件都提出了一個 數(shù)據(jù)庫連接池 的概念(例如DRUID數(shù)據(jù)庫連接池)。簡單來說就是session共用!

Session公用就會帶來一個并發(fā)問題:A用戶使用系統(tǒng),并初始化了該Session的環(huán)境變量為A用戶;當A用戶不用系統(tǒng)的時候,Session會閑置并放回連接池里面等待別的用戶使用。

這時候如果B的用戶很可能會使用該Session,如果不重新初始化環(huán)境變量的話,那B用戶使用系統(tǒng)的Session的環(huán)境變量還是A用戶,就會導(dǎo)致數(shù)據(jù)的bug!
如何處理該問題是開發(fā)該系統(tǒng)碰到的一個難題。

問題解決:

我目前的處理辦法是:在Service層,用AOP統(tǒng)一自動監(jiān)控Service層的這個參數(shù)AuthUser user

只要在Service層將參數(shù)AuthUser user放在最后,AOP會自動初始化Session的環(huán)境變量。(需要注意的是,我這個系統(tǒng)的數(shù)據(jù)庫Transaction在Service層啟用!)

另外,語言環(huán)境變量,登錄ID環(huán)境變量等,會一并自動初始化。因為AuthUser會攜帶該定義!

核心處理代碼如下:

private static final String SQL_GLOBAL_INIT
        = " DECLARE "
        + "    L_session_id NUMBER;L_user_id NUMBER;L_login_id NUMBER;L_LANG VARCHAR2(10); "
        + " BEGIN "
        + "    L_user_id:=:P_USER_ID; L_login_id:=:P_LOGIN_ID; L_LANG:=:P_LANG;"
        + "    APPS.fnd_global.INITIALIZE("
        + "       session_id=>L_session_id, user_id =>L_user_id, resp_id =>NULL, "
        + "       resp_appl_id=>NULL, security_group_id=>NULL, site_id=>NULL, login_id =>L_login_id, "
        + "       conc_login_id=>NULL, prog_appl_id=>NULL, conc_program_id=>NULL, conc_request_id=>NULL, "
        + "       conc_priority_request=>NULL"
        + "     ); "
        + "    IF NVL(L_LANG,"US") <> USERENV("LANG") THEN "
        + "        IF L_LANG="ZHS" THEN "
        + "            APPS.fnd_global.set_nls_context(p_nls_language => "SIMPLIFIED CHINESE"); "
        + "        ELSE "
        + "            APPS.fnd_global.set_nls_context(p_nls_language => "AMERICAN"); "
        + "         END IF;"
        + "    END IF;"
        + " END; ";
    
    /*** 
     * service層調(diào)用之前先自動初始化環(huán)境變量
     * 需要注意的是,用戶變量必須的參數(shù)放在最后!
     * 只要在Service層將參數(shù)AuthUser user放在最后,AOP會自動初始化Session的環(huán)境變量。
     * @throws Exception 
     */  
    @SuppressWarnings("static-access")
    @Before("execution(* com.jebms.*.service..*.*(..))  && args(..,user)")  
    public void oracleDBInit(JoinPoint joinPoint,AuthUser user) throws Exception{
        Long dbLoginId=devDao.getJdbcTemplate().queryForObject("SELECT FND_GLOBAL.LOGIN_ID FROM DUAL", Long.class);
if(user.getLoginId()!=null&&user.getLoginId()>0&&!user.getLoginId().equals(dbLoginId)){
            Map inParamMap=new HashMap();
            inParamMap.put("P_USER_ID", user.getUserId());
            inParamMap.put("P_LOGIN_ID", user.getLoginId());
            inParamMap.put("P_LANG", user.getLanguage());
            devDao.getDevJdbcTemplate().execute(this.SQL_GLOBAL_INIT, inParamMap);
        }
    }
源代碼在:com.jebms.comm.utils. AopUtil
3.6.2 解決EBS的用戶的登錄問題:統(tǒng)一用EBS系統(tǒng)的帳號密碼登錄API系統(tǒng)。

問題描述:

由于我這個是第三方的API系統(tǒng),所以,用戶名和密碼信息實際上并不是該API系統(tǒng)需要管理的事情。

相當于說,API系統(tǒng)無法按照正常的流程來驗證用戶名和密碼:輸入用戶名和密碼,系統(tǒng)驗證后臺數(shù)據(jù)庫的用戶名和密碼,再返回驗證結(jié)果。

而是:輸入用戶名和密碼,系統(tǒng) 調(diào)用ERP的用戶名密碼驗證包 進行驗證,再返回結(jié)果。
簡單來說:需要添加自定義驗證的邏輯。

還好Spring Security框架支持靈活的驗證邏輯。

添加步驟:

首先,寫一個自定義驗證的類:MyAuthenticationProvider

接著,在Spring Security框架的定義中,添加這個自定義的驗證。

AbstractWebSecurityConfig

private MyAuthenticationProvider provider;//自定義驗證
auth.authenticationProvider(provider);

即可以完美實現(xiàn)這個效果

核心代碼:

    /**
     * 自定義驗證方式
     */
    @Override
    public Authentication authenticate(Authentication authentication) throws AuthenticationException {
        String username = authentication.getName();
        String password = (String) authentication.getCredentials();
        AuthUser user = (AuthUser) userService.loadUserByUsername(username);
        System.out.println("username:"+username+",password:"+password);
        if(user == null){
            throw new BadCredentialsException("Username not found.");
        }

        //加密過程在這里體現(xiàn)
        if (!sysService.xygErpValidateLogin(username, password)) {
            throw new BadCredentialsException("Wrong password.");
        }
        
        user.setPassword(passwordEncoder.encode(password));

        Collection authorities = user.getAuthorities();
        return new UsernamePasswordAuthenticationToken(user, password, authorities);
    }
3.6.3 統(tǒng)一的開發(fā)風格。

1.Entity基類封裝。

封裝了5who欄位,以及類似Form的FND_SET_WHO的方法,可以很方便進行開發(fā)。

另外,為了防止丟失更新,這邊每次更新前實際上會先檢測數(shù)據(jù)的一致性,對應(yīng)的動作也有做了封裝。

2.查詢邏輯的封裝。

查詢功能相對來說還是會很多,對于復(fù)雜的查詢條件如何傳值是一個難題。

這里封裝了一個SearchInfo積累,可以統(tǒng)一將所有的查詢條件都放在這個類,然后在java的Controller層定義好查詢條件對應(yīng)匹配欄位,系統(tǒng)就可以自動產(chǎn)生對應(yīng)的and條件。

例如:

   @GetMapping(value = "/getPageLocator")
    @ApiOperation(value = "貨位分頁列表接口")
    public ResultEntity> getPageLocator(
            @ApiParam(value = "庫存組織ID",required = true) @RequestParam(required = true) int organizationId,
            @ApiParam(value = "庫別代碼",required = true) @RequestParam(required = true) String subinventoryCode,
            @ApiParam(value = "貨位代碼",required = false) @RequestParam(required = false) String locatorCode,
            SearchInfo searchInfo) throws Exception {
        searchInfo.getConditionMap().put("organizationId", organizationId);
        searchInfo.getConditionMap().put("subinventoryCode", subinventoryCode);
        searchInfo.getConditionMap().put("locatorCode", locatorCode);
        searchInfo.setAuthUser(this.authUser);
        searchInfo.initSqlCondition();
        searchInfo.andSqlCondition("MIL.ORGANIZATION_ID","organizationId");
        searchInfo.andSqlCondition("MIL.SUBINVENTORY_CODE","subinventoryCode");
        searchInfo.andSqlCondition("MIL.SEGMENT1","locatorCode");
        return eslipService.selectForPageLocator(searchInfo);
    }

3.統(tǒng)一的處理結(jié)果的封裝。

基本上任何一個處理,要不成功,要不失敗(警告其實也算失敗)。

這里封裝了一個返回結(jié)果的基類ResultEntity,可以進行有效的應(yīng)用端或者java端的交互。

@ApiModelProperty(value = "狀態(tài)碼,0表示成功 其他表示失敗", example = "0",position = 1)
private String code;

特別需要指出的是,前端獲取或者處理數(shù)據(jù),也是統(tǒng)一要用這個處理結(jié)果基類的返回。

簡單來說,就是數(shù)據(jù)處理成功/失敗,會有一個統(tǒng)一的返回結(jié)果標識。注意,這個標識和請求的響應(yīng)結(jié)果標識(200)是有所不同的!

請求響應(yīng)標識只是說明web服務(wù)器的響應(yīng)是正常,但,具體的處理結(jié)果可能是處理失敗。

下面是一個具體的例子(到時候?qū)嶋H開發(fā)處理的接口處理結(jié)果也是這樣子):

{
    "code": "0",
    "message": "",
    "description": "",
    "obj": [{
        "createdBy": -1,
        "creationDate": "2017-10-10 09:37:03",
        "lastUpdatedBy": 10,
        "lastUpdateDate": "2017-11-16 14:47:48",
        "lastUpdateLogin": 96,
        "valueUUID": null,
        "id": 2,
        "applId": 1,
        "respCode": "BASIC_SET",
        "menuId": 2,
        "startDate": "2017-10-10 09:37:03",
        "endDate": null,
        "respName": "系統(tǒng)設(shè)置職責",
        "description": "系統(tǒng)設(shè)置職責",
        "menuCode": "SYSTEM_SET",
        "menuName": "系統(tǒng)設(shè)置菜單",
        "enabled": true
    }],
    "param1": null,
    "param2": null,
    "param3": null,
    "param4": null,
    "param5": null,
    "ok": true
}
文檔參考鏈接:
https://juejin.im/entry/5a781...
http://blog.51cto.com/ityoukn...

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/69133.html

相關(guān)文章

  • 服務(wù)實戰(zhàn):從架構(gòu)到發(fā)布(一)

    摘要:微服務(wù)集成服務(wù)間通信微服務(wù)架構(gòu)下,應(yīng)用的服務(wù)直接相互獨立。微服務(wù)架構(gòu)傾向于降低中心消息總線類似于的依賴,將業(yè)務(wù)邏輯分布在每個具體的服務(wù)終端。 引言:微服務(wù)是當前軟件架構(gòu)領(lǐng)域非常熱門的詞匯,能找到很多關(guān)于微服務(wù)的定義、準則,以及如何從微服務(wù)中獲益的文章,在企業(yè)的實踐中去應(yīng)用微服務(wù)的資源卻很少。本篇文章中,會介紹微服務(wù)架構(gòu)(Microservices Architecture)的基礎(chǔ)概念,...

    libin19890520 評論0 收藏0
  • 服務(wù)實戰(zhàn):從架構(gòu)到發(fā)布(一)

    摘要:微服務(wù)集成服務(wù)間通信微服務(wù)架構(gòu)下,應(yīng)用的服務(wù)直接相互獨立。微服務(wù)架構(gòu)傾向于降低中心消息總線類似于的依賴,將業(yè)務(wù)邏輯分布在每個具體的服務(wù)終端。 引言:微服務(wù)是當前軟件架構(gòu)領(lǐng)域非常熱門的詞匯,能找到很多關(guān)于微服務(wù)的定義、準則,以及如何從微服務(wù)中獲益的文章,在企業(yè)的實踐中去應(yīng)用微服務(wù)的資源卻很少。本篇文章中,會介紹微服務(wù)架構(gòu)(Microservices Architecture)的基礎(chǔ)概念,...

    HtmlCssJs 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<