1. 基本HTTP POST发送方式

  • 表单提交协议规定:
    要先将 HTTP 要求的 Content-Type 设为 multipart/form-data,而且要设定一个 boundary 参数,
    这个参数是由应用程序自行产生,它会用来识别每一份资料的边界 (boundary),
    用以产生多重信息部份 (message part)。
    而 HTTP 服务器可以抓取 HTTP POST 的信息,
    基本内容:
    1. 每个信息部份都要用 --[BOUNDARY_NAME] 来包装,以分隔出信息的每个部份,而最后要再加上一个 --[BOUNDARY_NAME] 来表示结束。
    2. 每个信息部份都要有一个 Content-Disposition: form-data; name="",而 name 设定的就是 HTTP POST 的键值 (key)。
    3. 声明区和值区中间要隔两个新行符号(\r\n)。
    4. 中间可以夹入二进制资料,但二进制资料必须要格式化为二进制字符串。
    5. 若要设定不同信息段的资料型别 (Content-Type),则要在信息段内的声明区设定。

  • 因此两个form内容模板为
    boundary = "----" + DateTime.Now.Ticks.ToString("x");//程序生成
    1.文本内容
    "\r\n--" + boundary +
    "\r\nContent-Disposition: form-data; name=\"键\"; filename=\"文件名\"" +
    "\r\nContent-Type: application/octet-stream" +
    "\r\n\r\n";
    2.文件内容
    "\r\n--" + boundary +
    "\r\nContent-Disposition: form-data; name=\"键\"" +
    "\r\n\r\n内容";

  • Python方法

def upload_file(file_path, wx_upload_url):file_name = file_path.split("/")[-1]with open(file_path, 'rb') as f:length = os.path.getsize(file_path)data = f.read()headers = {"Content-Type": "application/octet-stream"}params = {"filename": file_name,"filelength": length,}file_data = copy(params)file_data['file'] = (file_path.split('/')[-1:][0], data)encode_data = encode_multipart_formdata(file_data)file_data = encode_data[0]headers['Content-Type'] = encode_data[1]print(headers)r = requests.post(wx_upload_url, data=file_data, headers=headers)print(r.text)media_id = r.json()['media_id']return media_id# media_id 通过上一步上传的方法获得
def qi_ye_wei_xin_file(wx_url, media_id):headers = {"Content-Type": "text/plain"}data = {"msgtype": "file","file": {"media_id": media_id}}r = requests.post(url=wx_url,headers=headers, json=data)
  • C#实现方式

  •    class FormItemModel{/// <summary>/// 表单键,request["key"]/// </summary>public string Key { set; get; }/// <summary>/// 表单值,上传文件时忽略,request["key"].value/// </summary>public string Value { set; get; }/// <summary>/// 是否是文件/// </summary>public bool IsFile{get{if (FileContent == null || FileContent.Length == 0)return false;if (FileContent != null && FileContent.Length > 0 && string.IsNullOrWhiteSpace(FileName))throw new Exception("上传文件时 FileName 属性值不能为空");return true;}}/// <summary>/// 上传的文件名/// </summary>public string FileName { set; get; }/// <summary>/// 上传的文件内容/// </summary>public byte[] FileContent { set; get; }}public static string PostForm(string url, List<FormItemModel> formItems, CookieContainer cookieContainer = null, string refererUrl = null, Encoding encoding = null, int timeOut = 20000){HttpWebRequest request = (HttpWebRequest)WebRequest.Create(url);#region 初始化请求对象request.Method = "POST";request.Timeout = timeOut;request.Accept = "text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8";request.KeepAlive = true;request.UserAgent = "Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/31.0.1650.57 Safari/537.36";if (!string.IsNullOrEmpty(refererUrl))request.Referer = refererUrl;if (cookieContainer != null)request.CookieContainer = cookieContainer;#endregionstring boundary = "----" + DateTime.Now.Ticks.ToString("x");//分隔符request.ContentType = string.Format("multipart/form-data; boundary={0}", boundary);//请求流var postStream = new MemoryStream();#region 处理Form表单请求内容//是否用Form上传文件var formUploadFile = formItems != null && formItems.Count > 0;if (formUploadFile){//文件数据模板string fileFormdataTemplate ="\r\n--" + boundary +"\r\nContent-Disposition: form-data; name=\"{0}\"; filename=\"{1}\"" +"\r\nContent-Type: application/octet-stream" +"\r\n\r\n";//文本数据模板string dataFormdataTemplate ="\r\n--" + boundary +"\r\nContent-Disposition: form-data; name=\"{0}\"" +"\r\n\r\n{1}";foreach (var item in formItems){string formdata = null;if (item.IsFile){//上传文件formdata = string.Format(fileFormdataTemplate,item.Key, //表单键item.FileName);}else{//上传键值对formdata = string.Format(dataFormdataTemplate,item.Key,item.Value);}//统一处理byte[] formdataBytes = null;//第一行不需要换行if (postStream.Length == 0)formdataBytes = Encoding.UTF8.GetBytes(formdata.Substring(2, formdata.Length - 2));elseformdataBytes = Encoding.UTF8.GetBytes(formdata);postStream.Write(formdataBytes, 0, formdataBytes.Length);//写入文件内容if (item.FileContent != null && item.FileContent.Length > 0){postStream.Write(item.FileContent, 0, item.FileContent.Length);}}//结尾var footer = Encoding.UTF8.GetBytes("\r\n--" + boundary + "--\r\n");postStream.Write(footer, 0, footer.Length);}else{request.ContentType = "application/x-www-form-urlencoded";}#endregionrequest.ContentLength = postStream.Length;#region 输入二进制流if (postStream != null){postStream.Position = 0;//直接写入流Stream requestStream = request.GetRequestStream();byte[] buffer = new byte[1024];int bytesRead = 0;while ((bytesRead = postStream.Read(buffer, 0, buffer.Length)) != 0){requestStream.Write(buffer, 0, bytesRead);}postStream.Close();//关闭文件访问}#endregionHttpWebResponse response = (HttpWebResponse)request.GetResponse();if (cookieContainer != null){response.Cookies = cookieContainer.GetCookies(response.ResponseUri);}using (Stream responseStream = response.GetResponseStream()){using (StreamReader myStreamReader = new StreamReader(responseStream, encoding ?? Encoding.UTF8)){string retString = myStreamReader.ReadToEnd();return retString;}}}public static string Send_file(string Url, string media_id){HttpWebRequest req = (HttpWebRequest)HttpWebRequest.Create(Url);req.Method = "POST";req.ContentType = "application/octet-stream";//设置对应的ContentTypestring postdata = "{\"msgtype\": \"file\",\"file\": {\"media_id\":\""+ media_id+"\"}}";byte[] data = Encoding.UTF8.GetBytes(postdata);//把字符串转换为字节Stream writer = req.GetRequestStream();//获取writer.Write(data, 0, data.Length);writer.Flush();HttpWebResponse response = (HttpWebResponse)req.GetResponse();//获取服务器返回的结果Stream getStream = response.GetResponseStream();StreamReader streamreader = new StreamReader(getStream);String result = streamreader.ReadToEnd();return result;}
    
  • 上传文档成功截图

C#、python企业微信群机器人自动发送文件相关推荐

  1. python 企业微信群机器人_企业微信群机器人应用:使用python从网站抓取行业资讯并定时推送...

    在企业经营过程中,及时了解行业相关信息(市场动态.竞品策略.行业数据等等)是非常必要的.通常情况下,商品部门.营销部门.市场部门可能都会安排专门的人员定期进行这些信息的搜集.整理,再进行内部的分享. ...

  2. Python企业微信群机器人推送消息,定时提醒。

    import time import schedule import datetime from WorkWeixinRobot.work_weixin_robot import WWXRobotww ...

  3. div区域内容抓取_企业微信群机器人应用:使用python从网站抓取行业资讯并定时推送...

    在企业经营过程中,及时了解行业相关信息(市场动态.竞品策略.行业数据等等)是非常必要的.通常情况下,商品部门.营销部门.市场部门可能都会安排专门的人员定期进行这些信息的搜集.整理,再进行内部的分享. ...

  4. Python实现企业微信群机器人自动化推送

    人工智能(Artificial Intelligence),英文缩写为AI.它是研究.开发用于模拟.延伸和扩展人的智能的理论.方法.技术及应用系统的一门新的技术科学. --<百度百科> 文 ...

  5. 运用python实现企业微信群机器人消息推送

    使用场景:将BI报表精准推送入(群),精准触达用户 目的:提高管理层对数据的感知度 工具:python+企业微信 步骤: 1.创建企业微信群机器人,提取Webhook地址(群机器人地址) 2.编写代码 ...

  6. 工作随记-Java利用企业微信群机器人定时发送消息

    hi,大家好,我是恰恰 阅读本文需要2分钟~ 最近利用企业微信群机器人做的需求主要有 1.返奖率通知与告警:抽奖箱能抽出垃圾也能抽出大货,每隔5分钟查询一下这个返奖率,如果用户频繁抽出大货,这个抽奖箱 ...

  7. 【SCF CLI实践】腾讯云serverless + 企业微信群机器人,轻松解决告警通知问题

    市面上有什么好用的从服务器推报警和日志的工具?之前私下用的是[Server酱]的服务,非常方便. 但是考虑到安全原因,这个服务如果用在生产环境心里还是有点慌(虽然我相信Server酱是很有节操的). ...

  8. 企业微信机器人脚本python_Python 操控企业微信群机器人

    目标 企业微信群机器人常用来作为通知工具,群发消息给群内成员,充当小助手的角色.但若按照官方 API 文档来构建请求,也确实不太方便.本文通过 Python 第三方库来控制企业微信群机器人发送消息. ...

  9. delphi 企业微信消息机器人_GitHub - guoxianlong/insight: Insight是一个可以管理企业微信群机器人的小工具,可以非常方便的往群里发布即时消息和定时消息。...

    最科幻的企业微信群机器人管理工具 非常方便的发布群即时消息和定时消息,解放双手,提升沟通效率 部署教程 更新日志(2020.05.31) 修复设置为智能跳过工作日时,周日依然提醒问题. 前端修复定时成 ...

最新文章

  1. php扩展mongodb模块安装
  2. services.xml应该放在项目的哪里_行车中手机支架到底应该放在哪里呢?出风口、方向盘、仪表台?...
  3. 让fedora18桌面显示图标
  4. 正则表达式提取器_C++11新特性7 - 正则表达式
  5. 动态SQL和PL/SQL的EXECUTE选项分析
  6. Json学习总结(4)——Json基础知识回顾
  7. 经典OA办公协同管理-原型UI设计-制作
  8. linux eof打印列表,Linux:结合cat和EOF输出到文本文件
  9. WPF基础(八)bitmapImage.EndInit()引发异常 未找到适用于完成此操作的图像处理组件:可能是收发图片格式不一致导致的。
  10. 遥感原理与应用总结——第三章:遥感传感器及成像原理
  11. K折交叉验证大集合(KFold 、Stratified k-fold、Group k-fold、StratifiedGroupKFold)
  12. surface4 笔盖失灵的解决方案
  13. Linux虚拟机怎么越狱,iOS 13永久越狱工具Linux/windows版进展突破,就快来了(附视频)...
  14. java基础,带参的方法进行客户姓名排序
  15. python binascii.b2a_hex_python标准模块介绍- binascii 二进制和ASCII转换
  16. 《高考前夕时间旅行的可行性研究报告》
  17. Web全栈开发1+x(中级)PHPMySQL知识
  18. FCoin回来了?起死回生凭借何种“妙法”?
  19. PyCham中安装包工具报错“Cannot set up a python SDK at Python 3.9”
  20. 科技碰撞:元宇宙与虚幻引擎,被掩盖的底层逻辑何在?

热门文章

  1. 项目日志在项目管理中的应用
  2. vs2013 无法创建项目 终极解决方案
  3. linux无法进入系统修复工具箱,ubuntu启动修复
  4. Push mail同步outlook文件夹设置
  5. Python之列表、矩阵、数组的相互转换
  6. 为什么C语言中int的表示范围是-32768~32767
  7. 用手机软件给ESP8266一键配网
  8. error Expected linebreaks to be ‘LF‘ but found ‘CRLF‘ linebreak-style
  9. CZXX 1+X测试
  10. 加载Flutter Assets中2倍图的一些细节