flexbiz-server 12.5.26 → 12.5.27
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/package.json +1 -1
- package/server/modules/lists/ls-checkin.js +336 -24
package/package.json
CHANGED
|
@@ -1,24 +1,336 @@
|
|
|
1
|
-
const model=global.getModel(
|
|
2
|
-
|
|
3
|
-
const
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
1
|
+
const model = global.getModel('checkin');
|
|
2
|
+
const customer = global.getModel('customer');
|
|
3
|
+
const moment = require("moment");
|
|
4
|
+
const dmnv = global.getModel('dmnv');
|
|
5
|
+
const chamcong = global.getModel('chamcong');
|
|
6
|
+
const User = global.getModel('user');
|
|
7
|
+
const controller = require('../../controllers/controller');
|
|
8
|
+
const PostBook = require('../../libs/post-book');
|
|
9
|
+
|
|
10
|
+
function deg2rad(deg) {
|
|
11
|
+
return deg * (Math.PI / 180);
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
function getDistanceFromLatLonInM(lat1, lon1, lat2, lon2) {
|
|
15
|
+
const R = 6371000; // Bán kính trái đất tính bằng mét
|
|
16
|
+
const dLat = deg2rad(lat2 - lat1);
|
|
17
|
+
const dLon = deg2rad(lon2 - lon1);
|
|
18
|
+
const a =
|
|
19
|
+
Math.sin(dLat / 2) * Math.sin(dLat / 2) +
|
|
20
|
+
Math.cos(deg2rad(lat1)) * Math.cos(deg2rad(lat2)) *
|
|
21
|
+
Math.sin(dLon / 2) * Math.sin(dLon / 2);
|
|
22
|
+
const c = 2 * Math.atan2(Math.sqrt(a), Math.sqrt(1 - a));
|
|
23
|
+
const d = R * c; // Khoảng cách theo mét
|
|
24
|
+
return d;
|
|
25
|
+
}
|
|
26
|
+
const postData = async (obj,callback)=>{
|
|
27
|
+
if(obj.trang_thai!="1") return callback();
|
|
28
|
+
if(global.configs?.required_device_id && !obj.device_id) return callback("Cần thông tin ID của thiết bị (device_id)");
|
|
29
|
+
//Tìm nhân viên theo device_user_id
|
|
30
|
+
const nv = await global.getModel("dmnv").findOne({
|
|
31
|
+
id_app:obj.id_app,
|
|
32
|
+
$or:[
|
|
33
|
+
{device_user_id:obj.device_user_id},
|
|
34
|
+
{user:obj.device_user_id}
|
|
35
|
+
]
|
|
36
|
+
}).lean();
|
|
37
|
+
if(!nv){
|
|
38
|
+
//console.error("Nhân viên không tồn tại",{id_app:obj.id_app,device_user_id:obj.device_user_id})
|
|
39
|
+
return callback(`Chưa khai báo thông tin nhân viên cho '${obj.device_user_id}'.`);
|
|
40
|
+
}
|
|
41
|
+
//Kiểm tra id thiết bị
|
|
42
|
+
if(nv.trusted_device_id){
|
|
43
|
+
if(nv.trusted_device_id!=obj.device_id){
|
|
44
|
+
return callback(`Thiết bị chấm công không hợp lệ. Bạn cần dùng thiết bị đã chấm công lần trước hoặc liên hệ với admin để đăng ký thiết bị mới.`);
|
|
45
|
+
}else{
|
|
46
|
+
nv.trusted_device_id = obj.device_id;
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
//Kiểm tra xem device_id này đã được sử dụng cho nhân viên nào khác không? nếu có thì không cho phép
|
|
50
|
+
if(obj.device_id){
|
|
51
|
+
const nv_device_id = await global.getModel("dmnv").findOne({id_app:obj.id_app,trusted_device_id:obj.device_id},{_id:1}).lean();
|
|
52
|
+
if(nv_device_id && nv_device_id._id!==nv._id){
|
|
53
|
+
return callback(`Thiết bị này đã được sử dụng để chấm cho nhân viên khác. Nếu bạn cần thay đổi hãy liên hệ với admin.`);
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
// Kiểm tra xem dữ liệu gửi lên có toạ độ không
|
|
57
|
+
if (obj.location && obj.location.latitude && obj.location.longitude) {
|
|
58
|
+
//Lấy thông tin chi nhánh của nhân viên đó
|
|
59
|
+
let chiNhanh = null;
|
|
60
|
+
if (nv.ma_cn) {
|
|
61
|
+
chiNhanh = await global.getModel("dmchinhanh").findOne({
|
|
62
|
+
id_app: obj.id_app,
|
|
63
|
+
ma_cn: nv.ma_cn,
|
|
64
|
+
status: true
|
|
65
|
+
}).lean();
|
|
66
|
+
}
|
|
67
|
+
if(!chiNhanh){
|
|
68
|
+
return callback("Chưa khai báo chi nhánh làm việc cho nhân viên này");
|
|
69
|
+
}
|
|
70
|
+
if(!chiNhanh.location || !chiNhanh.location.latitude || !chiNhanh.location.longitude){
|
|
71
|
+
return callback(`Chưa khai báo vị trí của chi nhánh ${chiNhanh.ten_cn}`);
|
|
72
|
+
}
|
|
73
|
+
//Tính khoảng cách
|
|
74
|
+
const distance = getDistanceFromLatLonInM(
|
|
75
|
+
obj.location.latitude,
|
|
76
|
+
obj.location.longitude,
|
|
77
|
+
chiNhanh.location.latitude,
|
|
78
|
+
chiNhanh.location.longitude
|
|
79
|
+
);
|
|
80
|
+
const max_radius = chiNhanh.location.allowed_radius || 50;
|
|
81
|
+
if (distance > max_radius) {
|
|
82
|
+
// Kiểm tra nếu KHÔNG có mã khách hàng (tức là chấm công tại văn phòng nhưng bị xa)
|
|
83
|
+
if (!obj.ma_kh) {
|
|
84
|
+
console.warn(`[CheckIn Fail] Distance: ${distance}m. User: ${obj.device_user_id}`);
|
|
85
|
+
return callback(`Vị trí chấm công không hợp lệ. Bạn đang cách công ty ${Math.round(distance)}m (Cho phép: ${max_radius}m).`);
|
|
86
|
+
}
|
|
87
|
+
// Nếu có obj.ma_kh -> Hợp lệ (đi công tác/gặp khách hàng), code chạy tiếp xuống dưới...
|
|
88
|
+
}
|
|
89
|
+
} else {
|
|
90
|
+
if(global.configs?.required_device_id ) return callback("Không xác định được vị trí của bạn.");
|
|
91
|
+
}
|
|
92
|
+
//Xác định thời gian vào ra
|
|
93
|
+
let cac_lan_cham_cong_trong_ngay = await model.find({
|
|
94
|
+
id_app: obj.id_app,
|
|
95
|
+
device_user_id: obj.device_user_id,
|
|
96
|
+
record_time:{
|
|
97
|
+
$gte:moment(obj.record_time).startOf("date").toDate(),
|
|
98
|
+
$lte:moment(obj.record_time).endOf("date").toDate(),
|
|
99
|
+
}
|
|
100
|
+
}).inTxn().sort({record_time:1}).lean();
|
|
101
|
+
//
|
|
102
|
+
let gio_vao,gio_ra,tong_gio_lam,ma_loai_cong;
|
|
103
|
+
if(cac_lan_cham_cong_trong_ngay.length>0){
|
|
104
|
+
gio_vao = cac_lan_cham_cong_trong_ngay[0].record_time;
|
|
105
|
+
}
|
|
106
|
+
if(cac_lan_cham_cong_trong_ngay.length>1){
|
|
107
|
+
gio_ra = cac_lan_cham_cong_trong_ngay[cac_lan_cham_cong_trong_ngay.length-1].record_time;
|
|
108
|
+
}
|
|
109
|
+
if(gio_vao && gio_ra){
|
|
110
|
+
tong_gio_lam = moment(gio_ra).diff(moment(gio_vao),"hours");
|
|
111
|
+
}else{
|
|
112
|
+
tong_gio_lam =0;
|
|
113
|
+
}
|
|
114
|
+
//Xác định mã loại công dựa vào thời gian làm việc
|
|
115
|
+
if (gio_vao) {
|
|
116
|
+
let ngay_trong_tuan = moment(obj.record_time).day().toString();
|
|
117
|
+
// 1. Query các điều kiện cứng (không dính đến so sánh giờ cụ thể)
|
|
118
|
+
const query_loai_cong = {
|
|
119
|
+
id_app: obj.id_app,
|
|
120
|
+
status: true,
|
|
121
|
+
$and: [
|
|
122
|
+
// Kiểm tra ngày trong tuần
|
|
123
|
+
{
|
|
124
|
+
$or: [
|
|
125
|
+
{ nhung_ngay_trong_tuan: { $exists: false } },
|
|
126
|
+
{ nhung_ngay_trong_tuan: ngay_trong_tuan }
|
|
127
|
+
]
|
|
128
|
+
},
|
|
129
|
+
// Kiểm tra tổng giờ làm tối đa
|
|
130
|
+
{
|
|
131
|
+
$or: [
|
|
132
|
+
{ tong_gio_lam_den: { $gte: tong_gio_lam } },
|
|
133
|
+
{ tong_gio_lam_den: 0 }
|
|
134
|
+
],
|
|
135
|
+
},
|
|
136
|
+
// Kiểm tra tổng giờ làm tối thiểu
|
|
137
|
+
{
|
|
138
|
+
$or: [
|
|
139
|
+
{ tong_gio_lam_tu: { $lte: tong_gio_lam } },
|
|
140
|
+
{ tong_gio_lam_tu: 0 }
|
|
141
|
+
]
|
|
142
|
+
}
|
|
143
|
+
]
|
|
144
|
+
};
|
|
145
|
+
|
|
146
|
+
// 2. Lấy danh sách các ứng viên (Candidates)
|
|
147
|
+
let candidates = await global.getModel("dmloaicong")
|
|
148
|
+
.find(query_loai_cong)
|
|
149
|
+
.sort({ tong_gio_lam_tu: -1, ngay_cong: -1 })
|
|
150
|
+
.lean();
|
|
151
|
+
|
|
152
|
+
// 3. Hàm helper để lấy số phút từ đầu ngày (00:00 -> Time)
|
|
153
|
+
const getMinutesFromMidnight = (dateStr) => {
|
|
154
|
+
if (!dateStr) return null;
|
|
155
|
+
const m = moment(dateStr);
|
|
156
|
+
return m.hours() * 60 + m.minutes();
|
|
157
|
+
};
|
|
158
|
+
|
|
159
|
+
const input_vao_mins = getMinutesFromMidnight(gio_vao);
|
|
160
|
+
const input_ra_mins = getMinutesFromMidnight(gio_ra || gio_vao);
|
|
161
|
+
|
|
162
|
+
// 4. Lọc lại bằng JS để so sánh giờ vào/ra (bỏ qua ngày)
|
|
163
|
+
let loaicong = candidates.find(lc => {
|
|
164
|
+
// Check giờ vào tối thiểu (DB >= Input)
|
|
165
|
+
// Nếu DB null thì pass, nếu có dữ liệu thì phải thỏa mãn
|
|
166
|
+
if (lc.gio_vao_toi_thieu) {
|
|
167
|
+
const db_vao_mins = getMinutesFromMidnight(lc.gio_vao_toi_thieu);
|
|
168
|
+
if (db_vao_mins < input_vao_mins) return false; // Logic của bạn: DB >= Input
|
|
169
|
+
}
|
|
170
|
+
|
|
171
|
+
// Check giờ ra tối thiểu (DB <= Input)
|
|
172
|
+
if (lc.gio_ra_toi_thieu) {
|
|
173
|
+
const db_ra_mins = getMinutesFromMidnight(lc.gio_ra_toi_thieu);
|
|
174
|
+
if (db_ra_mins > input_ra_mins) return false; // Logic của bạn: DB <= Input
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
return true;
|
|
178
|
+
});
|
|
179
|
+
|
|
180
|
+
// console.log("[checkin] tim loai cong", query_loai_cong, loaicong);
|
|
181
|
+
if (loaicong) {
|
|
182
|
+
ma_loai_cong = loaicong.ma_loai_cong;
|
|
183
|
+
}
|
|
184
|
+
}
|
|
185
|
+
//Tìm bộ phận của nhân viên
|
|
186
|
+
let ma_bp = nv.ma_bp;
|
|
187
|
+
if(nv.bo_phan && nv.bo_phan.length>0){
|
|
188
|
+
ma_bp = nv.bo_phan[0].ma_bp;
|
|
189
|
+
}
|
|
190
|
+
//Tạo công
|
|
191
|
+
let data = obj.toObject?obj.toObject():{...obj};
|
|
192
|
+
data.id_checkin = obj.device_user_id;
|
|
193
|
+
data.ngay = obj.record_time;
|
|
194
|
+
data.ma_loai_cong = ma_loai_cong||"UNKNOW";
|
|
195
|
+
data.ma_nv =nv.ma_nv;
|
|
196
|
+
data.ma_bp =ma_bp;
|
|
197
|
+
delete data._id;
|
|
198
|
+
delete data.trang_thai;
|
|
199
|
+
data.gio_vao = gio_vao;
|
|
200
|
+
data.gio_ra = gio_ra;
|
|
201
|
+
data.tong_gio_lam = tong_gio_lam;
|
|
202
|
+
//xoá post cũ
|
|
203
|
+
await chamcong.deleteMany({
|
|
204
|
+
id_app:obj.id_app,
|
|
205
|
+
id_checkin:obj.device_user_id,
|
|
206
|
+
ngay:{
|
|
207
|
+
$gte:moment(obj.record_time).startOf("date").toDate(),
|
|
208
|
+
$lte:moment(obj.record_time).endOf("date").toDate()}}
|
|
209
|
+
)
|
|
210
|
+
//tạo post mới
|
|
211
|
+
const postbangchamcong = new PostBook(obj, [data], chamcong);
|
|
212
|
+
postbangchamcong.run(async (e, rs)=>{
|
|
213
|
+
if(!e){
|
|
214
|
+
//lưu trusted_device_id
|
|
215
|
+
await global.getModel("dmnv").updateOne({_id:nv._id},{trusted_device_id:obj.device_id})
|
|
216
|
+
}
|
|
217
|
+
callback(e, rs);
|
|
218
|
+
});
|
|
219
|
+
|
|
220
|
+
|
|
221
|
+
}
|
|
222
|
+
module.exports = function(router) {
|
|
223
|
+
const contr = new controller(router, model, 'checkin', {
|
|
224
|
+
notNeedRight:true,
|
|
225
|
+
sort: {
|
|
226
|
+
record_time: -1
|
|
227
|
+
},
|
|
228
|
+
onView:async(user,items,next)=>{
|
|
229
|
+
//Cham cong
|
|
230
|
+
await items.filter(item=>item.trang_thai==="1").asyncJoinModel2(user.current_id_app,dmnv,{
|
|
231
|
+
where: (item=>{
|
|
232
|
+
return {$or: [
|
|
233
|
+
{device_user_id:item.device_user_id},
|
|
234
|
+
{user:item.device_user_id}
|
|
235
|
+
]}
|
|
236
|
+
}),
|
|
237
|
+
fields:["ma_nv","ten_nv","picture"]}
|
|
238
|
+
);
|
|
239
|
+
//Khach hang
|
|
240
|
+
await items.filter(item=>item.trang_thai==="2" || item.trang_thai==="3").asyncJoinModel2(user.current_id_app,customer,{
|
|
241
|
+
where: (item=>{
|
|
242
|
+
return {$or: [
|
|
243
|
+
{device_user_id:item.device_user_id},
|
|
244
|
+
{of_user:item.device_user_id}
|
|
245
|
+
]}
|
|
246
|
+
}),
|
|
247
|
+
fields:["ma_kh","ten_kh","picture"]}
|
|
248
|
+
);
|
|
249
|
+
//Ca khach hang va nhan vien
|
|
250
|
+
await items.filter(item=>item.trang_thai==="3" && item.device_user_id2).asyncJoinModel2(user.current_id_app,dmnv,{where:{device_user_id2:"device_user_id"},fields:["ma_nv","ten_nv",{picture_nv:"picture"}]});
|
|
251
|
+
//Get picture
|
|
252
|
+
await items.filter(item=>!item.picture).asyncJoinModel2(null,User,{where:{of_user:"email"},fields:["picture"]});
|
|
253
|
+
await items.filter(item=>!item.picture).asyncJoinModel2(null,User,{where:{user:"email"},fields:["picture"]});
|
|
254
|
+
await items.filter(item=>item.ma_kh).asyncJoinModel2(user.current_id_app,"customer",{where:"ma_kh",fields:["ten_kh"]});
|
|
255
|
+
next(null,items);
|
|
256
|
+
},
|
|
257
|
+
onCreating: async (user,obj,next)=>{
|
|
258
|
+
if(global.configs?.required_device_id && obj.location) obj.record_time = new Date();
|
|
259
|
+
await chamcong.deleteMany({
|
|
260
|
+
id_app:obj.id_app,
|
|
261
|
+
id_checkin:obj.device_user_id||user.email,
|
|
262
|
+
ngay:{
|
|
263
|
+
$gte:moment(obj.record_time).startOf("date").toDate(),
|
|
264
|
+
$lte:moment(obj.record_time).endOf("date").toDate()
|
|
265
|
+
}})
|
|
266
|
+
next(null,obj)
|
|
267
|
+
},
|
|
268
|
+
onUpdating:async (user,data,obj,next)=>{
|
|
269
|
+
data.device_user_id = obj.device_user_id;
|
|
270
|
+
delete data.record_time;
|
|
271
|
+
|
|
272
|
+
await chamcong.deleteMany({
|
|
273
|
+
id_app:obj.id_app,
|
|
274
|
+
id_checkin:obj.device_user_id,
|
|
275
|
+
ngay:{
|
|
276
|
+
$gte:moment(obj.record_time).startOf("date").toDate(),
|
|
277
|
+
$lte:moment(obj.record_time).endOf("date").toDate()
|
|
278
|
+
}}
|
|
279
|
+
)
|
|
280
|
+
next(null,data,obj)
|
|
281
|
+
},
|
|
282
|
+
onDeleting: async (user,obj,next)=>{
|
|
283
|
+
await chamcong.deleteMany({
|
|
284
|
+
id_app:obj.id_app,
|
|
285
|
+
id_checkin:obj.device_user_id,
|
|
286
|
+
ngay:{
|
|
287
|
+
$gte:moment(obj.record_time).startOf("date").toDate(),
|
|
288
|
+
$lte:moment(obj.record_time).endOf("date").toDate()
|
|
289
|
+
}})
|
|
290
|
+
next(null,obj)
|
|
291
|
+
},
|
|
292
|
+
onUpdated:async (user,obj,next)=>{
|
|
293
|
+
postData(obj,async (e)=>{
|
|
294
|
+
if(e){
|
|
295
|
+
console.log("lỗi lý checkin...");
|
|
296
|
+
await model.deleteOne({_id:obj._id})
|
|
297
|
+
return next(e);
|
|
298
|
+
}
|
|
299
|
+
next(null,obj);
|
|
300
|
+
})
|
|
301
|
+
},
|
|
302
|
+
onDeleted:async (user,obj,next)=>{
|
|
303
|
+
let query_checkin = {
|
|
304
|
+
id_app:obj.id_app,
|
|
305
|
+
device_user_id:obj.device_user_id,
|
|
306
|
+
record_time:{
|
|
307
|
+
$gte:moment(obj.record_time).startOf("date").toDate(),
|
|
308
|
+
$lte:moment(obj.record_time).endOf("date").toDate()
|
|
309
|
+
}
|
|
310
|
+
}
|
|
311
|
+
|
|
312
|
+
let obj_post = await model.findOne(query_checkin).lean();
|
|
313
|
+
//console.log("[checkin] checkin đã bị xoá. Tìm checkin khác cùng ngày để tạo lại chấm công",query_checkin,obj_post);
|
|
314
|
+
if(obj_post){
|
|
315
|
+
//post lại sau khi xoá checkin
|
|
316
|
+
//console.log("[checkin] tạo lại chấm công",obj_post)
|
|
317
|
+
postData(obj_post,(e)=>{
|
|
318
|
+
if(e) return next(e);
|
|
319
|
+
next(null,obj);
|
|
320
|
+
})
|
|
321
|
+
}else{
|
|
322
|
+
next(null,obj);
|
|
323
|
+
}
|
|
324
|
+
},
|
|
325
|
+
onCreated:async (user,obj,next)=>{
|
|
326
|
+
postData(obj,async (e)=>{
|
|
327
|
+
if(e){
|
|
328
|
+
await model.deleteOne({_id:obj._id});
|
|
329
|
+
return next(e);
|
|
330
|
+
}
|
|
331
|
+
next(null,obj);
|
|
332
|
+
})
|
|
333
|
+
},
|
|
334
|
+
});
|
|
335
|
+
contr.route();
|
|
336
|
+
};
|