亚洲一久久久久久久久,国产免费天天看高清影视在线,精品人妻伦九区久久aaa片,性荡视频播放在线视频

Docker 中的 .NET 異常了怎么抓 Dump_全球熱推薦

來源: 博客園2023-06-27 03:12:08
  一:背景1. 講故事

有很多朋友跟我說,在 Windows 上看過你文章知道了怎么抓 Crash, CPU爆高,內(nèi)存暴漲 等各種Dump,為什么你沒有寫在 Docker 中如何抓的相關(guān)文章呢?瞧不上嗎?


(資料圖片僅供參考)

哈哈,在DUMP的分析旅程中,跑在 Docker 中的 .NET 占比真的不多,大概10個(gè)dump有 1-2 個(gè)是 docker 中的,市場決定了我的研究方向,為了彌補(bǔ)這一塊的空洞,決定寫一篇文章來分享下這三大異常下的捕獲吧。

二:Docker 下的三大異常捕獲1. crash dump 捕獲

前不久我寫了一篇 Linux 上的 .NET 崩潰了怎么抓 Dump (https://www.cnblogs.com/huangxincheng/p/17440153.html)的文章,使用了微軟推薦的環(huán)境變量方式,其實(shí)這在 Docker 中是一樣適用的。

為了讓 webapi崩潰退出,我故意造一個(gè)棧溢出異常,參考代碼如下:

public class Program    {        public static void Main(string[] args)        {            var builder = WebApplication.CreateBuilder(args);            builder.Services.AddAuthorization();            var app = builder.Build();            app.UseAuthorization();            //1. crash            Task.Factory.StartNew(() =>            {                Test("a");            });            app.Run();        }        public static string Test(string a)        {            return Test("a" + a.Length);        }    }

有了代碼之后,接下來寫一個(gè) Dockerfile,主要就是把三個(gè)環(huán)境變量塞進(jìn)去。

FROM mcr.microsoft.com/dotnet/aspnet:6.0 AS runtimeWORKDIR /appCOPY ./ ./# 1. 使用中科大鏡像源RUN sed -i "s/deb.debian.org/mirrors.ustc.edu.cn/g" /etc/apt/sources.listENV COMPlus_DbgMiniDumpType 4ENV COMPlus_DbgMiniDumpName /dumps/%p-%e-%h-%t.dmpENV COMPlus_DbgEnableMiniDump 1ENTRYPOINT ["dotnet", "AspNetWebApi.dll"]

這里有一個(gè)細(xì)節(jié),為了能夠讓 Docker 中的 webapi 能夠訪問到,將 localhost 設(shè)置為 * ,修改 appsettings.json如下:

{  "urls": "http://*:5001",  "Logging": {    "LogLevel": {      "Default": "Information",      "Microsoft.AspNetCore": "Warning"    }  },  "AllowedHosts": "*"}

有了這些基礎(chǔ)最后就是 docker build & docker run 啦。

[root@localhost data]# docker build -t aspnetapp .[+] Building 0.3s (9/9) FINISHED                                                                          => [internal] load build definition from Dockerfile                                                0.0s => => transferring dockerfile: 447B                                                                0.0s => [internal] load .dockerignore                                                                   0.0s => => transferring context: 2B                                                                     0.0s => [internal] load metadata for mcr.microsoft.com/dotnet/aspnet:6.0                                0.3s => [1/4] FROM mcr.microsoft.com/dotnet/aspnet:6.0@sha256:a2a04325fdb2a871e964c89318921f82f6435b54  0.0s => [internal] load build context                                                                   0.0s => => transferring context: 860B                                                                   0.0s => CACHED [2/4] WORKDIR /app                                                                       0.0s => CACHED [3/4] COPY ./ ./                                                                         0.0s => CACHED [4/4] RUN sed -i "s/deb.debian.org/mirrors.ustc.edu.cn/g" /etc/apt/sources.list          0.0s => exporting to image                                                                              0.0s => => exporting layers                                                                             0.0s => => writing image sha256:be69203995c0e5423b2af913549e618d7ee8306fff3961118ff403b1359ae571        0.0s => => naming to docker.io/library/aspnetapp                                                        0.0s[root@localhost data]# docker run -itd  -p 5001:5001 --privileged -v /data2:/dumps --name aspnetcore_sample aspnetappca34c9274d998096f8562cbef3a43a7cbd9aa5ff2923e0f3e702b159e0b2f447[root@localhost data]# docker ps -aCONTAINER ID   IMAGE       COMMAND                  CREATED          STATUS                       PORTS     NAMESca34c9274d99   aspnetapp   "dotnet AspNetWebApi…"   20 seconds ago   Exited (139) 9 seconds ago             aspnetcore_sample[root@localhost data]# docker logs ca34c9274d99   ...   at AspNetWebApi.Program.Test(System.String)   at AspNetWebApi.Program.Test(System.String)   at AspNetWebApi.Program.Test(System.String)   at AspNetWebApi.Program.Test(System.String)   at AspNetWebApi.Program+<>c.
b__0_0() at System.Threading.Tasks.Task.InnerInvoke() at System.Threading.Tasks.Task+<>c.<.cctor>b__272_0(System.Object) at System.Threading.ExecutionContext.RunFromThreadPoolDispatchLoop(System.Threading.Thread, System.Threading.ExecutionContext, System.Threading.ContextCallback, System.Object) at System.Threading.Tasks.Task.ExecuteWithThreadLocal(System.Threading.Tasks.Task ByRef, System.Threading.Thread) at System.Threading.Tasks.Task.ExecuteEntryUnsafe(System.Threading.Thread) at System.Threading.ThreadPoolWorkQueue.Dispatch() at System.Threading.PortableThreadPool+WorkerThread.WorkerThreadStart() at System.Threading.Thread.StartCallback()[createdump] Gathering state for process 1 dotnet[createdump] Crashing thread 0017 signal 6 (0006)[createdump] Writing full dump to file /dumps/1-dotnet-ca34c9274d99-1687746929.dmp[createdump] Written 261320704 bytes (63799 pages) to core file[createdump] Target process is alive[createdump] Dump successfully written[root@localhost data2]# cd /data2[root@localhost data2]# ls -lntotal 255288-rw-------. 1 0 0 261414912 Jun 26 10:35 1-dotnet-ca34c9274d99-1687746929.dmp

上面的腳本已經(jīng)寫的非常清楚了,這里有幾個(gè)注意點(diǎn)提一下:

--privileged

一定要加上特殊權(quán)限,否則生成 dump 的時(shí)候會(huì)提示無權(quán)限。

-v /data2:/dumps

防止dump丟失,記得掛載到宿主機(jī)目錄 或者 共享容器 中。

2. 內(nèi)存暴漲 dump 捕獲

要想對(duì) docker 中的 .NET 程序內(nèi)存 進(jìn)行監(jiān)控,我一直都是極力推薦 procdump,目前最新的是版本是 1.5, github官網(wǎng)地址: https://github.com/Sysinternals/ProcDump-for-Linux 鑒于現(xiàn)在訪問 github 太慢,大家可以把 procdump_1.5-16239_amd64.deb下載到本地,為什么下載它,是因?yàn)槿萜髦惺?debain 系統(tǒng)。

下載好了之后放到項(xiàng)目中,使用默認(rèn)代碼骨架:

public class Program    {        public static void Main(string[] args)        {            var builder = WebApplication.CreateBuilder(args);            builder.Services.AddAuthorization();            var app = builder.Build();            app.UseAuthorization();            app.Run();        }    }

接下來就是寫 dockerfile 了,這里有一個(gè)細(xì)節(jié),就是如何在 Docker 中開啟多進(jìn)程,這里用 start.sh 腳本的方式開啟,參考代碼如下:

FROM mcr.microsoft.com/dotnet/aspnet:6.0 AS runtimeWORKDIR /appCOPY ./ ./# 1. 使用中科大鏡像源RUN sed -i "s/deb.debian.org/mirrors.ustc.edu.cn/g" /etc/apt/sources.list# 2. 安裝 gdb & procdumpRUN apt-get update && apt-get install -y gdbRUN dpkg -i procdump.debRUN echo "#!/bin/bash \n\procdump -m 30 -w dotnet /dumps & \n\dotnet \$1 \n\" > ./start.shRUN chmod +x ./start.shENTRYPOINT ["./start.sh", "AspNetWebApi.dll"]

有了這些設(shè)置后,接下來就是 publish 代碼用 docker 構(gòu)建啦,為了方便演示,這里就用 前臺(tái)模式開啟了哈。

[root@localhost data]# docker build -t aspnetapp .[+] Building 11.5s (13/13) FINISHED              [root@localhost data]# docker rm -f aspnetcore_sampleaspnetcore_sample[root@localhost data]# docker run -it --rm  -p 5001:5001 --privileged -v /data2:/dumps --name aspnetcore_sample aspnetappProcDump v1.5 - Sysinternals process dump utilityCopyright (C) 2023 Microsoft Corporation. All rights reserved. Licensed under the MIT license.Mark Russinovich, Mario Hewardt, John Salem, Javid HabibiSysinternals - www.sysinternals.comMonitors one or more processes and writes a core dump file when the processes exceeds thespecified criteria.[02:57:34 - INFO]: Waiting for processes "dotnet" to launch[02:57:34 - INFO]: Press Ctrl-C to end monitoring without terminating the process(es).Process Name:                           dotnetCPU Threshold:                          n/aCommit Threshold:                       >=30 MBThread Threshold:                       n/aFile Descriptor Threshold:              n/aSignal:                                 n/aException monitor                       OffPolling Interval (ms):                  1000Threshold (s):                          10Number of Dumps:                        1Output directory:                       /dumps[02:57:34 - INFO]: Starting monitor for process dotnet (9)info: Microsoft.Hosting.Lifetime[14]      Now listening on: http://[::]:5001info: Microsoft.Hosting.Lifetime[0]      Application started. Press Ctrl+C to shut down.info: Microsoft.Hosting.Lifetime[0]      Hosting environment: Productioninfo: Microsoft.Hosting.Lifetime[0]      Content root path: /app/[02:57:35 - INFO]: Trigger: Commit usage:48MB on process ID: 9[createdump] Gathering state for process 9 dotnet[createdump] Writing full dump to file /dumps/dotnet_commit_2023-06-26_02:57:35.9[createdump] Written 254459904 bytes (62124 pages) to core file[createdump] Target process is alive[createdump] Dump successfully written[02:57:35 - INFO]: Core dump 0 generated: /dumps/dotnet_commit_2023-06-26_02:57:35.9[02:57:36 - INFO]: Stopping monitors for process: dotnet (9)[root@localhost data2]# ls -lhtotal 243M-rw-------. 1 root root 243M Jun 26 10:57 dotnet_commit_2023-06-26_02:57:35.9

從腳本信息看,當(dāng)內(nèi)存到了 48MB的時(shí)候觸發(fā)的 dump 生成,也成功的進(jìn)入了 /dumps目錄中,太棒了。

3. cpu爆高 dump 捕獲

抓 cpu 爆高的dump最好的方式就是多抓幾個(gè),比如說:當(dāng) CPU >20% 連續(xù)超過 5s 抓 2個(gè)dump,這種方式抓的dump很容易就能找到真兇,為了方便演示,讓兩個(gè) cpu 直接打滿,參考代碼如下:

public static void Main(string[] args)        {            var builder = WebApplication.CreateBuilder(args);            builder.Services.AddAuthorization();            var app = builder.Build();            app.UseAuthorization();            //3. cpu            app.MapGet("/cpu", (HttpContext httpContext) =>            {                Task.Factory.StartNew(() => { bool b = true; while (true) { b = !b; } });                Task.Factory.StartNew(() => { bool b = true; while (true) { b = !b; } });                return new WeatherForecast();            });            app.Run();        }

接下來就是修改 dockerfile,因?yàn)槲业奶摂M機(jī)是 8 核心,如果兩個(gè)核心被打滿,那應(yīng)該會(huì)占用大概 24% 的 cpu 利用率,所以腳本中就設(shè)置 20% 吧。

FROM mcr.microsoft.com/dotnet/aspnet:6.0 AS runtimeWORKDIR /appCOPY ./ ./# 1. 使用中科大鏡像源RUN sed -i "s/deb.debian.org/mirrors.ustc.edu.cn/g" /etc/apt/sources.list# 2. 安裝 wgetRUN apt-get update && apt-get install -y gdbRUN dpkg -i procdump.debRUN echo "#!/bin/bash \n\procdump -c 20 -n 2 -s 5 -w dotnet /dumps & \n\dotnet \$1 \n\" > ./start.shRUN chmod +x ./start.shENTRYPOINT ["./start.sh", "AspNetWebApi.dll"]

最后就是 docker 構(gòu)建。

[root@localhost data]# docker build -t aspnetapp .[+] Building 0.4s (13/13) FINISHED[root@localhost data]# docker run -it --rm  -p 5001:5001 --privileged -v /data2:/dumps --name aspnetcore_sample aspnetappProcDump v1.5 - Sysinternals process dump utilityCopyright (C) 2023 Microsoft Corporation. All rights reserved. Licensed under the MIT license.Mark Russinovich, Mario Hewardt, John Salem, Javid HabibiSysinternals - www.sysinternals.comMonitors one or more processes and writes a core dump file when the processes exceeds thespecified criteria.[03:35:56 - INFO]: Waiting for processes "dotnet" to launch[03:35:56 - INFO]: Press Ctrl-C to end monitoring without terminating the process(es).Process Name:                           dotnetCPU Threshold:                          >= 20%Commit Threshold:                       n/aThread Threshold:                       n/aFile Descriptor Threshold:              n/aSignal:                                 n/aException monitor                       OffPolling Interval (ms):                  1000Threshold (s):                          5Number of Dumps:                        2Output directory:                       /dumps[03:35:56 - INFO]: Starting monitor for process dotnet (8)info: Microsoft.Hosting.Lifetime[14]      Now listening on: http://[::]:5001info: Microsoft.Hosting.Lifetime[0]      Application started. Press Ctrl+C to shut down.info: Microsoft.Hosting.Lifetime[0]      Hosting environment: Productioninfo: Microsoft.Hosting.Lifetime[0]      Content root path: /app/

看輸出是正在監(jiān)控,接下來我們訪問下網(wǎng)址: http://192.168.17.129:5001/cpu,稍等片刻之后就會(huì)生成兩個(gè)dump 文件。

三:總結(jié)

雖然Docker中的 .NET 程序占比較少,但把經(jīng)驗(yàn)總結(jié)出來還是很值得的,以后有人問怎么抓,可以把這篇文章直接丟過去啦!

關(guān)鍵詞:

責(zé)任編輯:sdnew003

相關(guān)新聞

版權(quán)與免責(zé)聲明:

1 本網(wǎng)注明“來源:×××”(非商業(yè)周刊網(wǎng))的作品,均轉(zhuǎn)載自其它媒體,轉(zhuǎn)載目的在于傳遞更多信息,并不代表本網(wǎng)贊同其觀點(diǎn)和對(duì)其真實(shí)性負(fù)責(zé),本網(wǎng)不承擔(dān)此類稿件侵權(quán)行為的連帶責(zé)任。

2 在本網(wǎng)的新聞頁面或BBS上進(jìn)行跟帖或發(fā)表言論者,文責(zé)自負(fù)。

3 相關(guān)信息并未經(jīng)過本網(wǎng)站證實(shí),不對(duì)您構(gòu)成任何投資建議,據(jù)此操作,風(fēng)險(xiǎn)自擔(dān)。

4 如涉及作品內(nèi)容、版權(quán)等其它問題,請(qǐng)?jiān)?0日內(nèi)同本網(wǎng)聯(lián)系。

熱文排行
  • 財(cái)經(jīng)
  • 綜合
  • 黃金360

蜜芽亚洲av无码精品色午夜| 男人扒开女人双腿猛进视频| 国产精品久久无码一区二区三区网| 成人aaa片一区国产精品| 天堂8在线新版官网| 麻豆精产一二三产区| 亚洲有码转帖| 亚洲综合欧美在线一区在线播放 | 国产成人亚洲精品无码青| 成人午夜国产内射主播| 极品美女扒开粉嫩小泬| 品色永久免费| 久久香蕉国产线熟妇人妻| 一女被多男玩喷潮视频| 浓毛老太交欧美老妇热爱乱| 又色又爽又高潮免费观看| 男女边吃奶边做边爱视频| 老色鬼永久精品网站| 2020年国产精品| 亚洲裸男gv网站| 亚洲精品成人片在线观看精品字幕 | 人妻人人做人做人人爱| 亚洲粉嫩高潮的18p| 精品亚洲成a人7777在线观看| 天天做天天爱天天综合网2021| 国产成人无码av| 无码a级毛片免费视频内谢| 亚洲av成人无码久久精品老人| 永久黄网站色视频免费直播| 中文字幕人成无码人妻 | 成人免费无遮挡在线播放| 日韩欧美在线综合网另类| 亚洲av无码一区东京热蜜芽| 无码中文av有码中文av| 国产精品一区在线观看你懂的| 国产成a人亚洲精v品无码| 精精国产xxxx视频在线| 大地资源中文第三页| 欧美一区二区三区成人片在线 | 久久99精品久久久久久噜噜| 强开小婷嫩苞又嫩又紧视频|