使用webservice自定義註解處理參數加解密問題
webservice自定義註解處理參數加解密
前一段項目中用到瞭webservice,正好自己之前也瞭解過一點apache的cxf框架,所以就采用瞭cxf來實現webservice服務端,本身實現並沒技術難點,但是項目為瞭保證安全性,采用瞭傳輸加密的過程,所以大部分請求參數需要加密,返回參數也需要加密,大致流程是:
- 請求參數對稱加密+對稱秘鑰非對稱加密
- 返回參數堆成加密+對稱秘鑰非對稱加密
參數加密本身並不復雜,但是有些服務並不需要加密,有些則需要,加密工具類並不通用
string,datahandler等不能重用,雖然本質都是對字節數據加密,但是要寫許多方法,還要方法參數,檢查需要加解密處理的參數,然後尋找對應的類型處理方法調用.
邏輯很清晰,但是過程實現很惡心,最終這樣實現瞭一版,但是並不合我的意.
如果能在攔截器中獲取到參數列表,然後尋找對應的解碼器解碼,這樣就比較簡單瞭.但是難點是,如果標記參數,並指定被標記參數的解碼器尼?
我想到瞭註解,註解就可以標記,參數,然後通過反射獲取到註解,解析出內容.
本來想標記到參數上,但是一個個標記太麻煩,而且獲取也不太方便,索性就放在方法上,用數組來接收.
問題解決.
代碼實現
自定義註解:
import java.lang.annotation.ElementType; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target; /** * @author webservice請求參數自定義註解 * */ @Target(ElementType.METHOD) @Retention(RetentionPolicy.RUNTIME) public @interface RequestHanleAnnotation { Class[] handler() default {}; int[] index() default {}; }
參數解碼器接口
import java.util.Map; import java.util.concurrent.ConcurrentHashMap; /** * @author taoyuan *參數處理器 * @param <T> */ public abstract class ParamHandler<T> { //解碼器實例緩存 public static final Map<Class,ParamHandler> hanlers=new ConcurrentHashMap<>(); //處理方法 abstract T handle(T t); }
解碼器實現,這裡並沒有真正解碼,隻是在參數後面加瞭123
public class StringHandler extends ParamHandler<String> { @Override public String handle(String t) { return t+"123"; } }
服務方法註解使用
//表示第一個參數需要StringHandler處理 @RequestHanleAnnotation(index=0,handler=StringHandler.class) public String test(String test, String test2) throws Exception { System.out.println(test); System.out.println(test2); return "cesshi"; }
攔截器實現
import java.io.File; import java.lang.reflect.Method; import java.util.Map; import org.apache.commons.io.FileUtils; import org.apache.cxf.binding.soap.SoapMessage; import org.apache.cxf.interceptor.Fault; import org.apache.cxf.message.MessageContentsList; import org.apache.cxf.phase.AbstractPhaseInterceptor; import org.apache.cxf.phase.Phase; import org.apache.log4j.Logger; import com.alibaba.fastjson.JSON; import com.alibaba.fastjson.JSONObject; /** * @author 都市桃源 */ public class EcrInInterceptor extends AbstractPhaseInterceptor<SoapMessage>{ private Logger log = Logger.getLogger(ContractLogicImpl.class); public EcrInInterceptor(){ // 在調用方法之前調用攔截器 super(Phase.PRE_INVOKE); } @Override public void handleMessage(SoapMessage msg) throws Fault { /*獲取請求ip,攔截器中可以做下統一日志處理 HttpServletRequest httprequest = (HttpServletRequest)msg.get(AbstractHTTPDestination.HTTP_REQUEST); * */ //獲取正在執行的方法 Method method = MsgUtil.getSoapMethod(msg); //解析註解,並處理參數 MessageContentsList contentsList = MessageContentsList.getContentsList(msg); MsgUtil.handle(method,contentsList); } @Override public void handleFault(SoapMessage message) { super.handleFault(message); }
註解解析工具類實現
import java.lang.reflect.Method; import java.util.HashMap; import java.util.Map; import org.apache.cxf.binding.soap.SoapMessage; import org.apache.cxf.message.Exchange; import org.apache.cxf.message.MessageContentsList; import org.apache.cxf.service.Service; import org.apache.cxf.service.invoker.MethodDispatcher; import org.apache.cxf.service.model.BindingOperationInfo; /** * @author ll * 獲取方法名稱 * */ public class MsgUtil { /**根據消息獲取調用方法 * @param msg * @return */ public static Method getSoapMethod(SoapMessage msg){ Exchange exchange = msg.getExchange(); BindingOperationInfo bop = exchange.get(BindingOperationInfo.class); MethodDispatcher md = (MethodDispatcher) exchange.get(Service.class) .get(MethodDispatcher.class.getName()); Method method = md.getMethod(bop); return method; } public static void handle(Method method, MessageContentsList contentsList) { if(method==null)return; RequestHanleAnnotation reqAnno= method.getAnnotation(RequestHanleAnnotation.class); int[] indexs = reqAnno.index(); if(indexs==null||indexs.length==0)return; Class[] handlers = reqAnno.handler(); try { //處理器實例 ParamHandler handler=null; for(int i=0,len=indexs.length;i<len;i++){ //獲取需要處理的參數 Object obj = contentsList.get(indexs[i]); //從緩存中獲取處理器實例 handler=ParamHandler.hanlers.get(handlers[i]); if(handler==null){ //創建處理器實例 handler=(ParamHandler) handlers[i].newInstance(); //緩存處理器實例 ParamHandler.hanlers.put(handlers[i], handler); } contentsList.set(indexs[i], handler.handle(obj)); } }catch (Exception e) { e.printStackTrace(); } } }
返回參數同樣也可以註解實現,這裡就不在贅述瞭,實現也比較簡單.
**需要註意的是,攔截器攔截順序不一樣 攔截器統一繼承 AbstractPhaseInterceptor類, 請求攔截 :super(Phase.PRE_INVOKE); 返回攔截: super(Phase.PRE_STREAM);**
webservice註解匯總
@WebService
serviceName
:對外發佈的服務名,指定 Web Service 的服務名稱:wsdl:service。缺省值為 Java 類的簡單名稱 + Service。(字符串)endpointInterface
:服務接口全路徑, 指定做SEI(Service EndPoint Interface)服務端點接口name
:此屬性的值包含XML Web Service的名稱。在默認情況下,該值是實現XML Web Service的類的名稱,wsdl:portType 的名稱。缺省值為 Java 類或接口的非限定名稱。(字符串portName
:wsdl:portName。缺省值為 WebService.name+Port。targetNamespace
:指定你想要的名稱空間,默認是使用接口實現類的包名的反序wsdlLocation
:指定用於定義 Web Service 的 WSDL 文檔的 Web 地址。Web 地址可以是相對路徑或絕對路徑。(字符串)註意:實現類上可以不添加Webservice註解
@WebMethod
註釋表示作為一項 Web Service 操作的方法,將此註釋應用於客戶機或服務器服務端點接口(SEI)上的方法,或者應用於 JavaBeans 端點的服務器端點實現類。
要點:
僅支持在使用 @WebService 註釋來註釋的類上使用 @WebMethod 註釋
operationName
:指定與此方法相匹配的wsdl:operation 的名稱。缺省值為 Java 方法的名稱。(字符串)action
:定義此操作的行為。對於 SOAP 綁定,此值將確定 SOAPAction 頭的值。缺省值為 Java 方法的名稱。(字符串)exclude
:指定是否從 Web Service 中排除某一方法。缺省值為 false。(佈爾值)
@Oneway
註釋將一個方法表示為隻有輸入消息而沒有輸出消息的 Web Service 單向操作。
將此註釋應用於客戶機或服務器服務端點接口(SEI)上的方法,或者應用於 JavaBeans 端點的服務器端點實現類
@WebParam
註釋用於定制從單個參數至 Web Service 消息部件和 XML 元素的映射。
將此註釋應用於客戶機或服務器服務端點接口(SEI)上的方法,或者應用於 JavaBeans 端點的服務器端點實現類。
name
:參數的名稱。如果操作是遠程過程調用(RPC)類型並且未指定partName 屬性,那麼這是用於表示參數的 wsdl:part 屬性的名稱。如果操作是文檔類型或者參數映射至某個頭,那麼 -name 是用於表示該參數的 XML 元素的局部名稱。如果操作是文檔類型、參數類型為 BARE 並且方式為 OUT 或 INOUT,那麼必須指定此屬性。(字符串)partName
:定義用於表示此參數的 wsdl:part屬性的名稱。僅當操作類型為 RPC 或者操作是文檔類型並且參數類型為BARE 時才使用此參數。(字符串)targetNamespace
:指定參數的 XML 元素的 XML 名稱空間。當屬性映射至 XML 元素時,僅應用於文檔綁定。缺省值為 Web Service的targetNamespace。(字符串)mode
:此值表示此方法的參數流的方向。有效值為 IN、INOUT 和 OUT。(字符串)header
:指定參數是在消息頭還是消息體中。缺省值為 false。(佈爾值)
@WebResult
註釋用於定制從返回值至 WSDL 部件或 XML 元素的映射。將此註釋應用於客戶機或服務器服務端點接口(SEI)上的方法,或者應用於 JavaBeans 端點的服務器端點實現類。
name
:當返回值列示在 WSDL 文件中並且在連接上的消息中找到該返回值時,指定該返回值的名稱。對於 RPC 綁定,這是用於表示返回值的 wsdl:part屬性的名稱。對於文檔綁定,-name參數是用於表示返回值的 XML 元素的局部名。對於 RPC 和 DOCUMENT/WRAPPED 綁定,缺省值為 return。對於 DOCUMENT/BARE 綁定,缺省值為方法名 + Response。(字符串)targetNamespace
:指定返回值的 XML 名稱空間。僅當操作類型為 RPC 或者操作是文檔類型並且參數類型為 BARE 時才使用此參數。(字符串)header
:指定頭中是否附帶結果。缺省值為false。(佈爾值)partName
:指定 RPC 或 DOCUMENT/BARE 操作的結果的部件名稱。缺省值為@WebResult.name。(字符串)
@HandlerChain
註釋用於使 Web Service 與外部定義的處理程序鏈相關聯。隻能通過對 SEI 或實現類使用 @HandlerChain 註釋來配置服務器端的處理程序。但是可以使用多種方法來配置客戶端的處理程序。可以通過對生成的服務類或者 SEI 使用 @HandlerChain 註釋來配置客戶端的處理程序。此外,可以按程序在服務上註冊您自己的 HandlerResolver 接口實現,或者按程序在綁定對象上設置處理程序鏈。
file
:指定處理程序鏈文件所在的位置。文件位置可以是采用外部格式的絕對 java.net.URL,也可以是類文件中的相對路徑。(字符串)name
:指定配置文件中處理程序鏈的名稱。
以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。
推薦閱讀:
- SpringBoot整合WebService服務的實現代碼
- java調用WebService服務的四種方法總結
- SpringBoot項目使用 axis 調用webservice接口的實踐記錄
- Python中WebService客戶端接口調用及身份驗證的問題
- 使用Postman和SoapUI工具測試WebService接口