我是微软Dynamics 365 & Power Platform方面的工程师罗勇,也是2015年7月到2018年6月连续三年Dynamics CRM/Business Solutions方面的微软最有价值专家(Microsoft MVP),欢迎关注我的微信公众号 MSFTDynamics365erLuoYong ,回复380或者20191124可方便获取本文,同时可以在第一间得到我发布的最新博文信息,follow me!

前面的博文讲了些客户端编程,但是对于重要的数据,一般需要服务器端再做一次校验,常用的手段就是插件和实时工作流,今天我讲一下插件,官方文档请参考 Write a plug-in 和 Register a plug-in 等。

首先需要创建一个 .NET Framework 的 Class Library项目,记得 Framework选择 .NET Framework 4.6.2 。当然,为不同版本的Dynamics 365 Customer Engagement选择的Framework不尽相同,请根据官方文档说明,我这个示例是为 V9.X 版本做的示例。

首先通过NuGet添加对Microsoft.CrmSdk.CoreAssemblies的引用,如下图,当然也要选择合适的版本。如果不能上网的话,就需要添加对 Microsoft.Xrm.Sdk.dll 和 Microsoft.Crm.Sdk.Proxy.dll 的引用。

一般我会删除生成的Class1.cs文件,而是使用固定的命名方式来命名。比如我这个插件步骤将会注册在ly_WorkOrder实体的Create消息的Pre阶段,我就会新建一个 PreWorkOrderCreate.cs 的文件来撰写业务逻辑,当然这个类必须继承 Microsoft.Xrm.Sdk.IPlugin 接口,我使用的代码如下:常见的是获取组织服务和跟踪服务,其中插件日志的使用请参考我的博文:Dynamics CRM 2015/2016新特性之三十四:有了插件日志,调试插件so easy! ,若要使用映像(Image),请参考我的博文:Dynamics 365 CE Update消息PostOperation阶段Image的尝试  和  Dynamics 365 CE在Pre Delete插件中应用Image 。记得若要抛出用户可见可理解的异常文本,请使用throw new InvalidPluginExecutionException。

using Microsoft.Xrm.Sdk;
using Microsoft.Xrm.Sdk.Query;
using System;
using System.ServiceModel;
using System.Text; namespace PluginDemo
{
public class PreWorkOrderCreate : IPlugin
{
public void Execute(IServiceProvider serviceProvider)
{
//获取日志服务
ITracingService tracingService = (ITracingService)serviceProvider.GetService(typeof(ITracingService));
//写一些日志,方便跟踪
tracingService.Trace($"Enter PreWorkOrderCreate on {DateTime.UtcNow.ToString()}");
IPluginExecutionContext context = (IPluginExecutionContext)serviceProvider.GetService(typeof(IPluginExecutionContext));
if (context.InputParameters.Contains("Target") && context.InputParameters["Target"] is Entity)
{
//插件针对的当前实体记录,对于Pre Create来讲,该对象包括了所有设置的字段值,若字段没有设置值,在该对象中会不存在
Entity currentEntity = (Entity)context.InputParameters["Target"];
//获取组织服务
IOrganizationServiceFactory serviceFactory = (IOrganizationServiceFactory)serviceProvider.GetService(typeof(IOrganizationServiceFactory));
IOrganizationService orgSvc = serviceFactory.CreateOrganizationService(context.UserId);
int provinceValue = Int32.MaxValue;
int cityValue = Int32.MaxValue;
string buName = string.Empty;
//首先获取要做校验的值
//记得先确认该属性存在值再获取其值,否则容易引发异常
if (currentEntity.Contains("ly_province"))
{
provinceValue = currentEntity.GetAttributeValue<OptionSetValue>("ly_province").Value;
}
if (currentEntity.Contains("ly_city"))
{
cityValue = currentEntity.GetAttributeValue<OptionSetValue>("ly_city").Value;
}
var userEntity = orgSvc.Retrieve("systemuser", context.UserId, new ColumnSet("businessunitid"));
//每个系统用户肯定都设置了业务部门,我这里只是例行检查这个字段存在值
if (userEntity.Contains("businessunitid"))
{
buName = userEntity.GetAttributeValue<EntityReference>("businessunitid").Name;
if(buName == "Demo")
{
//省份字段用户选择了值才校验
if(provinceValue != Int32.MaxValue)
{
if(provinceValue != )
{
throw new InvalidPluginExecutionException($"省份字段值选择有误!");
}
else
{
//城市字段用户选择了值才做校验
if(cityValue != Int32.MaxValue)
{
if (cityValue >= provinceValue && cityValue < provinceValue + )
{
tracingService.Trace($"城市字段选择了值,且属于正确的省份!");
}
else
{
throw new InvalidPluginExecutionException($"城市字段值选择有误!");
}
}
}
}
}
else if(buName == "Sub Unit")
{
//省份字段用户选择了值才校验
if (provinceValue != Int32.MaxValue)
{
if (provinceValue != )
{
throw new InvalidPluginExecutionException($"省份字段值选择有误!");
}
else
{
//城市字段用户选择了值才做校验
if (cityValue != Int32.MaxValue)
{
if (cityValue >= provinceValue && cityValue < provinceValue + )
{
tracingService.Trace($"城市字段选择了值,且属于正确的省份!");
}
else
{
throw new InvalidPluginExecutionException($"城市字段值选择有误!");
}
}
}
}
}
}
else
{
throw new InvalidPluginExecutionException($"系统异常,当前用户(userid={context.UserId})的业务部门没有设置!");
}
}
tracingService.Trace($"Leave PreWorkOrderCreate on {DateTime.UtcNow.ToString()}");
}
}
}

