feat (person): add check nick flow, and add nik validation
This commit is contained in:
@@ -3,11 +3,13 @@ package nurse
|
||||
import (
|
||||
ecore "simrs-vx/internal/domain/base-entities/core"
|
||||
ee "simrs-vx/internal/domain/main-entities/employee"
|
||||
eu "simrs-vx/internal/domain/main-entities/unit"
|
||||
)
|
||||
|
||||
type CreateDto struct {
|
||||
Employee_Id *uint `json:"employee_id"`
|
||||
IHS_Number *string `json:"ihs_number"`
|
||||
Unit_Id *uint16 `json:"unit_id"`
|
||||
}
|
||||
|
||||
type ReadListDto struct {
|
||||
@@ -19,6 +21,7 @@ type ReadListDto struct {
|
||||
type FilterDto struct {
|
||||
Employee_Id *uint `json:"employee_id"`
|
||||
IHS_Number *string `json:"ihs_number"`
|
||||
Unit_Id *uint16 `json:"unit_id"`
|
||||
|
||||
Page int `json:"page"`
|
||||
PageSize int `json:"page_size"`
|
||||
@@ -50,6 +53,8 @@ type ResponseDto struct {
|
||||
Employee_Id *uint `json:"employee_id"`
|
||||
Employee *ee.Employee `json:"employee,omitempty"`
|
||||
IHS_Number *string `json:"ihs_number"`
|
||||
Unit_Id *uint16 `json:"unit_id"`
|
||||
Unit *eu.Unit `json:"unit,omitempty"`
|
||||
}
|
||||
|
||||
func (d Nurse) ToResponse() ResponseDto {
|
||||
@@ -57,6 +62,8 @@ func (d Nurse) ToResponse() ResponseDto {
|
||||
Employee_Id: d.Employee_Id,
|
||||
Employee: d.Employee,
|
||||
IHS_Number: d.IHS_Number,
|
||||
Unit_Id: d.Unit_Id,
|
||||
Unit: d.Unit,
|
||||
}
|
||||
resp.Main = d.Main
|
||||
return resp
|
||||
|
||||
@@ -19,7 +19,6 @@ type CreateDto struct {
|
||||
PersonRelatives []epr.UpdateDto `json:"personRelatives"`
|
||||
RegisteredAt *time.Time `json:"registeredAt"`
|
||||
Status_Code erc.ActiveStatusCode `json:"status_code"`
|
||||
Number *string `json:"number"`
|
||||
}
|
||||
|
||||
type ReadListDto struct {
|
||||
|
||||
@@ -16,7 +16,7 @@ type CreateDto struct {
|
||||
PhoneNumber *string `json:"phoneNumber"`
|
||||
Education_Code *erp.EducationCode `json:"education_code"`
|
||||
Occupation_Code *erp.OcupationCode `json:"occupation_code"`
|
||||
Occupation_Name *string `json:"occupation_name"`
|
||||
Occupation_Name *string `json:"occupation_name" validate:"maxLength=50"`
|
||||
}
|
||||
|
||||
type ReadListDto struct {
|
||||
|
||||
@@ -17,13 +17,13 @@ type CreateDto struct {
|
||||
BirthDate *time.Time `json:"birthDate,omitempty"`
|
||||
BirthRegency_Code *string `json:"birthRegency_code"`
|
||||
Gender_Code *erp.GenderCode `json:"gender_code"`
|
||||
ResidentIdentityNumber *string `json:"residentIdentityNumber"`
|
||||
ResidentIdentityNumber *string `json:"residentIdentityNumber" validate:"nik"`
|
||||
PassportNumber *string `json:"passportNumber"`
|
||||
DrivingLicenseNumber *string `json:"drivingLicenseNumber"`
|
||||
Religion_Code *erp.ReligionCode `json:"religion_code"`
|
||||
Education_Code *erp.EducationCode `json:"education_code"`
|
||||
Ocupation_Code *erp.OcupationCode `json:"occupation_code"`
|
||||
Ocupation_Name *string `json:"occupation_name"`
|
||||
Ocupation_Name *string `json:"occupation_name" validate:"maxLength=50"`
|
||||
Ethnic_Code *string `json:"ethnic_code"`
|
||||
Language_Code *string `json:"language_code"`
|
||||
}
|
||||
@@ -60,6 +60,8 @@ type ReadDetailDto struct {
|
||||
Id uint `json:"id"`
|
||||
Name *string `json:"name"`
|
||||
ResidentIdentityNumber *string `json:"residentIdentityNumber"`
|
||||
PassportNumber *string `json:"passportNumber"`
|
||||
DrivingLicenseNumber *string `json:"drivingLicenseNumber"`
|
||||
}
|
||||
|
||||
type UpdateDto struct {
|
||||
|
||||
@@ -35,3 +35,10 @@ type Person struct {
|
||||
Language_Code *string `json:"language_code" gorm:"size:10"`
|
||||
Language *el.Language `json:"language,omitempty" gorm:"foreignKey:Language_Code;references:Code"`
|
||||
}
|
||||
|
||||
func (d Person) IsSameResidentIdentityNumber(input *string) bool {
|
||||
if input == nil {
|
||||
return false
|
||||
}
|
||||
return d.ResidentIdentityNumber == input
|
||||
}
|
||||
|
||||
@@ -0,0 +1,9 @@
|
||||
package validation
|
||||
|
||||
import (
|
||||
s "github.com/karincake/serabi"
|
||||
)
|
||||
|
||||
func RegisterValidation() {
|
||||
s.AddTagForRegex("nik", "^(1[1-9]|21|[37][1-6]|5[1-3]|6[1-5]|[89][12])\\d{2}\\d{2}([04][1-9]|[1256][0-9]|[37][01])(0[1-9]|1[0-2])\\d{2}\\d{4}$", "must be a valid nik format")
|
||||
}
|
||||
@@ -69,6 +69,7 @@ import (
|
||||
village "simrs-vx/internal/interface/main-handler/village"
|
||||
|
||||
///// Internal
|
||||
validation "simrs-vx/internal/interface/main-handler/helper/validation"
|
||||
"simrs-vx/internal/interface/main-handler/home"
|
||||
)
|
||||
|
||||
@@ -79,6 +80,7 @@ func SetRoutes() http.Handler {
|
||||
a.RegisterExtCall(zlc.Adjust)
|
||||
a.RegisterExtCall(ssdb.Init)
|
||||
a.RegisterExtCall(lh.Populate)
|
||||
a.RegisterExtCall(validation.RegisterValidation)
|
||||
|
||||
r := http.NewServeMux()
|
||||
|
||||
|
||||
@@ -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