feat (person): add check nick flow, and add nik validation
This commit is contained in:
@@ -19,4 +19,5 @@ func setData[T *e.CreateDto | *e.UpdateDto](input T, data *e.Nurse) {
|
||||
|
||||
data.Employee_Id = inputSrc.Employee_Id
|
||||
data.IHS_Number = inputSrc.IHS_Number
|
||||
data.Unit_Id = inputSrc.Unit_Id
|
||||
}
|
||||
|
||||
@@ -5,19 +5,14 @@ Any functions that are used internally by the use-case
|
||||
package patient
|
||||
|
||||
import (
|
||||
"errors"
|
||||
"fmt"
|
||||
e "simrs-vx/internal/domain/main-entities/patient"
|
||||
ep "simrs-vx/internal/domain/main-entities/person"
|
||||
"strconv"
|
||||
|
||||
up "simrs-vx/internal/use-case/main-use-case/person"
|
||||
|
||||
pl "simrs-vx/pkg/logger"
|
||||
|
||||
"gorm.io/gorm"
|
||||
dg "github.com/karincake/apem/db-gorm-pg"
|
||||
)
|
||||
|
||||
func setData[T *e.CreateDto | *e.UpdateDto](input T, data *e.Patient) {
|
||||
func setData[T *e.CreateDto | *e.UpdateDto](input T, data *e.Patient) error {
|
||||
var inputSrc *e.CreateDto
|
||||
if inputT, ok := any(input).(*e.CreateDto); ok {
|
||||
inputSrc = inputT
|
||||
@@ -26,40 +21,63 @@ func setData[T *e.CreateDto | *e.UpdateDto](input T, data *e.Patient) {
|
||||
inputSrc = &inputTemp.CreateDto
|
||||
}
|
||||
|
||||
data.Person_Id = inputSrc.Person_Id
|
||||
data.RegisteredAt = inputSrc.RegisteredAt
|
||||
data.Status_Code = inputSrc.Status_Code
|
||||
data.Number = inputSrc.Number
|
||||
}
|
||||
|
||||
func createOrUpdatePerson(input *e.CreateDto, event *pl.Event, tx *gorm.DB) error {
|
||||
if input.Person.Id == 0 {
|
||||
person, err := up.CreateData(input.Person.CreateDto, event, tx)
|
||||
if data.Id == 0 {
|
||||
medRecNum, err := GenerateNextMedicalRecordNumber()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
input.Person_Id = &person.Id
|
||||
return nil
|
||||
data.Number = &medRecNum
|
||||
}
|
||||
|
||||
person, err := up.ReadDetailData(ep.ReadDetailDto{Id: input.Person.Id}, event, tx)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
if person == nil {
|
||||
event.Status = "failed"
|
||||
event.ErrInfo = pl.ErrorInfo{
|
||||
Code: "data-notFound",
|
||||
Detail: "person with ID " + strconv.Itoa(int(*input.Person_Id)) + " not found",
|
||||
Raw: errors.New("person with ID " + strconv.Itoa(int(*input.Person_Id)) + " not found"),
|
||||
}
|
||||
return pl.SetLogError(event, input)
|
||||
}
|
||||
|
||||
if err := up.UpdateData(ep.UpdateDto{CreateDto: input.Person.CreateDto}, person, event, tx); err != nil {
|
||||
return err
|
||||
}
|
||||
input.Person_Id = &person.Id
|
||||
data.Person_Id = inputSrc.Person_Id
|
||||
data.RegisteredAt = inputSrc.RegisteredAt
|
||||
data.Status_Code = inputSrc.Status_Code
|
||||
return nil
|
||||
}
|
||||
|
||||
func GenerateNextMedicalRecordNumber() (string, error) {
|
||||
var last string
|
||||
err := dg.I.
|
||||
Table("\"Patient\"").
|
||||
Select("\"Number\"").
|
||||
Where("\"Number\" IS NOT NULL AND \"Number\" ~ '^[0-9]+$'"). // Only numeric strings
|
||||
Order("\"Number\"::bigint DESC").
|
||||
Limit(1).
|
||||
Scan(&last).Error
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
|
||||
var nextInt int64
|
||||
var format string
|
||||
|
||||
if last == "" {
|
||||
// No existing records, start with 10 digits
|
||||
nextInt = 1
|
||||
format = "%010d"
|
||||
} else {
|
||||
n, err := strconv.ParseInt(last, 10, 64)
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
nextInt = n + 1
|
||||
|
||||
// Dynamically determine format based on existing number
|
||||
digitCount := len(last)
|
||||
|
||||
// If the incremented number needs more digits, expand format
|
||||
nextStr := strconv.FormatInt(nextInt, 10)
|
||||
if len(nextStr) > digitCount {
|
||||
digitCount = len(nextStr)
|
||||
}
|
||||
|
||||
// Ensure minimum 10 digits as per requirement
|
||||
if digitCount < 10 {
|
||||
digitCount = 10
|
||||
}
|
||||
|
||||
format = fmt.Sprintf("%%0%dd", digitCount)
|
||||
}
|
||||
|
||||
return fmt.Sprintf(format, nextInt), nil
|
||||
}
|
||||
|
||||
@@ -15,7 +15,9 @@ func CreateData(input e.CreateDto, event *pl.Event, dbx ...*gorm.DB) (*e.Patient
|
||||
pl.SetLogInfo(event, nil, "started", "DBCreate")
|
||||
|
||||
data := e.Patient{}
|
||||
setData(&input, &data)
|
||||
if err := setData(&input, &data); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
var tx *gorm.DB
|
||||
if len(dbx) > 0 {
|
||||
@@ -114,7 +116,9 @@ func ReadDetailData(input e.ReadDetailDto, event *pl.Event, dbx ...*gorm.DB) (*e
|
||||
|
||||
func UpdateData(input e.UpdateDto, data *e.Patient, event *pl.Event, dbx ...*gorm.DB) error {
|
||||
pl.SetLogInfo(event, data, "started", "DBUpdate")
|
||||
setData(&input, data)
|
||||
if err := setData(&input, data); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
var tx *gorm.DB
|
||||
if len(dbx) > 0 {
|
||||
|
||||
@@ -8,6 +8,7 @@ import (
|
||||
"strconv"
|
||||
|
||||
dg "github.com/karincake/apem/db-gorm-pg"
|
||||
d "github.com/karincake/dodol"
|
||||
gh "github.com/karincake/getuk"
|
||||
"gorm.io/gorm"
|
||||
)
|
||||
@@ -100,7 +101,21 @@ func ReadDetailData(input e.ReadDetailDto, event *pl.Event, dbx ...*gorm.DB) (*e
|
||||
|
||||
tx = tx.Preload("Contacts").
|
||||
Preload("Addresses")
|
||||
if err := tx.First(&data, input.Id).Error; err != nil {
|
||||
|
||||
if input.ResidentIdentityNumber != nil {
|
||||
tx = tx.Where("\"ResidentIdentityNumber\" = ?", *input.ResidentIdentityNumber)
|
||||
}
|
||||
|
||||
if input.PassportNumber != nil {
|
||||
tx = tx.Where("\"PassportNumber\" = ?", *input.PassportNumber)
|
||||
}
|
||||
if input.DrivingLicenseNumber != nil {
|
||||
tx = tx.Where("\"DrivingLicenseNumber\" = ?", *input.DrivingLicenseNumber)
|
||||
}
|
||||
if input.Id != 0 {
|
||||
tx = tx.Where("\"Id\" = ?", input.Id)
|
||||
}
|
||||
if err := tx.First(&data).Error; err != nil {
|
||||
if processedErr := pu.HandleReadError(err, event, source, input.Id, data); processedErr != nil {
|
||||
return nil, processedErr
|
||||
}
|
||||
@@ -159,6 +174,7 @@ func DeleteData(data *e.Person, event *pl.Event, dbx ...*gorm.DB) error {
|
||||
}
|
||||
|
||||
func CreateOrUpdatePerson(input *e.UpdateDto, event *pl.Event, tx *gorm.DB) (*uint, error) {
|
||||
inputCreateData := input.CreateDto
|
||||
if input.Id == 0 {
|
||||
person, err := CreateData(input.CreateDto, event, tx)
|
||||
if err != nil {
|
||||
@@ -183,9 +199,30 @@ func CreateOrUpdatePerson(input *e.UpdateDto, event *pl.Event, tx *gorm.DB) (*ui
|
||||
return nil, pl.SetLogError(event, input)
|
||||
}
|
||||
|
||||
// check nik jika sama ok; jika tidak sama cari person dengan nik tersebut [yg idnya bukan id orang tersebut]
|
||||
// jika ketemu return erroor nik sudah digunakan; jika tidak ketemu berarti update nik
|
||||
if err := UpdateData(e.UpdateDto{CreateDto: input.CreateDto}, person, event, tx); err != nil {
|
||||
if person.IsSameResidentIdentityNumber(input.ResidentIdentityNumber) {
|
||||
if err := UpdateData(e.UpdateDto{CreateDto: input.CreateDto}, person, event, tx); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
return &person.Id, nil
|
||||
}
|
||||
|
||||
dup, err := ReadDetailData(e.ReadDetailDto{ResidentIdentityNumber: input.ResidentIdentityNumber}, event, tx)
|
||||
if err != nil {
|
||||
if fieldErr, ok := err.(d.FieldError); ok && fieldErr.Code == "data-notFound" {
|
||||
} else {
|
||||
return nil, err
|
||||
}
|
||||
} else if dup != nil && dup.Id != person.Id {
|
||||
event.Status = "failed"
|
||||
event.ErrInfo = pl.ErrorInfo{
|
||||
Code: "data-duplicate",
|
||||
Detail: "person with ResidentIdentityNumber " + *input.ResidentIdentityNumber + " already exists",
|
||||
Raw: errors.New("person with ResidentIdentityNumber " + *input.ResidentIdentityNumber + " already exists"),
|
||||
}
|
||||
return nil, pl.SetLogError(event, input)
|
||||
}
|
||||
|
||||
if err := UpdateData(e.UpdateDto{CreateDto: inputCreateData}, person, event, tx); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
|
||||
Reference in New Issue
Block a user