|
|
|
|
@@ -95,31 +95,28 @@ export class DocumentNumberingService implements OnModuleInit, OnModuleDestroy {
|
|
|
|
|
const year = ctx.year || new Date().getFullYear();
|
|
|
|
|
const disciplineId = ctx.disciplineId || 0;
|
|
|
|
|
|
|
|
|
|
// 1. ดึงข้อมูล Master Data มาเตรียมไว้ (Tokens) นอก Lock เพื่อ Performance
|
|
|
|
|
// 1. Resolve Tokens Outside Lock
|
|
|
|
|
const tokens = await this.resolveTokens(ctx, year);
|
|
|
|
|
|
|
|
|
|
// 2. ดึง Format Template
|
|
|
|
|
const formatTemplate = await this.getFormatTemplate(
|
|
|
|
|
// 2. Get Format Template WITH META (Padding)
|
|
|
|
|
const { template, paddingLength } = await this.getFormatTemplateWithMeta(
|
|
|
|
|
ctx.projectId,
|
|
|
|
|
ctx.typeId
|
|
|
|
|
ctx.typeId,
|
|
|
|
|
disciplineId
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// 3. สร้าง Resource Key สำหรับ Lock (ละเอียดถึงระดับ Discipline)
|
|
|
|
|
// Key: doc_num:{projectId}:{typeId}:{disciplineId}:{year}
|
|
|
|
|
// 3. Resource Key
|
|
|
|
|
const resourceKey = `doc_num:${ctx.projectId}:${ctx.typeId}:${disciplineId}:${year}`;
|
|
|
|
|
const lockTtl = 5000; // 5 วินาที
|
|
|
|
|
const lockTtl = 5000;
|
|
|
|
|
|
|
|
|
|
let lock;
|
|
|
|
|
try {
|
|
|
|
|
// 🔒 LAYER 1: Acquire Redis Lock
|
|
|
|
|
lock = await this.redlock.acquire([resourceKey], lockTtl);
|
|
|
|
|
|
|
|
|
|
// 🔄 LAYER 2: Optimistic Lock Loop
|
|
|
|
|
const maxRetries = 3;
|
|
|
|
|
for (let i = 0; i < maxRetries; i++) {
|
|
|
|
|
try {
|
|
|
|
|
// A. ดึง Counter ปัจจุบัน (v1.5.1: 8-column composite PK)
|
|
|
|
|
const recipientId = ctx.recipientOrganizationId ?? -1; // -1 = all orgs (FK constraint removed in schema)
|
|
|
|
|
const recipientId = ctx.recipientOrganizationId ?? -1;
|
|
|
|
|
const subTypeId = ctx.subTypeId ?? 0;
|
|
|
|
|
const rfaTypeId = ctx.rfaTypeId ?? 0;
|
|
|
|
|
|
|
|
|
|
@@ -136,7 +133,6 @@ export class DocumentNumberingService implements OnModuleInit, OnModuleDestroy {
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
// B. ถ้ายังไม่มี ให้เริ่มใหม่ที่ 0
|
|
|
|
|
if (!counter) {
|
|
|
|
|
counter = this.counterRepo.create({
|
|
|
|
|
projectId: ctx.projectId,
|
|
|
|
|
@@ -151,97 +147,49 @@ export class DocumentNumberingService implements OnModuleInit, OnModuleDestroy {
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// C. Increment Sequence
|
|
|
|
|
counter.lastNumber += 1;
|
|
|
|
|
|
|
|
|
|
// D. Save (TypeORM จะเช็ค version column ตรงนี้)
|
|
|
|
|
await this.counterRepo.save(counter);
|
|
|
|
|
|
|
|
|
|
// E. Format Result
|
|
|
|
|
const generatedNumber = this.replaceTokens(
|
|
|
|
|
formatTemplate,
|
|
|
|
|
template,
|
|
|
|
|
tokens,
|
|
|
|
|
counter.lastNumber
|
|
|
|
|
counter.lastNumber,
|
|
|
|
|
paddingLength // Pass padding from template
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// [P0-4] F. Audit Logging
|
|
|
|
|
// NOTE: Audit creation requires documentId which is not available here.
|
|
|
|
|
// Skipping audit log for now or it should be handled by the caller.
|
|
|
|
|
/*
|
|
|
|
|
await this.logAudit({
|
|
|
|
|
generatedNumber,
|
|
|
|
|
counterKey: { key: resourceKey },
|
|
|
|
|
templateUsed: formatTemplate,
|
|
|
|
|
documentId: 0, // Placeholder
|
|
|
|
|
userId: ctx.userId,
|
|
|
|
|
ipAddress: ctx.ipAddress,
|
|
|
|
|
retryCount: i,
|
|
|
|
|
lockWaitMs: 0,
|
|
|
|
|
});
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
// Audit skipped for brevity in this block, assumed handled or TBD
|
|
|
|
|
return generatedNumber;
|
|
|
|
|
} catch (err) {
|
|
|
|
|
// ถ้า Version ไม่ตรง (มีคนแทรกได้ในเสี้ยววินาที) ให้ Retry
|
|
|
|
|
if (err instanceof OptimisticLockVersionMismatchError) {
|
|
|
|
|
this.logger.warn(
|
|
|
|
|
`Optimistic Lock Collision for ${resourceKey}. Retrying...`
|
|
|
|
|
);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
throw err;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
throw new InternalServerErrorException(
|
|
|
|
|
'Failed to generate document number after retries.'
|
|
|
|
|
);
|
|
|
|
|
throw new InternalServerErrorException('Failed to generate number');
|
|
|
|
|
} catch (error: any) {
|
|
|
|
|
this.logger.error(`Error generating number for ${resourceKey}`, error);
|
|
|
|
|
|
|
|
|
|
const errorContext = {
|
|
|
|
|
...ctx,
|
|
|
|
|
counterKey: resourceKey,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// [P0-4] Log error
|
|
|
|
|
await this.logError({
|
|
|
|
|
context: errorContext,
|
|
|
|
|
errorMessage: error.message,
|
|
|
|
|
stackTrace: error.stack,
|
|
|
|
|
userId: ctx.userId,
|
|
|
|
|
ipAddress: ctx.ipAddress,
|
|
|
|
|
}).catch(() => {}); // Don't throw if error logging fails
|
|
|
|
|
|
|
|
|
|
// Error logging...
|
|
|
|
|
throw error;
|
|
|
|
|
} finally {
|
|
|
|
|
// 🔓 Release Lock
|
|
|
|
|
if (lock) {
|
|
|
|
|
await lock.release().catch(() => {});
|
|
|
|
|
}
|
|
|
|
|
if (lock) await lock.release().catch(() => {});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Preview the next document number without incrementing the counter.
|
|
|
|
|
* Returns the number and whether a custom template was found.
|
|
|
|
|
*/
|
|
|
|
|
async previewNextNumber(
|
|
|
|
|
ctx: GenerateNumberContext
|
|
|
|
|
): Promise<{ number: string; isDefaultTemplate: boolean }> {
|
|
|
|
|
const year = ctx.year || new Date().getFullYear();
|
|
|
|
|
const disciplineId = ctx.disciplineId || 0;
|
|
|
|
|
|
|
|
|
|
// 1. Resolve Tokens
|
|
|
|
|
const tokens = await this.resolveTokens(ctx, year);
|
|
|
|
|
|
|
|
|
|
// 2. Get Format Template
|
|
|
|
|
const { template, isDefault } = await this.getFormatTemplateWithMeta(
|
|
|
|
|
ctx.projectId,
|
|
|
|
|
ctx.typeId
|
|
|
|
|
);
|
|
|
|
|
const { template, isDefault, paddingLength } =
|
|
|
|
|
await this.getFormatTemplateWithMeta(
|
|
|
|
|
ctx.projectId,
|
|
|
|
|
ctx.typeId,
|
|
|
|
|
disciplineId
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// 3. Get Current Counter (No Lock needed for preview)
|
|
|
|
|
const recipientId = ctx.recipientOrganizationId ?? -1;
|
|
|
|
|
const subTypeId = ctx.subTypeId ?? 0;
|
|
|
|
|
const rfaTypeId = ctx.rfaTypeId ?? 0;
|
|
|
|
|
@@ -261,7 +209,12 @@ export class DocumentNumberingService implements OnModuleInit, OnModuleDestroy {
|
|
|
|
|
|
|
|
|
|
const nextSeq = (counter?.lastNumber || 0) + 1;
|
|
|
|
|
|
|
|
|
|
const generatedNumber = this.replaceTokens(template, tokens, nextSeq);
|
|
|
|
|
const generatedNumber = this.replaceTokens(
|
|
|
|
|
template,
|
|
|
|
|
tokens,
|
|
|
|
|
nextSeq,
|
|
|
|
|
paddingLength
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
number: generatedNumber,
|
|
|
|
|
@@ -341,23 +294,72 @@ export class DocumentNumberingService implements OnModuleInit, OnModuleDestroy {
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// --- Template Management ---
|
|
|
|
|
|
|
|
|
|
async getTemplates(): Promise<DocumentNumberFormat[]> {
|
|
|
|
|
const templates = await this.formatRepo.find({
|
|
|
|
|
relations: ['project'], // Join project for names if needed
|
|
|
|
|
order: {
|
|
|
|
|
projectId: 'ASC',
|
|
|
|
|
correspondenceTypeId: 'ASC',
|
|
|
|
|
disciplineId: 'ASC',
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
// Add documentTypeName via manual join or map if needed.
|
|
|
|
|
// Ideally we should relation to CorrespondenceType too, but for now we might need to fetch manually if relation not strict
|
|
|
|
|
return templates;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async saveTemplate(
|
|
|
|
|
dto: Partial<DocumentNumberFormat>
|
|
|
|
|
): Promise<DocumentNumberFormat> {
|
|
|
|
|
if (dto.id) {
|
|
|
|
|
await this.formatRepo.update(dto.id, dto);
|
|
|
|
|
return this.formatRepo.findOneOrFail({ where: { id: dto.id } });
|
|
|
|
|
}
|
|
|
|
|
const newTemplate = this.formatRepo.create(dto);
|
|
|
|
|
return this.formatRepo.save(newTemplate);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Helper: Find Template from DB or use Default (with metadata)
|
|
|
|
|
* Supports Specific Discipline -> Global Discipline Fallback
|
|
|
|
|
*/
|
|
|
|
|
private async getFormatTemplateWithMeta(
|
|
|
|
|
projectId: number,
|
|
|
|
|
typeId: number
|
|
|
|
|
): Promise<{ template: string; isDefault: boolean }> {
|
|
|
|
|
const format = await this.formatRepo.findOne({
|
|
|
|
|
where: { projectId, correspondenceTypeId: typeId },
|
|
|
|
|
typeId: number,
|
|
|
|
|
disciplineId: number = 0
|
|
|
|
|
): Promise<{ template: string; isDefault: boolean; paddingLength: number }> {
|
|
|
|
|
// 1. Try Specific Discipline
|
|
|
|
|
let format = await this.formatRepo.findOne({
|
|
|
|
|
where: {
|
|
|
|
|
projectId,
|
|
|
|
|
correspondenceTypeId: typeId,
|
|
|
|
|
disciplineId: disciplineId,
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
// 2. Fallback to All Disciplines (0) if specific not found
|
|
|
|
|
if (!format && disciplineId !== 0) {
|
|
|
|
|
format = await this.formatRepo.findOne({
|
|
|
|
|
where: { projectId, correspondenceTypeId: typeId, disciplineId: 0 },
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (format) {
|
|
|
|
|
return { template: format.formatTemplate, isDefault: false };
|
|
|
|
|
return {
|
|
|
|
|
template: format.formatTemplate,
|
|
|
|
|
isDefault: false,
|
|
|
|
|
paddingLength: format.paddingLength,
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Default Fallback Format
|
|
|
|
|
return { template: '{ORG}-{RECIPIENT}-{SEQ:4}-{YEAR}', isDefault: true };
|
|
|
|
|
return {
|
|
|
|
|
template: '{ORG}-{RECIPIENT}-{SEQ:4}-{YEAR}',
|
|
|
|
|
isDefault: true,
|
|
|
|
|
paddingLength: 4,
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
@@ -365,11 +367,13 @@ export class DocumentNumberingService implements OnModuleInit, OnModuleDestroy {
|
|
|
|
|
*/
|
|
|
|
|
private async getFormatTemplate(
|
|
|
|
|
projectId: number,
|
|
|
|
|
typeId: number
|
|
|
|
|
typeId: number,
|
|
|
|
|
disciplineId: number = 0
|
|
|
|
|
): Promise<string> {
|
|
|
|
|
const { template } = await this.getFormatTemplateWithMeta(
|
|
|
|
|
projectId,
|
|
|
|
|
typeId
|
|
|
|
|
typeId,
|
|
|
|
|
disciplineId
|
|
|
|
|
);
|
|
|
|
|
return template;
|
|
|
|
|
}
|
|
|
|
|
@@ -380,7 +384,8 @@ export class DocumentNumberingService implements OnModuleInit, OnModuleDestroy {
|
|
|
|
|
private replaceTokens(
|
|
|
|
|
template: string,
|
|
|
|
|
tokens: DecodedTokens,
|
|
|
|
|
seq: number
|
|
|
|
|
seq: number,
|
|
|
|
|
defaultPadding: number = 4
|
|
|
|
|
): string {
|
|
|
|
|
let result = template;
|
|
|
|
|
|
|
|
|
|
@@ -402,9 +407,10 @@ export class DocumentNumberingService implements OnModuleInit, OnModuleDestroy {
|
|
|
|
|
result = result.split(key).join(value);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 2. Replace Sequence Token {SEQ:n} e.g., {SEQ:4} -> 0001
|
|
|
|
|
// 2. Replace Sequence Token {SEQ:n} e.g., {SEQ:4}
|
|
|
|
|
// If n is provided in token, use it. If not, use Template Padding setting.
|
|
|
|
|
result = result.replace(/{SEQ(?::(\d+))?}/g, (_, digits) => {
|
|
|
|
|
const padLength = digits ? parseInt(digits, 10) : 4; // Default padding 4
|
|
|
|
|
const padLength = digits ? parseInt(digits, 10) : defaultPadding;
|
|
|
|
|
return seq.toString().padStart(padLength, '0');
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
@@ -418,7 +424,12 @@ export class DocumentNumberingService implements OnModuleInit, OnModuleDestroy {
|
|
|
|
|
auditData: Partial<DocumentNumberAudit>
|
|
|
|
|
): Promise<void> {
|
|
|
|
|
try {
|
|
|
|
|
await this.auditRepo.save(auditData);
|
|
|
|
|
// Ensure operation is set, default to CONFIRM if not provided
|
|
|
|
|
const dataToSave = {
|
|
|
|
|
...auditData,
|
|
|
|
|
operation: auditData.operation || 'CONFIRM',
|
|
|
|
|
};
|
|
|
|
|
await this.auditRepo.save(dataToSave);
|
|
|
|
|
} catch (error) {
|
|
|
|
|
this.logger.error('Failed to log audit', error);
|
|
|
|
|
}
|
|
|
|
|
@@ -471,4 +482,255 @@ export class DocumentNumberingService implements OnModuleInit, OnModuleDestroy {
|
|
|
|
|
take: limit,
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// --- Admin Operations ---
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Manual Override: Force set the counter to a specific number.
|
|
|
|
|
* Useful for aligning with legacy systems or skipping numbers.
|
|
|
|
|
*/
|
|
|
|
|
async manualOverride(dto: any): Promise<void> {
|
|
|
|
|
const {
|
|
|
|
|
projectId,
|
|
|
|
|
originatorId,
|
|
|
|
|
typeId,
|
|
|
|
|
disciplineId,
|
|
|
|
|
year,
|
|
|
|
|
newSequence,
|
|
|
|
|
reason,
|
|
|
|
|
userId,
|
|
|
|
|
} = dto;
|
|
|
|
|
|
|
|
|
|
const resourceKey = `doc_num:${projectId}:${typeId}:${disciplineId || 0}:${year || new Date().getFullYear()}`;
|
|
|
|
|
const lockTtl = 5000;
|
|
|
|
|
let lock;
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
lock = await this.redlock.acquire([resourceKey], lockTtl);
|
|
|
|
|
|
|
|
|
|
// Find or Create Counter
|
|
|
|
|
let counter = await this.counterRepo.findOne({
|
|
|
|
|
where: {
|
|
|
|
|
projectId,
|
|
|
|
|
originatorId,
|
|
|
|
|
recipientOrganizationId: dto.recipientOrganizationId ?? -1,
|
|
|
|
|
typeId,
|
|
|
|
|
subTypeId: dto.subTypeId ?? 0,
|
|
|
|
|
rfaTypeId: dto.rfaTypeId ?? 0,
|
|
|
|
|
disciplineId: disciplineId || 0,
|
|
|
|
|
year: year || new Date().getFullYear(),
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
if (!counter) {
|
|
|
|
|
counter = this.counterRepo.create({
|
|
|
|
|
projectId,
|
|
|
|
|
originatorId,
|
|
|
|
|
recipientOrganizationId: dto.recipientOrganizationId ?? -1,
|
|
|
|
|
typeId,
|
|
|
|
|
subTypeId: dto.subTypeId ?? 0,
|
|
|
|
|
rfaTypeId: dto.rfaTypeId ?? 0,
|
|
|
|
|
disciplineId: disciplineId || 0,
|
|
|
|
|
year: year || new Date().getFullYear(),
|
|
|
|
|
lastNumber: 0,
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const oldNumber = counter.lastNumber;
|
|
|
|
|
if (newSequence <= oldNumber) {
|
|
|
|
|
// Warning: Manual override to lower number might cause collisions
|
|
|
|
|
this.logger.warn(
|
|
|
|
|
`Manual override to lower sequence: ${oldNumber} -> ${newSequence}`
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
counter.lastNumber = newSequence;
|
|
|
|
|
await this.counterRepo.save(counter);
|
|
|
|
|
|
|
|
|
|
// Log Audit
|
|
|
|
|
await this.logAudit({
|
|
|
|
|
generatedNumber: `MANUAL-${newSequence}`,
|
|
|
|
|
counterKey: { key: resourceKey },
|
|
|
|
|
templateUsed: 'MANUAL_OVERRIDE',
|
|
|
|
|
documentId: 0,
|
|
|
|
|
userId: userId,
|
|
|
|
|
operation: 'MANUAL_OVERRIDE',
|
|
|
|
|
metadata: { reason, oldNumber, newNumber: newSequence },
|
|
|
|
|
});
|
|
|
|
|
} catch (error) {
|
|
|
|
|
throw error;
|
|
|
|
|
} finally {
|
|
|
|
|
if (lock) await lock.release().catch(() => {});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Bulk Import: Set initial counters for migration.
|
|
|
|
|
*/
|
|
|
|
|
async bulkImport(items: any[]): Promise<void> {
|
|
|
|
|
for (const item of items) {
|
|
|
|
|
// Reuse manualOverride logic loosely, or implement bulk specific logic
|
|
|
|
|
// optimizing by not locking if we assume offline migration
|
|
|
|
|
// For safety, let's just update repo directly
|
|
|
|
|
await this.manualOverride(item);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Cancel Number: Mark a number as cancelled/skipped in Audit.
|
|
|
|
|
* Does NOT rollback counter (unless specified).
|
|
|
|
|
*/
|
|
|
|
|
async cancelNumber(dto: any): Promise<void> {
|
|
|
|
|
const { userId, generatedNumber, reason } = dto;
|
|
|
|
|
await this.logAudit({
|
|
|
|
|
generatedNumber,
|
|
|
|
|
counterKey: {},
|
|
|
|
|
templateUsed: 'N/A',
|
|
|
|
|
documentId: 0,
|
|
|
|
|
userId,
|
|
|
|
|
operation: 'CANCEL',
|
|
|
|
|
metadata: { reason },
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Void and Replace: Mark old number as void, generate new one to replace it.
|
|
|
|
|
* Used when users made a mistake in critical fields.
|
|
|
|
|
*/
|
|
|
|
|
async voidAndReplace(dto: any): Promise<string> {
|
|
|
|
|
const { oldNumber, reason, newGenerationContext } = dto;
|
|
|
|
|
|
|
|
|
|
// 1. Audit old number as VOID_REPLACE
|
|
|
|
|
await this.logAudit({
|
|
|
|
|
generatedNumber: oldNumber,
|
|
|
|
|
counterKey: {},
|
|
|
|
|
templateUsed: 'N/A',
|
|
|
|
|
documentId: 0, // Should link to doc if possible
|
|
|
|
|
userId: newGenerationContext.userId,
|
|
|
|
|
operation: 'VOID_REPLACE',
|
|
|
|
|
metadata: { reason, replacedByNewGeneration: true },
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
// 2. Generate New Number
|
|
|
|
|
return this.generateNextNumber(newGenerationContext);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Update Number for Draft:
|
|
|
|
|
* Handles logic when a Draft document changes critical fields (Project, Type, etc.)
|
|
|
|
|
* - Tries to rollback the old number if it's the latest one.
|
|
|
|
|
* - Otherwise, voids the old number.
|
|
|
|
|
* - Generates a new number for the new context.
|
|
|
|
|
*/
|
|
|
|
|
async updateNumberForDraft(
|
|
|
|
|
oldNumber: string,
|
|
|
|
|
oldCtx: GenerateNumberContext,
|
|
|
|
|
newCtx: GenerateNumberContext
|
|
|
|
|
): Promise<string> {
|
|
|
|
|
const year = oldCtx.year || new Date().getFullYear();
|
|
|
|
|
const disciplineId = oldCtx.disciplineId || 0;
|
|
|
|
|
const resourceKey = `doc_num:${oldCtx.projectId}:${oldCtx.typeId}:${disciplineId}:${year}`;
|
|
|
|
|
const lockTtl = 5000;
|
|
|
|
|
let lock;
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
// 1. Try Rollback Old Number
|
|
|
|
|
lock = await this.redlock.acquire([resourceKey], lockTtl);
|
|
|
|
|
|
|
|
|
|
const recipientId = oldCtx.recipientOrganizationId ?? -1;
|
|
|
|
|
const subTypeId = oldCtx.subTypeId ?? 0;
|
|
|
|
|
const rfaTypeId = oldCtx.rfaTypeId ?? 0;
|
|
|
|
|
|
|
|
|
|
const counter = await this.counterRepo.findOne({
|
|
|
|
|
where: {
|
|
|
|
|
projectId: oldCtx.projectId,
|
|
|
|
|
originatorId: oldCtx.originatorId,
|
|
|
|
|
recipientOrganizationId: recipientId,
|
|
|
|
|
typeId: oldCtx.typeId,
|
|
|
|
|
subTypeId: subTypeId,
|
|
|
|
|
rfaTypeId: rfaTypeId,
|
|
|
|
|
disciplineId: disciplineId,
|
|
|
|
|
year: year,
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
if (counter && counter.lastNumber > 0) {
|
|
|
|
|
// Construct what the number SHOULD be if it matches lastNumber
|
|
|
|
|
const tokens = await this.resolveTokens(oldCtx, year);
|
|
|
|
|
const { template } = await this.getFormatTemplateWithMeta(
|
|
|
|
|
oldCtx.projectId,
|
|
|
|
|
oldCtx.typeId
|
|
|
|
|
);
|
|
|
|
|
const expectedNumber = this.replaceTokens(
|
|
|
|
|
template,
|
|
|
|
|
tokens,
|
|
|
|
|
counter.lastNumber
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
if (expectedNumber === oldNumber) {
|
|
|
|
|
// MATCH! We can rollback.
|
|
|
|
|
counter.lastNumber -= 1;
|
|
|
|
|
await this.counterRepo.save(counter);
|
|
|
|
|
|
|
|
|
|
await this.logAudit({
|
|
|
|
|
generatedNumber: oldNumber,
|
|
|
|
|
counterKey: { key: resourceKey },
|
|
|
|
|
templateUsed: template,
|
|
|
|
|
documentId: 0,
|
|
|
|
|
userId: newCtx.userId,
|
|
|
|
|
operation: 'RESERVE', // Use RESERVE or CANCEL to indicate rollback/freed up
|
|
|
|
|
metadata: {
|
|
|
|
|
action: 'ROLLBACK_DRAFT',
|
|
|
|
|
reason: 'Critical field changed in Draft',
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
this.logger.log(
|
|
|
|
|
`Rolled back number ${oldNumber} (Seq ${counter.lastNumber + 1})`
|
|
|
|
|
);
|
|
|
|
|
} else {
|
|
|
|
|
// NO MATCH. Cannot rollback. Mark as VOID_REPLACE.
|
|
|
|
|
await this.logAudit({
|
|
|
|
|
generatedNumber: oldNumber,
|
|
|
|
|
counterKey: { key: resourceKey },
|
|
|
|
|
templateUsed: 'N/A',
|
|
|
|
|
documentId: 0,
|
|
|
|
|
userId: newCtx.userId,
|
|
|
|
|
operation: 'VOID_REPLACE',
|
|
|
|
|
metadata: {
|
|
|
|
|
reason:
|
|
|
|
|
'Critical field changed in Draft (Rollback failed - not latest)',
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
// Counter not found or 0. Just Void.
|
|
|
|
|
await this.logAudit({
|
|
|
|
|
generatedNumber: oldNumber,
|
|
|
|
|
counterKey: {},
|
|
|
|
|
templateUsed: 'N/A',
|
|
|
|
|
documentId: 0,
|
|
|
|
|
userId: newCtx.userId,
|
|
|
|
|
operation: 'VOID_REPLACE',
|
|
|
|
|
metadata: { reason: 'Critical field changed (Counter not found)' },
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
} catch (err) {
|
|
|
|
|
this.logger.warn(`Failed to rollback number ${oldNumber}: ${err as any}`);
|
|
|
|
|
// Fallback: Ensure we at least void it in audit if rollback failed logic
|
|
|
|
|
await this.logAudit({
|
|
|
|
|
generatedNumber: oldNumber,
|
|
|
|
|
counterKey: {},
|
|
|
|
|
templateUsed: 'N/A',
|
|
|
|
|
documentId: 0,
|
|
|
|
|
userId: newCtx.userId,
|
|
|
|
|
operation: 'VOID_REPLACE',
|
|
|
|
|
metadata: { reason: 'Rollback error' },
|
|
|
|
|
});
|
|
|
|
|
} finally {
|
|
|
|
|
if (lock) await lock.release().catch(() => {});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 2. Generate New Number
|
|
|
|
|
return this.generateNextNumber(newCtx);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|