前言

上一篇我们通过实战分享了使用Go推送钉钉消息,由于技痒,笔者现在也编写了一个.NET Core的Demo,作为简单的对照和说明。

最后,由于精力有限,笔者希望有兴趣的朋友可以分享下使用CoreRT将.NET Core编译成机器代码这块的实践。

目录

  • 使用.NET Core推送钉钉消息

  • 获取参数

  • 设置消息数据格式

  • 发送请求

  • 设置Dockerfile

  • 运行并设置环境变量推送消息

使用.NET Core推送钉钉消息

这里我们使用.NET Core来完成相关需求,注意,这里是.NET Core,而不是ASP.NET Core。需求和上面类似,工程相关依赖如下所示:

<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"/>

  

以下是相关的主体代码:

获取参数

从环境变量或者命令行参数获取配置:

       ///<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

在之前我们已经讲述过,使用了分阶段构建。整个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

效果如图:

Docker最全教程之使用.NET Core推送钉钉消息(十九)的更多相关文章

  1. Docker最全教程之使用Node.js搭建团队技术文档站(二十三)

    前言 各种编程语言均有其优势和生态,有兴趣的朋友完全可以涉猎多门语言.在平常的工作之中,也可以尝试选择相对适合的编程语言来完成相关的工作. 在团队技术文档站搭建这块,笔者尝试了许多框架,最终还是选择了 ...

  2. Docker最全教程——从理论到实战(八)

    在本系列教程中,笔者希望将必要的知识点围绕理论.流程(工作流程).方法.实践来进行讲解,而不是单纯的为讲解知识点而进行讲解.也就是说,笔者希望能够让大家将理论.知识.思想和指导应用到工作的实际场景和实 ...

  3. Docker最全教程之使用Tencent Hub来完成CI(九)

    使用Tencent Hub来完成CI 关于Tencent Hub Tencent Hub是腾讯出品的DevOps服务.主要提供多存储格式的版本管理,支持Docker Image.Binary.Helm ...

  4. Docker最全教程——从理论到实战(七)

    在本系列教程中,笔者希望将必要的知识点围绕理论.流程(工作流程).方法.实践来进行讲解,而不是单纯的为讲解知识点而进行讲解.也就是说,笔者希望能够让大家将理论.知识.思想和指导应用到工作的实际场景和实 ...

  5. Docker最全教程——从理论到实战(六)

    托管到腾讯云容器服务 托管到腾讯云容器服务,我们的公众号“magiccodes”已经发布了相关的录屏教程,大家可以结合本篇教程一起查阅.   自建还是托管? 在开始之前,我们先来讨论一个问题——是自建 ...

  6. Docker最全教程——从理论到实战(五)

    往期内容链接 Docker最全教程——从理论到实战(一) Docker最全教程——从理论到实战(二) Docker最全教程——从理论到实战(三) Docker最全教程——从理论到实战(四) 本篇教程持 ...

  7. Docker最全教程——从理论到实战

    Docker最全教程——从理论到实战(一) Docker最全教程——从理论到实战(二) Docker最全教程——从理论到实战(三) Docker最全教程——从理论到实战(四) Docker最全教程—— ...

  8. Docker最全教程

    摘自雪雁大佬的博客,地址:https://www.cnblogs.com/codelove/default.html 目录: Docker最全教程——从理论到实战(一) Docker最全教程——从理论 ...

  9. Docker最全教程——从理论到实战(十七)

    前言 上一篇我们通过实战分享了使用Go推送钉钉消息,由于技痒,笔者现在也编写了一个.NET Core的Demo,作为简单的对照和说明. 最后,由于精力有限,笔者希望有兴趣的朋友可以分享下使用CoreR ...

随机推荐

  1. SQL转化为MapReduce的过程

    转载:http://www.cnblogs.com/yaojingang/p/5446310.html 在了解了MapReduce实现SQL基本操作之后,我们来看看Hive是如何将SQL转化为MapR ...

  2. springmvc 请求经过controller后静态资源无法访问的问题

    经过RequestMapping(“xx”)后 转发请求时会在url里面附带地址, 导致访问静态资源文件失败, 解决办法是在 spring-mvc.xml文件中加上 <mvc:default-s ...

  3. python笔试题(1)

            为了充实自己,小编决定上传自己见到的笔试题和面试题.可能要写好长时间,一时半会写不了多少,只能说遇到多少写多少吧,但是只要小编有时间,会持续上传(但是答案却不能保证,所以有看到错误的及 ...

  4. OVMF基础

    什么是OVMF The Open Virtual Machine Firmware (OVMF) project aims to support firmware for Virtual Machin ...

  5. VMware 非简易安装centos6(静态ip配置)

    1.选择新建虚拟机 在弹出框中我选择推荐安装 然后点击下一步 2.选择稍后安装操作系统(不然会默认简易安装) 接着一直下一步就可以了,最后到这个页面 4.如下图选择事先下载好的安装文件 5.配置桥接模 ...

  6. app后端设计(4)-- 通讯的安全性

    在app的后台设计中,一个很重要的因素是考虑通讯的安全性. 因此,我们需要考虑的要点有: 1. 在app和后台,都不能保存任何用户密码的明文 2. 在app和后台通讯的过程中,怎么保证用户信息的安全性 ...

  7. Kafka基础

    简介 #概念:消息中间件(消息系统)      //消息系统分类:         点对点 消息队列(peer-to-peer)         发布/订阅 消息队列 消费者在消费时,是通过pull ...

  8. Python 模块详解及import本质

    同在当前目录下的模块和包导入 模块定义 本质就是.py结尾的python文件. 用来从逻辑上组织python代码(变量,函数,类,逻辑) 文件名: test.py;  对应的模块名 : test 模块 ...

  9. Java 读书笔记 (六) 引用类型

    Java里使用long类型的数据要在数值后面加上L,否则会作为整型解析. 引用类型 引用类型是一个对象类型,它的值是指向内存空间的引用,就是地址, 所指向的内存中保存着变量所表示的一个值或一组值. i ...

  10. 第四天 Java语言基础

    一.函数的概念 1)什么函数 函数就是定义在类中的具有特定功能的一段独立小程序,并能被多次使用. 2)问题引入 在昨天讲述使用循环嵌套画出矩形.但有问题,每次要画矩形都要写很多重复性的代码,能不能将这 ...