Merge branch 'develop' into adiDev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 1m3s
All checks were successful
Build & Deploy on Dev / build (push) Successful in 1m3s
This commit is contained in:
commit
8c9a62a378
6 changed files with 602 additions and 22 deletions
|
|
@ -325,7 +325,123 @@ export class ApiManageController extends Controller {
|
|||
];
|
||||
|
||||
private readonly DEFAULT_PAGE_SIZE = 10; // ขนาดหน้าเริ่มต้น
|
||||
private readonly EXCLUDED_COLUMNS = ["createdUserId", "lastUpdateUserId"]; // ฟิลด์ที่ไม่ต้องการแสดงในผลลัพธ์
|
||||
private readonly EXCLUDED_COLUMNS = [
|
||||
"createdUserId",
|
||||
"lastUpdateUserId",
|
||||
"createdAt",
|
||||
"createdFullName",
|
||||
"lastUpdateFullName",
|
||||
"avatarName",
|
||||
"profileId",
|
||||
"prefixId",
|
||||
"profileEmployeeId",
|
||||
"documentId",
|
||||
"orgRevisionId",
|
||||
"posMasterId",
|
||||
"orgRootId",
|
||||
"orgChild1Id",
|
||||
"orgChild2Id",
|
||||
"orgChild3Id",
|
||||
"orgChild4Id",
|
||||
"ancestorDNA",
|
||||
"keycloak",
|
||||
"commandId",
|
||||
"prefixMain",
|
||||
"authRoleId",
|
||||
"next_holderId",
|
||||
"current_holderId",
|
||||
]; // ฟิลด์ที่ไม่ต้องการแสดงในผลลัพธ์
|
||||
|
||||
// การแทนที่ฟิลด์ ID ด้วยฟิลด์ Name สำหรับ Profile entity
|
||||
private readonly PROFILE_FIELD_REPLACEMENTS: Record<
|
||||
string,
|
||||
{ propertyName: string; type: string; comment: string; joinTable: string; joinField: string }
|
||||
> = {
|
||||
posLevelId: {
|
||||
propertyName: "posLevelName",
|
||||
type: "string",
|
||||
comment: "ระดับตำแหน่ง",
|
||||
joinTable: "PosLevel",
|
||||
joinField: "posLevelName",
|
||||
},
|
||||
posTypeId: {
|
||||
propertyName: "posTypeName",
|
||||
type: "string",
|
||||
comment: "ประเภทตำแหน่ง",
|
||||
joinTable: "PosType",
|
||||
joinField: "posTypeName",
|
||||
},
|
||||
registrationProvinceId: {
|
||||
propertyName: "registrationProvinceName",
|
||||
type: "string",
|
||||
comment: "จังหวัดตามทะเบียนบ้าน",
|
||||
joinTable: "Province",
|
||||
joinField: "name",
|
||||
},
|
||||
registrationDistrictId: {
|
||||
propertyName: "registrationDistrictName",
|
||||
type: "string",
|
||||
comment: "เขตตามทะเบียนบ้าน",
|
||||
joinTable: "District",
|
||||
joinField: "name",
|
||||
},
|
||||
registrationSubDistrictId: {
|
||||
propertyName: "registrationSubDistrictName",
|
||||
type: "string",
|
||||
comment: "แขวงตามทะเบียนบ้าน",
|
||||
joinTable: "SubDistrict",
|
||||
joinField: "name",
|
||||
},
|
||||
currentProvinceId: {
|
||||
propertyName: "currentProvinceName",
|
||||
type: "string",
|
||||
comment: "จังหวัดตามปัจจุบัน",
|
||||
joinTable: "Province",
|
||||
joinField: "name",
|
||||
},
|
||||
currentDistrictId: {
|
||||
propertyName: "currentDistrictName",
|
||||
type: "string",
|
||||
comment: "เขตตามปัจจุบัน",
|
||||
joinTable: "District",
|
||||
joinField: "name",
|
||||
},
|
||||
currentSubDistrictId: {
|
||||
propertyName: "currentSubDistrictName",
|
||||
type: "string",
|
||||
comment: "แขวงตามปัจจุบัน",
|
||||
joinTable: "SubDistrict",
|
||||
joinField: "name",
|
||||
},
|
||||
};
|
||||
|
||||
// การแทนที่ฟิลด์ ID ด้วยฟิลด์ Name สำหรับ Position entity
|
||||
private readonly POSITION_FIELD_REPLACEMENTS: Record<
|
||||
string,
|
||||
{ propertyName: string; type: string; comment: string; joinTable: string; joinField: string }
|
||||
> = {
|
||||
posTypeId: {
|
||||
propertyName: "posTypeName",
|
||||
type: "string",
|
||||
comment: "ประเภทตำแหน่ง",
|
||||
joinTable: "PosType",
|
||||
joinField: "posTypeName",
|
||||
},
|
||||
posLevelId: {
|
||||
propertyName: "posLevelName",
|
||||
type: "string",
|
||||
comment: "ระดับตำแหน่ง",
|
||||
joinTable: "PosLevel",
|
||||
joinField: "posLevelName",
|
||||
},
|
||||
posExecutiveId: {
|
||||
propertyName: "posExecutiveName",
|
||||
type: "string",
|
||||
comment: "ตำแหน่งทางการบริหาร",
|
||||
joinTable: "PosExecutive",
|
||||
joinField: "posExecutiveName",
|
||||
},
|
||||
};
|
||||
|
||||
private validateSuperAdminRole(user: any): void {
|
||||
if (!user.role.includes("SUPER_ADMIN")) {
|
||||
|
|
@ -364,11 +480,8 @@ export class ApiManageController extends Controller {
|
|||
|
||||
const result = this.entities
|
||||
.filter((entity) => entity.system.includes(system))
|
||||
.map(({ name, repository, description, isMain }) => ({
|
||||
tb: name,
|
||||
description,
|
||||
isMain: isMain || false,
|
||||
propertys: repository.metadata.columns
|
||||
.map(({ name, repository, description, isMain }) => {
|
||||
let columns = repository.metadata.columns
|
||||
.filter(
|
||||
(column: any) =>
|
||||
!column.isPrimary && !this.EXCLUDED_COLUMNS.includes(column.propertyName),
|
||||
|
|
@ -378,8 +491,74 @@ export class ApiManageController extends Controller {
|
|||
type: typeof column.type === "string" ? column.type : "string",
|
||||
comment: column.comment,
|
||||
key: column.propertyName,
|
||||
})),
|
||||
}));
|
||||
}));
|
||||
|
||||
// Special handling for Profile entity - replace ID fields with name fields
|
||||
if (name === "Profile") {
|
||||
const replacementKeys = Object.keys(this.PROFILE_FIELD_REPLACEMENTS);
|
||||
|
||||
// Remove ID fields that should be replaced
|
||||
columns = columns.filter(
|
||||
(col: { propertyName: string }) => !replacementKeys.includes(col.propertyName),
|
||||
);
|
||||
|
||||
// Add the corresponding name fields
|
||||
const nameFields = replacementKeys.map((key) => ({
|
||||
propertyName: this.PROFILE_FIELD_REPLACEMENTS[key].propertyName,
|
||||
type: "string",
|
||||
comment: this.PROFILE_FIELD_REPLACEMENTS[key].comment,
|
||||
key: this.PROFILE_FIELD_REPLACEMENTS[key].propertyName,
|
||||
}));
|
||||
|
||||
columns = [...columns, ...nameFields];
|
||||
}
|
||||
|
||||
// Special handling for Position entity - replace ID fields with name fields
|
||||
if (name === "Position") {
|
||||
const replacementKeys = Object.keys(this.POSITION_FIELD_REPLACEMENTS);
|
||||
|
||||
// Remove ID fields that should be replaced
|
||||
columns = columns.filter(
|
||||
(col: { propertyName: string }) => !replacementKeys.includes(col.propertyName),
|
||||
);
|
||||
|
||||
// Add the corresponding name fields
|
||||
const nameFields = replacementKeys.map((key) => ({
|
||||
propertyName: this.POSITION_FIELD_REPLACEMENTS[key].propertyName,
|
||||
type: "string",
|
||||
comment: this.POSITION_FIELD_REPLACEMENTS[key].comment,
|
||||
key: this.POSITION_FIELD_REPLACEMENTS[key].propertyName,
|
||||
}));
|
||||
|
||||
columns = [...columns, ...nameFields];
|
||||
}
|
||||
|
||||
// Special handling for PosMaster entity - add Profile fields for holder information
|
||||
if (name === "PosMaster") {
|
||||
// Add Profile fields that are accessible via current_holder relation
|
||||
const profileFields = ["prefix", "rank", "firstName", "lastName", "citizenId"];
|
||||
const profileRepository = AppDataSource.getRepository(Profile);
|
||||
const profileColumns = profileRepository.metadata.columns
|
||||
.filter(
|
||||
(column: any) => !column.isPrimary && profileFields.includes(column.propertyName),
|
||||
)
|
||||
.map((column: any) => ({
|
||||
propertyName: `Profile.${column.propertyName}`,
|
||||
type: typeof column.type === "string" ? column.type : "string",
|
||||
comment: column.comment,
|
||||
key: `Profile.${column.propertyName}`,
|
||||
}));
|
||||
|
||||
columns = [...columns, ...profileColumns];
|
||||
}
|
||||
|
||||
return {
|
||||
tb: name,
|
||||
description,
|
||||
isMain: isMain || false,
|
||||
propertys: columns,
|
||||
};
|
||||
});
|
||||
|
||||
return new HttpSuccess(result);
|
||||
} catch (error) {
|
||||
|
|
|
|||
|
|
@ -20,6 +20,153 @@ export class ApiWebServiceController extends Controller {
|
|||
private apiNameRepository = AppDataSource.getRepository(ApiName);
|
||||
private orgRevisionRepository = AppDataSource.getRepository(OrgRevision);
|
||||
private apiHistoryRepository = AppDataSource.getRepository(ApiHistory);
|
||||
private currentRevisionId: string = "";
|
||||
|
||||
// การแทนที่ฟิลด์ ID ด้วยฟิลด์ Name สำหรับ Profile entity
|
||||
private readonly PROFILE_FIELD_REPLACEMENTS: Record<
|
||||
string,
|
||||
{ propertyName: string; joinRelation: string; joinField: string }
|
||||
> = {
|
||||
posLevelName: {
|
||||
propertyName: "posLevelId",
|
||||
joinRelation: "posLevel",
|
||||
joinField: "posLevelName",
|
||||
},
|
||||
posTypeName: {
|
||||
propertyName: "posTypeId",
|
||||
joinRelation: "posType",
|
||||
joinField: "posTypeName",
|
||||
},
|
||||
registrationProvinceName: {
|
||||
propertyName: "registrationProvinceId",
|
||||
joinRelation: "registrationProvince",
|
||||
joinField: "name",
|
||||
},
|
||||
registrationDistrictName: {
|
||||
propertyName: "registrationDistrictId",
|
||||
joinRelation: "registrationDistrict",
|
||||
joinField: "name",
|
||||
},
|
||||
registrationSubDistrictName: {
|
||||
propertyName: "registrationSubDistrictId",
|
||||
joinRelation: "registrationSubDistrict",
|
||||
joinField: "name",
|
||||
},
|
||||
currentProvinceName: {
|
||||
propertyName: "currentProvinceId",
|
||||
joinRelation: "currentProvince",
|
||||
joinField: "name",
|
||||
},
|
||||
currentDistrictName: {
|
||||
propertyName: "currentDistrictId",
|
||||
joinRelation: "currentDistrict",
|
||||
joinField: "name",
|
||||
},
|
||||
currentSubDistrictName: {
|
||||
propertyName: "currentSubDistrictId",
|
||||
joinRelation: "currentSubDistrict",
|
||||
joinField: "name",
|
||||
},
|
||||
};
|
||||
|
||||
// การแทนที่ฟิลด์ ID ด้วยฟิลด์ Name สำหรับ Position entity
|
||||
private readonly POSITION_FIELD_REPLACEMENTS: Record<
|
||||
string,
|
||||
{ propertyName: string; joinRelation: string; joinField: string }
|
||||
> = {
|
||||
posTypeName: {
|
||||
propertyName: "posTypeId",
|
||||
joinRelation: "posType",
|
||||
joinField: "posTypeName",
|
||||
},
|
||||
posLevelName: {
|
||||
propertyName: "posLevelId",
|
||||
joinRelation: "posLevel",
|
||||
joinField: "posLevelName",
|
||||
},
|
||||
posExecutiveName: {
|
||||
propertyName: "posExecutiveId",
|
||||
joinRelation: "posExecutive",
|
||||
joinField: "posExecutiveName",
|
||||
},
|
||||
};
|
||||
|
||||
/**
|
||||
* build posMaster permission condition
|
||||
* @summary สร้างเงื่อนไขการกรองข้อมูลตามสิทธิ์การเข้าถึง
|
||||
*/
|
||||
private buildPosMasterPermissionCondition(
|
||||
accessType: string | undefined,
|
||||
dnaIds: {
|
||||
dnaRootId?: string | null;
|
||||
dnaChild1Id?: string | null;
|
||||
dnaChild2Id?: string | null;
|
||||
dnaChild3Id?: string | null;
|
||||
dnaChild4Id?: string | null;
|
||||
},
|
||||
tableAlias: string = "posMaster",
|
||||
): string {
|
||||
// ALL - no filtering
|
||||
if (accessType === "ALL") {
|
||||
return "1=1";
|
||||
}
|
||||
|
||||
// No access type specified but has DNA IDs - default to NORMAL behavior
|
||||
const conditions: string[] = [];
|
||||
|
||||
if (accessType === "ROOT" && dnaIds.dnaRootId) {
|
||||
// All organizations under this root
|
||||
conditions.push(
|
||||
`${tableAlias}.orgRootId IN (SELECT id FROM orgRoot WHERE orgRevisionId = "${this.currentRevisionId}" AND ancestorDNA LIKE "${dnaIds.dnaRootId}%")`,
|
||||
);
|
||||
} else if (accessType === "CHILD" || accessType === "NORMAL") {
|
||||
// Build conditions based on which DNA level is specified
|
||||
if (dnaIds.dnaChild4Id) {
|
||||
conditions.push(
|
||||
`${tableAlias}.orgChild4Id IN (SELECT id FROM orgChild4 WHERE orgRevisionId = "${this.currentRevisionId}" AND ancestorDNA = "${dnaIds.dnaChild4Id}")`,
|
||||
);
|
||||
} else if (dnaIds.dnaChild3Id) {
|
||||
conditions.push(
|
||||
`${tableAlias}.orgChild3Id IN (SELECT id FROM orgChild3 WHERE orgRevisionId = "${this.currentRevisionId}" AND ancestorDNA = "${dnaIds.dnaChild3Id}")`,
|
||||
);
|
||||
// For CHILD type, include all descendants
|
||||
if (accessType === "CHILD") {
|
||||
conditions.push(
|
||||
`(${tableAlias}.orgChild3Id IN (SELECT id FROM orgChild3 WHERE orgRevisionId = "${this.currentRevisionId}" AND ancestorDNA LIKE "${dnaIds.dnaChild3Id}%") OR ${tableAlias}.orgChild4Id IS NOT NULL)`,
|
||||
);
|
||||
}
|
||||
} else if (dnaIds.dnaChild2Id) {
|
||||
conditions.push(
|
||||
`${tableAlias}.orgChild2Id IN (SELECT id FROM orgChild2 WHERE orgRevisionId = "${this.currentRevisionId}" AND ancestorDNA = "${dnaIds.dnaChild2Id}")`,
|
||||
);
|
||||
if (accessType === "CHILD") {
|
||||
conditions.push(
|
||||
`(${tableAlias}.orgChild2Id IN (SELECT id FROM orgChild2 WHERE orgRevisionId = "${this.currentRevisionId}" AND ancestorDNA LIKE "${dnaIds.dnaChild2Id}%") OR ${tableAlias}.orgChild3Id IS NOT NULL)`,
|
||||
);
|
||||
}
|
||||
} else if (dnaIds.dnaChild1Id) {
|
||||
conditions.push(
|
||||
`${tableAlias}.orgChild1Id IN (SELECT id FROM orgChild1 WHERE orgRevisionId = "${this.currentRevisionId}" AND ancestorDNA = "${dnaIds.dnaChild1Id}")`,
|
||||
);
|
||||
if (accessType === "CHILD") {
|
||||
conditions.push(
|
||||
`(${tableAlias}.orgChild1Id IN (SELECT id FROM orgChild1 WHERE orgRevisionId = "${this.currentRevisionId}" AND ancestorDNA LIKE "${dnaIds.dnaChild1Id}%") OR ${tableAlias}.orgChild2Id IS NOT NULL)`,
|
||||
);
|
||||
}
|
||||
} else if (dnaIds.dnaRootId) {
|
||||
conditions.push(
|
||||
`${tableAlias}.orgRootId IN (SELECT id FROM orgRoot WHERE orgRevisionId = "${this.currentRevisionId}" AND ancestorDNA = "${dnaIds.dnaRootId}")`,
|
||||
);
|
||||
if (accessType === "CHILD") {
|
||||
conditions.push(
|
||||
`(${tableAlias}.orgRootId IN (SELECT id FROM orgRoot WHERE orgRevisionId = "${this.currentRevisionId}" AND ancestorDNA LIKE "${dnaIds.dnaRootId}%") OR ${tableAlias}.orgChild1Id IS NOT NULL)`,
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return conditions.length > 0 ? `(${conditions.join(" OR ")})` : "1=1";
|
||||
}
|
||||
|
||||
/**
|
||||
* list fields by systems
|
||||
|
|
@ -50,7 +197,7 @@ export class ApiWebServiceController extends Controller {
|
|||
}
|
||||
await isPermissionRequest(request, apiName.id);
|
||||
const offset = (page - 1) * pageSize;
|
||||
const propertyKey = apiName.apiAttributes.map((attr) => `${attr.tbName}.${attr.propertyKey}`);
|
||||
let propertyKey = apiName.apiAttributes.map((attr) => `${attr.tbName}.${attr.propertyKey}`);
|
||||
|
||||
let tbMain: string = "";
|
||||
let condition: string = "1=1";
|
||||
|
|
@ -78,6 +225,104 @@ export class ApiWebServiceController extends Controller {
|
|||
condition = `PosMaster.orgRevisionId = "${revision?.id}"`;
|
||||
}
|
||||
|
||||
let posMasterCondition: string = "";
|
||||
let posMasterAlias: string = "";
|
||||
|
||||
// Special handling for Profile and ProfileEmployee systems with permission filtering
|
||||
if (system == "registry") {
|
||||
// Get current revision
|
||||
const revision = await this.orgRevisionRepository.findOne({
|
||||
select: ["id"],
|
||||
where: { orgRevisionIsCurrent: true, orgRevisionIsDraft: false },
|
||||
});
|
||||
|
||||
// Store for use in permission building
|
||||
this.currentRevisionId = revision?.id || "";
|
||||
posMasterAlias = "posMaster";
|
||||
|
||||
// Build permission condition
|
||||
posMasterCondition = this.buildPosMasterPermissionCondition(
|
||||
request.user.accessType,
|
||||
{
|
||||
dnaRootId: request.user.dnaRootId,
|
||||
dnaChild1Id: request.user.dnaChild1Id,
|
||||
dnaChild2Id: request.user.dnaChild2Id,
|
||||
dnaChild3Id: request.user.dnaChild3Id,
|
||||
dnaChild4Id: request.user.dnaChild4Id,
|
||||
},
|
||||
posMasterAlias,
|
||||
);
|
||||
} else if (system == "registry_emp") {
|
||||
// Get current revision
|
||||
const revision = await this.orgRevisionRepository.findOne({
|
||||
select: ["id"],
|
||||
where: { orgRevisionIsCurrent: true, orgRevisionIsDraft: false },
|
||||
});
|
||||
|
||||
// Store for use in permission building
|
||||
this.currentRevisionId = revision?.id || "";
|
||||
posMasterAlias = "employeePosMaster";
|
||||
|
||||
// Build permission condition
|
||||
posMasterCondition = this.buildPosMasterPermissionCondition(
|
||||
request.user.accessType,
|
||||
{
|
||||
dnaRootId: request.user.dnaRootId,
|
||||
dnaChild1Id: request.user.dnaChild1Id,
|
||||
dnaChild2Id: request.user.dnaChild2Id,
|
||||
dnaChild3Id: request.user.dnaChild3Id,
|
||||
dnaChild4Id: request.user.dnaChild4Id,
|
||||
},
|
||||
posMasterAlias,
|
||||
);
|
||||
} else if (system == "registry_temp") {
|
||||
// Get current revision
|
||||
const revision = await this.orgRevisionRepository.findOne({
|
||||
select: ["id"],
|
||||
where: { orgRevisionIsCurrent: true, orgRevisionIsDraft: false },
|
||||
});
|
||||
|
||||
// Store for use in permission building
|
||||
this.currentRevisionId = revision?.id || "";
|
||||
posMasterAlias = "employeeTempPosMaster";
|
||||
|
||||
// Build permission condition
|
||||
posMasterCondition = this.buildPosMasterPermissionCondition(
|
||||
request.user.accessType,
|
||||
{
|
||||
dnaRootId: request.user.dnaRootId,
|
||||
dnaChild1Id: request.user.dnaChild1Id,
|
||||
dnaChild2Id: request.user.dnaChild2Id,
|
||||
dnaChild3Id: request.user.dnaChild3Id,
|
||||
dnaChild4Id: request.user.dnaChild4Id,
|
||||
},
|
||||
posMasterAlias,
|
||||
);
|
||||
} else if (system == "position") {
|
||||
// Get current revision
|
||||
const revision = await this.orgRevisionRepository.findOne({
|
||||
select: ["id"],
|
||||
where: { orgRevisionIsCurrent: true, orgRevisionIsDraft: false },
|
||||
});
|
||||
|
||||
// Store for use in permission building
|
||||
this.currentRevisionId = revision?.id || "";
|
||||
posMasterAlias = "PosMaster"; // Note: Uses PascalCase to match tbMain alias
|
||||
|
||||
// Build permission condition
|
||||
posMasterCondition = this.buildPosMasterPermissionCondition(
|
||||
request.user.accessType,
|
||||
{
|
||||
dnaRootId: request.user.dnaRootId,
|
||||
dnaChild1Id: request.user.dnaChild1Id,
|
||||
dnaChild2Id: request.user.dnaChild2Id,
|
||||
dnaChild3Id: request.user.dnaChild3Id,
|
||||
dnaChild4Id: request.user.dnaChild4Id,
|
||||
},
|
||||
posMasterAlias,
|
||||
);
|
||||
}
|
||||
|
||||
const repo = AppDataSource.getRepository(tbMain);
|
||||
const metadata = repo.metadata;
|
||||
|
||||
|
|
@ -92,6 +337,40 @@ export class ApiWebServiceController extends Controller {
|
|||
...new Set(propertyKey.map((x) => x.split(".")[0]).filter((tb) => tb !== tbMain)),
|
||||
];
|
||||
|
||||
// สำหรับ Profile: ตรวจสอบฟิลด์ที่ต้องการ join และแปลง propertyKey
|
||||
const profileFieldJoins: Record<string, string> = {}; // alias -> relationName
|
||||
if (tbMain === "Profile") {
|
||||
propertyKey = propertyKey.map((key) => {
|
||||
const [table, field] = key.split(".");
|
||||
if (table === "Profile") {
|
||||
const replacement = this.PROFILE_FIELD_REPLACEMENTS[field];
|
||||
if (replacement) {
|
||||
const alias = `${table}_${replacement.joinRelation}`;
|
||||
profileFieldJoins[alias] = replacement.joinRelation;
|
||||
return `${alias}.${replacement.joinField}`;
|
||||
}
|
||||
}
|
||||
return key;
|
||||
});
|
||||
}
|
||||
|
||||
// สำหรับ Position: ตรวจสอบฟิลด์ที่ต้องการ join และแปลง propertyKey
|
||||
const positionFieldJoins: Record<string, string> = {}; // alias -> relationName
|
||||
if (tbMain === "Position") {
|
||||
propertyKey = propertyKey.map((key) => {
|
||||
const [table, field] = key.split(".");
|
||||
if (table === "Position") {
|
||||
const replacement = this.POSITION_FIELD_REPLACEMENTS[field];
|
||||
if (replacement) {
|
||||
const alias = `${table}_${replacement.joinRelation}`;
|
||||
positionFieldJoins[alias] = replacement.joinRelation;
|
||||
return `${alias}.${replacement.joinField}`;
|
||||
}
|
||||
}
|
||||
return key;
|
||||
});
|
||||
}
|
||||
|
||||
const queryBuilder = repo.createQueryBuilder(tbMain);
|
||||
|
||||
// join กับตารารอง
|
||||
|
|
@ -107,6 +386,49 @@ export class ApiWebServiceController extends Controller {
|
|||
});
|
||||
}
|
||||
|
||||
// join สำหรับฟิลด์ Profile ที่ต้องการดึงค่าจากตารางอื่น
|
||||
if (tbMain === "Profile" && Object.keys(profileFieldJoins).length > 0) {
|
||||
Object.entries(profileFieldJoins).forEach(([alias, relationName]) => {
|
||||
queryBuilder.leftJoin(`${tbMain}.${relationName}`, alias);
|
||||
});
|
||||
}
|
||||
|
||||
// join สำหรับฟิลด์ Position ที่ต้องการดึงค่าจากตารางอื่น
|
||||
if (tbMain === "Position" && Object.keys(positionFieldJoins).length > 0) {
|
||||
Object.entries(positionFieldJoins).forEach(([alias, relationName]) => {
|
||||
queryBuilder.leftJoin(`${tbMain}.${relationName}`, alias);
|
||||
});
|
||||
}
|
||||
|
||||
// join สำหรับ PosMaster เมื่อต้องการดึงค่าจาก Profile (ข้อมูลคนครอง)
|
||||
const posMasterProfileFields: string[] = [];
|
||||
if (tbMain === "PosMaster") {
|
||||
propertyKey.forEach((key) => {
|
||||
if (key.startsWith("Profile.")) {
|
||||
posMasterProfileFields.push(key);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
// join PosMaster กับ Profile เมื่อมีการขอ Profile fields
|
||||
if (tbMain === "PosMaster" && posMasterProfileFields.length > 0) {
|
||||
queryBuilder.leftJoin("PosMaster.current_holder", "Profile");
|
||||
}
|
||||
|
||||
// join กับ posMaster/employeePosMaster/employeeTempPosMaster เพื่อกรองตามสิทธิ์การเข้าถึง
|
||||
if ((tbMain === "Profile" || tbMain === "ProfileEmployee") && posMasterCondition !== "1=1") {
|
||||
if (tbMain === "Profile") {
|
||||
queryBuilder.leftJoin("Profile.current_holders", "posMaster");
|
||||
} else if (tbMain === "ProfileEmployee") {
|
||||
// Use the correct relation based on posMasterAlias
|
||||
if (posMasterAlias === "employeeTempPosMaster") {
|
||||
queryBuilder.leftJoin("ProfileEmployee.current_holderTemps", "employeeTempPosMaster");
|
||||
} else {
|
||||
queryBuilder.leftJoin("ProfileEmployee.current_holders", "employeePosMaster");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// // เพิ่ม Main.id เพราะจะใช้ pk ในการแมบและนับจำนวน
|
||||
// if (!propertyKey.includes(`${Main}.id`)) {
|
||||
// propertyKey.push(`${Main}.id`);
|
||||
|
|
@ -125,6 +447,7 @@ export class ApiWebServiceController extends Controller {
|
|||
const [items, total] = await queryBuilder
|
||||
.select(propertyKey)
|
||||
.where(condition)
|
||||
.andWhere(posMasterCondition)
|
||||
.orderBy(propertyKey[0], "ASC")
|
||||
.skip(offset)
|
||||
.take(pageSize)
|
||||
|
|
@ -141,8 +464,55 @@ export class ApiWebServiceController extends Controller {
|
|||
|
||||
// split object id ออกก่อน return
|
||||
const data = items.map((item) => {
|
||||
const { [pk]: removedPk, ...x } = item;
|
||||
return x;
|
||||
const { [pk]: removedPk, ...rest } = item;
|
||||
|
||||
// สำหรับ Profile: แปลงฟิลด์ที่มาจาก join กลับเป็นชื่อเดิม
|
||||
if (tbMain === "Profile") {
|
||||
const flattened: any = { ...rest };
|
||||
Object.entries(this.PROFILE_FIELD_REPLACEMENTS).forEach(([nameField, config]) => {
|
||||
const alias = `${tbMain}_${config.joinRelation}`;
|
||||
if (rest[alias] && rest[alias][config.joinField] !== undefined) {
|
||||
flattened[nameField] = rest[alias][config.joinField];
|
||||
delete flattened[alias];
|
||||
}
|
||||
});
|
||||
return flattened;
|
||||
}
|
||||
|
||||
// สำหรับ Position: แปลงฟิลด์ที่มาจาก join กลับเป็นชื่อเดิม
|
||||
if (tbMain === "Position") {
|
||||
const flattened: any = { ...rest };
|
||||
Object.entries(this.POSITION_FIELD_REPLACEMENTS).forEach(([nameField, config]) => {
|
||||
// Remove the original ID field
|
||||
delete flattened[config.propertyName];
|
||||
// Add the name field from joined table
|
||||
const alias = `${tbMain}_${config.joinRelation}`;
|
||||
if (rest[alias] && rest[alias][config.joinField] !== undefined) {
|
||||
flattened[nameField] = rest[alias][config.joinField];
|
||||
}
|
||||
// Remove the joined table object
|
||||
delete flattened[alias];
|
||||
});
|
||||
return flattened;
|
||||
}
|
||||
|
||||
// สำหรับ PosMaster: แปลงฟิลด์ Profile ที่มาจาก join กลับเป็นฟิลด์ระดับบน
|
||||
if (tbMain === "PosMaster" && posMasterProfileFields.length > 0) {
|
||||
const flattened: any = { ...rest };
|
||||
// Extract Profile fields and add them at top level with "profile_" prefix to avoid conflicts
|
||||
if (rest["Profile"]) {
|
||||
flattened["profile_prefix"] = rest["Profile"].prefix;
|
||||
flattened["profile_rank"] = rest["Profile"].rank;
|
||||
flattened["profile_firstName"] = rest["Profile"].firstName;
|
||||
flattened["profile_lastName"] = rest["Profile"].lastName;
|
||||
flattened["profile_citizenId"] = rest["Profile"].citizenId;
|
||||
// Remove the nested Profile object
|
||||
delete flattened["Profile"];
|
||||
}
|
||||
return flattened;
|
||||
}
|
||||
|
||||
return rest;
|
||||
});
|
||||
|
||||
// console.log("queryBuilder ===> ", queryBuilder.getQuery());
|
||||
|
|
|
|||
|
|
@ -49,7 +49,7 @@ export class ScriptProfileOrgController extends Controller {
|
|||
* @summary Update org structure for profiles updated within a certain time window and sync to Keycloak
|
||||
*/
|
||||
@Post("update-org")
|
||||
public async cronjobUpdateOrg(@Request() request: RequestWithUser) {
|
||||
public async cronjobUpdateOrg(@Request() _request: RequestWithUser) {
|
||||
// Idempotency check - prevent concurrent runs
|
||||
if (this.isRunning) {
|
||||
console.log("cronjobUpdateOrg: Job already running, skipping this execution");
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue