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

之前的文章 使用JS通过Web API执行批量操作,多个操作是一个事务! 讲的是JavaScript的做法,今天我实验了一阵子终于搞定了C#做法。

不多说,上代码,这个代码的用途简单,就是新建一个注释,然后将某个注释的stepid字段值设置为Y,两个操作做成一个事务:

        private static async Task<string> ExecuteBatch(string ODataBaseUrl)
{
Guid batchId = Guid.NewGuid();
Guid changesetId = Guid.NewGuid();
string returnVal = string.Empty;
StringBuilder requestBody = new StringBuilder();
requestBody.Append($"--batch_{batchId}");
requestBody.Append("\n");
requestBody.Append($"Content-Type: multipart/mixed;boundary=changeset_{changesetId}");
requestBody.Append("\n");
requestBody.Append("\n");
requestBody.Append($"--changeset_{changesetId}");
requestBody.Append("\n");
requestBody.Append("Content-Type: application/http");
requestBody.Append("\n");
requestBody.Append("Content-Transfer-Encoding:binary");
requestBody.Append("\n");
requestBody.Append("Content-ID: 1");
requestBody.Append("\n");
requestBody.Append("\n");
requestBody.Append($"POST {ODataBaseUrl}annotations HTTP/1.1");
requestBody.Append("\n");
requestBody.Append("Content-Type: application/json;type=entry");
requestBody.Append("\n");
requestBody.Append("\n");
JObject jObject = new JObject(
new JProperty("subject", "克隆出来的记录"),
new JProperty("filename", "MSFTDynamics365erLuoYong.jpg"),
new JProperty("filesize", ),
new JProperty("documentbody", "/9j/4AAQSkZJRgAB2cFFABRRRQAUUUUAf//Z"
requestBody.Append("\n");
requestBody.Append($"--changeset_{changesetId}");
requestBody.Append("\n");
requestBody.Append("Content-Type: application/http");
requestBody.Append("\n");
requestBody.Append("Content-Transfer-Encoding:binary");
requestBody.Append("\n");
requestBody.Append("Content-ID: 2");
requestBody.Append("\n");
requestBody.Append("\n");
requestBody.Append($"PUT {ODataBaseUrl}annotations(4B502B89-4520-E911-B0C6-E05D5152C120)/stepid HTTP/1.1");
requestBody.Append("\n");
requestBody.Append("Content-Type: application/json;type=entry");
requestBody.Append("\n");
requestBody.Append("\n");
requestBody.Append("{\"value\":\"Y\"}");
requestBody.Append("\n");
requestBody.Append("\n");
requestBody.Append($"--changeset_{changesetId}--");
requestBody.Append("\n");
requestBody.Append("\n");
requestBody.Append($"--batch_{batchId}--");
HttpWebRequest request = (HttpWebRequest)HttpWebRequest.Create($"{ODataBaseUrl}$batch");
request.Credentials = new NetworkCredential(ConfigurationManager.AppSettings["userName"], ConfigurationManager.AppSettings["passWord"]);
request.Method = "POST";
request.ContentType = $"multipart/mixed;boundary=batch_{batchId}";
request.Accept = "application/json";
request.Headers.Add("OData-MaxVersion", "4.0");
request.Headers.Add("OData-Version", "4.0");
byte[] buffer = Encoding.UTF8.GetBytes(requestBody.ToString());
request.ContentLength = buffer.Length;
using (Stream stream = await request.GetRequestStreamAsync())
{
stream.Write(buffer, , buffer.Length);
stream.Flush();
}
using (WebResponse response = await request.GetResponseAsync())
{
var webResponse = response as HttpWebResponse;
if (webResponse.StatusCode == HttpStatusCode.OK)
{
Stream Answer = response.GetResponseStream();
StreamReader _Answer = new StreamReader(Answer);
returnVal = _Answer.ReadToEnd();
}
else
{
throw new Exception($"Error. {webResponse.StatusCode}");
}
}
return returnVal;
}

我的执行效果如下,我这里是执行成功HTTP STATUS CODE = 200)后显示了返回内容:

如果改成用 HttpClient 来发起请求,代码如下,个人推荐使用这种:

        private static async Task<string> ExecuteBatch(string ODataBaseUrl)
{
Guid batchId = Guid.NewGuid();
Guid changesetId = Guid.NewGuid();
string returnVal = string.Empty;
StringBuilder requestBody = new StringBuilder();
requestBody.Append($"--batch_{batchId}");
requestBody.Append("\n");
requestBody.Append($"Content-Type: multipart/mixed;boundary=changeset_{changesetId}");
requestBody.Append("\n");
requestBody.Append("\n");
requestBody.Append($"--changeset_{changesetId}");
requestBody.Append("\n");
requestBody.Append("Content-Type: application/http");
requestBody.Append("\n");
requestBody.Append("Content-Transfer-Encoding:binary");
requestBody.Append("\n");
requestBody.Append("Content-ID: 1");
requestBody.Append("\n");
requestBody.Append("\n");
requestBody.Append($"POST {ODataBaseUrl}annotations HTTP/1.1");
requestBody.Append("\n");
requestBody.Append("Content-Type: application/json;type=entry");
requestBody.Append("\n");
requestBody.Append("\n");
JObject jObject = new JObject(
new JProperty("subject", "克隆出来的记录"),
new JProperty("filename", "MSFTDynamics365erLuoYong.jpg"),
new JProperty("filesize", ),
new JProperty("documentbody", "/9j/4AAQSkZJRRRQAUUUUAf//Z"),
new JProperty("isdocument", true),
new JProperty("mimetype", "image/jpeg"),
new JProperty("notetext", "罗勇测试用的"),
new JProperty("objectid_account@odata.bind", "/accounts(C543D891-9FBD-E911-B0D1-8280A40FB795)")
);
requestBody.Append(JsonConvert.SerializeObject(jObject));
requestBody.Append("\n");
requestBody.Append("\n");
requestBody.Append($"--changeset_{changesetId}");
requestBody.Append("\n");
requestBody.Append("Content-Type: application/http");
requestBody.Append("\n");
requestBody.Append("Content-Transfer-Encoding:binary");
requestBody.Append("\n");
requestBody.Append("Content-ID: 2");
requestBody.Append("\n");
requestBody.Append("\n");
requestBody.Append($"PUT {ODataBaseUrl}annotations(85412E8C-B08D-E911-B0C9-C8187530CEF1)/stepid HTTP/1.1");
requestBody.Append("\n");
requestBody.Append("Content-Type: application/json;type=entry");
requestBody.Append("\n");
requestBody.Append("\n");
requestBody.Append("{\"value\":\"Y\"}");
requestBody.Append("\n");
requestBody.Append("\n");
requestBody.Append($"--changeset_{changesetId}--");
NetworkCredential credentials = new NetworkCredential(ConfigurationManager.AppSettings["userName"], ConfigurationManager.AppSettings["passWord"]);
HttpMessageHandler messageHandler = new HttpClientHandler()
{
Credentials = credentials
};
using (HttpClient httpClient = new HttpClient(messageHandler))
{
httpClient.DefaultRequestHeaders.Add("OData-MaxVersion", "4.0");
httpClient.DefaultRequestHeaders.Add("OData-Version", "4.0");
httpClient.DefaultRequestHeaders.Add("Accept", "application/json");
MultipartContent mainContent = new MultipartContent("mixed", $"batch_{batchId.ToString().Replace("\"","")}");
StringContent sc = new StringContent(requestBody.ToString());
sc.Headers.Clear();
sc.Headers.Add("Content-Type", $"multipart/mixed;boundary={changesetId.ToString()}");
mainContent.Add(sc);
var response = await httpClient.PostAsync($"{ODataBaseUrl}$batch", mainContent);
if (response.IsSuccessStatusCode)
{
returnVal = await response.Content.ReadAsStringAsync();
}
else
{
var errorMsg = await response.Content.ReadAsStringAsync();
throw new Exception(errorMsg);
}
return returnVal;
}
}

