fix: time calculation customer
parent
4eedca12e7
commit
60b5bcf638
|
@ -136,4 +136,6 @@ export class QueueModel extends BaseModel<Queue> implements Queue {
|
||||||
|
|
||||||
@Column('int')
|
@Column('int')
|
||||||
qty: number;
|
qty: number;
|
||||||
|
|
||||||
|
average = 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -24,16 +24,9 @@ export class QueueBucketReadService extends BaseReadService<QueueBucketModel> {
|
||||||
const start = moment().startOf('day').valueOf();
|
const start = moment().startOf('day').valueOf();
|
||||||
const end = moment().endOf('day').valueOf();
|
const end = moment().endOf('day').valueOf();
|
||||||
|
|
||||||
const queueItem = await this.item.findOne({
|
|
||||||
relations: ['item'],
|
|
||||||
where: {
|
|
||||||
id: item_id,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
const queue = await this.repo.findOne({
|
const queue = await this.repo.findOne({
|
||||||
where: {
|
where: {
|
||||||
queue_item_id: queueItem.item.item_queue_id ?? queueItem.item.id,
|
queue_item_id: item_id,
|
||||||
date: Between(start, end),
|
date: Between(start, end),
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
@ -42,7 +35,7 @@ export class QueueBucketReadService extends BaseReadService<QueueBucketModel> {
|
||||||
const regularNumber = vip ? 0 : 1;
|
const regularNumber = vip ? 0 : 1;
|
||||||
const vipNumber = vip ? 1 : 0;
|
const vipNumber = vip ? 1 : 0;
|
||||||
await this.repo.save({
|
await this.repo.save({
|
||||||
queue_item_id: queueItem.item.item_queue_id ?? queueItem.item.id,
|
queue_item_id: item_id,
|
||||||
date: start,
|
date: start,
|
||||||
regular: regularNumber,
|
regular: regularNumber,
|
||||||
vip: vipNumber,
|
vip: vipNumber,
|
||||||
|
|
|
@ -13,6 +13,7 @@ import { BaseDataService } from 'src/core/modules/data/service/base-data.service
|
||||||
import { ItemQueueModel } from 'src/modules/item-related/item-queue/data/models/item-queue.model';
|
import { ItemQueueModel } from 'src/modules/item-related/item-queue/data/models/item-queue.model';
|
||||||
import * as moment from 'moment';
|
import * as moment from 'moment';
|
||||||
import * as math from 'mathjs';
|
import * as math from 'mathjs';
|
||||||
|
import { ItemModel } from 'src/modules/item-related/item/data/models/item.model';
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class QueueDataService extends BaseReadService<QueueModel> {
|
export class QueueDataService extends BaseReadService<QueueModel> {
|
||||||
|
@ -136,6 +137,9 @@ export class QueueService extends BaseDataService<QueueModel> {
|
||||||
@InjectRepository(QueueItemModel, CONNECTION_NAME.DEFAULT)
|
@InjectRepository(QueueItemModel, CONNECTION_NAME.DEFAULT)
|
||||||
private item: Repository<QueueItemModel>,
|
private item: Repository<QueueItemModel>,
|
||||||
|
|
||||||
|
@InjectRepository(ItemModel, CONNECTION_NAME.DEFAULT)
|
||||||
|
private itemMaster: Repository<ItemModel>,
|
||||||
|
|
||||||
@InjectDataSource(CONNECTION_NAME.DEFAULT)
|
@InjectDataSource(CONNECTION_NAME.DEFAULT)
|
||||||
private dataSource: DataSource,
|
private dataSource: DataSource,
|
||||||
) {
|
) {
|
||||||
|
@ -145,7 +149,13 @@ export class QueueService extends BaseDataService<QueueModel> {
|
||||||
async queues(ids: string[]) {
|
async queues(ids: string[]) {
|
||||||
const start = moment().startOf('day').valueOf();
|
const start = moment().startOf('day').valueOf();
|
||||||
const end = moment().endOf('day').valueOf();
|
const end = moment().endOf('day').valueOf();
|
||||||
return this.repo.find({
|
const playEstimations = {};
|
||||||
|
|
||||||
|
for (const id of ids) {
|
||||||
|
playEstimations[id] = await this.itemAverageTimeEstimation(id);
|
||||||
|
}
|
||||||
|
|
||||||
|
const queues = await this.repo.find({
|
||||||
where: {
|
where: {
|
||||||
item_queue_id: In(ids),
|
item_queue_id: In(ids),
|
||||||
time: Between(start, end),
|
time: Between(start, end),
|
||||||
|
@ -154,6 +164,24 @@ export class QueueService extends BaseDataService<QueueModel> {
|
||||||
time: 'ASC',
|
time: 'ASC',
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
queues.forEach((queue) => {
|
||||||
|
queue.average = playEstimations[queue.item_queue_id];
|
||||||
|
});
|
||||||
|
|
||||||
|
return queues;
|
||||||
|
}
|
||||||
|
|
||||||
|
async itemAverageTimeEstimation(item_queue_id: string) {
|
||||||
|
const items = await this.itemMaster.find({
|
||||||
|
where: {
|
||||||
|
item_queue_id,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const times = items.map((item) => item.play_estimation ?? 0);
|
||||||
|
const average = math.mean(times) * 60 * 1000; // change average minute to milliseconds
|
||||||
|
return average;
|
||||||
}
|
}
|
||||||
|
|
||||||
async getTicketItems(ticket_id: string, item_id: string) {
|
async getTicketItems(ticket_id: string, item_id: string) {
|
||||||
|
|
|
@ -1,6 +1,8 @@
|
||||||
import { QueueModel } from 'src/modules/queue/data/models/queue.model';
|
import { QueueModel } from 'src/modules/queue/data/models/queue.model';
|
||||||
import { toTime } from '../../helpers/time.helper';
|
import { toTime } from '../../helpers/time.helper';
|
||||||
import * as math from 'mathjs';
|
import * as math from 'mathjs';
|
||||||
|
import { QueueTimeFormula } from './queue-time.formula';
|
||||||
|
import * as moment from 'moment';
|
||||||
|
|
||||||
export class QueueCondition {
|
export class QueueCondition {
|
||||||
private ticketItems = {};
|
private ticketItems = {};
|
||||||
|
@ -14,23 +16,59 @@ export class QueueCondition {
|
||||||
|
|
||||||
condition(item_id: string) {
|
condition(item_id: string) {
|
||||||
const queues: QueueModel[] = this.ticketItems[item_id] ?? [];
|
const queues: QueueModel[] = this.ticketItems[item_id] ?? [];
|
||||||
const time = queues[0];
|
const playEstimation = queues[0]?.average ?? 0;
|
||||||
const nearest = time ? toTime(time.time) : 0;
|
const [time, last] = this.queueTime(queues, playEstimation);
|
||||||
|
const nearest = time ? toTime(time) : 0;
|
||||||
|
const lastTime = last ? toTime(last + playEstimation) : 0;
|
||||||
|
|
||||||
|
const queuePeople = this.queuePeople(queues);
|
||||||
|
|
||||||
const last = [...queues].pop();
|
|
||||||
const lastTime = last ? toTime(last.time) : 0;
|
|
||||||
return {
|
return {
|
||||||
available: queues.length == 0,
|
available: queuePeople == 0,
|
||||||
average: this.averageTime(queues),
|
average: this.averageTime(queues),
|
||||||
nearest: nearest,
|
nearest: nearest,
|
||||||
crowded_level: queues.length,
|
crowded_level: queuePeople,
|
||||||
available_time: lastTime,
|
available_time: lastTime,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
queuePeople(queues: QueueModel[]): number {
|
||||||
|
const queue = this.activeQueue(queues);
|
||||||
|
const queuePeople = queue.reduce((acc, q) => {
|
||||||
|
return acc + q.qty;
|
||||||
|
}, 0);
|
||||||
|
return queuePeople;
|
||||||
|
}
|
||||||
|
|
||||||
|
activeQueue(queues: QueueModel[]) {
|
||||||
|
return queues.filter((q) => q.status == 'waiting');
|
||||||
|
}
|
||||||
|
|
||||||
|
queueTime(queues: QueueModel[], average = 0): number[] {
|
||||||
|
const calledQueue = queues.filter((q) =>
|
||||||
|
['called', 'done'].includes(q.status),
|
||||||
|
);
|
||||||
|
const lastCalledQueue = calledQueue[calledQueue.length - 1];
|
||||||
|
const activeQueues = this.activeQueue(queues);
|
||||||
|
|
||||||
|
const queueTimes = QueueTimeFormula.queueTime(
|
||||||
|
activeQueues,
|
||||||
|
lastCalledQueue,
|
||||||
|
average,
|
||||||
|
);
|
||||||
|
|
||||||
|
const queueEstimation = Object.values<number>(queueTimes);
|
||||||
|
|
||||||
|
const first = queueEstimation[0] ?? moment().valueOf();
|
||||||
|
const last = queueEstimation[queueEstimation.length - 1] ?? 0;
|
||||||
|
return [first, last];
|
||||||
|
}
|
||||||
|
|
||||||
averageTime(queues: QueueModel[]) {
|
averageTime(queues: QueueModel[]) {
|
||||||
if (queues.length == 0) return 0;
|
if (queues.length == 0) return 0;
|
||||||
const calledQueue = queues.filter((q) => q.status === 'called');
|
const calledQueue = queues.filter((q) =>
|
||||||
|
['called', 'done'].includes(q.status),
|
||||||
|
);
|
||||||
|
|
||||||
if (calledQueue.length < 1) return 0;
|
if (calledQueue.length < 1) return 0;
|
||||||
|
|
||||||
|
|
|
@ -5,6 +5,7 @@ import { QueueDataService } from 'src/modules/queue/data/services/queue.service'
|
||||||
import { Repository } from 'typeorm';
|
import { Repository } from 'typeorm';
|
||||||
import * as moment from 'moment';
|
import * as moment from 'moment';
|
||||||
import * as math from 'mathjs';
|
import * as math from 'mathjs';
|
||||||
|
import { QueueModel } from 'src/modules/queue/data/models/queue.model';
|
||||||
|
|
||||||
export class QueueTimeFormula {
|
export class QueueTimeFormula {
|
||||||
constructor(
|
constructor(
|
||||||
|
@ -31,9 +32,18 @@ export class QueueTimeFormula {
|
||||||
|
|
||||||
const calledQueue = await this.queueDataService.lastQueue(item_queue_id);
|
const calledQueue = await this.queueDataService.lastQueue(item_queue_id);
|
||||||
|
|
||||||
|
return QueueTimeFormula.queueTime(queues, calledQueue, average);
|
||||||
|
}
|
||||||
|
|
||||||
|
static queueTime(
|
||||||
|
queues: QueueModel[],
|
||||||
|
lastQueue: QueueModel,
|
||||||
|
average: number,
|
||||||
|
) {
|
||||||
const queueTimes = {};
|
const queueTimes = {};
|
||||||
|
if (queues.length < 1) return queueTimes;
|
||||||
const timeNow = moment().valueOf();
|
const timeNow = moment().valueOf();
|
||||||
const lastCall = calledQueue?.call_time ?? timeNow - average;
|
const lastCall = lastQueue?.call_time ?? timeNow - average;
|
||||||
|
|
||||||
const callTime = +lastCall + average;
|
const callTime = +lastCall + average;
|
||||||
const currentQueueCallTime = timeNow > callTime ? timeNow : callTime;
|
const currentQueueCallTime = timeNow > callTime ? timeNow : callTime;
|
||||||
|
|
|
@ -106,7 +106,7 @@ export class CustomerQueueManager {
|
||||||
const queueItem = item.item.item_queue ?? item.item;
|
const queueItem = item.item.item_queue ?? item.item;
|
||||||
return {
|
return {
|
||||||
id: item.item_id,
|
id: item.item_id,
|
||||||
queue_item_id: item.id,
|
queue_item_id: queueItem.id,
|
||||||
title: queueItem.name,
|
title: queueItem.name,
|
||||||
image_url: item.item.image_url,
|
image_url: item.item.image_url,
|
||||||
qty: item_qty,
|
qty: item_qty,
|
||||||
|
|
|
@ -10,10 +10,14 @@ import { QueueItemModel, QueueModel } from '../../data/models/queue.model';
|
||||||
import { padCode } from 'src/modules/transaction/vip-code/domain/usecases/managers/helpers/generate-random.helper';
|
import { padCode } from 'src/modules/transaction/vip-code/domain/usecases/managers/helpers/generate-random.helper';
|
||||||
import { QueueBucketReadService } from '../../data/services/queue-bucket';
|
import { QueueBucketReadService } from '../../data/services/queue-bucket';
|
||||||
import { ItemModel } from 'src/modules/item-related/item/data/models/item.model';
|
import { ItemModel } from 'src/modules/item-related/item/data/models/item.model';
|
||||||
|
import { QueueTimeFormula } from './formula/queue-time.formula';
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class RegisterQueueManager extends BaseCreateManager<QueueModel> {
|
export class RegisterQueueManager extends BaseCreateManager<QueueModel> {
|
||||||
constructor(private readonly bucketService: QueueBucketReadService) {
|
constructor(
|
||||||
|
private readonly bucketService: QueueBucketReadService,
|
||||||
|
private readonly queueTimeFormula: QueueTimeFormula,
|
||||||
|
) {
|
||||||
super();
|
super();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -22,11 +26,22 @@ export class RegisterQueueManager extends BaseCreateManager<QueueModel> {
|
||||||
return item.play_estimation;
|
return item.play_estimation;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async queueTime(queue_id: string): Promise<number[]> {
|
||||||
|
const queueTimes = await this.queueTimeFormula.items(queue_id);
|
||||||
|
const queues = Object.values<number>(queueTimes);
|
||||||
|
|
||||||
|
const first = queues[0];
|
||||||
|
const last = queues[queues.length - 1];
|
||||||
|
return [first, last];
|
||||||
|
}
|
||||||
|
|
||||||
async beforeProcess(): Promise<void> {
|
async beforeProcess(): Promise<void> {
|
||||||
const vip = this.data.vip ?? false;
|
const vip = this.data.vip ?? false;
|
||||||
const item = await this.getItemMaster();
|
const item = await this.getItemMaster();
|
||||||
|
const [, end] = await this.queueTime(item.item_queue_id);
|
||||||
|
|
||||||
const queueNumber = await this.bucketService.getQueue(
|
const queueNumber = await this.bucketService.getQueue(
|
||||||
this.data.item_id,
|
item.item_queue_id,
|
||||||
vip,
|
vip,
|
||||||
);
|
);
|
||||||
const prefix = vip ? 'B' : 'A';
|
const prefix = vip ? 'B' : 'A';
|
||||||
|
@ -34,7 +49,7 @@ export class RegisterQueueManager extends BaseCreateManager<QueueModel> {
|
||||||
|
|
||||||
Object.assign(this.data, {
|
Object.assign(this.data, {
|
||||||
status: STATUS.WAITING,
|
status: STATUS.WAITING,
|
||||||
time: new Date().getTime(),
|
time: end,
|
||||||
item_queue_id: item.item_queue_id,
|
item_queue_id: item.item_queue_id,
|
||||||
vip,
|
vip,
|
||||||
code,
|
code,
|
||||||
|
|
|
@ -15,6 +15,7 @@ import { RegisterQueueDto } from '../controllers/dto/register-queue.dto';
|
||||||
import { QueueService } from '../../data/services/queue.service';
|
import { QueueService } from '../../data/services/queue.service';
|
||||||
import { TABLE_NAME } from 'src/core/strings/constants/table.constants';
|
import { TABLE_NAME } from 'src/core/strings/constants/table.constants';
|
||||||
import { QueueBucketReadService } from '../../data/services/queue-bucket';
|
import { QueueBucketReadService } from '../../data/services/queue-bucket';
|
||||||
|
import { QueueTimeFormula } from '../../domain/usecases/formula/queue-time.formula';
|
||||||
|
|
||||||
@EventsHandler(TransactionChangeStatusEvent, TransactionCreateQueueEvent)
|
@EventsHandler(TransactionChangeStatusEvent, TransactionCreateQueueEvent)
|
||||||
export class QueueTransactionHandler
|
export class QueueTransactionHandler
|
||||||
|
@ -25,6 +26,7 @@ export class QueueTransactionHandler
|
||||||
private readonly ticketService: TicketDataService,
|
private readonly ticketService: TicketDataService,
|
||||||
private readonly queueService: QueueService,
|
private readonly queueService: QueueService,
|
||||||
private readonly bucketService: QueueBucketReadService,
|
private readonly bucketService: QueueBucketReadService,
|
||||||
|
private readonly queueTimeFormula: QueueTimeFormula,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
async handle(event: TransactionChangeStatusEvent) {
|
async handle(event: TransactionChangeStatusEvent) {
|
||||||
|
@ -125,7 +127,10 @@ export class QueueTransactionHandler
|
||||||
item_id: queue.id,
|
item_id: queue.id,
|
||||||
vip: true,
|
vip: true,
|
||||||
};
|
};
|
||||||
const registerQueueManager = new RegisterQueueManager(this.bucketService);
|
const registerQueueManager = new RegisterQueueManager(
|
||||||
|
this.bucketService,
|
||||||
|
this.queueTimeFormula,
|
||||||
|
);
|
||||||
registerQueueManager.setData(queueRequest);
|
registerQueueManager.setData(queueRequest);
|
||||||
registerQueueManager.setService(this.queueService, TABLE_NAME.QUEUE);
|
registerQueueManager.setService(this.queueService, TABLE_NAME.QUEUE);
|
||||||
await registerQueueManager.execute();
|
await registerQueueManager.execute();
|
||||||
|
|
Loading…
Reference in New Issue