flexbiz-server 12.5.26 → 12.5.28
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/libs/post-sokho.js +351 -25
- package/server/modules/lists/ls-checkin.js +336 -24
package/package.json
CHANGED
|
@@ -1,25 +1,351 @@
|
|
|
1
|
-
const async=require("async")
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
1
|
+
const async = require("async");
|
|
2
|
+
const utils = require("./utils");
|
|
3
|
+
const tontucthoi = global.getModel("tontucthoi");
|
|
4
|
+
const Dmvt = global.getModel("dmvt");
|
|
5
|
+
const round = 4;
|
|
6
|
+
const tinhgiatb = require("../libs/tinhgiatb1vt");
|
|
7
|
+
const moment = require("moment");
|
|
8
|
+
const dinhmucsx = async (id_app,ma_sp,ma_tt1,sl_xuat,ma_dvt)=>{
|
|
9
|
+
let vts=[];
|
|
10
|
+
let dm = await global.getModel("dinhmucsx").findOne({id_app,ma_sp,ma_tt1:ma_tt1||undefined},{details:1}).lean();
|
|
11
|
+
if(!dm || !dm.details || dm.details.length===0) return vts;
|
|
12
|
+
dm.details = dm.details.map(d=>{
|
|
13
|
+
if(d.toObject) d = d.toObject();
|
|
14
|
+
return d;
|
|
15
|
+
})
|
|
16
|
+
//tinh so luong san pham quy doi
|
|
17
|
+
let he_so_qd=1;
|
|
18
|
+
let qd = await global.getModel("dmqddvt").findOne({id_app,ma_vt:ma_sp,ma_dvt}).lean();
|
|
19
|
+
if(qd) he_so_qd = (qd.mau?(qd.tu/qd.mau): qd.ty_le_qd);
|
|
20
|
+
let sl_xuat_qd = sl_xuat * he_so_qd;
|
|
21
|
+
//
|
|
22
|
+
await dm.details.asyncJoinModel2(id_app,Dmvt,{where:"ma_vt",setFields:(item,ref)=>{
|
|
23
|
+
if(!item.tk_vt) item.tk_vt = ref.tk_vt;
|
|
24
|
+
if(!item.tk_gv) item.tk_gv = ref.tk_gv;
|
|
25
|
+
item.tg_tk = ref.tg_tk;
|
|
26
|
+
item.xk_nvl = ref.xk_nvl;
|
|
27
|
+
}});
|
|
28
|
+
for(let i=0;i<dm.details.length;i++){
|
|
29
|
+
let vt = dm.details[i];
|
|
30
|
+
let sl = 0;
|
|
31
|
+
if(vt.loai_dinh_muc==1){
|
|
32
|
+
sl = vt.sl_dinh_muc * sl_xuat_qd * (1+(vt.ty_le_hao_hut||0)/100);
|
|
33
|
+
}else{
|
|
34
|
+
sl = vt.sl_dinh_muc * (1+vt.ty_le_hao_hut/100);
|
|
35
|
+
}
|
|
36
|
+
if(vt.xk_nvl){
|
|
37
|
+
let dm_of_vt = await dinhmucsx(id_app,vt.ma_vt,vt.ma_tt1||undefined,sl,vt.ma_dvt);
|
|
38
|
+
//console.log("dinh muc cua ban thanh pham",vt,dm_of_vt);
|
|
39
|
+
vts = [...vts,...dm_of_vt];
|
|
40
|
+
}else{
|
|
41
|
+
vt.sl_xuat = sl;
|
|
42
|
+
vts.push(vt);
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
return vts;
|
|
46
|
+
|
|
47
|
+
}
|
|
48
|
+
const post = function(master, details, prepare,options={}) {
|
|
49
|
+
//
|
|
50
|
+
if (!details) {
|
|
51
|
+
details =[]
|
|
52
|
+
}
|
|
53
|
+
//console.log("details",details)
|
|
54
|
+
//data
|
|
55
|
+
this.master = {...master}
|
|
56
|
+
this.details = details.map(detail=>{
|
|
57
|
+
return {...detail};
|
|
58
|
+
});
|
|
59
|
+
//prepare function
|
|
60
|
+
this.prepare = prepare;
|
|
61
|
+
this.options = options;
|
|
62
|
+
};
|
|
63
|
+
post.prototype.runPrepare = async (prepareFunction,details)=>{
|
|
64
|
+
details = details.map(d=>{
|
|
65
|
+
if(d.toObject) d = d.toObject();
|
|
66
|
+
return {...d}
|
|
67
|
+
})
|
|
68
|
+
return (new Promise(resolve=>{
|
|
69
|
+
if (prepareFunction) {
|
|
70
|
+
async.map(details,(detail,callback)=>{
|
|
71
|
+
setImmediate(()=>{
|
|
72
|
+
detail.toObject = function(){
|
|
73
|
+
return Object.assign({},detail)
|
|
74
|
+
}
|
|
75
|
+
prepareFunction(detail,(detail)=>{
|
|
76
|
+
detail.toObject = function(){
|
|
77
|
+
return Object.assign({},detail)
|
|
78
|
+
}
|
|
79
|
+
callback(null,detail);
|
|
80
|
+
})
|
|
81
|
+
})
|
|
82
|
+
|
|
83
|
+
},(e,details)=>{
|
|
84
|
+
resolve(details);
|
|
85
|
+
})
|
|
86
|
+
}else{
|
|
87
|
+
resolve(details);
|
|
88
|
+
}
|
|
89
|
+
}))
|
|
90
|
+
}
|
|
91
|
+
post.prototype.remove = function(callback = ()=>{}){
|
|
92
|
+
let master = this.master;
|
|
93
|
+
if(master && master.toObject) master = master.toObject();
|
|
94
|
+
if(!master._id) return callback();
|
|
95
|
+
let query_delete = {id_ct: master._id.toString(),id_app:master.id_app};
|
|
96
|
+
if(this.options.queryDelete){
|
|
97
|
+
query_delete = {...query_delete,...this.options.queryDelete,id_app:master.id_app}
|
|
98
|
+
}
|
|
99
|
+
async.series({
|
|
100
|
+
sks:(callback)=>{
|
|
101
|
+
setImmediate(()=>{
|
|
102
|
+
global.getModel("sokho").find({id_ct:master._id.toString()}).lean().then(function(sks){
|
|
103
|
+
callback(null,sks);
|
|
104
|
+
}).catch(e=>{
|
|
105
|
+
callback(e);
|
|
106
|
+
})
|
|
107
|
+
})
|
|
108
|
+
|
|
109
|
+
}
|
|
110
|
+
},async (e,rs)=>{
|
|
111
|
+
if(e) return callback(e);
|
|
112
|
+
//delete sokho congty
|
|
113
|
+
await global.getModel("sokho").deleteMany(query_delete);
|
|
114
|
+
//delete sokhonpp
|
|
115
|
+
await global.getModel("sokhonpp").deleteMany(query_delete);
|
|
116
|
+
//delete sokhokhongton
|
|
117
|
+
await global.getModel("sokhokhongton").deleteMany(query_delete);
|
|
118
|
+
//Tính tồn kho tức thời
|
|
119
|
+
if(rs.sks.length>0){
|
|
120
|
+
//tinh lai ton kho tuc thoi khi xoa sokho
|
|
121
|
+
tontucthoi.tinhTonTucThoi(rs.sks,(e)=>{
|
|
122
|
+
if(e) console.error("[post-sokho] Tính tồn tức thời sau khi xoá sokho của vật tư",e);
|
|
123
|
+
})
|
|
124
|
+
}
|
|
125
|
+
callback();
|
|
126
|
+
})
|
|
127
|
+
}
|
|
128
|
+
post.prototype.run = async function(callback=()=>{}) {
|
|
129
|
+
if (!this.details) {
|
|
130
|
+
console.error("details not exists")
|
|
131
|
+
return callback(null);
|
|
132
|
+
}
|
|
133
|
+
if (!this.master || !this.master._id) {
|
|
134
|
+
console.error("masters not exists",this.master)
|
|
135
|
+
return callback(null);
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
let master = this.master;
|
|
139
|
+
if(master.toObject) master = master.toObject();
|
|
140
|
+
|
|
141
|
+
let details = await this.runPrepare(this.prepare,this.details);
|
|
142
|
+
await details.asyncJoinModel2(master.id_app,Dmvt,{where:"ma_vt",setFields:(item,ref)=>{
|
|
143
|
+
if(!item.tk_vt) item.tk_vt = ref.tk_vt;
|
|
144
|
+
if(!item.tk_gv) item.tk_gv = ref.tk_gv;
|
|
145
|
+
item.tg_tk = ref.tg_tk;
|
|
146
|
+
item.xk_nvl = ref.xk_nvl;
|
|
147
|
+
}});
|
|
148
|
+
//kiểm tra kho bị khoá. loại trừ phiếu kiểm kê
|
|
149
|
+
if((master.ma_ct||"").toUpperCase()!=="PKK"){
|
|
150
|
+
let ds_kho = [...new Set(details.map(d=>d.ma_kho||master.ma_kho))]
|
|
151
|
+
const kho_locked = await global.getModel('dmkho').findOne({id_app:master.id_app,lock:true,ma_kho:{$in:ds_kho}}).lean();
|
|
152
|
+
if(kho_locked) return callback(`Kho ${kho_locked.ten_kho} đã bị khoá`);
|
|
153
|
+
}
|
|
154
|
+
//dinh muc san pham
|
|
155
|
+
let sp_xk_nvl = details.filter(p=>p.xk_nvl && p.nxt==2);
|
|
156
|
+
for(let i =0;i<sp_xk_nvl.length;i++){
|
|
157
|
+
sp_xk_nvl[i].dinh_muc_nvl = await dinhmucsx(master.id_app,sp_xk_nvl[i].ma_vt,sp_xk_nvl[i].ma_tt1,sp_xk_nvl[i].sl_xuat,sp_xk_nvl[i].ma_dvt);
|
|
158
|
+
}
|
|
159
|
+
//
|
|
160
|
+
details = details.filter(d=>d.tg_tk || (d.dinh_muc_nvl && d.dinh_muc_nvl.length>0));
|
|
161
|
+
const sp_co_dm = details.filter(p=>p.xk_nvl && p.dinh_muc_nvl && p.dinh_muc_nvl.length>0);
|
|
162
|
+
for(let i =0;i<sp_co_dm.length;i++){
|
|
163
|
+
//Xác định nguyên vật liệu cho sản phẩm
|
|
164
|
+
const p = sp_co_dm[i];
|
|
165
|
+
//Tính số lượng sản phẩm sản xuất (quy đổi sang đơn vị tính chuẩn)
|
|
166
|
+
let he_so_qd=1;
|
|
167
|
+
let qd = await global.getModel("dmqddvt").findOne({id_app:master.id_app,ma_vt:p.ma_vt,ma_dvt:p.ma_dvt}).lean();
|
|
168
|
+
if(qd) he_so_qd = (qd.mau?(qd.tu/qd.mau): qd.ty_le_qd);
|
|
169
|
+
let sl_sp = p.sl_xuat * he_so_qd;
|
|
170
|
+
//
|
|
171
|
+
p.dinh_muc_nvl = p.dinh_muc_nvl.map(d=>{
|
|
172
|
+
d.nxt = p.nxt;
|
|
173
|
+
d.ma_sp = p.ma_vt;
|
|
174
|
+
d.sl_sp = sl_sp;
|
|
175
|
+
|
|
176
|
+
d.ma_tt1_sp = p.ma_tt1;
|
|
177
|
+
d.ma_tt2_sp = p.ma_tt2;
|
|
178
|
+
d.ma_tt3_sp = p.ma_tt3;
|
|
179
|
+
d.ma_kho = p.ma_kho;
|
|
180
|
+
/*if(d.loai_dinh_muc==1){
|
|
181
|
+
d.sl_xuat = d.sl_dinh_muc * p.sl_xuat * (1+(d.ty_le_hao_hut||0)/100);
|
|
182
|
+
}else{
|
|
183
|
+
d.sl_xuat = d.sl_dinh_muc * (1+p.ty_le_hao_hut/100);
|
|
184
|
+
}*/
|
|
185
|
+
return d;
|
|
186
|
+
})
|
|
187
|
+
/*await p.dinh_muc_nvl.asyncJoinModel2(master.id_app,Dmvt,{where:"ma_vt",setFields:(item,ref)=>{
|
|
188
|
+
if(!item.tk_vt) item.tk_vt = ref.tk_vt;
|
|
189
|
+
if(!item.tk_gv) item.tk_gv = ref.tk_gv;
|
|
190
|
+
item.tg_tk = ref.tg_tk;
|
|
191
|
+
item.xk_nvl = ref.xk_nvl;
|
|
192
|
+
}});*/
|
|
193
|
+
p.dinh_muc_nvl = p.dinh_muc_nvl.filter(p=>p.tg_tk);
|
|
194
|
+
//dinh muc ban thanh pham
|
|
195
|
+
}
|
|
196
|
+
//
|
|
197
|
+
let ps_kg_dm = details.filter(p=>!p.xk_nvl || !p.dinh_muc_nvl || p.dinh_muc_nvl.length===0);
|
|
198
|
+
let ps_dm = details.filter(p=>p.xk_nvl && p.dinh_muc_nvl && p.dinh_muc_nvl.length>0).map(p=>p.dinh_muc_nvl).reduce((a,b)=>a.concat(b),[]);
|
|
199
|
+
//
|
|
200
|
+
details = [...ps_kg_dm,...ps_dm];
|
|
201
|
+
//
|
|
202
|
+
await details.asyncJoinModel2(master.id_app,global.getModel("dmqddvt"),{where:{ma_vt:"ma_vt",ma_dvt:"ma_dvt"},fields:["tu","mau",{he_so_qd:"ty_le_qd"}]});
|
|
203
|
+
details.forEach((detail)=>{
|
|
204
|
+
detail.ma_kho = detail.ma_kho || master.ma_kho;
|
|
205
|
+
detail.he_so_qd = detail.he_so_qd || 1;
|
|
206
|
+
if(detail.mau && detail.tu){
|
|
207
|
+
detail.he_so_qd = detail.tu/detail.mau;
|
|
208
|
+
detail.sl_xuat_qd = utils.round((detail.sl_xuat||0) * detail.tu/detail.mau,round);
|
|
209
|
+
detail.sl_nhap_qd = utils.round((detail.sl_nhap||0) * detail.tu/detail.mau,round);
|
|
210
|
+
}else{
|
|
211
|
+
detail.sl_xuat_qd =utils.round( (detail.sl_xuat||0) * detail.he_so_qd,round);
|
|
212
|
+
detail.sl_nhap_qd =utils.round( (detail.sl_nhap||0) * detail.he_so_qd,round);
|
|
213
|
+
}
|
|
214
|
+
})
|
|
215
|
+
//kiểm tra các kho bị khoá
|
|
216
|
+
|
|
217
|
+
//tu dong xac dinh gia von
|
|
218
|
+
let nam = new Date(master.ngay_ct).getFullYear();
|
|
219
|
+
let thang = new Date(master.ngay_ct).getMonth() + 1;
|
|
220
|
+
await new Promise((rs)=>{
|
|
221
|
+
async.map( details.filter(p=>p.tg_tk && ((p.sl_xuat && !p.px_gia_dd) || (p.sl_nhap && p.pn_gia_tb)) && p.sl_xuat),(p,callback)=>{
|
|
222
|
+
setImmediate(async ()=>{
|
|
223
|
+
//xac dinh he so quy doi
|
|
224
|
+
let he_so_qd=1;
|
|
225
|
+
let qd = await global.getModel("dmqddvt").findOne({id_app:master.id_app,ma_vt:p.ma_vt,ma_dvt:p.ma_dvt}).lean();
|
|
226
|
+
if(qd) he_so_qd = (qd.mau?(qd.tu/qd.mau): qd.ty_le_qd);
|
|
227
|
+
//tim gia tb
|
|
228
|
+
let query = {ma_vt:p.ma_vt,nam,thang,ma_kho:{$in:[p.ma_kho,"",null]}}
|
|
229
|
+
let giatb = await global.getModel("giatb").findOne(query).sort({ma_kho:-1}).lean();
|
|
230
|
+
if(giatb){
|
|
231
|
+
p.gia_von_nt = p.gia_von = (giatb.gia||0) * he_so_qd;
|
|
232
|
+
p.tien_xuat_nt = p.tien_xuat = (p.sl_xuat||0) * p.gia_von_nt;
|
|
233
|
+
p.tien_nhap_nt = p.tien_nhap = (p.sl_nhap||0) * p.gia_von_nt;
|
|
234
|
+
callback();
|
|
235
|
+
}else{
|
|
236
|
+
let condition = {id_app:master.id_app,ma_vt:p.ma_vt,ma_kho:p.ma_kho};
|
|
237
|
+
condition.tu_ngay = moment(master.ngay_ct).startOf("month").toDate();
|
|
238
|
+
condition.den_ngay = moment(master.ngay_ct).endOf("month").toDate();
|
|
239
|
+
tinhgiatb(condition,(e,giatb)=>{
|
|
240
|
+
if(e) console.error("[postSokho] Không thể tính giá trung bình",e.message||e.error||e,condition);
|
|
241
|
+
if(rs){
|
|
242
|
+
p.gia_von_nt = p.gia_von = (giatb.gia||0) * he_so_qd;
|
|
243
|
+
}else{
|
|
244
|
+
p.gia_von_nt = p.gia_von =0;
|
|
245
|
+
}
|
|
246
|
+
p.tien_xuat_nt = p.tien_xuat = (p.sl_xuat||0) * p.gia_von_nt;
|
|
247
|
+
p.tien_nhap_nt = p.tien_nhap = (p.sl_nhap||0) * p.gia_von_nt;
|
|
248
|
+
callback();
|
|
249
|
+
})
|
|
250
|
+
}
|
|
251
|
+
})
|
|
252
|
+
},()=>{
|
|
253
|
+
rs();
|
|
254
|
+
})
|
|
255
|
+
})
|
|
256
|
+
//sync master and detail
|
|
257
|
+
details = details.map((detail,line)=>{
|
|
258
|
+
detail.id_ct = master._id.toString();
|
|
259
|
+
detail.ma_ct = master.ma_ct;
|
|
260
|
+
detail.so_ct = master.so_ct;
|
|
261
|
+
detail.ngay_ct = master.ngay_ct;
|
|
262
|
+
for (let attr in global.getModel("sokho").schema.paths) {
|
|
263
|
+
if (attr != 'id_ct' && attr != '_id' && attr != '__v' && attr != 'date_updated' && attr != 'date_created') {
|
|
264
|
+
let v = detail[attr];
|
|
265
|
+
if((v===undefined || v===null || v==="") && v!==0 && (master[attr]!==undefined)){
|
|
266
|
+
detail[attr] = master[attr]
|
|
267
|
+
}
|
|
268
|
+
}
|
|
269
|
+
}
|
|
270
|
+
detail.line = line;
|
|
271
|
+
delete detail.toObject;
|
|
272
|
+
delete detail._id;
|
|
273
|
+
delete detail.__v;
|
|
274
|
+
return detail;
|
|
275
|
+
})
|
|
276
|
+
//Xác định loại kho
|
|
277
|
+
await details.asyncJoinModel2(master.id_app,"dmkho",{where:"ma_kho",fields:"la_kho_npp"});
|
|
278
|
+
//xoá dư liệu cũ
|
|
279
|
+
const self = this;
|
|
280
|
+
self.remove(async (error)=> {
|
|
281
|
+
if (error){
|
|
282
|
+
console.error("[postSokho] error delete data post",error.message||error.error||error,master.so_ct,master.ma_ct)
|
|
283
|
+
return callback(error)
|
|
284
|
+
}
|
|
285
|
+
if(details.length==0) return callback(null,[]);
|
|
286
|
+
try{
|
|
287
|
+
//post sokho
|
|
288
|
+
let data_sokho = details.filter(d=>d.tg_tk && !d.la_kho_npp);
|
|
289
|
+
let rs_sokho = await global.getModel("sokho").create(data_sokho);
|
|
290
|
+
//console.log("post sokho",rs_sokho);
|
|
291
|
+
//post sokhonpp
|
|
292
|
+
let data_sokhonpp = details.filter(d=>d.tg_tk && d.la_kho_npp);
|
|
293
|
+
let rs_sokhnpp = await global.getModel("sokhonpp").create(data_sokhonpp);
|
|
294
|
+
//post sokhokhongton
|
|
295
|
+
let data_sokhokhongton = details.filter(d=>d.tg_tk==false);
|
|
296
|
+
let rs_sokhokhongton = await global.getModel("sokhokhongton").create(data_sokhokhongton);
|
|
297
|
+
/*if(master.ma_ct){
|
|
298
|
+
let contr = global.controllers[master.ma_ct.toUpperCase()];
|
|
299
|
+
if(contr){
|
|
300
|
+
let ct = await contr.model.findById(master._id).inTxt();
|
|
301
|
+
if(!ct){
|
|
302
|
+
console.error("[postSokho] Chứng từ không tồn tại. Xóa sổ sách",master._id,master.so_ct,master.ma_ct);
|
|
303
|
+
self.remove(()=> {
|
|
304
|
+
callback(`Chứng từ không tồn tại`);
|
|
305
|
+
})
|
|
306
|
+
}else{
|
|
307
|
+
callback(null,{rs_sokho,rs_sokhnpp,rs_sokhokhongton});
|
|
308
|
+
}
|
|
309
|
+
}else{
|
|
310
|
+
console.log("[postSokho] not found controller",master.ma_ct.toUpperCase());
|
|
311
|
+
callback(null,{rs_sokho,rs_sokhnpp,rs_sokhokhongton});
|
|
312
|
+
}
|
|
313
|
+
}else{
|
|
314
|
+
callback(null,{rs_sokho,rs_sokhnpp,rs_sokhokhongton});
|
|
315
|
+
}*/
|
|
316
|
+
callback(null,{rs_sokho,rs_sokhnpp,rs_sokhokhongton});
|
|
317
|
+
//tinh ton kho tuc thoi
|
|
318
|
+
if(rs_sokho){
|
|
319
|
+
setImmediate(()=>{
|
|
320
|
+
tontucthoi.tinhTonTucThoi(rs_sokho,(e)=>{
|
|
321
|
+
if(e) console.error("Lỗi tính tồn tức thời vật tư",rs_sokho,e.message||e.error||e,master.so_ct,master.ma_ct)
|
|
322
|
+
})
|
|
323
|
+
})
|
|
324
|
+
}
|
|
325
|
+
}catch(e){
|
|
326
|
+
console.error("[postSokho] Đã có lỗi khi post sokho",e);
|
|
327
|
+
callback(e);
|
|
328
|
+
/*let msg_error = e.message || e.error || e.toString();
|
|
329
|
+
if(msg_error.indexOf("E11000 duplicate")>=0 && msg_error.indexOf("id_ct_1_line_1")>=0){
|
|
330
|
+
let rs_sokho = await global.getModel("sokho").find({id_ct: master._id.toString()});
|
|
331
|
+
let rs_sokhnpp = await global.getModel("sokhonpp").find({id_ct: master._id.toString()});
|
|
332
|
+
let rs_sokhokhongton = await global.getModel("sokhokhongton").find({id_ct: master._id.toString()});
|
|
333
|
+
if(rs_sokho.length>0 || rs_sokhnpp.length>0 || rs_sokhokhongton.length>0 ){
|
|
334
|
+
callback(null, {rs_sokho,rs_sokhnpp,rs_sokhokhongton});
|
|
335
|
+
}else{
|
|
336
|
+
console.error("[postSokho] Error save data post",e.message||e.error||e,master.so_ct,master.ma_ct)
|
|
337
|
+
self.remove(()=> {
|
|
338
|
+
callback(e);
|
|
339
|
+
})
|
|
340
|
+
}
|
|
341
|
+
}else{
|
|
342
|
+
console.error("[postSokho] Error save data post",e.message||e.error||e,master.so_ct,master.ma_ct)
|
|
343
|
+
self.remove(()=> {
|
|
344
|
+
callback(e);
|
|
345
|
+
})
|
|
346
|
+
}*/
|
|
347
|
+
|
|
348
|
+
}
|
|
349
|
+
});
|
|
350
|
+
};
|
|
351
|
+
module.exports = post;
|
|
@@ -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
|
+
};
|