通过C#代码调用Dynamics 365 Web API执行批量操作的更多相关文章

  1. 使用JS通过Web API执行批量操作,多个操作是一个事务!

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

  2. Dynamics 365 Web Api之基于single-valued navigation property的filter查询

    本篇要讲的是dynamics 新版本中web api的一个改进功能,虽然改进的很有限,但至少是改进了. 举个例子,我们现在知道联系人的名字vic,我们想找出客户记录中主要联系人名字为vic的所有客户, ...

  3. 利用Fiddler修改请求信息通过Web API执行Dynamics 365操作(Action)实例

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

  4. 不借助工具在浏览器中通过Web API执行Dynamics 365操作(Action)实例

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

  5. Dynamics CRM Web API中的and和or组合的正确方式!

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

  6. MVC项目实践,在三层架构下实现SportsStore-09,ASP.NET MVC调用ASP.NET Web API的查询服务

    ASP.NET Web API和WCF都体现了REST软件架构风格.在REST中,把一切数据视为资源,所以也是一种面向资源的架构风格.所有的资源都可以通过URI来唯一标识,通过对资源的HTTP操作(G ...

  7. Dynamics CRM2016 Web Api之分页查询

    在dynamics crm web api还没出现前,我们是通过fetchxml来实现的,当然这种方式依旧可行,那既然web api来了我们就拥抱新的方式. web api中我们通过指定查询的条数来实 ...

  8. 延迟调用或多次调用第三方的Web API服务

    当我们调用第三方的Web API服务的时候,不一定每次都是成功的.这时候,我们可能会再多尝试几次,也有可能延迟一段时间再去尝试调用服务. Task的静态方法Delay允许我们延迟执行某个Task,此方 ...

  9. 利用Fiddler修改请求信息通过Web API执行操作(Action)实例

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

随机推荐

  1. 不看好 git ,也看不懂为什么那么多人去使用 git

    上来就亮明观点,符合我的性格.呵呵呵. 为什么不看好 git 呢? 首先,我们来看看 git 产生的背景. git 是 Linus 开发的,最初的目的,是为了管理 Linux 系统的源代码.这是一个分 ...

  2. Java多线程面试问答

    今天,我们将讨论Java 多线程面试问答. 线程是Java面试问题中的热门话题之一.在这里,我从面试的角度列出了大多数重要的Java多线程面试问题,但是您应该对Java线程有足够的知识来处理后续问题. ...

  3. mysql分表之水平分割

    一.背景 老大安排要对某张大容量表进行分割,根据年份分割成多张表. 二.水平分割 水平拆分是指数据表行的拆分,表的行数超过百万行时,就会变慢,这时可以把一张的表的数据拆成多张表来存放. 定义规则分表后 ...

  4. zuul网关

    Zuul路由网关简介及基本使用 简介 Zuul API路由网关服务简介 请看上图,这里的API 路由网关服务 由Zuul实现,主要就是对外提供服务接口的时候,起到了请求的路由和过滤作用,也因此能够隐藏 ...

  5. 【杭研大咖说】温正湖:6年,从不会写SQL到数据库专家

    他是业界主流数据库技术会议的明星讲师,开源社区各种分享活动的活跃分子:他累计申请了10多个技术发明专利,已授权8个:他近一年发布60多篇高质量技术博客文章,阅读量数十万:他和团队对MyRocks的优化 ...

  6. JCC 指令

    JCC跳转指令 JCC指条件跳转指令,CC就是指条件码. JCC指令 中文含义 英文原意 检查符号位 典型C应用 JZ/JE 若为0则跳转:若相等则跳转 jump if zero;jump if eq ...

  7. C# Replace字符替换函数

    它可以将字串内的字符替换为别的字符,可以嵌套使用,如下: 需要注意的是,它可以把字符替换为空,但不可以替换空字符,当不确定字符串是否为空时,可以进行以下判断,再替换: 示例的完整代码: string ...

  8. GitLab-使用SSH的方式拉取和推送项目

    场景 Docker Compose部署GitLab服务,搭建自己的代码托管平台(图文教程): https://blog.csdn.net/BADAO_LIUMANG_QIZHI/article/det ...

  9. eCharts二三维地图总结

    文章版权由作者李晓晖和博客园共有,若转载请于明显处标明出处:http://www.cnblogs.com/naaoveGIS/ 1.背景 最近多个项目中的登录页面陆续提出了不少地图需求,主要围绕地图的 ...

  10. 桶排序(C语言)

    #include <stdio.h> int main(void) { int arr[5]={2,5,1,3,3}; //定义需要排序的数组 int res[6]={0}; //初始化& ...