TensorFlow已經(jīng)發(fā)展成為事實上的ML(機器學(xué)習(xí))平臺,在業(yè)界和研究領(lǐng)域都很流行。對TensorFlow的需求和支持促成了一系列圍繞培訓(xùn)和服務(wù)ML模型的OSS庫、工具和框架。TensorFlow Serving是一個構(gòu)建在分布式生產(chǎn)環(huán)境中、主要為ML模型提供推理服務(wù)的項目。
Mux在其基礎(chǔ)設(shè)施的幾個部分中使用TensorFlow Serving,我們之前已經(jīng)討論過使用TensorFlow來實現(xiàn)按標題編碼(per-title-encoding)功能。今天,我們將重點關(guān)注通過優(yōu)化預(yù)測服務(wù)器和客戶端來改善延遲的技術(shù)。模型預(yù)測通常是“在線”操作(在關(guān)鍵的應(yīng)用請求路徑上),因此我們的主要優(yōu)化目標是以盡可能低的延遲處理大量的請求。
首先,讓我們簡要介紹一下TensorFlow Serving。
什么是TensorFlow Serving?
TensorFlow Serving提供了靈活的服務(wù)器架構(gòu),旨在部署和服務(wù)ML模型。一旦模型經(jīng)過訓(xùn)練并可以用于預(yù)測,TensorFlow Serving需要將模型導(dǎo)出到可服務(wù)類(Servable)兼容格式。
Servable是包裝TensorFlow對象的中心抽象。例如,模型可以表示為一個或多個Servable。因此,Servable是客戶端用于執(zhí)行計算(如推理)的底層對象。Servable的大小很重要,由于較小的模型使用更少的內(nèi)存、更少的存儲,因此將有更快的加載時間(加載時間更短)。Servable要求模型以SavedModel格式加載,并使用PerdictAPI提供服務(wù)。
TensorFlow Serving將核心服務(wù)組件組合在一起,構(gòu)建一個GRPC/HTTP服務(wù)器。該服務(wù)器可以服務(wù)多個ML模型(或多個版本),并提供監(jiān)測組件和可配置的體系結(jié)構(gòu)。
Tensorflow Serving and Docker
讓我們使用標準的TensorFlow Serving獲得基線預(yù)測性能延遲指標(未使用CPU優(yōu)化)。
首先,從TensorFlow Docker hub提取最新的服務(wù)映像文件:
dockerpulltensorflow/serving:latest
在這篇文章中,所有的容器都運行在一個4核、15 GB、Ubuntu 16.04主機上。
將TensorFlow模型導(dǎo)出到SavedModel格式
使用TensorFlow訓(xùn)練模型時,可以將輸出保存為可變檢查點(磁盤上的文件)。推理可以通過恢復(fù)模型檢查點或在其轉(zhuǎn)換的靜態(tài)圖(二進制)上直接運行。
為了使用TensorFlow服務(wù)這些模型,必須靜態(tài)圖導(dǎo)出到SavedModel格式。TensorFlow文檔可查詢以SavedModel格式導(dǎo)出預(yù)訓(xùn)練模型的示例。
TensorFlow還提供了一系列官方和研究模型作為實驗、研究或生產(chǎn)的入門資料。
例如,我們將使用深殘余網(wǎng)絡(luò)(ResNet)模型,可用于對ImageNet的1000個類的數(shù)據(jù)集進行分類。下載預(yù)訓(xùn)練ResNet-50 v2模型,特別是channels_last (NHWC)卷積SavedModel,這對于CPU來說通常更好。
在以下結(jié)構(gòu)中復(fù)制RestNet模型目錄:
models/1/saved_model.pbvariables/variables.data-00000-of-00001variables.index
TensorFlow Serving按數(shù)字排序的目錄結(jié)構(gòu)管理模型版本。在本例中,目錄1/對應(yīng)于模型版本1,其中包含模型體系結(jié)構(gòu)。saved_model.pb以及模型權(quán)重(變量)的快照。
加載和服務(wù)SavedModel
下面的命令在docker容器中啟動一個TensorFlow模型服務(wù)器。為了加載SavedModel,需要將模型的主機目錄mount到預(yù)期的容器目錄中。
dockerrun-d-p9000:8500\-v$(pwd)/models:/models/resnet-eMODEL_NAME=resnet\-ttensorflow/serving:latest
檢查容器日志,確定ModelServer正在運行并可以在GRPC和HTTP端點上為resnet模型提供服務(wù):
Itensorflow_serving/core/loader_harness.cc:86]Successfullyloadedservableversion{name:resnetversion:1}Itensorflow_serving/model_servers/server.cc:286]RunninggRPCModelServerat0.0.0.0:8500...Itensorflow_serving/model_servers/server.cc:302]ExportingHTTP/RESTAPIat:localhost:8501...
預(yù)測客戶端
TensorFlow將API服務(wù)模式定義為協(xié)議緩沖器(protobuf)。預(yù)測API的gRPC客戶端用例會被打包為tensorflow_serving.apis Python包??紤]到實用功能,我們還需要tensorflow Python包。
讓我們安裝倚賴項(dependencies)創(chuàng)建一個簡單的客戶端:
virtualenv.env&&source.env/bin/activate&&\pipinstallnumpygrpcioopencv-pythontensorflowtensorflow-serving-api
ResNet-50 v2模型要求浮點Tensor輸入應(yīng)采用Channel_last(NHWC)格式的數(shù)據(jù)結(jié)構(gòu)。因此,輸入圖像是使用OpenCV-python讀取的,它以32位浮點數(shù)據(jù)類型加載到numpy數(shù)組(高度x寬度x通道)中。下面的腳本創(chuàng)建預(yù)測客戶端存根,并將JPEG圖像數(shù)據(jù)加載到numpy數(shù)組中,然后轉(zhuǎn)換為TensorProto以發(fā)出GRPC預(yù)測請求:
#!/usr/bin/envpythonfrom__future__importprint_functionimportargparseimportnumpyasnpimporttimett=time.time()importcv2importtensorflowastffromgrpc.betaimportimplementationsfromtensorflow_serving.apisimportpredict_pb2fromtensorflow_serving.apisimportprediction_service_pb2parser=argparse.ArgumentParser(description='incetiongrpcclientflags.')parser.add_argument('--host',default='0.0.0.0',help='inceptionservinghost')parser.add_argument('--port',default='9000',help='inceptionservingport')parser.add_argument('--image',default='',help='pathtoJPEGimagefile')FLAGS=parser.parse_args()defmain():#createpredictionserviceclientstubchannel=implementations.insecure_channel(FLAGS.host,int(FLAGS.port))stub=prediction_service_pb2.beta_create_PredictionService_stub(channel)#createrequestrequest=predict_pb2.PredictRequest()request.model_spec.name='resnet'request.model_spec.signature_name='serving_default'#readimageintonumpyarrayimg=cv2.imread(FLAGS.image).astype(np.float32)#converttotensorprotoandmakerequest#shapeisinNHWC(num_samplesxheightxwidthxchannels)formattensor=tf.contrib.util.make_tensor_proto(img,shape=[1]+list(img.shape))request.inputs['input'].CopyFrom(tensor)resp=stub.Predict(request,30.0)print('totaltime:{}s'.format(time.time()-tt))if__name__=='__main__':main()
使用JPEG圖像作為輸入,運行客戶端的輸出如下所示:
pythontf_serving_client.py--image=images/pupper.jpgtotaltime:2.56152906418s
輸出Tensor的預(yù)測結(jié)果為一個整數(shù)值和各特征的概率。
outputs{key:"classes"value{dtype:DT_INT64tensor_shape{dim{size:1}}int64_val:238}}outputs{key:"probabilities"...
對于單個請求,這種預(yù)測延遲是不可接受的。然而,這并非完全出乎意料;默認的TensorFlow二進制程序的目標是支持最廣泛的硬件以覆蓋絕大多數(shù)用例。您可能已經(jīng)從標準的TensorFlow容器日志中注意到:
Iexternal/org_tensorflow/tensorflow/core/platform/cpu_feature_guard.cc:141]YourCPUsupportsinstructionsthatthisTensorFlowbinarywasnotcompiledtouse:AVX2FMA
這表明TensorFlow二進制程序運行在不兼容的CPU平臺上,而該平臺沒有為其進行優(yōu)化。
構(gòu)建CPU優(yōu)化服務(wù)二進制代碼
根據(jù) TensorFlow文獻資料,建議從源代碼編譯TensorFlow,并采用運行二進制文件的主機平臺的CPU提供的所有優(yōu)化。TensorFlow給出了Build選項標志,以便為特定于平臺的CPU指令集構(gòu)建二進制代碼:
指令集 | 標志 |
AVX | -Copt=-mavx |
AVX 2 | -Copt=-mavx 2 |
FMA | --copt=-mfma |
SSE 4.1 | -Copt=-msse4.1 |
SSE 4.2 | -Copt=-msse4.2 |
全部由處理器支持 | --copt=-march=native |
克隆TensorFlow固定到特定版本。在這種情況下,我們將使用1.13(發(fā)表此文時的最新版本):
USER=$1TAG=$2TF_SERVING_VERSION_GIT_BRANCH="r1.13"gitclone--branch="$TF_SERVING_VERSION_GIT_BRANCH"https://github.com/tensorflow/serving
TensorFlow Serving映像使用Bazel作為Build工具。針對特定處理器CPU指令集的生成目標可以用如下方式指定:
TF_SERVING_BUILD_OPTIONS="--copt=-mavx--copt=-mavx2--copt=-mfma--copt=-msse4.1--copt=-msse4.2"
如果內(nèi)存有限制,則使用--local_resources=2048,.5,1.0標志。參考TensorFlow與Docker聯(lián)合服務(wù)和Bazel docs的文獻作為這些構(gòu)建標志上的資源。
使用開發(fā)圖像建立serving image作為服務(wù)基類:
#!/bin/bashUSER=$1TAG=$2TF_SERVING_VERSION_GIT_BRANCH="r1.13"gitclone--branch="${TF_SERVING_VERSION_GIT_BRANCH}"https://github.com/tensorflow/servingTF_SERVING_BUILD_OPTIONS="--copt=-mavx--copt=-mavx2--copt=-mfma--copt=-msse4.1--copt=-msse4.2"cdserving&&\dockerbuild--pull-t$USER/tensorflow-serving-devel:$TAG\--build-argTF_SERVING_VERSION_GIT_BRANCH="${TF_SERVING_VERSION_GIT_BRANCH}"\--build-argTF_SERVING_BUILD_OPTIONS="${TF_SERVING_BUILD_OPTIONS}"\-ftensorflow_serving/tools/docker/Dockerfile.devel.cdserving&&\dockerbuild-t$USER/tensorflow-serving:$TAG\--build-argTF_SERVING_BUILD_IMAGE=$USER/tensorflow-serving-devel:$TAG\-ftensorflow_serving/tools/docker/Dockerfile.
ModelServer可以用TensorFlow專用標志配置以啟用會話并行性。以下選項配置兩個線程池并行執(zhí)行:
intra_op_parallelism_threads
控制用于單個操作并行執(zhí)行的最大線程數(shù)。
用于具有固有非倚賴子操作的并行執(zhí)行。
inter_op_parallelism_threads
控制獨立的不同操作并行執(zhí)行的最大線程數(shù)。
對TensorFlow Graph的操作彼此獨立,因此可以在不同的線程上運行。
這兩個選項的默認值設(shè)置為0。這意味著,系統(tǒng)選擇一個適當(dāng)?shù)臄?shù)字,這通常需要每個CPU核心有一個線程可用。但是,對于多核CPU并行性,我們可以手動控制。
接下來,按照與之前類似的方式啟動服務(wù)容器,這一次使用從源代碼編譯的docker映像,并使用TensorFlow特定的CPU優(yōu)化標志:
dockerrun-d-p9000:8500\-v$(pwd)/models:/models/resnet-eMODEL_NAME=resnet\-t$USER/tensorflow-serving:$TAG\--tensorflow_intra_op_parallelism=4\--tensorflow_inter_op_parallelism=4
容器日志應(yīng)該不顯示CPU guard告警。在不更改任何代碼的情況下,運行相同的預(yù)測請求會使預(yù)測延遲降低約35.8%:
pythontf_serving_client.py--image=images/pupper.jpgtotaltime:1.64234706879s
提高預(yù)測客戶端的速度
我們能做得更好嗎?服務(wù)器端已經(jīng)為其CPU平臺進行了優(yōu)化,但超過1s的預(yù)測延遲似乎仍然太高。
問題是,加載tensorflow_serving和tensorflow庫確實會導(dǎo)致這個問題。每次調(diào)用tf.contrib.util.make_tensor_proto同時也增加了不必要的延遲開銷。
“等等”,你可能在想?!半y道我不需要TensorFlow Python包向TensorFlow Server提出預(yù)測請求嗎?”
答案很簡單:確實如此,實際上不需要tensorflow或tensorflow_serving包發(fā)出預(yù)測請求。
如前所述,TensorFlow預(yù)測API被定義為Protobufs。因此,可以通過生成必要的tensorflow和tensorflow_serving protobuf python 存根(stubs)。這就避免了對客戶機本身整個(巨大)TensorFlow庫的調(diào)用。
首先,舍棄tensorflow和tensorflow_serving依賴項,添加grpcio-tools包。
pipuninstalltensorflowtensorflow-serving-api&&\pipinstallgrpcio-tools==1.0.0
復(fù)制tensorflow/tensorflow和tensorflow/serving存儲庫,并將以下Protobuf文件復(fù)制到客戶端項目中:
tensorflow/serving/tensorflow_serving/apis/model.prototensorflow_serving/apis/predict.prototensorflow_serving/apis/prediction_service.prototensorflow/tensorflow/tensorflow/core/framework/resource_handle.prototensorflow/core/framework/tensor_shape.prototensorflow/core/framework/tensor.prototensorflow/core/framework/types.proto
將上述原型文件復(fù)制到protos/目錄并保存原始路徑:
protos/tensorflow_serving/apis/*.prototensorflow/core/framework/*.proto
為了簡單起見,prediction_service.proto(預(yù)測服務(wù))可以簡化為只實現(xiàn)Predict RPC。這避免了引入服務(wù)中定義的其他RPC的嵌套依賴關(guān)系。這里是簡化后的prediction_service.proto.
使用grpcio.tools.protoc:
PROTOC_OUT=protos/PROTOS=$(find.|grep"\.proto$")forpin$PROTOS;dopython-mgrpc.tools.protoc-I.--python_out=$PROTOC_OUT--grpc_python_out=$PROTOC_OUT$pdone
現(xiàn)在可以刪除整個tensorflow_serving模塊:
fromtensorflow_serving.apisimportpredict_pb2fromtensorflow_serving.apisimportprediction_service_pb2
中生成的Probufs替換為protos/tensorflow_serving/apis:
fromprotos.tensorflow_serving.apisimportpredict_pb2fromprotos.tensorflow_serving.apisimportprediction_service_pb2
為了使用幫助函數(shù)make_tensor_proto,導(dǎo)入了TensorFlow庫,也就是用于包裝python/numpy對象成為TensorProto對象。
因此,我們可以替換以下依賴項和代碼片段:
importtensorflowastf...tensor=tf.contrib.util.make_tensor_proto(features)request.inputs['inputs'].CopyFrom(tensor)
導(dǎo)入Protobuf,構(gòu)建TensorProto對象:
fromprotos.tensorflow.core.frameworkimporttensor_pb2fromprotos.tensorflow.core.frameworkimporttensor_shape_pb2fromprotos.tensorflow.core.frameworkimporttypes_pb2...#ensureNHWCshapeandbuildtensorprototensor_shape=[1]+list(img.shape)dims=[tensor_shape_pb2.TensorShapeProto.Dim(size=dim)fordimintensor_shape]tensor_shape=tensor_shape_pb2.TensorShapeProto(dim=dims)tensor=tensor_pb2.TensorProto(dtype=types_pb2.DT_FLOAT,tensor_shape=tensor_shape,float_val=list(img.reshape(-1)))request.inputs['inputs'].CopyFrom(tensor)
完整的python腳本請訪問這里。運行更新后的初始客戶端,該客戶端向優(yōu)化的TensorFlow發(fā)出預(yù)測請求:
pythontf_inception_grpc_client.py--image=images/pupper.jpgtotaltime:0.58314920859s
下圖顯示了針對標準、優(yōu)化的TensorFlow和客戶端10次運行的預(yù)測請求的延遲:
顯然,從標準TensorFlow到優(yōu)化版的平均延遲下降了約70.4%。
優(yōu)化預(yù)測吞吐量
TensorFlow Serving也可以配置為高吞吐量處理。對吞吐量的優(yōu)化通常用于“脫機”批處理,在這些批處理中,不需要有嚴格的延遲閥值。
(1) 服務(wù)器端批處理
服務(wù)器端批處理由TensorFlow Serving支持開箱即用。.
延遲和吞吐量之間的權(quán)衡取決于所支持的批處理參數(shù)。TensorFlow批處理最有效地利用硬件加速器承諾(保證)的高吞吐量。
若要啟用批處理,請設(shè)置--enable_batching和--batching_parameters_file標志。可以將批處理參數(shù)設(shè)置為SessionBundleConfig。對于只使用CPU的系統(tǒng),請考慮設(shè)置num_batch_threads可以使用的核心數(shù)量。批處理配置方法可訪問這里,使用支持GPU的系統(tǒng)。
當(dāng)服務(wù)器端的批處理請求全部到達時,推理請求在內(nèi)部合并為單個大請求(Tensor),并在合并請求上運行一個TensorFlow會話。在單個會話上運行批量請求,可以真正利用CPU/GPU并行性。
批處理過程中需要考慮的Tensorflow Serving Batching進程:
在客戶端使用異步請求,以在服務(wù)器端進行批處理
在CPU/GPU上加入模型圖組件,加速批處理
在同一服務(wù)器服務(wù)多個模型時,對預(yù)測請求進行交織處理
對于“脫機”大容量推理處理,強烈推薦批處理。
(2) 客戶端批處理
客戶端的批處理是將多個輸入組合在一起,以發(fā)出單個請求。
由于ResNet模型要求以NHWC格式輸入(第一個維度是輸入的數(shù)量),所以我們可以將多個輸入圖像聚合到一個RPC請求中:
...batch=[]forjpeginos.listdir(FLAGS.images_path):path=os.path.join(FLAGS.images_path,jpeg)img=cv2.imread(path).astype(np.float32)batch.append(img)...batch_np=np.array(batch).astype(np.float32)dims=[tensor_shape_pb2.TensorShapeProto.Dim(size=dim)fordiminbatch_np.shape]t_shape=tensor_shape_pb2.TensorShapeProto(dim=dims)tensor=tensor_pb2.TensorProto(dtype=types_pb2.DT_FLOAT,tensor_shape=t_shape,float_val=list(batched_np.reshape(-1)))request.inputs['inputs'].CopyFrom(tensor)
對N個圖像的批處理,響應(yīng)(相應(yīng))的輸出Tensor對于請求批處理中相同數(shù)量的輸入具有預(yù)測結(jié)果,在這種情況下,N=2(以下是N=2的情況):
outputs{key:"classes"value{dtype:DT_INT64tensor_shape{dim{size:2}}int64_val:238int64_val:121}}...
硬件加速
關(guān)于GPU:
對于訓(xùn)練,GPU可以更直觀地利用并行化,因為構(gòu)建深層神經(jīng)網(wǎng)絡(luò)需要大量的計算才能得到最優(yōu)解。
然而,推理的情況并不總是如此。很多時候,當(dāng)圖形執(zhí)行步驟放置在GPU設(shè)備上時,CNN的推理就會加快。然而,挑選能夠優(yōu)化性價比的硬件,需要進行嚴格的測試、深入的技術(shù)和成本分析。硬件加速并行化對于“脫機”推理批處理(海量卷)更有價值。
在引入GPU處理之前,要考慮業(yè)務(wù)需求,并對收益(嚴格延遲、高吞吐量)進行徹底的成本(貨幣、操作、技術(shù))分析。
-
cpu
+關(guān)注
關(guān)注
68文章
10804瀏覽量
210829 -
二進制
+關(guān)注
關(guān)注
2文章
786瀏覽量
41564 -
tensorflow
+關(guān)注
關(guān)注
13文章
328瀏覽量
60473
原文標題:如何將TensorFlow Serving的性能提高超過70%?
文章出處:【微信號:rgznai100,微信公眾號:rgznai100】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。
發(fā)布評論請先 登錄
相關(guān)推薦
評論