上一篇咱們經過實戰分享了使用Go推送釘釘消息,因爲技癢,筆者如今也編寫了一個.NET Core的Demo,做爲簡單的對照和說明。web
最後,因爲精力有限,筆者但願有興趣的朋友能夠分享下使用CoreRT將.NET Core編譯成機器代碼這塊的實踐。docker
獲取參數json
設置消息數據格式api
發送請求markdown
設置Dockerfileapp
運行並設置環境變量推送消息async
這裏咱們使用.NET Core來完成相關需求,注意,這裏是.NET Core,而不是ASP.NET Core。需求和上面相似,工程相關依賴以下所示:ui
<PackageReferenceInclude="Microsoft.Extensions.Configuration"Version="2.2.0"/> <PackageReferenceInclude="Microsoft.Extensions.Configuration.CommandLine"Version="2.2.0"/> <PackageReferenceInclude="Microsoft.Extensions.Configuration.EnvironmentVariables"Version="2.2.0"/> <PackageReferenceInclude="Microsoft.Extensions.Http"Version="2.2.0"/> <PackageReferenceInclude="Newtonsoft.Json"Version="12.0.1"/>
如下是相關的主體代碼:url
從環境變量或者命令行參數獲取配置:spa
///<summary> ///環境變量列表 ///</summary> privatestaticreadonlystring[] EnvList = { //釘釘機器人地址 "WEBHOOK", //@的手機號碼 "AT_MOBILES", //@全部人 "IS_AT_ALL", //消息內容 "MESSAGE", //消息類型(僅支持文本和markdown) "MSG_TYPE" }; privatestaticvoid Main(string[] args) { var config = newConfigurationBuilder() //支持命令行參數 .AddCommandLine(args) //支持環境變量 .AddEnvironmentVariables() .Build(); #region參數檢查 foreach (var envName in EnvList) { var value =config[envName]; if (string.IsNullOrWhiteSpace(value)&& envName != "AT_MOBILES" && envName !="IS_AT_ALL") { Console.WriteLine($"{envName}不能爲空!"); return; } } if (string.IsNullOrWhiteSpace(config["AT_MOBILES"]) &&string.IsNullOrWhiteSpace(config["IS_AT_ALL"])) { Console.WriteLine("必須設置參數 AT_MOBILES 和 IS_AT_ALL 二者之一!"); return; } #endregion try { //推送消息 SetDataAndSendWebhooks(config).Wait(); } catch (Exception ex) { Console.WriteLine(ex.ToString()); } }
設置消息格式,爲了簡單,這裏咱們使用匿名類:
///<summary> ///設置消息並調用Webhook ///</summary> ///<param name="config"></param> ///<returns></returns> privatestaticasync Task SetDataAndSendWebhooks(IConfigurationRoot config) { var at = new { AtMobiles = config["AT_MOBILES"]?.Split(','), IsAtAll = Convert.ToBoolean(config["IS_AT_ALL"] ?? "false") }; switch (config["MSG_TYPE"]) { case"text": { var data = new { Msgtype = "text", Text = new { Content =config["MESSAGE"] }, At = at }; awaitSendWebhooks(config["WEBHOOK"], data); break; } case"markdown": { var data = new { Msgtype = "markdown", Markdown = new { Title = "釘釘通知", Text = config["MESSAGE"] }, At = at }; awaitSendWebhooks(config["WEBHOOK"], data); break; } default: { Console.WriteLine($"不支持的格式:{config["MSG_TYPE"]}"); break; } } }
此處代碼使用Newtonsoft.Json作JSON序列化,而後使用Microsoft.Extensions.Http的HttpClient庫來發送Post請求。
在數據格式這塊,咱們經過配置作了如下設置:
忽略Null值。也就是爲null的屬性不作JSON序列化。
設置屬性命名規則爲Camel-Case駝峯式命名法,首字母小寫。
主體代碼以下所示:
///<summary> ///調用webhook ///</summary> ///<typeparamname="T"></typeparam> ///<param name="url">webhook地址</param> ///<param name="data">消息</param> ///<returns></returns> privatestaticasync Task SendWebhooks<T>(string url, T data) where T : class { JsonConvert.DefaultSettings = newFunc<JsonSerializerSettings>(() =>newJsonSerializerSettings() { NullValueHandling =NullValueHandling.Ignore, ContractResolver = newCamelCasePropertyNamesContractResolver() }); var jsonData =JsonConvert.SerializeObject(data); Console.WriteLine(jsonData); using (var httpClient = new HttpClient()) { var content = newStringContent(jsonData); content.Headers.ContentType = newMediaTypeHeaderValue("application/json"); var result = awaithttpClient.PostAsync(url, content); result.EnsureSuccessStatusCode(); Console.WriteLine($"Send webhook succeed. StatusCode:{result.StatusCode}"); } }
在以前咱們已經講述過,使用了分階段構建。整個Dockerfile基本上使用VS Docker tool生成:
FROMmicrosoft/dotnet:2.2-runtime AS base WORKDIR /app FROMmicrosoft/dotnet:2.2-sdk AS build WORKDIR /src COPY DingTalk.NET/DingTalk.NET.csprojDingTalk.NET/ RUN dotnet restoreDingTalk.NET/DingTalk.NET.csproj COPY . . WORKDIR /src/DingTalk.NET RUN dotnet buildDingTalk.NET.csproj -c Release -o /app FROM build AS publish RUN dotnet publish DingTalk.NET.csproj-c Release -o /app FROM base AS final WORKDIR /app COPY --from=publish/app . ENTRYPOINT ["dotnet", "DingTalk.NET.dll"] # 注意不要單獨使用 MAINTAINER 指令,MAINTAINER已被Label標籤代替 LABEL MAINTAINER ="xinlai@xin-lai.com" # LABEL指令用於將元數據添加到鏡像,支持鍵值對和JSON,咱們可使用 docker inspect 命令來查看 LABELDingtalkComponent='{\ "description": "使用釘釘發送通知消息.",\ "input": [\ {"name": "WEBHOOK","desc": "必填, 釘釘機器人Webhook地址"},\ {"name":"AT_MOBILES", "desc": "非必填,被@人的手機號"},\ {"name":"IS_AT_ALL", "desc": "非必填,@全部人時:true, 不然爲:false"},\ {"name": "MESSAGE","desc": "必填,自定義發送的消息內容"},\ {"name":"MSG_TYPE", "desc": "必填,自定義發送的消息類型,目前僅支持text和markdown"}\ ]\ }'
編譯完成後,咱們來查看下鏡像大小:
注意:
經過上圖咱們能夠看到,鏡像大小不到200M,相比GO體重大了許多,可是相比其餘語言卻輕了很多。不過,咱們能夠經過官方開源庫CoreRT將.NET Core編譯成機器代碼,也就是.NET Core也能夠作到編譯完成後只有幾M大小。有興趣的朋友能夠分享下這塊的實踐。
咱們使用PowerShell編寫簡單腳本以下所示:
docker build --rm-f "Dockerfile" -t dingtalk.net:latest . docker run --rm -e"WEBHOOK=https://oapi.dingtalk.com/robot/send?access_token={yourAccess Token}" ` -e "MESSAGE=*使用.NET Core發送釘釘消息。*" ` -e "IS_AT_ALL=true" ` -e "MSG_TYPE=markdown" ` -d dingtalk.net
效果如圖: