WinForm调用钉钉获取考勤结果
关注点:
- 1、钉钉AccessToken的获取和防止过期
- 2、使用TPL并行编程调用钉钉接口
需求详解
公司前台有个大屏,领导想显示全部员工的考勤结果统计情况和车间的实时监控视频,还有车间的看板。简单说就是把大屏分割成几个区域。现在遇到的难题是钉钉获取考勤结果的api是只有明细记录,比如你公司1000人,那么可能回给你2000条考勤结果。分别是上班考勤和下班考勤的。没有整个公司的,我就需要这么一条数据就行了。但人家没有这样的接口提供。卷起袖子,干!
趟坑过程
考勤打卡数据开放
业务场景
该接口仅限企业接入使用,用于返回企业内员工的实际打卡结果。比如,企业给一个员工设定的排班是上午9点和下午6点各打一次卡,即使员工在这期间打了多次,该接口也只会返回两条记录,包括上午的打卡结果和下午的打卡结果
考勤打卡数据开放
请求说明(ISV无调用权限)
如果你是ISV(应用服务商,将开发的应用上架到钉钉应用市场,提供给钉钉其他企业用户使用),则无调用权限
如果你是企业内部开发者(将自己公司的HR、OA、客户管理、业务管理等系统接入钉钉),有权限调用
2017-10-16更新:新增用户userId列表参数(userIdList)和分页参数(offset,limit),提升接口稳定性。
Https请求方式: POST
https://oapi.dingtalk.com/attendance/list?access_token=ACCESS_TOKEN
请求包结构体
|
1
2
3
4
5
6
7
|
{ "workDateFrom": "yyyy-MM-dd hh:mm:ss", "workDateTo": "yyyy-MM-dd hh:mm:ss", "userIdList":["员工UserId列表"], // 必填,与offset和limit配合使用,不传表示分页获取全员的数据 "offset":0, // 必填,第一次传0,如果还有多余数据,下次传之前的offset加上limit的值 "limit":1, // 必填,表示数据条数,最大不能超过50条} |
参数说明
| 参数 | 参数类型 | 必须 | 说明 |
|---|---|---|---|
| access_token | String | 是 | 调用接口凭证 |
| workDateFrom | String | 是 | 查询考勤打卡记录的起始工作日 |
| workDateTo | String | 是 | 查询考勤打卡记录的结束工作日 |
| userIdList | List | 是 | 员工在企业内的UserID列表,企业用来唯一标识用户的字段 |
| offset | Long | 是 | 表示获取考勤数据的起始点,第一次传0,如果还有多余数据,下次获取传的offset值为之前的offset+limit |
| limit | Long | 是 | 表示获取考勤数据的条数,最大不能超过50条 |
1)获取AccessToken
钉钉的服务器很牛X可以并行访问的。但是获取回来的数据集有点乱,其中有个关于分页的参数需要顺序读取,就是“hasMore”还有没有下一页。所以最终没有使用TPL。
/// <summary>
/// 获取AccessToken
/// </summary>
/// <returns></returns>
private AccessToken GetAccessToken()
{
string URL_GetToken = "https://oapi.dingtalk.com/gettoken?corpid={0}&corpsecret={1}";
string fileName = AppDomain.CurrentDomain.BaseDirectory + "AccessToken.xml";
if (!System.IO.File.Exists(fileName))
{
string respon = Program.HttpGetRequest(string.Format(URL_GetToken, Program.CorpId, Program.CorpSecret));
DingTalkRespond obj = JsonConvert.DeserializeObject<DingTalkRespond>(respon);
if (obj != null && obj.errcode == )
{
var result = new AccessToken
{
access_token = obj.access_token,
expires_in = GetCurrentTimeStamp()
};
SerializerHelper.SerializerToXML(fileName, result);
return result;
}
else
return new AccessToken
{
access_token = obj.access_token,
expires_in = GetCurrentTimeStamp()
};
}
else
{
var fileResult = SerializerHelper.LoadFromXML<AccessToken>(fileName);
long ts = (GetCurrentTimeStamp() - fileResult.expires_in) / ;
if (ts >= )
{
string respon = Program.HttpGetRequest(string.Format(URL_GetToken, Program.CorpId, Program.CorpSecret));
DingTalkRespond obj = JsonConvert.DeserializeObject<DingTalkRespond>(respon);
if (obj != null && obj.errcode == )
{
fileResult.access_token = obj.access_token;
fileResult.expires_in = GetCurrentTimeStamp();
SerializerHelper.SerializerToXML(fileName, fileResult);
}
}
return fileResult;
}
}
2)获取企业总人数
/// <summary>
/// 获取公司总人数
/// </summary>
/// <returns></returns>
private int GetFactoryEmployeeCount(AccessToken token)
{
int result = ;
string url = string.Format("https://oapi.dingtalk.com/user/get_org_user_count?access_token={0}&onlyActive=0", token.access_token);
string jsonObj = Program.HttpGetRequest(url);
var objResult = JsonConvert.DeserializeObject<DingTalkRespond>(jsonObj);
if (objResult != null && objResult.errcode == )
result = objResult.count;
return result;
}
3)获取全体员工的考勤结果
/// <summary>
/// 获取指定页的考勤结果
/// </summary>
/// <param name="offset"></param>
/// <returns></returns>
private void GetAttendance(AccessToken token)
{
EmployeeAttendanceList.Clear();
string url = string.Format("https://oapi.dingtalk.com/attendance/list?access_token={0}", token.access_token);
int offset = ;
bool bHasMore = true;
while (bHasMore)
{
var request = new AttendanceRequest
{
workDateFrom = DateTime.Now.ToString("yyyy-MM-dd") + " 00:00:00",
workDateTo = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"),
limit = ,
userIdList = null,
offset = offset *
};
string jsonRequest = JsonConvert.SerializeObject(request).Replace("null", "[]");
string jsonResult = Program.PostJsonData(url, jsonRequest);
var objResult = JsonConvert.DeserializeObject<DingTalkRespond>(jsonResult);
if (objResult != null && objResult.errcode == )
{
foreach (var item in objResult.recordresult)
{
if (!EmployeeAttendanceList.Any(a => a.id == item.id))
EmployeeAttendanceList.Add(item);
}
offset++;
bHasMore = objResult.hasMore;
}
else
break;
}
EmployeeAttendanceList = EmployeeAttendanceList.Where(a => a.checkType == "OnDuty").OrderBy(e => e.id).ToList();
}
private void backgroundWorker1_DoWork(object sender, System.ComponentModel.DoWorkEventArgs e)
{
AccessToken token = GetAccessToken();
GetAttendance(token);
var report = from q in EmployeeAttendanceList
group q by q.timeResult into g
select new AttendanceReportItemDTO
{
TimeResult = g.Key,
PersonNumber = g.Count()
};
BulletinBoardInit();
#region 考勤统计
attendanceReport.Total = GetFactoryEmployeeCount(token);
var normalObj = report.Where(p => p.TimeResult == "正常").FirstOrDefault();
attendanceReport.Normal = normalObj == null ? 0 : normalObj.PersonNumber;
var lateObj = report.Where(p => p.TimeResult == "迟到").FirstOrDefault();
attendanceReport.Late = lateObj == null ? 0 : lateObj.PersonNumber;
var earlyObj = report.Where(p => p.TimeResult == "早退").FirstOrDefault();
attendanceReport.Early = earlyObj == null ? 0 : earlyObj.PersonNumber;
var seriousLateObj = report.Where(p => p.TimeResult == "严重迟到").FirstOrDefault();
attendanceReport.SeriousLate = seriousLateObj == null ? 0 : seriousLateObj.PersonNumber;
var absenteeismObj = report.Where(p => p.TimeResult == "旷工").FirstOrDefault();
attendanceReport.Absenteeism = absenteeismObj == null ? 0 : absenteeismObj.PersonNumber;
int total_temp = report.Sum(p => p.PersonNumber);
var notSignedObj = report.Where(p => p.TimeResult == "未打卡").FirstOrDefault();
attendanceReport.NotSigned = notSignedObj == null ? 0 : notSignedObj.PersonNumber;
attendanceReport.NotSigned = attendanceReport.NotSigned + (attendanceReport.Total - total_temp);
#endregion
#region 判断有没有公告信息
if (BulletinImages != null && BulletinImages.Length > 0)
{
if (BulletinImages.Length == 1)
{
pbxNotity.Image = Image.FromFile(BulletinImages[0]);
}
if (BulletinImages.Length > 1)
{
if (notifyShowIndex == BulletinImages.Length - 1)
{
notifyShowIndex = 0;
}
pbxNotity.Image = Image.FromFile(BulletinImages[notifyShowIndex]);
notifyShowIndex++;
}
}
else
{
Action task = () =>
{
BulletinBoardInit();
};
task.BeginInvoke(null, null);
string dir = AppDomain.CurrentDomain.BaseDirectory + "公告栏图片\\";
if (!Directory.Exists(dir))
{
Directory.CreateDirectory(dir);
}
string[] bgFiles = Directory.GetFiles(dir);
if (bgFiles != null && bgFiles.Length > 0)
{
int imgIndex = rd.Next(bgFiles.Length);
pbxNotity.BackgroundImage = Image.FromFile(bgFiles[imgIndex]);
}
}
#endregion
}
效果展示

WinForm调用钉钉获取考勤结果的更多相关文章
- npm钉钉脚手架,支持考勤信息获取
钉钉官方并未提供nodejs包,第一次调用接口的时候非常费事,而且尝试去寻找相关的钉钉考勤数据模块的时候只找到了一些消息啊,只能办公啊,免登啊之类的模块,有关考勤数据的似乎没有 关于dd的npm包中一 ...
- 钉钉开发中post异步调用问题
最近项目上在做钉钉开发中,经常会遇到使用post方式调用钉钉内部的方法(微信也有一样),这里涉及到跨域的post调用,但跨域一般都是用jsonp格式,而这个格式只支持get方式.尝试了挺多方法都没有返 ...
- 钉钉相关功能介入开发系列一:获取access_token
获取access_token的基本代码,与微信不同的是钉钉的token正常情况下有效期为7200秒,有效期内重复获取返回相同结果,并自动续期,比微信方便多了 //基本信息 string appkey ...
- 钉钉开发入门,微应用识别用户身份,获取用户免登授权码code,获取用户userid,获取用户详细信息
最近有个需求,在钉钉内,点击微应用,获取用户身份,根据获取到的用户身份去企业内部的用户中心做校验,校验通过,相关子系统直接登陆; 就是在获取这个用户身份的时候,网上的资料七零八落的,找的人烦躁的很,所 ...
- 使用python对mysql主从进行监控,并调用钉钉发送报警信息
1.编写python的监控脚本 A.通过获取mysql库中的状态值来判断这个mysql主从状态是否正常 B.进行两个状态值的判断 C.进行调取钉钉机器人,发送消息 2.设置定时任务进行脚本运行 cro ...
- .NET平台下,钉钉微应用开发之:获取userid
工作需求,开发钉钉微应用和小程序,之前有接触过支付宝小程序和生活号的开发,流程没有很大的差别,这里记录下我用ASP.NET MVC实现钉钉微应用的开发,并实现获取用户的userid.小弟我技术有限,本 ...
- 钉钉开发第三方H5微应用入门详细教程[ISV][免登流程][授权码][HTTP回调推送][识别用户身份][获取用户信息]
转载请注明原文地址:https://www.cnblogs.com/applerosa/p/11509512.html (by lnexin@aliyun.com 世间草木) 此教程注意点: 适用于第 ...
- 141_Power Query之获取钉钉审批流自动刷新Power BI报告
博客:www.jiaopengzi.com 焦棚子的文章目录 请点击下载附件 一.背景 钉钉办公给很多企业带来了很多方便,比如审批流线上化,通用化.线上化填写后,数据自动获取又是一个硬伤了,虽然数据可 ...
- 131_Power Query之获取钉钉日志自动刷新Power BI报告
博客:www.jiaopengzi.com 焦棚子的文章目录 请点击下载附件 一.背景 最近在玩钉钉日志,企业填写简单数据后方便汇总到一起比较实用的工具,但数据填写以后还是需要下载日志报表,比较麻烦. ...
随机推荐
- DSAPI HTTP监听服务端与客户端_指令版
前面介绍了DSAPI多功能组件编程应用-HTTP监听服务端与客户端的内容,这里介绍一个适用于更高效更快速的基于HTTP监听的服务端.客户端. 在本篇,你将见到前所未有的超简化超傻瓜式的HTTP监听服务 ...
- 使用Common.Logging+log4net规范日志管理【转载】
使用Common.Logging+log4net规范日志管理 Common.Logging+(log4net/NLog/) common logging是一个通用日志接口,log4net是一个强大 ...
- C# 批量删除Word超链接
对于Word文档中包含较多的超链接,如果一个个来删除很花费时间和精力,本篇文章将提供一种可用于批量删除Word中的超链接的方法.这里的超链接可以是页眉页脚处的超链接.正文中的超链接.表格中的超链接.文 ...
- Java开发笔记(四十八)类的简单继承
前面介绍了类的基本用法,主要是如何封装一个类的各项要素,包括成员属性.成员方法.构造方法等,想必大家对类的简单运用早已驾轻就熟.所谓“物以类聚,人以群分”,之所以某些事物会聚在一起,乃是因为它们拥有类 ...
- nodejs 搭建简易服务器
www文件夹下 template文件夹下 server.js代码: const express=require('express'); const static=require('express-st ...
- JavaScript大师必须掌握的12个知识点
既然你对这篇文章感兴趣,我想你应该是一位前端开发,也许你有一份不错的工作.自主创业甚至是一位自由从业者.不知你的前端技术如何,也许你是一位新手,亦或是一位资深开发. 如果你想让自己成为一个 JavaS ...
- JS之This的用法
This的用法 This作为JavaScript中的关键字,在函数中具有四种用法. 一.直接在函数中使用,谁调用这个函数,this就指向谁 例如: var n = "指我"; fu ...
- 免费开源ERP-成功案例分析(2)
Odoo用户案例 Odoo用户概要 关于Odoo全球的用户,我们来看一些数据: Odoo目前全球有300万使用者 Odoo系统上每天新创建的数据库超过1000个 Odoo和Word.Excel.Pow ...
- 《Flask Web开发》学习笔记
第一部分 Flask简介 前言:想熟练掌握一门web框架,为以后即将诞生的测试工具集做准备.为什么选择flask要做熟练掌握的一门框架,而不是其他的,最主要的原因是可以随意定制. 特别提醒:这本书的代 ...
- 2019Java查漏补缺(一)
看到一个总结的知识: 感觉很全面的知识梳理,自己在github上总结了计算机网络笔记就很累了,猜想思维导图的方式一定花费了作者很大的精力,特共享出来.原文:java基础思维导图 自己学习的查漏补缺如下 ...