记得一定要给该插件/自定义工作流活动程序集签名,在Visual Studio中右击该项目,选择属性(Properties) >  签名(Signing),选中 Sign the assembly,我这里新建一个Key file。

Key file我的设置如下,为了简便,我就不设置密码保护了,保存后编译插件项目,确定没有编译错误。

然后需要使用插件注册工具将其注册到Dynamics 365中,工具的下载请参考我的博文 下载Dynamics 365 Customer Engagement 工具 。双击其中的 PluginRegistration\PluginRegistration.exe 文件。点击【CREATE NEW CONNECTION】,以便连接到Dynamics 365,下面这个截图是连接到我自己的做了面向互联网部署(IFD)的环境示例。

如果是连接到Dynamics 365 Customer Engagement Online,请参考下图:

点击【Register】 > 【Register New Assembly】。

选择前面步骤编译生成的插件程序集,Isolation Mode一般建议选择Sandbox,而且Dynamics Customer Engagement Onine也只能选择Sandbox,强烈建议选择存储到Database,点击【Register Selected Plugins】按钮,如果看不到该按钮,是你的电脑分辨率太低所致,就用Tab键盘来辅助操作吧。

如果没有错误的话会弹出类似如下对话框:

还需要右击创建的Plugin,选择【Register New Step】按钮。

我这里设置如下,是注册在lw_workorder的Create消息的PreOperation阶段,其余的就不一一解释了,请参考官方文档,点击【Register New Step】按钮。

注册成功后可以看到最终的结果类似如下:

剩下的工作就是测试了,使用InvalidPluginExecutionException抛出的异常信息在界面显示效果如下:

