diff --git a/assets/docs/logo-provinsi-jawa-timur.png b/assets/docs/logo-provinsi-jawa-timur.png new file mode 100644 index 00000000..88903ac0 Binary files /dev/null and b/assets/docs/logo-provinsi-jawa-timur.png differ diff --git a/assets/docs/logo-rssa.png b/assets/docs/logo-rssa.png new file mode 100644 index 00000000..4af06927 Binary files /dev/null and b/assets/docs/logo-rssa.png differ diff --git a/assets/docs/resume.html b/assets/docs/resume.html index 8715f1ba..1b1d3ff2 100644 --- a/assets/docs/resume.html +++ b/assets/docs/resume.html @@ -48,7 +48,7 @@
-
+
|
diff --git a/cmd/main-migration/migrations/20251202063820.sql b/cmd/main-migration/migrations/20251202063820.sql
deleted file mode 100644
index 52b2689e..00000000
--- a/cmd/main-migration/migrations/20251202063820.sql
+++ /dev/null
@@ -1,17 +0,0 @@
--- Create "VaccineData" table
-CREATE TABLE "public"."VaccineData" (
- "Id" bigserial NOT NULL,
- "CreatedAt" timestamptz NULL,
- "UpdatedAt" timestamptz NULL,
- "DeletedAt" timestamptz NULL,
- "Vaccine_type_Code" text NULL,
- "Encounter_Id" bigint NULL,
- "Vaccine_batch_number" bigint NULL,
- "Vaccine_dose" bigint NULL,
- "Dose_order" bigint NULL,
- "Injection_location" text NULL,
- "Vaccination_datetime" timestamptz NULL,
- "Vaccine_expiration_date" timestamptz NULL,
- PRIMARY KEY ("Id"),
- CONSTRAINT "fk_VaccineData_Encounter" FOREIGN KEY ("Encounter_Id") REFERENCES "public"."Encounter" ("Id") ON UPDATE NO ACTION ON DELETE NO ACTION
-);
diff --git a/cmd/main-migration/migrations/20251217043504.sql b/cmd/main-migration/migrations/20251217043504.sql
new file mode 100644
index 00000000..890819fc
--- /dev/null
+++ b/cmd/main-migration/migrations/20251217043504.sql
@@ -0,0 +1,24 @@
+-- Create "KFR" table
+CREATE TABLE "public"."KFR" (
+ "Id" bigserial NOT NULL,
+ "CreatedAt" timestamptz NULL,
+ "UpdatedAt" timestamptz NULL,
+ "DeletedAt" timestamptz NULL,
+ "Encounter_Id" bigint NULL,
+ "CreatedBy_Employee_Id" bigint NULL,
+ "Type" character varying(15) NULL,
+ "Subjective" text NULL,
+ "Objective" text NULL,
+ "Assessment" text NULL,
+ "TreatmentGoals" text NULL,
+ "Education" text NULL,
+ "Action" text NULL,
+ "Frequency" bigint NULL,
+ "IntervalUnit_Code" character varying(10) NULL,
+ "FollowUpType" character varying(10) NULL,
+ "FollowUpNote" text NULL,
+ "Status_Code" character varying(10) NOT NULL,
+ PRIMARY KEY ("Id"),
+ CONSTRAINT "fk_KFR_CreatedBy_Employee" FOREIGN KEY ("CreatedBy_Employee_Id") REFERENCES "public"."Employee" ("Id") ON UPDATE NO ACTION ON DELETE NO ACTION,
+ CONSTRAINT "fk_KFR_Encounter" FOREIGN KEY ("Encounter_Id") REFERENCES "public"."Encounter" ("Id") ON UPDATE NO ACTION ON DELETE NO ACTION
+);
diff --git a/cmd/main-migration/migrations/atlas.sum b/cmd/main-migration/migrations/atlas.sum
index 4b0c4948..763c31aa 100644
--- a/cmd/main-migration/migrations/atlas.sum
+++ b/cmd/main-migration/migrations/atlas.sum
@@ -1,4 +1,4 @@
-h1:hG7tbR8A3hp2zgbUt9pwONdJrHiSfrgcVBWoRB8Cx9k=
+h1:w6tFa/NMDQdQDXl91m3KqSaPSa9ktGQH7la07Gu6eoc=
20250904105930.sql h1:MEM6blCgke9DzWQSTnLzasbPIrcHssNNrJqZpSkEo6k=
20250904141448.sql h1:J8cmYNk4ZrG9fhfbi2Z1IWz7YkfvhFqTzrLFo58BPY0=
20250908062237.sql h1:Pu23yEW/aKkwozHoOuROvHS/GK4ngARJGdO7FB7HZuI=
@@ -75,92 +75,93 @@ h1:hG7tbR8A3hp2zgbUt9pwONdJrHiSfrgcVBWoRB8Cx9k=
20251106040137.sql h1:ppcqkVoT0o9jZcjI/TN7LuaPxXhJQhnIXEJtloP/46o=
20251106041333.sql h1:2JkxyelQ/EeB+boL5bfpnzefw32ttEGKvKchtQjWmAU=
20251106042006.sql h1:ruppYa1kAJQUU3ufQBbKGMcXrGbGJJiRPclT+dNc/YQ=
-20251106050412.sql h1:1002KYtHd8AwrQTMewbs/PPHDylHDghigE/3S7PVdMA=
-20251106063418.sql h1:jPW/gBnbFl4RO39lQ0ZMDtYA6xbhyD6CgQupT50HmaY=
-20251106071906.sql h1:leYGKxR3EQn794aOehf0sd/ZPmOnvBMZPy5/anGmRB4=
-20251106073157.sql h1:KASMzjjjk5UB7Zj8lCRtM1utc4ZnDjlnpZbtTe3vONE=
-20251106074218.sql h1:Z5q5deOvLaZDPhiVTN9st3/s56RepBa2YOyrMXBdj4A=
-20251106081846.sql h1:P+VsWwhGt60adDIZuE/Aa38JVp/yX1rnsdpXpxASodw=
-20251106082844.sql h1:Dmi5A8i9frQZvdXYPwc7f8CisZtBH8liSXq1rI6z1iM=
-20251106090021.sql h1:4JwdKgO8T46YhyWVJUxpRIwudBDlG8QN1brSOYmgQ20=
-20251106144745.sql h1:nqnQCzGrVJaq8ilOEOGXeRUL1dolj+OPWKuP8A92FRA=
-20251107012049.sql h1:Pff4UqltGS3clSlGr0qq8CQM56L29wyxY0FC/N/YAhU=
-20251107064812.sql h1:GB9a0ZfMYTIoGNmKUG+XcYUsTnRMFfT4/dAD71uCPc4=
-20251107064937.sql h1:IC5pw1Ifj30hiE6dr5NMHXaSHoQI+vRd40N5ABgBHRI=
-20251107071420.sql h1:9NO3iyLEXEtWa2kSRjM/8LyzuVIk6pdFL2SuheWjB08=
-20251107074318.sql h1:7fHbSRrdjOmHh/xwnjCLwoiB5cW5zeH+uxLV0vZbkIA=
-20251107075050.sql h1:np+3uTOnU9QNtK7Knaw8eRMhkyB9AwrtSNHphOBxbHY=
-20251107080604.sql h1:cXDBLPJDVWLTG6yEJqkJsOQ7p7VYxLM2SY+mwO8qSHo=
-20251107081830.sql h1:/S7OQZo4ZnK80t28g/JyiOTZtmWG/dP5Wg2zXNMQ/iE=
-20251107091033.sql h1:/cbkF1nO/IjNSIfDJJx456KJtQ9rWFXOBFAkR/M2xiE=
-20251107091209.sql h1:jrLQOUeV8ji2fg0pnEcs1bw4ANUxzTSMXC/rrHLIY+M=
-20251107091541.sql h1:6UqbhQQRmzA2+eKu5lIvkwOkk+lH70QLZC8Pjpjcq68=
-20251110012217.sql h1:C9HpX0iyHzKjyNv/5DSAn2MCHj6MX4p5UQ/NrY7QD0w=
-20251110012306.sql h1:J54yb27d30LBbYp9n1P66gFVRlxPguKu0kxmWIBBG8g=
-20251110052049.sql h1:232T2x8xTczJl9nk4jxJpZXhoOGYthhxjJ7nK8Jd8vg=
-20251110062042.sql h1:WnfVUXrzYoj8qdkkjO9/JQQ8agGd4GfSHQdMjo7LDAg=
-20251110063202.sql h1:hSzGfwVMWa6q3vwIQZUkxKgBNCzHjB+6GKy54zfV+oQ=
-20251110063633.sql h1:/VpofIAqNS1CnazEnpW/+evbzn9Kew3xDW48r57M+Xg=
-20251110085551.sql h1:bFZwSmfvVbTUr/enWB82WqjG88gpqcZ6s45btUvO0uo=
-20251110091516.sql h1:KkJMwPQuaZQhiqnKrNQrgP12gw9rV8T3P2o3mtGTcvY=
-20251110091948.sql h1:I4odAYrJdvNf1jPw6ppDC0XdI7v6vKBACg/ABwUgA7I=
-20251110092729.sql h1:l1out8soEmVP6dNjaIOtGYo6QDcoJZRI8X1sjZ5ZGmo=
-20251110093522.sql h1:nsz8jCxGjEdr/bz9g+4ozfZzIP803xONjVmucad1GMc=
-20251110100258.sql h1:IBqt1VZj5WjQ+l9aAFGHOCCBtzb03KlLLihFLut7itg=
-20251110100545.sql h1:6/LV7751iyKxE2xI6vO1zly+aHUwxXD/IBwLcVpKxqM=
-20251110155448.sql h1:kFPobJB+cpflsXBAWUwy3lohuWvrb/VRlXnhJWl7i3Y=
-20251111072601.sql h1:ch8F+yVhsSM5xY+TwMLY3PxdLa4Wuhtj76oyw79R7Js=
-20251111073546.sql h1:cCv0NPscADAOBahRVqtDWFs6G2t7n+4a+RwlF8vk/c4=
-20251111074148.sql h1:70TsV83u1gQ5TktI13K7NQiyCCa35Td2aR6CNtKUa4U=
-20251111074652.sql h1:ddfQ/sRKMezPM75xBFTGytUQX5AwZ3znrJVpg73gKPA=
-20251111082257.sql h1:ZsdLY1ROouos0l3oS0lkeSiuKLEUGbVvBhpcM2AVhkw=
-20251111111017.sql h1:qrJ93dNtQwcuAvpsP/lAK/H63C4cinXrsVaPmWsTqkU=
-20251113101344.sql h1:xaOZvAUP1fFfnO+syEFOzJUIg5lTfBe5AWHPbBWuCLA=
-20251113120533.sql h1:f3/U1Ve2yF2zSMhkt+xtwF8wUYfUKYwgbNeGfE37EW4=
-20251114062746.sql h1:FInLaEFQByESEwFJKuKnuUSTKmcDpi3ZXaxkKwz2+D8=
-20251117005942.sql h1:wD3BWrUSmo1HlW16V3lkaBkJvbAZ0fNk77te7J9NhOc=
-20251117075427.sql h1:TqU9VKZa3I8YNXUGQWY3WVBYN+1FvyyaKy0hB1jgAho=
-20251118074929.sql h1:p1KsWqCuR1JXA/jVO5BmOhCcaQ8clT7t0YRszAhPFbg=
-20251119063438.sql h1:NVGM0X/LHD37EaPl8SNzkNiZDJ7AB1QR+LLwLh6WRdg=
-20251119065730.sql h1:U5lzk1WvMB0bw3kwckou7jkEt4bwdYItwHr2Vxqe7w4=
-20251119072302.sql h1:qCuI2WMEMF/XNbjV+RXPjBnuCKLu1Fia+mR9HiLWBIs=
-20251119072450.sql h1:Xg+bTwqGyKPNFEQhJylvpz1wifdfmDJvcAq6vmNf0Ng=
-20251120005512.sql h1:Ek6qpacAI/qVuTYxKno+uJyzn7s5z9pf3t7VA8gTzm4=
-20251120074415.sql h1:NNUeJVA03EeBHJhHqPXEZoDv/PnC6yK1/cRhmukyaJo=
-20251121033803.sql h1:/vfvFX/3pzSCIHnSbMUT9EMBDykOpVkvyfeTEle9Vas=
-20251124071457.sql h1:qg2dhCL9YwD13xnfJ175lW/p6MGfzFKaBqd908FByRc=
-20251125125303.sql h1:4JSFv1Pmhbe9tqpLXgO63OwYnGsalStgUXKVWPyc1YE=
-20251126064057.sql h1:vAdhz5Nn/gGJy0UKZAEldeXv8HpHtJU/t8ygDVIbTsU=
-20251201081333.sql h1:PoC8ADRdwDuohDTB74yW/DaB42igdYa4B6humbrEJBk=
-20251201104439.sql h1:tpqdrOf9d2aGwZshqm62nG6SXnfVaO/g6A7z0efPS14=
-20251201113804.sql h1:kIEmVoETJXBkab2Q+b3y/pP84eF8W2BdQ47amHCnc+c=
-20251201113858.sql h1:KLXKZO5XTQPoEU0YLHE8Fhg9WPKpSN3wNgYPJ+RGFcg=
-20251201114751.sql h1:HM17diiPknfSHAmP+kJGP6GzToaPU9/NT+KQBpf3Jq0=
-20251201114913.sql h1:gqucFgHFFLA6n/Rdz486cZH5xkaJuwefESLJMJLDue8=
-20251202030445.sql h1:QWBVfTepT7DaXP5E0BYoxNM0JwKIQ2qIMXzI4kbz/qE=
-20251202044430.sql h1:4QZH9lz0GrQ9rzP1AJ+hJgGalNpp+8FCRckNK7xaTbU=
-20251202064000.sql h1:/EN7sT1ol/91qW1aXWrzX+Mc3XOC/7f/LtfA0JRHpbg=
-20251202130629.sql h1:9mvalqfhqGCdkcJepJDzHprU2xb0i5sYys1Htf62ioo=
-20251202160848.sql h1:Kd2/TziKSMezrt4XgbjQcYvY/Lo9rX0qw7/Lz0/oyKk=
-20251202180207.sql h1:IHmSMIO3ia+YV5GULixbdlV1joaUAWtnjQHPd8+HKiM=
-20251202231005.sql h1:lua0KKoeBptSfs/6ehZE6Azo6YUlNkOJwGFyb1HQWkY=
-20251203205052.sql h1:nk0VK2Uv4bHE3SBfHo/aevVZxtHzr7zAzvmMU8TCCtk=
-20251205073858.sql h1:46qqXnArgJmzGE/WO7v7Ev8Jh7BudDiGbdANexq/5Dk=
-20251205211957.sql h1:3fvtZ/mBWsTIXllXMFOuCLJsp4MivVP56dunehlU0yo=
-20251205214433.sql h1:rn3++FEfX7ntcJcOmCEuOMnr27TZqH0KMGRppzFwFTc=
-20251205221124.sql h1:CRruUvGZqlVDBmbQSVEl4wFm+uq30AurMMDI6sb8jxg=
-20251206021053.sql h1:bpuEocu4lOhZ7oLuxd//22dzjfNgU2iaWEqSD1mVwoU=
-20251207020537.sql h1:m6uh4NHVF3EKNTVMqOmuBSDFD9oCQk5mAwo05fT46G4=
-20251207212015.sql h1:UPelYGTeUR6rm8mU8dfNzgRDEDun0UQ4tkgsaDljn30=
-20251207221222.sql h1:bTfUCvCf2UPh+BA2IY2PHQafb9DwY9nhH5FRuMEHy+0=
-20251209022744.sql h1:y5/PAiZH/fYCpDJpkQdNRJwWICHH2iNIwM1V+S1P6KA=
-20251209025908.sql h1:p3kZA8kyEj+mQZSrdY3k2K1NojQzFJh/MlZJ0Oy6t/k=
-20251209030538.sql h1:zltV6/Fu2zJW0/lVBl7MdnWuJcqNTUIRcqYYZ8Fi1wo=
-20251209051742.sql h1:BBNSmWfkamWrcKdxWjPiBS9yJ8yyQQUQIj3kip53nuE=
-20251209064304.sql h1:Xs73yQbuJvuQ0OnW1FAZpeytmUl/bGTlJFrwGOsTF4w=
-20251209070128.sql h1:fPGE6xOV6uCiVOqnvwn2L/GsBbgp2wxgmZOhF3bSGGM=
-20251209084929.sql h1:u4LPMvkGAH4RfGC2IlBTIm7T7paMHoBSvTQ0w5Br7d0=
-20251210145148.sql h1:rejGrnTpaygxPv06v0vxMytF4rk1OJBXaw3ttSmidgc=
-20251211101547.sql h1:+jT5yRCEsSRExzoawrqymS/I7lVfwUQQSgSzbxCxgRk=
-20251211113942.sql h1:hRwiVZnXGzgUbqOk5TZ6ZHzGs1GebIFeIKkJNb+6+f0=
-20251216074834.sql h1:OJkATzCGB8NQlhaNblRMRhBLuT5XV1YSUWhtYjJzxl0=
+20251106050412.sql h1:MiEMJ1HCFYnalKuq3Z38xJeogfBAMqsTv2sG4EF8dDw=
+20251106063418.sql h1:y3veDJPjKekOWLCZek/LgQwXPRhZtOppTfUXiqoL95s=
+20251106071906.sql h1:/TUZA3XpMY23qEJXdkTwlzrNMvSSl6JJniPcgAttBaw=
+20251106073157.sql h1:78txeibJ602DMD7huD618ZSMt6phSRzDNPTlo0PGyrc=
+20251106074218.sql h1:8Xz7WywrtUnSxOHhlal53gG9rE7r86LFUt5zBFe/mIs=
+20251106081846.sql h1:jp91Bf5bxGXMiUB1VIuN6y768vb2iWwow44WfCE5J5k=
+20251106082844.sql h1:RHYzRO4G1fSWwf+xc/3QezZ/Iil67cZPIgNpNz3TNhQ=
+20251106090021.sql h1:dFDk6mq+zjbYWmfWIrHf9DiKvvoXHjrr0++zssMTWP8=
+20251106144745.sql h1:aHcr23iBFqCHer5D/SsPMXBCLjGqUYvWYfRU8jSJgIw=
+20251107012049.sql h1:hu/7NHhnAkT4xK0RNtqmMDdH1Bo5EZbl7itDRjiCT+g=
+20251107064812.sql h1:sfCXDQYnMf0ddrQ9oYljWJLLSt9NJjJV6o8VS3p7aZE=
+20251107064937.sql h1:DlYGJ9LZFwZyR7jBP5zaGB128aIc4HAixBKPYCz9EkY=
+20251107071420.sql h1:ynCdZAd2utLl+FhtWZwtahNXgIVOvuk3s/rOq7lfXA4=
+20251107074318.sql h1:WE9cPhibWtZ0dbu1VEGirTeY6ijFYGMNhHdBtM32kOc=
+20251107075050.sql h1:8tvneruqdynDOaJK1+0z4CH7YXZStZpGdqwIeOMLik4=
+20251107080604.sql h1:8c4jd4Tql7tcdhbI9NS0tgvN+ADu9FnCf8wMUbmW7A0=
+20251107081830.sql h1:SAAe3lmsm9vGXuSEsDdl7ad0EAxP5CMmFRDEgp9M7yY=
+20251107091033.sql h1:JLdX/u7GUdBfjrPrMSNAqc8HtSoj0YA9iW9Vc6FJZdw=
+20251107091209.sql h1:CzhYtwAwT+GHrbqcagnJE+v3mbl/rObf1IJaLCKlzrs=
+20251107091541.sql h1:+3ZyWJTftDY2JeWThXuIxGWpUBnyMPyOyY4jBjdWYJI=
+20251110012217.sql h1:f4Z8TuGc+XMSJ+Ekn4/PeHRE2FlHWkc5gKPJB0hAX7c=
+20251110012306.sql h1:ENPyI6Kjdk6qKJQb0yJ6MCTDPAmO1WD/uhKuCSl+jYo=
+20251110052049.sql h1:OrQ0acnyoQLKnTitZfnBcVr5jDslF59OFLaqT7SpdVs=
+20251110062042.sql h1:9KwldQt0NpVPR86L0T4hlkfHAGau+7CiZYgu5rF+yhg=
+20251110063202.sql h1:A117DuZmZ6U0jWHA3DISnr+yvBjKIr1ObrUr047YezQ=
+20251110063633.sql h1:qTiC0F19JnhUIXF4LGJQ21jEV6kKGyhTr1x2kimFqPQ=
+20251110085551.sql h1:HZcJM0RSC6HBaUSjKBE8MgDG8Vn9f3LmwA/OnT9Cp7I=
+20251110091516.sql h1:W3AQhQLgirEWuCObbLl+Prdrbq6k6EEY1xcoWsmbog4=
+20251110091948.sql h1:3tsITMrZr/T+L4wqUMz8sHS229jCJl4T0Nu3dMccxH8=
+20251110092729.sql h1:uU+k88RH/e0Ns4/SmJl03RVYPscBAPuiLfxR6CJqaf0=
+20251110093522.sql h1:O7upSj8VNjzvroL4IU59bfxKATOkAVGBArcUbVNq9aM=
+20251110100258.sql h1://JSArUMNI3/gAtYDx2VN94C198SFW0ANjgs+p6eCRM=
+20251110100545.sql h1:ENPOqeJYRpMI4e8VCKwaQgaql8se6pIidAhG2cjskBg=
+20251110155448.sql h1:VW9KE0jxWy4flZ28sdXmhY6JWd6eKAX/cVL8KWRVii4=
+20251111072601.sql h1:99WWzGjcDDFNC2cHRfPu4MBLWNrgPMY5HAYUbmIcVRA=
+20251111073546.sql h1:iIGwFNfUgStdczw/PXTH3SD84xAyxrbZICofc3M8TMk=
+20251111074148.sql h1:mzkezSKwCV2bTZ/0BHiTCX5qAy4uHpwar1xzwAH15Ko=
+20251111074652.sql h1:lYh4/3BHV24pgPC0pP4RUKb+XtL/6AsZGPItRpnzBiQ=
+20251111082257.sql h1:+cIZ1lf8HYGSL6t6U88plLKk8nOO1YVdV7h3YOM84ds=
+20251111111017.sql h1:xzlhR2xLfOj4ddYlesS+bpEeDrxiAf5ILNOspsbZjBU=
+20251113101344.sql h1:vSzThY3p6XYTj0dJ2uFVkHmlytyrFAnGE58CJLx364I=
+20251113120533.sql h1:lfjgdqRGo/EohrVw8sWlFbDjh3ASTsfQ6pr3qjafqvc=
+20251114062746.sql h1:6DLYjfJ60PkAABZTXvOwSE+xrU25oyoX7gpYlBnA9cw=
+20251117005942.sql h1:0XoJKca8IOaB9QKFVF/qPY7jKcIgh6m/LODQuE06SAs=
+20251117075427.sql h1:LhY2urosfoSu1/vkHmgsNP4JA4DuWBs9gL59yMqcF8M=
+20251118074929.sql h1:3RWBD6BziQVw6WSfthgoVuhTELHER9NrIuZm4hY/F1A=
+20251119063438.sql h1:EVTG3ZrHjCmM95YPASZTRPiwHrG6e33A2vO4hLSAaBQ=
+20251119065730.sql h1:s98wnZOCW6NbnwDS3H53XIQ60u6B9bDwBLNvy5+Rn64=
+20251119072302.sql h1:ZL+VHekLYqgNtOFKlj02WHrAk11dtTxQkmyTKE8IOzY=
+20251119072450.sql h1:SiJy2vpSvoPFw4J1SW7HjGSJzrqR62NsjRYAeabV+kc=
+20251120005512.sql h1:hJUaWXYJ3HiSquBTw8OKhymjA4O43ehAMGfiOY8W87Q=
+20251120074415.sql h1:dNmcqZHqfZwi/wtYvO/pSFgImN2scXL0p/7g0+HLp0s=
+20251121033803.sql h1:onlddYGo+tQdGaEdJPqdsx3sncGnwEvqMSCksF6vjjI=
+20251124071457.sql h1:ikQLIXFikUbkUd55uJBmEvqLGEC9t0db+Om4TQsWP7M=
+20251125125303.sql h1:OLKbNPO36AHtIDursW9AeBvf60sahQKC1iOjHmpx0MA=
+20251126064057.sql h1:6al3PTWbw/WGiBcrRrVWppOMLtG+eRaH/qSLbnmh1kQ=
+20251201081333.sql h1:dD+jcisoUYnxRYWdtVcnxbDeYjUW12/R/qarYQr+utg=
+20251201104439.sql h1:h4tz5uetbCKeOI3agSVmAeh6jUfECYy0LhQ7HooYhzg=
+20251201113804.sql h1:DdUzqfLdy4AUpVSRFDrIJXntGmtKgZrtvv8MAT4mMxs=
+20251201113858.sql h1:XSpicm4aWjrfeY5EYIiw8Ios9v7BGfzZJxpW/57Qwqs=
+20251201114751.sql h1:qx+ShdHLacVFcRwwGrW1zMKFHOjMGdnusk7CEfyV8ig=
+20251201114913.sql h1:6l/m2/6fmTIjFmKio0QmB8oTwBDCHTIkbuYTODQ7Gwk=
+20251202030445.sql h1:5+6ss5KdN6pll8Qtf0RGGyy/BURNO3dBgIBJOzDb624=
+20251202044430.sql h1:nJtlQDGj7ZJtJ07NYcF41JCn9ck50GyDPoitTOe8P/s=
+20251202063820.sql h1:dXGXAp2Aq1PUuwAT5FtBquZUeEv5Pdb81/bFt+KR274=
+20251202064000.sql h1:+W5j/AipU7qnFLCTaS7jVd/SIbIPo4k5rRScAvu0L44=
+20251202130629.sql h1:rB/+hjwgQnHoA/uOv4pJbB0iXAeaPMinlzRI/755raY=
+20251202160848.sql h1:KbTqHicChkfszdW9pQzoPlnvtECKDPmwpV3Rc7+5a8o=
+20251202180207.sql h1:4DMAex+Go7MuMt15Di/J1C/ag15KwgWW1u8KDFmbJ9M=
+20251202231005.sql h1:4GcnNRFZnjY1nASBj1IUkzLYCYIfnSE9s6kcWXZSiO8=
+20251203205052.sql h1:0TZ4D3YKO2lmOAa60nhkJAQSx8QHgd+Zj4R39bPdU8Y=
+20251205073858.sql h1:F4o2SDJ/deNjvKaimBGccZ9t4Fir01YocT6emhEoqiU=
+20251205211957.sql h1:rJLemdhIo7NQRgsl5DANVuz8pAee7HciyGVET7V/DMk=
+20251205214433.sql h1:1axU9bZaXW+sASNXQn08dKBX+fDEj1qHGiigEoJQXjY=
+20251205221124.sql h1:TEy3NdLtM8+BkHwVT11kKvX3V28PP1bDMocZ0404Tm4=
+20251206021053.sql h1:cEmwbMKPz8DY2OUsr3ym3blyp823418qMnx/AJFFYqY=
+20251207020537.sql h1:hCENQuZ6jXSp3LrmZ7FgLxvI1gJNgBa+F39/RosKfgk=
+20251207212015.sql h1:uhNmVl9qXZnqwy3GXfxh5JJjjXK4mztmL19ioezZwVQ=
+20251207221222.sql h1:QMxCwIwb4M9jA9z8BJ9aNsOcMVwuZ1ZSsx0FjOlQjek=
+20251209022744.sql h1:BBb5VV+hUGtKW+nZhu/C6OhBgiQhs84vMYfLM51V+VA=
+20251209025908.sql h1:42z/3wPoOt/nkS92xZ9PHeu7m0IWgVICETkf6Ugtx7w=
+20251209030538.sql h1:M2zy92ZTBTHLzbKs7cv1S5epLbJPQPVxME5E8nrPL2w=
+20251209051742.sql h1:AKCu0xgzUlH5MjjBQLiHq6Gw54HA0j++Eknj1i6FHAw=
+20251209064304.sql h1:MSMSfF2HVU7Qg/nyHmh89i7+oAj3V2wbPv/5AQcYNYM=
+20251209070128.sql h1:CU3U33jEhzNFa2yc7aBGHv/XcyFEHRkjHUJFLTKZRjc=
+20251209084929.sql h1:5H1z+0hOmDHQastabHLaxN5d6YWiD2p4+nUf1Fmthrc=
+20251210145148.sql h1:tAs3iNEHnASYEQ4/h8XDXmKPvVASgfP6oTjnFdmBqFA=
+20251211101547.sql h1:hGxvjf2XLoHJIQQlCoSOgqOrIkGjk1A7CLdDV1R4pgc=
+20251211113942.sql h1:uxNunwWePW7UpR+8KtdvqQ4mUofJascWOOUEyRCKIEI=
+20251216074834.sql h1:jja2bz2dtaC4327V2iqs9LP7MsW6ycz68BM0KPld9xs=
diff --git a/internal/domain/main-entities/kfr/dto.go b/internal/domain/main-entities/kfr/dto.go
new file mode 100644
index 00000000..d8b92c1a
--- /dev/null
+++ b/internal/domain/main-entities/kfr/dto.go
@@ -0,0 +1,126 @@
+package kfr
+
+import (
+ ecore "simrs-vx/internal/domain/base-entities/core"
+ erc "simrs-vx/internal/domain/references/common"
+ ere "simrs-vx/internal/domain/references/encounter"
+
+ eem "simrs-vx/internal/domain/main-entities/employee"
+ ee "simrs-vx/internal/domain/main-entities/encounter"
+
+ pa "simrs-vx/internal/lib/auth"
+
+ pu "simrs-vx/pkg/use-case-helper"
+)
+
+type CreateDto struct {
+ Encounter_Id *uint `json:"encounter_id"`
+ CreatedBy_Employee_Id *uint `json:"createdBy_employee_id"`
+ Type ere.KFRTypeCode `json:"type" gorm:"size:15"`
+ Subjective string `json:"subjective"`
+ Objective string `json:"objective"`
+ Assessment string `json:"assessment"`
+ TreatmentGoals string `json:"treatmentGoals"`
+ Education string `json:"education"`
+ Action string `json:"action"`
+ Frequency *uint `json:"frequency"`
+ IntervalUnit_Code *erc.TimeUnitCode `json:"intervalUnit_code" gorm:"size:10"`
+ FollowUpType ere.KFRFollowUpTypeCode `json:"followUpType" gorm:"size:10"`
+ FollowUpNote string `json:"followUpNote"`
+ Status_Code erc.DataVerifiedCode `json:"status_code" gorm:"not null;size:10"`
+
+ pa.AuthInfo
+}
+
+type ReadListDto struct {
+ FilterDto
+ Includes string `json:"includes"`
+ Pagination ecore.Pagination
+}
+
+type FilterDto struct {
+ Encounter_Id *uint `json:"encounter-id"`
+ CreatedBy_Employee_Id *uint `json:"createdBy_employee-id"`
+ Type ere.KFRTypeCode `json:"type" gorm:"size:15"`
+}
+
+type ReadDetailDto struct {
+ Id uint `json:"id"`
+ Includes string `json:"includes"`
+}
+
+type UpdateDto struct {
+ Id uint `json:"id"`
+ CreateDto
+}
+
+type DeleteDto struct {
+ Id uint `json:"id"`
+
+ pa.AuthInfo
+}
+
+type MetaDto struct {
+ PageNumber int `json:"page_number"`
+ PageSize int `json:"page_size"`
+ Count int `json:"count"`
+}
+
+type ResponseDto struct {
+ ecore.Main
+ Encounter_Id *uint `json:"encounter_id"`
+ Encounter *ee.Encounter `json:"encounter,omitempty" gorm:"foreignKey:Encounter_Id;references:Id"`
+ CreatedBy_Employee_Id *uint `json:"createdBy_employee_id"`
+ CreatedBy_Employee *eem.Employee `json:"createdBy_employee,omitempty" gorm:"foreignKey:CreatedBy_Employee_Id;references:Id"`
+ Type ere.KFRTypeCode `json:"type" gorm:"size:15"`
+ Subjective string `json:"subjective"`
+ Objective string `json:"objective"`
+ Assessment string `json:"assessment"`
+ TreatmentGoals string `json:"treatmentGoals"`
+ Education string `json:"education"`
+ Action string `json:"action"`
+ Frequency *uint `json:"frequency"`
+ IntervalUnit_Code *erc.TimeUnitCode `json:"intervalUnit_code" gorm:"size:10"`
+ FollowUpType ere.KFRFollowUpTypeCode `json:"followUpType" gorm:"size:10"`
+ FollowUpNote string `json:"followUpNote"`
+ Status_Code erc.DataVerifiedCode `json:"status_code" gorm:"not null;size:10"`
+}
+
+func (d KFR) ToResponse() ResponseDto {
+ resp := ResponseDto{
+ Encounter_Id: d.Encounter_Id,
+ Encounter: d.Encounter,
+ CreatedBy_Employee_Id: d.CreatedBy_Employee_Id,
+ CreatedBy_Employee: d.CreatedBy_Employee,
+ Type: d.Type,
+ Subjective: d.Subjective,
+ Objective: d.Objective,
+ Assessment: d.Assessment,
+ TreatmentGoals: d.TreatmentGoals,
+ Education: d.Education,
+ Action: d.Action,
+ Frequency: d.Frequency,
+ IntervalUnit_Code: d.IntervalUnit_Code,
+ FollowUpType: d.FollowUpType,
+ FollowUpNote: d.FollowUpNote,
+ Status_Code: d.Status_Code,
+ }
+ resp.Main = d.Main
+ return resp
+}
+
+func ToResponseList(data []KFR) []ResponseDto {
+ resp := make([]ResponseDto, len(data))
+ for i, u := range data {
+ resp[i] = u.ToResponse()
+ }
+ return resp
+}
+
+func (c CreateDto) IsCorrectType() bool {
+ return pu.Contains([]ere.KFRTypeCode{ere.KFRAssessment, ere.KFRTherapy, ere.KFRReAssessment}, c.Type)
+}
+
+func (c CreateDto) IsCorrectFollowUpType() bool {
+ return pu.Contains([]ere.KFRFollowUpTypeCode{ere.KFUTCEva, ere.KFUTCRef, ere.KFUTCFin}, c.FollowUpType)
+}
diff --git a/internal/domain/main-entities/kfr/entitiy.go b/internal/domain/main-entities/kfr/entitiy.go
new file mode 100644
index 00000000..dea1f2cf
--- /dev/null
+++ b/internal/domain/main-entities/kfr/entitiy.go
@@ -0,0 +1,42 @@
+package kfr
+
+import (
+ ecore "simrs-vx/internal/domain/base-entities/core"
+ erc "simrs-vx/internal/domain/references/common"
+ ere "simrs-vx/internal/domain/references/encounter"
+
+ eem "simrs-vx/internal/domain/main-entities/employee"
+ ee "simrs-vx/internal/domain/main-entities/encounter"
+)
+
+type KFR struct {
+ ecore.Main
+ Encounter_Id *uint `json:"encounter_id"`
+ Encounter *ee.Encounter `json:"encounter,omitempty" gorm:"foreignKey:Encounter_Id;references:Id"`
+ CreatedBy_Employee_Id *uint `json:"createdBy_employee_id"`
+ CreatedBy_Employee *eem.Employee `json:"createdBy_employee,omitempty" gorm:"foreignKey:CreatedBy_Employee_Id;references:Id"`
+ Type ere.KFRTypeCode `json:"type" gorm:"size:15"`
+ Subjective string `json:"subjective"`
+ Objective string `json:"objective"`
+ Assessment string `json:"assessment"`
+ TreatmentGoals string `json:"treatmentGoals"`
+ Education string `json:"education"`
+ Action string `json:"action"`
+ Frequency *uint `json:"frequency"`
+ IntervalUnit_Code *erc.TimeUnitCode `json:"intervalUnit_code" gorm:"size:10"`
+ FollowUpType ere.KFRFollowUpTypeCode `json:"followUpType" gorm:"size:10"`
+ FollowUpNote string `json:"followUpNote"`
+ Status_Code erc.DataVerifiedCode `json:"status_code" gorm:"not null;size:10"`
+}
+
+func (d KFR) IsNew() bool {
+ return d.Status_Code == erc.DVCNew
+}
+
+func (d KFR) IsVerified() bool {
+ return d.Status_Code == erc.DVCVerified
+}
+
+func (d KFR) IsValidated() bool {
+ return d.Status_Code == erc.DVCValidated
+}
diff --git a/internal/domain/references/encounter/encounter.go b/internal/domain/references/encounter/encounter.go
index 846b11be..8114f365 100644
--- a/internal/domain/references/encounter/encounter.go
+++ b/internal/domain/references/encounter/encounter.go
@@ -23,6 +23,8 @@ type (
DocTypeCode string
EntityTypeCode string
StatusProtocolChemo string
+ KFRTypeCode string
+ KFRFollowUpTypeCode string
)
const (
@@ -133,6 +135,14 @@ const (
SPCComplete StatusProtocolChemo = "complete" // Terealisasi
SPCPlanned StatusProtocolChemo = "planned" // Terencana
SPCSchedule StatusProtocolChemo = "schedule" // Terjadwal
+
+ KFRAssessment KFRTypeCode = "assessment"
+ KFRTherapy KFRTypeCode = "therapy"
+ KFRReAssessment KFRTypeCode = "reassessment"
+
+ KFUTCEva KFRFollowUpTypeCode = "evaluation"
+ KFUTCRef KFRFollowUpTypeCode = "referral"
+ KFUTCFin KFRFollowUpTypeCode = "finish"
)
func (ec EncounterClassCode) Code() string {
diff --git a/internal/interface/main-handler/kfr/handler.go b/internal/interface/main-handler/kfr/handler.go
new file mode 100644
index 00000000..bceead6f
--- /dev/null
+++ b/internal/interface/main-handler/kfr/handler.go
@@ -0,0 +1,146 @@
+package kfr
+
+import (
+ "net/http"
+
+ d "github.com/karincake/dodol"
+ rw "github.com/karincake/risoles"
+ sf "github.com/karincake/semprit"
+
+ // ua "github.com/karincake/tumpeng/auth/svc"
+
+ e "simrs-vx/internal/domain/main-entities/kfr"
+ u "simrs-vx/internal/use-case/main-use-case/kfr"
+
+ erc "simrs-vx/internal/domain/references/common"
+
+ pa "simrs-vx/internal/lib/auth"
+)
+
+func Create(w http.ResponseWriter, r *http.Request) {
+ authInfo, err := pa.GetAuthInfo(r)
+ if err != nil {
+ rw.WriteJSON(w, http.StatusUnauthorized, d.IS{"message": err.Error()}, nil)
+ }
+ dto := e.CreateDto{}
+ if res := rw.ValidateStructByIOR(w, r.Body, &dto); !res {
+ return
+ }
+
+ if !dto.IsCorrectType() {
+ rw.WriteJSON(w, http.StatusBadRequest, d.IS{"message": "invalid kfr type"}, nil)
+ return
+ }
+
+ if !dto.IsCorrectFollowUpType() {
+ rw.WriteJSON(w, http.StatusBadRequest, d.IS{"message": "invalid kfr followUpType"}, nil)
+ return
+ }
+
+ dto.AuthInfo = *authInfo
+ res, err := u.Create(dto)
+ rw.DataResponse(w, res, err)
+}
+
+func GetList(w http.ResponseWriter, r *http.Request) {
+ dto := e.ReadListDto{}
+ sf.UrlQueryParam(&dto, *r.URL)
+ res, err := u.ReadList(dto)
+ rw.DataResponse(w, res, err)
+}
+
+func GetDetail(w http.ResponseWriter, r *http.Request) {
+ id := rw.ValidateInt(w, "id", r.PathValue("id"))
+ if id <= 0 {
+ return
+ }
+ dto := e.ReadDetailDto{}
+ sf.UrlQueryParam(&dto, *r.URL)
+ dto.Id = uint(id)
+ res, err := u.ReadDetail(dto)
+ rw.DataResponse(w, res, err)
+}
+
+func Update(w http.ResponseWriter, r *http.Request) {
+ authInfo, err := pa.GetAuthInfo(r)
+ if err != nil {
+ rw.WriteJSON(w, http.StatusUnauthorized, d.IS{"message": err.Error()}, nil)
+ }
+ id := rw.ValidateInt(w, "id", r.PathValue("id"))
+ if id <= 0 {
+ return
+ }
+
+ dto := e.UpdateDto{}
+ if res := rw.ValidateStructByIOR(w, r.Body, &dto); !res {
+ return
+ }
+
+ if !dto.IsCorrectType() {
+ rw.WriteJSON(w, http.StatusBadRequest, d.IS{"message": "invalid kfr type"}, nil)
+ return
+ }
+
+ if !dto.IsCorrectFollowUpType() {
+ rw.WriteJSON(w, http.StatusBadRequest, d.IS{"message": "invalid kfr followUpType"}, nil)
+ return
+ }
+
+ dto.Id = uint(id)
+ dto.AuthInfo = *authInfo
+ res, err := u.Update(dto)
+ rw.DataResponse(w, res, err)
+}
+
+func Delete(w http.ResponseWriter, r *http.Request) {
+ authInfo, err := pa.GetAuthInfo(r)
+ if err != nil {
+ rw.WriteJSON(w, http.StatusUnauthorized, d.IS{"message": err.Error()}, nil)
+ }
+ id := rw.ValidateInt(w, "id", r.PathValue("id"))
+ if id <= 0 {
+ return
+ }
+
+ dto := e.DeleteDto{}
+ dto.Id = uint(id)
+ dto.AuthInfo = *authInfo
+ res, err := u.Delete(dto)
+ rw.DataResponse(w, res, err)
+}
+
+func Verify(w http.ResponseWriter, r *http.Request) {
+ authInfo, err := pa.GetAuthInfo(r)
+ if err != nil {
+ rw.WriteJSON(w, http.StatusUnauthorized, d.IS{"message": err.Error()}, nil)
+ }
+ id := rw.ValidateInt(w, "id", r.PathValue("id"))
+ if id <= 0 {
+ return
+ }
+
+ dto := e.UpdateDto{}
+ dto.Id = uint(id)
+ dto.Status_Code = erc.DVCVerified
+ dto.AuthInfo = *authInfo
+ res, err := u.UpdateStatusCode(dto)
+ rw.DataResponse(w, res, err)
+}
+
+func Validate(w http.ResponseWriter, r *http.Request) {
+ authInfo, err := pa.GetAuthInfo(r)
+ if err != nil {
+ rw.WriteJSON(w, http.StatusUnauthorized, d.IS{"message": err.Error()}, nil)
+ }
+ id := rw.ValidateInt(w, "id", r.PathValue("id"))
+ if id <= 0 {
+ return
+ }
+
+ dto := e.UpdateDto{}
+ dto.Id = uint(id)
+ dto.Status_Code = erc.DVCValidated
+ dto.AuthInfo = *authInfo
+ res, err := u.UpdateStatusCode(dto)
+ rw.DataResponse(w, res, err)
+}
diff --git a/internal/interface/main-handler/main-handler.go b/internal/interface/main-handler/main-handler.go
index a4249e08..150c419d 100644
--- a/internal/interface/main-handler/main-handler.go
+++ b/internal/interface/main-handler/main-handler.go
@@ -19,6 +19,7 @@ import (
generalconsent "simrs-vx/internal/interface/main-handler/general-consent"
generatefile "simrs-vx/internal/interface/main-handler/generate-file"
internalreference "simrs-vx/internal/interface/main-handler/internal-reference"
+ kfr "simrs-vx/internal/interface/main-handler/kfr"
materialorder "simrs-vx/internal/interface/main-handler/material-order"
materialorderitem "simrs-vx/internal/interface/main-handler/material-order-item"
mcuorder "simrs-vx/internal/interface/main-handler/mcu-order"
@@ -347,6 +348,15 @@ func SetRoutes() http.Handler {
"GET /": screening.O.GetList,
"GET /{id}": screening.O.GetDetail,
})
+ hk.GroupRoutes("/v1/kfr", r, auth.GuardMW, hk.MapHandlerFunc{
+ "POST /": kfr.Create,
+ "GET /": kfr.GetList,
+ "GET /{id}": kfr.GetDetail,
+ "PATCH /{id}": kfr.Update,
+ "DELETE /{id}": kfr.Delete,
+ "PATCH /{id}/verify": kfr.Verify,
+ "PATCH /{id}/validate": kfr.Validate,
+ })
/******************** actor ********************/
hc.RegCrud(r, "/v1/person", person.O)
hc.RegCrud(r, "/v1/person-address", personaddress.O)
diff --git a/internal/interface/migration/main-entities.go b/internal/interface/migration/main-entities.go
index 6e92950e..f8eb48c0 100644
--- a/internal/interface/migration/main-entities.go
+++ b/internal/interface/migration/main-entities.go
@@ -46,6 +46,7 @@ import (
internalreference "simrs-vx/internal/domain/main-entities/internal-reference"
item "simrs-vx/internal/domain/main-entities/item"
itemprice "simrs-vx/internal/domain/main-entities/item-price"
+ kfr "simrs-vx/internal/domain/main-entities/kfr"
laborant "simrs-vx/internal/domain/main-entities/laborant"
language "simrs-vx/internal/domain/main-entities/language"
material "simrs-vx/internal/domain/main-entities/material"
@@ -251,5 +252,6 @@ func getMainEntities() []any {
&screening.Screening{},
&procedurereport.ProcedureReport{},
&chemoplan.ChemoPlan{},
+ &kfr.KFR{},
}
}
diff --git a/internal/use-case/main-use-case/kfr/case.go b/internal/use-case/main-use-case/kfr/case.go
new file mode 100644
index 00000000..56d0f35e
--- /dev/null
+++ b/internal/use-case/main-use-case/kfr/case.go
@@ -0,0 +1,397 @@
+package kfr
+
+import (
+ "errors"
+ "strconv"
+
+ erc "simrs-vx/internal/domain/references/common"
+
+ // main entities
+ ee "simrs-vx/internal/domain/main-entities/encounter"
+ e "simrs-vx/internal/domain/main-entities/kfr"
+
+ ue "simrs-vx/internal/use-case/main-use-case/encounter"
+
+ pl "simrs-vx/pkg/logger"
+ pu "simrs-vx/pkg/use-case-helper"
+
+ dg "github.com/karincake/apem/db-gorm-pg"
+ d "github.com/karincake/dodol"
+ "gorm.io/gorm"
+)
+
+const source = "kfr"
+
+func Create(input e.CreateDto) (*d.Data, error) {
+ data := e.KFR{}
+
+ event := pl.Event{
+ Feature: "Create",
+ Source: source,
+ }
+
+ // Start log
+ pl.SetLogInfo(&event, input, "started", "create")
+
+ err := dg.I.Transaction(func(tx *gorm.DB) error {
+ if !input.AuthInfo.IsDoctor() {
+ return errors.New("user is not a doctor")
+ }
+
+ mwRunner := newMiddlewareRunner(&event, tx)
+ mwRunner.setMwType(pu.MWTPre)
+ // Run pre-middleware
+ if err := mwRunner.RunCreateMiddleware(createPreMw, &input, &data); err != nil {
+ return err
+ }
+
+ // check if encounter is done
+ if ue.IsDone(*input.Encounter_Id, &event, tx) {
+ return errors.New("encounter is already done")
+ }
+
+ if input.AuthInfo.Employee_Id == nil {
+ return errors.New("user is not an employee")
+ }
+
+ input.CreatedBy_Employee_Id = input.AuthInfo.Employee_Id
+
+ if resData, err := CreateData(input, &event, tx); err != nil {
+ return err
+ } else {
+ data = *resData
+ }
+
+ mwRunner.setMwType(pu.MWTPost)
+ // Run post-middleware
+ if err := mwRunner.RunCreateMiddleware(createPostMw, &input, &data); err != nil {
+ return err
+ }
+
+ pl.SetLogInfo(&event, nil, "complete")
+
+ return nil
+ })
+
+ if err != nil {
+ return nil, err
+ }
+
+ return &d.Data{
+ Meta: d.II{
+ "source": source,
+ "structure": "single-data",
+ "status": "created",
+ },
+ Data: data.ToResponse(),
+ }, nil
+}
+
+func ReadList(input e.ReadListDto) (*d.Data, error) {
+ var data *e.KFR
+ var dataList []e.KFR
+ var metaList *e.MetaDto
+ var err error
+
+ event := pl.Event{
+ Feature: "ReadList",
+ Source: source,
+ }
+
+ // Start log
+ pl.SetLogInfo(&event, input, "started", "readList")
+
+ err = dg.I.Transaction(func(tx *gorm.DB) error {
+ mwRunner := newMiddlewareRunner(&event, tx)
+ mwRunner.setMwType(pu.MWTPre)
+ // Run pre-middleware
+ if err := mwRunner.RunReadListMiddleware(readListPreMw, &input, data); err != nil {
+ return err
+ }
+
+ if dataList, metaList, err = ReadListData(input, &event, tx); err != nil {
+ return err
+ }
+
+ mwRunner.setMwType(pu.MWTPost)
+ // Run post-middleware
+ if err := mwRunner.RunReadListMiddleware(readListPostMw, &input, data); err != nil {
+ return err
+ }
+
+ return nil
+ })
+
+ if err != nil {
+ return nil, err
+ }
+
+ return &d.Data{
+ Meta: d.IS{
+ "source": source,
+ "structure": "list-data",
+ "status": "fetched",
+ "page_number": strconv.Itoa(metaList.PageNumber),
+ "page_size": strconv.Itoa(metaList.PageSize),
+ "record_totalCount": strconv.Itoa(metaList.Count),
+ "record_currentCount": strconv.Itoa(len(dataList)),
+ },
+ Data: e.ToResponseList(dataList),
+ }, nil
+}
+
+func ReadDetail(input e.ReadDetailDto) (*d.Data, error) {
+ var data *e.KFR
+ var err error
+
+ event := pl.Event{
+ Feature: "ReadDetail",
+ Source: source,
+ }
+
+ // Start log
+ pl.SetLogInfo(&event, input, "started", "readDetail")
+
+ err = dg.I.Transaction(func(tx *gorm.DB) error {
+ mwRunner := newMiddlewareRunner(&event, tx)
+ mwRunner.setMwType(pu.MWTPre)
+ // Run pre-middleware
+ if err := mwRunner.RunReadDetailMiddleware(readDetailPreMw, &input, data); err != nil {
+ return err
+ }
+
+ if data, err = ReadDetailData(input, &event, tx); err != nil {
+ return err
+ }
+
+ mwRunner.setMwType(pu.MWTPost)
+ // Run post-middleware
+ if err := mwRunner.RunReadDetailMiddleware(readDetailPostMw, &input, data); err != nil {
+ return err
+ }
+
+ return nil
+ })
+
+ if err != nil {
+ return nil, err
+ }
+
+ return &d.Data{
+ Meta: d.IS{
+ "source": source,
+ "structure": "single-data",
+ "status": "fetched",
+ },
+ Data: data.ToResponse(),
+ }, nil
+}
+
+func Update(input e.UpdateDto) (*d.Data, error) {
+ rdDto := e.ReadDetailDto{Id: input.Id}
+ var data *e.KFR
+ var err error
+
+ event := pl.Event{
+ Feature: "Update",
+ Source: source,
+ }
+
+ // Start log
+ pl.SetLogInfo(&event, input, "started", "update")
+
+ err = dg.I.Transaction(func(tx *gorm.DB) error {
+ if !input.AuthInfo.IsDoctor() {
+ return errors.New("user is not a doctor")
+ }
+
+ pl.SetLogInfo(&event, rdDto, "started", "DBReadDetail")
+ if data, err = ReadDetailData(rdDto, &event, tx); err != nil {
+ return err
+ }
+
+ mwRunner := newMiddlewareRunner(&event, tx)
+ mwRunner.setMwType(pu.MWTPre)
+ // Run pre-middleware
+ if err := mwRunner.RunUpdateMiddleware(readDetailPreMw, &rdDto, data); err != nil {
+ return err
+ }
+
+ // check if encounter is done
+ if ue.IsDone(*input.Encounter_Id, &event, tx) {
+ return errors.New("encounter is already done")
+ }
+
+ if err := UpdateData(input, data, &event, tx); err != nil {
+ return err
+ }
+
+ // Get Updated Data
+ if data, err = ReadDetailData(rdDto, &event, tx); err != nil {
+ return err
+ }
+
+ pl.SetLogInfo(&event, nil, "complete")
+
+ mwRunner.setMwType(pu.MWTPost)
+ // Run post-middleware
+ if err := mwRunner.RunUpdateMiddleware(readDetailPostMw, &rdDto, data); err != nil {
+ return err
+ }
+
+ return nil
+ })
+
+ if err != nil {
+ return nil, err
+ }
+
+ return &d.Data{
+ Meta: d.IS{
+ "source": source,
+ "structure": "single-data",
+ "status": "updated",
+ },
+ Data: data.ToResponse(),
+ }, nil
+
+}
+
+func Delete(input e.DeleteDto) (*d.Data, error) {
+ rdDto := e.ReadDetailDto{Id: uint(input.Id)}
+ var data *e.KFR
+ var err error
+
+ event := pl.Event{
+ Feature: "Delete",
+ Source: source,
+ }
+
+ // Start log
+ pl.SetLogInfo(&event, input, "started", "delete")
+
+ err = dg.I.Transaction(func(tx *gorm.DB) error {
+ if !input.AuthInfo.IsDoctor() {
+ return errors.New("user is not a doctor")
+ }
+
+ pl.SetLogInfo(&event, rdDto, "started", "DBReadDetail")
+ if data, err = ReadDetailData(rdDto, &event, tx); err != nil {
+ return err
+ }
+
+ mwRunner := newMiddlewareRunner(&event, tx)
+ mwRunner.setMwType(pu.MWTPre)
+ // Run pre-middleware
+ if err := mwRunner.RunDeleteMiddleware(readDetailPreMw, &rdDto, data); err != nil {
+ return err
+ }
+
+ if err := DeleteData(data, &event, tx); err != nil {
+ return err
+ }
+
+ mwRunner.setMwType(pu.MWTPost)
+ // Run post-middleware
+ if err := mwRunner.RunDeleteMiddleware(readDetailPostMw, &rdDto, data); err != nil {
+ return err
+ }
+
+ return nil
+ })
+
+ if err != nil {
+ return nil, err
+ }
+
+ return &d.Data{
+ Meta: d.IS{
+ "source": source,
+ "structure": "single-data",
+ "status": "deleted",
+ },
+ Data: data.ToResponse(),
+ }, nil
+
+}
+
+func UpdateStatusCode(input e.UpdateDto) (*d.Data, error) {
+ rdDto := e.ReadDetailDto{Id: input.Id}
+ var data *e.KFR
+ var err error
+
+ event := pl.Event{
+ Feature: "UpdateStatusCode",
+ Source: source,
+ }
+
+ // Start log
+ pl.SetLogInfo(&event, input, "started", "updateStatusCode")
+
+ err = dg.I.Transaction(func(tx *gorm.DB) error {
+ if !input.AuthInfo.IsDoctor() {
+ return errors.New("user is not a doctor")
+ }
+
+ pl.SetLogInfo(&event, rdDto, "started", "DBReadDetail")
+ if data, err = ReadDetailData(rdDto, &event, tx); err != nil {
+ return err
+ }
+
+ enc, err := ue.ReadDetailData(ee.ReadDetailDto{Id: *data.Encounter_Id}, &event, tx)
+ if err != nil {
+ return err
+ }
+
+ // check if encounter is done
+ if enc.IsDone() {
+ return errors.New("encounter is already done")
+ }
+
+ switch input.Status_Code {
+ case erc.DVCValidated:
+ if data.IsNew() {
+ return errors.New("resume need to be verified first")
+ }
+ if data.IsValidated() {
+ return errors.New("resume already validated")
+ }
+ if !enc.IsSameResponsibleDoctor(input.AuthInfo.Doctor_Code) {
+ return errors.New("validation doctor is not the same as encounter responsible doctor")
+ }
+
+ data.Status_Code = erc.DVCValidated
+ err = tx.Save(&data).Error
+ if err != nil {
+ return err
+ }
+ case erc.DVCVerified:
+ if data.IsValidated() {
+ return errors.New("resume already validated")
+ }
+ if data.IsVerified() {
+ return errors.New("resume already verified")
+ }
+ data.Status_Code = erc.DVCVerified
+ err = tx.Save(&data).Error
+ if err != nil {
+ return err
+ }
+ }
+
+ return nil
+ })
+
+ if err != nil {
+ return nil, err
+ }
+
+ return &d.Data{
+ Meta: d.IS{
+ "source": source,
+ "structure": "single-data",
+ "status": "updated",
+ },
+ Data: data.ToResponse(),
+ }, nil
+}
diff --git a/internal/use-case/main-use-case/kfr/helper.go b/internal/use-case/main-use-case/kfr/helper.go
new file mode 100644
index 00000000..84cdd70f
--- /dev/null
+++ b/internal/use-case/main-use-case/kfr/helper.go
@@ -0,0 +1,36 @@
+/*
+DESCRIPTION:
+Any functions that are used internally by the use-case
+*/
+package kfr
+
+import (
+ e "simrs-vx/internal/domain/main-entities/kfr"
+
+ erc "simrs-vx/internal/domain/references/common"
+)
+
+func setData[T *e.CreateDto | *e.UpdateDto](input T, data *e.KFR) {
+ var inputSrc *e.CreateDto
+ if inputT, ok := any(input).(*e.CreateDto); ok {
+ inputSrc = inputT
+ data.Status_Code = erc.DVCNew
+ } else {
+ inputTemp := any(input).(*e.UpdateDto)
+ inputSrc = &inputTemp.CreateDto
+ }
+
+ data.Encounter_Id = inputSrc.Encounter_Id
+ data.CreatedBy_Employee_Id = inputSrc.CreatedBy_Employee_Id
+ data.Type = inputSrc.Type
+ data.Subjective = inputSrc.Subjective
+ data.Objective = inputSrc.Objective
+ data.Assessment = inputSrc.Assessment
+ data.TreatmentGoals = inputSrc.TreatmentGoals
+ data.Education = inputSrc.Education
+ data.Action = inputSrc.Action
+ data.Frequency = inputSrc.Frequency
+ data.IntervalUnit_Code = inputSrc.IntervalUnit_Code
+ data.FollowUpType = inputSrc.FollowUpType
+ data.FollowUpNote = inputSrc.FollowUpNote
+}
diff --git a/internal/use-case/main-use-case/kfr/lib.go b/internal/use-case/main-use-case/kfr/lib.go
new file mode 100644
index 00000000..b332f5c2
--- /dev/null
+++ b/internal/use-case/main-use-case/kfr/lib.go
@@ -0,0 +1,140 @@
+package kfr
+
+import (
+ "errors"
+ e "simrs-vx/internal/domain/main-entities/kfr"
+ plh "simrs-vx/pkg/lib-helper"
+ pl "simrs-vx/pkg/logger"
+ pu "simrs-vx/pkg/use-case-helper"
+
+ dg "github.com/karincake/apem/db-gorm-pg"
+ gh "github.com/karincake/getuk"
+ "gorm.io/gorm"
+)
+
+func CreateData(input e.CreateDto, event *pl.Event, dbx ...*gorm.DB) (*e.KFR, error) {
+ pl.SetLogInfo(event, nil, "started", "DBCreate")
+
+ data := e.KFR{}
+ setData(&input, &data)
+
+ var tx *gorm.DB
+ if len(dbx) > 0 {
+ tx = dbx[0]
+ } else {
+ tx = dg.I
+ }
+
+ if err := tx.Create(&data).Error; err != nil {
+ return nil, plh.HandleCreateError(input, event, err)
+ }
+
+ pl.SetLogInfo(event, nil, "complete")
+ return &data, nil
+}
+
+func ReadListData(input e.ReadListDto, event *pl.Event, dbx ...*gorm.DB) ([]e.KFR, *e.MetaDto, error) {
+ pl.SetLogInfo(event, input, "started", "DBReadList")
+ data := []e.KFR{}
+ pagination := gh.Pagination{}
+ count := int64(0)
+ meta := e.MetaDto{}
+
+ var tx *gorm.DB
+ if len(dbx) > 0 {
+ tx = dbx[0]
+ } else {
+ tx = dg.I
+ }
+
+ tx = tx.
+ Model(&e.KFR{}).
+ Scopes(gh.Preload(input.Includes)).
+ Scopes(gh.Filter(input.FilterDto)).
+ Count(&count).
+ Scopes(gh.Paginate(input, &pagination)).
+ Order("\"CreatedAt\" DESC")
+
+ if err := tx.Find(&data).Error; err != nil {
+ if errors.Is(err, gorm.ErrRecordNotFound) {
+ return nil, &meta, nil
+ }
+ return nil, nil, plh.HandleListError(input, event, err)
+ }
+
+ meta.Count = int(count)
+ meta.PageNumber = pagination.PageNumber
+ meta.PageSize = pagination.PageSize
+
+ pl.SetLogInfo(event, nil, "complete")
+ return data, &meta, nil
+}
+
+func ReadDetailData(input e.ReadDetailDto, event *pl.Event, dbx ...*gorm.DB) (*e.KFR, error) {
+ pl.SetLogInfo(event, input, "started", "DBReadDetail")
+ data := e.KFR{}
+
+ var tx *gorm.DB
+ if len(dbx) > 0 {
+ tx = dbx[0]
+ } else {
+ tx = dg.I
+ }
+
+ if err := tx.Scopes(gh.Preload(input.Includes)).First(&data, input.Id).Error; err != nil {
+ if processedErr := pu.HandleReadError(err, event, source, input.Id, data); processedErr != nil {
+ return nil, processedErr
+ }
+ }
+
+ pl.SetLogInfo(event, nil, "complete")
+ return &data, nil
+}
+
+func UpdateData(input e.UpdateDto, data *e.KFR, event *pl.Event, dbx ...*gorm.DB) error {
+ pl.SetLogInfo(event, data, "started", "DBUpdate")
+ setData(&input, data)
+
+ var tx *gorm.DB
+ if len(dbx) > 0 {
+ tx = dbx[0]
+ } else {
+ tx = dg.I
+ }
+
+ if err := tx.Save(&data).Error; err != nil {
+ event.Status = "failed"
+ event.ErrInfo = pl.ErrorInfo{
+ Code: "data-update-fail",
+ Detail: "Database update failed",
+ Raw: err,
+ }
+ return pl.SetLogError(event, input)
+ }
+
+ pl.SetLogInfo(event, nil, "complete")
+ return nil
+}
+
+func DeleteData(data *e.KFR, event *pl.Event, dbx ...*gorm.DB) error {
+ pl.SetLogInfo(event, data, "started", "DBDelete")
+ var tx *gorm.DB
+ if len(dbx) > 0 {
+ tx = dbx[0]
+ } else {
+ tx = dg.I
+ }
+
+ if err := tx.Delete(&data).Error; err != nil {
+ event.Status = "failed"
+ event.ErrInfo = pl.ErrorInfo{
+ Code: "data-delete-fail",
+ Detail: "Database delete failed",
+ Raw: err,
+ }
+ return pl.SetLogError(event, data)
+ }
+
+ pl.SetLogInfo(event, nil, "complete")
+ return nil
+}
diff --git a/internal/use-case/main-use-case/kfr/middleware-runner.go b/internal/use-case/main-use-case/kfr/middleware-runner.go
new file mode 100644
index 00000000..0ffaaad3
--- /dev/null
+++ b/internal/use-case/main-use-case/kfr/middleware-runner.go
@@ -0,0 +1,103 @@
+package kfr
+
+import (
+ e "simrs-vx/internal/domain/main-entities/kfr"
+ pl "simrs-vx/pkg/logger"
+ pu "simrs-vx/pkg/use-case-helper"
+
+ "gorm.io/gorm"
+)
+
+type middlewareRunner struct {
+ Event *pl.Event
+ Tx *gorm.DB
+ MwType pu.MWType
+}
+
+// NewMiddlewareExecutor creates a new middleware executor
+func newMiddlewareRunner(event *pl.Event, tx *gorm.DB) *middlewareRunner {
+ return &middlewareRunner{
+ Event: event,
+ Tx: tx,
+ }
+}
+
+// ExecuteCreateMiddleware executes create middleware
+func (me *middlewareRunner) RunCreateMiddleware(middlewares []createMw, input *e.CreateDto, data *e.KFR) error {
+ for _, middleware := range middlewares {
+ logData := pu.GetLogData(input, data)
+
+ pl.SetLogInfo(me.Event, logData, "started", middleware.Name)
+
+ if err := middleware.Func(input, data, me.Tx); err != nil {
+ return pu.HandleMiddlewareError(me.Event, string(me.MwType), middleware.Name, logData, err)
+ }
+
+ pl.SetLogInfo(me.Event, nil, "complete")
+ }
+ return nil
+}
+
+func (me *middlewareRunner) RunReadListMiddleware(middlewares []readListMw, input *e.ReadListDto, data *e.KFR) error {
+ for _, middleware := range middlewares {
+ logData := pu.GetLogData(input, data)
+
+ pl.SetLogInfo(me.Event, logData, "started", middleware.Name)
+
+ if err := middleware.Func(input, data, me.Tx); err != nil {
+ return pu.HandleMiddlewareError(me.Event, string(me.MwType), middleware.Name, logData, err)
+ }
+
+ pl.SetLogInfo(me.Event, nil, "complete")
+ }
+ return nil
+}
+
+func (me *middlewareRunner) RunReadDetailMiddleware(middlewares []readDetailMw, input *e.ReadDetailDto, data *e.KFR) error {
+ for _, middleware := range middlewares {
+ logData := pu.GetLogData(input, data)
+
+ pl.SetLogInfo(me.Event, logData, "started", middleware.Name)
+
+ if err := middleware.Func(input, data, me.Tx); err != nil {
+ return pu.HandleMiddlewareError(me.Event, string(me.MwType), middleware.Name, logData, err)
+ }
+
+ pl.SetLogInfo(me.Event, nil, "complete")
+ }
+ return nil
+}
+
+func (me *middlewareRunner) RunUpdateMiddleware(middlewares []readDetailMw, input *e.ReadDetailDto, data *e.KFR) error {
+ for _, middleware := range middlewares {
+ logData := pu.GetLogData(input, data)
+
+ pl.SetLogInfo(me.Event, logData, "started", middleware.Name)
+
+ if err := middleware.Func(input, data, me.Tx); err != nil {
+ return pu.HandleMiddlewareError(me.Event, string(me.MwType), middleware.Name, logData, err)
+ }
+
+ pl.SetLogInfo(me.Event, nil, "complete")
+ }
+ return nil
+}
+
+func (me *middlewareRunner) RunDeleteMiddleware(middlewares []readDetailMw, input *e.ReadDetailDto, data *e.KFR) error {
+ for _, middleware := range middlewares {
+ logData := pu.GetLogData(input, data)
+
+ pl.SetLogInfo(me.Event, logData, "started", middleware.Name)
+
+ if err := middleware.Func(input, data, me.Tx); err != nil {
+ return pu.HandleMiddlewareError(me.Event, string(me.MwType), middleware.Name, logData, err)
+ }
+
+ pl.SetLogInfo(me.Event, nil, "complete")
+ }
+ return nil
+}
+
+func (me *middlewareRunner) setMwType(mwType pu.MWType) {
+ me.MwType = mwType
+}
diff --git a/internal/use-case/main-use-case/kfr/middleware.go b/internal/use-case/main-use-case/kfr/middleware.go
new file mode 100644
index 00000000..d3329f28
--- /dev/null
+++ b/internal/use-case/main-use-case/kfr/middleware.go
@@ -0,0 +1,9 @@
+package kfr
+
+// example of middleware
+// func init() {
+// createPreMw = append(createPreMw,
+// CreateMw{Name: "modif-input", Func: pm.ModifInput},
+// CreateMw{Name: "check-data", Func: pm.CheckData},
+// )
+// }
diff --git a/internal/use-case/main-use-case/kfr/tycovar.go b/internal/use-case/main-use-case/kfr/tycovar.go
new file mode 100644
index 00000000..ec0319d5
--- /dev/null
+++ b/internal/use-case/main-use-case/kfr/tycovar.go
@@ -0,0 +1,44 @@
+/*
+DESCRIPTION:
+A sample, part of the package that contains type, constants, and/or variables.
+
+In this sample it also provides type and variable regarding the needs of the
+middleware to separate from main use-case which has the basic CRUD
+functionality. The purpose of this is to make the code more maintainable.
+*/
+package kfr
+
+import (
+ "gorm.io/gorm"
+
+ e "simrs-vx/internal/domain/main-entities/kfr"
+)
+
+type createMw struct {
+ Name string
+ Func func(input *e.CreateDto, data *e.KFR, tx *gorm.DB) error
+}
+
+type readListMw struct {
+ Name string
+ Func func(input *e.ReadListDto, data *e.KFR, tx *gorm.DB) error
+}
+
+type readDetailMw struct {
+ Name string
+ Func func(input *e.ReadDetailDto, data *e.KFR, tx *gorm.DB) error
+}
+
+type UpdateMw = readDetailMw
+type DeleteMw = readDetailMw
+
+var createPreMw []createMw // preprocess middleware
+var createPostMw []createMw // postprocess middleware
+var readListPreMw []readListMw // ..
+var readListPostMw []readListMw // ..
+var readDetailPreMw []readDetailMw
+var readDetailPostMw []readDetailMw
+var updatePreMw []readDetailMw
+var updatePostMw []readDetailMw
+var deletePreMw []readDetailMw
+var deletePostMw []readDetailMw
|