feat (person): add check nick flow, and add nik validation
This commit is contained in:
@@ -3,11 +3,13 @@ package nurse
|
|||||||
import (
|
import (
|
||||||
ecore "simrs-vx/internal/domain/base-entities/core"
|
ecore "simrs-vx/internal/domain/base-entities/core"
|
||||||
ee "simrs-vx/internal/domain/main-entities/employee"
|
ee "simrs-vx/internal/domain/main-entities/employee"
|
||||||
|
eu "simrs-vx/internal/domain/main-entities/unit"
|
||||||
)
|
)
|
||||||
|
|
||||||
type CreateDto struct {
|
type CreateDto struct {
|
||||||
Employee_Id *uint `json:"employee_id"`
|
Employee_Id *uint `json:"employee_id"`
|
||||||
IHS_Number *string `json:"ihs_number"`
|
IHS_Number *string `json:"ihs_number"`
|
||||||
|
Unit_Id *uint16 `json:"unit_id"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type ReadListDto struct {
|
type ReadListDto struct {
|
||||||
@@ -19,6 +21,7 @@ type ReadListDto struct {
|
|||||||
type FilterDto struct {
|
type FilterDto struct {
|
||||||
Employee_Id *uint `json:"employee_id"`
|
Employee_Id *uint `json:"employee_id"`
|
||||||
IHS_Number *string `json:"ihs_number"`
|
IHS_Number *string `json:"ihs_number"`
|
||||||
|
Unit_Id *uint16 `json:"unit_id"`
|
||||||
|
|
||||||
Page int `json:"page"`
|
Page int `json:"page"`
|
||||||
PageSize int `json:"page_size"`
|
PageSize int `json:"page_size"`
|
||||||
@@ -50,6 +53,8 @@ type ResponseDto struct {
|
|||||||
Employee_Id *uint `json:"employee_id"`
|
Employee_Id *uint `json:"employee_id"`
|
||||||
Employee *ee.Employee `json:"employee,omitempty"`
|
Employee *ee.Employee `json:"employee,omitempty"`
|
||||||
IHS_Number *string `json:"ihs_number"`
|
IHS_Number *string `json:"ihs_number"`
|
||||||
|
Unit_Id *uint16 `json:"unit_id"`
|
||||||
|
Unit *eu.Unit `json:"unit,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func (d Nurse) ToResponse() ResponseDto {
|
func (d Nurse) ToResponse() ResponseDto {
|
||||||
@@ -57,6 +62,8 @@ func (d Nurse) ToResponse() ResponseDto {
|
|||||||
Employee_Id: d.Employee_Id,
|
Employee_Id: d.Employee_Id,
|
||||||
Employee: d.Employee,
|
Employee: d.Employee,
|
||||||
IHS_Number: d.IHS_Number,
|
IHS_Number: d.IHS_Number,
|
||||||
|
Unit_Id: d.Unit_Id,
|
||||||
|
Unit: d.Unit,
|
||||||
}
|
}
|
||||||
resp.Main = d.Main
|
resp.Main = d.Main
|
||||||
return resp
|
return resp
|
||||||
|
|||||||
@@ -19,7 +19,6 @@ type CreateDto struct {
|
|||||||
PersonRelatives []epr.UpdateDto `json:"personRelatives"`
|
PersonRelatives []epr.UpdateDto `json:"personRelatives"`
|
||||||
RegisteredAt *time.Time `json:"registeredAt"`
|
RegisteredAt *time.Time `json:"registeredAt"`
|
||||||
Status_Code erc.ActiveStatusCode `json:"status_code"`
|
Status_Code erc.ActiveStatusCode `json:"status_code"`
|
||||||
Number *string `json:"number"`
|
|
||||||
}
|
}
|
||||||
|
|
||||||
type ReadListDto struct {
|
type ReadListDto struct {
|
||||||
|
|||||||
@@ -16,7 +16,7 @@ type CreateDto struct {
|
|||||||
PhoneNumber *string `json:"phoneNumber"`
|
PhoneNumber *string `json:"phoneNumber"`
|
||||||
Education_Code *erp.EducationCode `json:"education_code"`
|
Education_Code *erp.EducationCode `json:"education_code"`
|
||||||
Occupation_Code *erp.OcupationCode `json:"occupation_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 {
|
type ReadListDto struct {
|
||||||
|
|||||||
@@ -17,13 +17,13 @@ type CreateDto struct {
|
|||||||
BirthDate *time.Time `json:"birthDate,omitempty"`
|
BirthDate *time.Time `json:"birthDate,omitempty"`
|
||||||
BirthRegency_Code *string `json:"birthRegency_code"`
|
BirthRegency_Code *string `json:"birthRegency_code"`
|
||||||
Gender_Code *erp.GenderCode `json:"gender_code"`
|
Gender_Code *erp.GenderCode `json:"gender_code"`
|
||||||
ResidentIdentityNumber *string `json:"residentIdentityNumber"`
|
ResidentIdentityNumber *string `json:"residentIdentityNumber" validate:"nik"`
|
||||||
PassportNumber *string `json:"passportNumber"`
|
PassportNumber *string `json:"passportNumber"`
|
||||||
DrivingLicenseNumber *string `json:"drivingLicenseNumber"`
|
DrivingLicenseNumber *string `json:"drivingLicenseNumber"`
|
||||||
Religion_Code *erp.ReligionCode `json:"religion_code"`
|
Religion_Code *erp.ReligionCode `json:"religion_code"`
|
||||||
Education_Code *erp.EducationCode `json:"education_code"`
|
Education_Code *erp.EducationCode `json:"education_code"`
|
||||||
Ocupation_Code *erp.OcupationCode `json:"occupation_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"`
|
Ethnic_Code *string `json:"ethnic_code"`
|
||||||
Language_Code *string `json:"language_code"`
|
Language_Code *string `json:"language_code"`
|
||||||
}
|
}
|
||||||
@@ -60,6 +60,8 @@ type ReadDetailDto struct {
|
|||||||
Id uint `json:"id"`
|
Id uint `json:"id"`
|
||||||
Name *string `json:"name"`
|
Name *string `json:"name"`
|
||||||
ResidentIdentityNumber *string `json:"residentIdentityNumber"`
|
ResidentIdentityNumber *string `json:"residentIdentityNumber"`
|
||||||
|
PassportNumber *string `json:"passportNumber"`
|
||||||
|
DrivingLicenseNumber *string `json:"drivingLicenseNumber"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type UpdateDto struct {
|
type UpdateDto struct {
|
||||||
|
|||||||
@@ -35,3 +35,10 @@ type Person struct {
|
|||||||
Language_Code *string `json:"language_code" gorm:"size:10"`
|
Language_Code *string `json:"language_code" gorm:"size:10"`
|
||||||
Language *el.Language `json:"language,omitempty" gorm:"foreignKey:Language_Code;references:Code"`
|
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"
|
village "simrs-vx/internal/interface/main-handler/village"
|
||||||
|
|
||||||
///// Internal
|
///// Internal
|
||||||
|
validation "simrs-vx/internal/interface/main-handler/helper/validation"
|
||||||
"simrs-vx/internal/interface/main-handler/home"
|
"simrs-vx/internal/interface/main-handler/home"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -79,6 +80,7 @@ func SetRoutes() http.Handler {
|
|||||||
a.RegisterExtCall(zlc.Adjust)
|
a.RegisterExtCall(zlc.Adjust)
|
||||||
a.RegisterExtCall(ssdb.Init)
|
a.RegisterExtCall(ssdb.Init)
|
||||||
a.RegisterExtCall(lh.Populate)
|
a.RegisterExtCall(lh.Populate)
|
||||||
|
a.RegisterExtCall(validation.RegisterValidation)
|
||||||
|
|
||||||
r := http.NewServeMux()
|
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.Employee_Id = inputSrc.Employee_Id
|
||||||
data.IHS_Number = inputSrc.IHS_Number
|
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
|
package patient
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
"fmt"
|
||||||
e "simrs-vx/internal/domain/main-entities/patient"
|
e "simrs-vx/internal/domain/main-entities/patient"
|
||||||
ep "simrs-vx/internal/domain/main-entities/person"
|
|
||||||
"strconv"
|
"strconv"
|
||||||
|
|
||||||
up "simrs-vx/internal/use-case/main-use-case/person"
|
dg "github.com/karincake/apem/db-gorm-pg"
|
||||||
|
|
||||||
pl "simrs-vx/pkg/logger"
|
|
||||||
|
|
||||||
"gorm.io/gorm"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
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
|
var inputSrc *e.CreateDto
|
||||||
if inputT, ok := any(input).(*e.CreateDto); ok {
|
if inputT, ok := any(input).(*e.CreateDto); ok {
|
||||||
inputSrc = inputT
|
inputSrc = inputT
|
||||||
@@ -26,40 +21,63 @@ func setData[T *e.CreateDto | *e.UpdateDto](input T, data *e.Patient) {
|
|||||||
inputSrc = &inputTemp.CreateDto
|
inputSrc = &inputTemp.CreateDto
|
||||||
}
|
}
|
||||||
|
|
||||||
data.Person_Id = inputSrc.Person_Id
|
if data.Id == 0 {
|
||||||
data.RegisteredAt = inputSrc.RegisteredAt
|
medRecNum, err := GenerateNextMedicalRecordNumber()
|
||||||
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 err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
input.Person_Id = &person.Id
|
data.Number = &medRecNum
|
||||||
return nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
person, err := up.ReadDetailData(ep.ReadDetailDto{Id: input.Person.Id}, event, tx)
|
data.Person_Id = inputSrc.Person_Id
|
||||||
if err != nil {
|
data.RegisteredAt = inputSrc.RegisteredAt
|
||||||
return err
|
data.Status_Code = inputSrc.Status_Code
|
||||||
}
|
|
||||||
|
|
||||||
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
|
|
||||||
return nil
|
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")
|
pl.SetLogInfo(event, nil, "started", "DBCreate")
|
||||||
|
|
||||||
data := e.Patient{}
|
data := e.Patient{}
|
||||||
setData(&input, &data)
|
if err := setData(&input, &data); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
var tx *gorm.DB
|
var tx *gorm.DB
|
||||||
if len(dbx) > 0 {
|
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 {
|
func UpdateData(input e.UpdateDto, data *e.Patient, event *pl.Event, dbx ...*gorm.DB) error {
|
||||||
pl.SetLogInfo(event, data, "started", "DBUpdate")
|
pl.SetLogInfo(event, data, "started", "DBUpdate")
|
||||||
setData(&input, data)
|
if err := setData(&input, data); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
var tx *gorm.DB
|
var tx *gorm.DB
|
||||||
if len(dbx) > 0 {
|
if len(dbx) > 0 {
|
||||||
|
|||||||
@@ -8,6 +8,7 @@ import (
|
|||||||
"strconv"
|
"strconv"
|
||||||
|
|
||||||
dg "github.com/karincake/apem/db-gorm-pg"
|
dg "github.com/karincake/apem/db-gorm-pg"
|
||||||
|
d "github.com/karincake/dodol"
|
||||||
gh "github.com/karincake/getuk"
|
gh "github.com/karincake/getuk"
|
||||||
"gorm.io/gorm"
|
"gorm.io/gorm"
|
||||||
)
|
)
|
||||||
@@ -100,7 +101,21 @@ func ReadDetailData(input e.ReadDetailDto, event *pl.Event, dbx ...*gorm.DB) (*e
|
|||||||
|
|
||||||
tx = tx.Preload("Contacts").
|
tx = tx.Preload("Contacts").
|
||||||
Preload("Addresses")
|
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 {
|
if processedErr := pu.HandleReadError(err, event, source, input.Id, data); processedErr != nil {
|
||||||
return nil, processedErr
|
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) {
|
func CreateOrUpdatePerson(input *e.UpdateDto, event *pl.Event, tx *gorm.DB) (*uint, error) {
|
||||||
|
inputCreateData := input.CreateDto
|
||||||
if input.Id == 0 {
|
if input.Id == 0 {
|
||||||
person, err := CreateData(input.CreateDto, event, tx)
|
person, err := CreateData(input.CreateDto, event, tx)
|
||||||
if err != nil {
|
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)
|
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]
|
if person.IsSameResidentIdentityNumber(input.ResidentIdentityNumber) {
|
||||||
// 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 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
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user