Dynamics 365中开发和注册插件介绍的更多相关文章

  1. Dynamics 365中的应用程序介绍

    本人微信和易信公众号:微软动态CRM专家罗勇 ,回复275或者20180630可方便获取本文,同时可以在第一间得到我发布的最新的博文信息,follow me!我的网站是 www.luoyong.me ...

  2. Dynamics 365中自定义工作流活动获取的上下文分析及注意事项

    关注本人微信和易信公众号: 微软动态CRM专家罗勇 ,回复244或者20170306可方便获取本文,同时可以在第一间得到我发布的最新的博文信息,follow me!我的网站是 www.luoyong. ...

  3. Dynamics 365 CRM 开发架构简介

    Dynamics 365 CRM提供了多种编程模型,你可以灵活地按需选用最佳模式. 本文是对Dynamics 365 CRM编程模型的综述. 概览 下图表明了Dynamics 365 CRM的主要可编 ...

  4. 自定义工作流活动报错:您无法登陆系统。原因可能是您的用户记录或您所属的业务部门在Microsoft Dynamics 365中已被禁用。

    本人微信和易信公众号: 微软动态CRM专家罗勇 ,回复265或者20170926可方便获取本文,同时可以在第一间得到我发布的最新的博文信息,follow me!我的网站是 www.luoyong.me ...

  5. 将Dynamics 365中的用户及其角色、角色导出到Excel中

    关注本人微信和易信公众号: 微软动态CRM专家罗勇 ,回复240或者20161204可方便获取本文,同时可以在第一间得到我发布的最新的博文信息,follow me!我的网站是 www.luoyong. ...

  6. Dynamics 365中配置和使用文件夹级别的跟踪(folder-level tracking)

    本人微信和易信公众号:微软动态CRM专家罗勇 ,回复274或者20180630可方便获取本文,同时可以在第一间得到我发布的最新的博文信息,follow me!我的网站是 www.luoyong.me ...

  7. Dynamics 365工作流报错:您无法登陆系统。原因可能是您的用户记录或您所属的业务部门在Microsoft Dynamics 365中已被禁用。

    本人微信和易信公众号: 微软动态CRM专家罗勇 ,回复265或者20170926可方便获取本文,同时可以在第一间得到我发布的最新的博文信息,follow me!我的网站是 www.luoyong.me ...

  8. Dynamics 365中的非交互式账号(Non-interactive User)介绍

    摘要: 本人微信和易信公众号: 微软动态CRM专家罗勇 ,回复272或者20180616可方便获取本文,同时可以在第一间得到我发布的最新的博文信息,follow me!我的网站是 www.luoyon ...

  9. Dynamics 365中的事件框架与事件执行管道(Event execution pipeline)

    本文介绍了Microsoft Dynamics 365(以下简称D365)中的两个概念,事件框架(Event Framework)与事件执行管道(Event execution pipeline). ...

随机推荐

  1. java.security.NoSuchProviderException: no such provider: BC 的问题解决

    第一种方式 1.修改以下两个文件 %JDK_Home%\jre\lib\security\java.security %JRE_Home%\jre\lib\security\java.security ...

  2. luogu P2650 弹幕考察

    题意简化:求某个区间在一组区间中覆盖的数量 对于这个问题,我们很容易想到线段树,或者树状数组,但是maxlongint不能让我们这么做 30分思路: 对于m个区间,枚举n个区间判断与它是否重合 但是O ...

  3. luogu P1044 栈

    题目背景 栈是计算机中经典的数据结构,简单的说,栈就是限制在一端进行插入删除操作的线性表. 栈有两种最重要的操作,即poppop(从栈顶弹出一个元素)和pushpush(将一个元素进栈). 栈的重要性 ...

  4. APP 框架搭建

    在开发一款app前必须先把框架搭建好,这样能避免代码混乱,多人开发时遇到bug难以解决. 构建MVC整体框架的缺点是会导致VC代码量过大,也存在一些不足.借鉴了网上牛人的思路后,https://www ...

  5. git 使用详解(6)—— 3种撤消操作

    接下来,我们会介绍一些基本的撤消操作相关的命令.请注意,有些操作并不总是可以撤消的,所以请务必谨慎小心,一旦失误,就有可能丢失部分工作成果. 修改最后一次提交 git commit --amend 有 ...

  6. nginx部署基于http负载均衡器

    nginx跨多个应用程序实例的负载平衡是一种用于优化资源利用率,最大化吞吐量,减少延迟和确保容错配置的常用技术. 环境介绍 配置nginx负载均衡器因会用到多台服务器来进行,所以下面我会用到docke ...

  7. 谈一谈AOP面向切面编程

    AOP是什么 : AOP面向切面编程他是一种编程思想,是指在程序运行期间,将某段代码动态的切入到指定方法的指定位置,将这种编程方式称为面向切面编程 AOP使用场景 : 日志 事务 使用AOP的好处是: ...

  8. IIS配置svc(IIS8中添加WCF支持几种方法小结)

    方法一 最近在做Silverlight,Windows Phone应用移植到Windows 8平台,在IIS8中测试一些传统WCF服务应用,发现IIS8不支持WCF服务svc请求,后来发现IIS8缺少 ...

  9. java8-date和timeAPI

    一 我们为什么要学习 java.timeAPI 原先的Date and Calendar 类的api比较复杂,不易于理解,应用起来不是很灵活. Calendar 是个线程不安全的类会导致SimpleD ...

  10. Python爬虫入门CentOS环境安装

    前言本文的文字及图片来源于网络,仅供学习.交流使用,不具有任何商业用途,版权归原作者所有,如有问题请及时联系我们以作处理.作者:梦想橡皮擦 CentOS环境安装-简介你好,当你打开这个文档的时候,我知 ...