flexbiz-server 12.5.5 → 12.5.6

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