Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Add not-null constraint for column trackedentitytypeid in trackedentity table [DHIS2-15066] #19112

Open
wants to merge 39 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
39 commits
Select commit Hold shift + click to select a range
4627513
feat: Add not-null constraint for column trackedentitytypeid in track…
zubaira Nov 9, 2024
adf3d6a
Merge branch 'master' into DHIS2-15066
zubaira Nov 15, 2024
962dae2
fix test
zubaira Nov 15, 2024
972f8d5
fix unit test
zubaira Nov 16, 2024
e642c35
fix unit test
zubaira Nov 16, 2024
290ad52
fix unit test
zubaira Nov 16, 2024
bee4104
remove unused dependency
zubaira Nov 16, 2024
900f3d7
fix integration test
zubaira Nov 16, 2024
62ef7d6
fix controller tests
zubaira Nov 16, 2024
ea2402b
fix tests
zubaira Nov 16, 2024
81daef6
Merge branch 'master' into DHIS2-15066
zubaira Nov 21, 2024
4f26b27
update flyway to cover existin not null rows
zubaira Nov 21, 2024
adc0326
update flyway
zubaira Nov 21, 2024
4ac4a44
update flyway migration
zubaira Dec 4, 2024
13024a3
minor
zubaira Dec 4, 2024
2baca3a
minor
zubaira Dec 4, 2024
38edde3
Merge branch 'master' into DHIS2-15066
zubaira Dec 4, 2024
b0797bc
expcetion message
zubaira Dec 5, 2024
e62c46b
correction in markdown
zubaira Dec 5, 2024
72868f4
Find inconstent trackedentities, fix them and set trackentitytypeid n…
zubaira Dec 5, 2024
746ec8b
update
zubaira Dec 5, 2024
1484545
add exception
zubaira Dec 5, 2024
c3359ce
add exception
zubaira Dec 5, 2024
b265862
Merge branch 'master' into DHIS2-15066
zubaira Dec 5, 2024
d885171
code reviews
zubaira Dec 6, 2024
964ed52
code review
zubaira Dec 6, 2024
91189c9
minor
zubaira Dec 6, 2024
6dcdf05
fix unit test
zubaira Dec 6, 2024
f2adebf
set operation instead of looping through the records
zubaira Dec 6, 2024
cdb5bf8
Merge branch 'master' into DHIS2-15066
zubaira Dec 6, 2024
659a986
Merge branch 'master' into DHIS2-15066
zubaira Dec 11, 2024
cfea998
update flyway
zubaira Dec 11, 2024
2682751
Merge branch 'master' into DHIS2-15066
zubaira Dec 11, 2024
583f488
Merge branch 'master' into DHIS2-15066
zubaira Jan 14, 2025
d3c34d3
fix test
zubaira Jan 14, 2025
ac7c201
Merge branch 'master' into DHIS2-15066
ameenhere Jan 16, 2025
d8b1f52
Merge branch 'master' into DHIS2-15066
zubaira Jan 20, 2025
49359f7
code style
zubaira Jan 20, 2025
c62cbe7
fix test
zubaira Jan 20, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -58,7 +58,7 @@
foreign-key="fk_trackedentityinstance_organisationunitid" not-null="true" lazy="false"/>

<many-to-one name="trackedEntityType" class="org.hisp.dhis.trackedentity.TrackedEntityType"
column="trackedentitytypeid" foreign-key="fk_trackedentityinstance_trackedentitytypeid" lazy="false"/>
column="trackedentitytypeid" foreign-key="fk_trackedentityinstance_trackedentitytypeid" lazy="false" not-null="true"/>
</class>

<sql-query name="updateTrackedEntitiesLastUpdated">update trackedentity set lastUpdated = :lastUpdated, lastupdatedbyuserinfo = CAST(:lastupdatedbyuserinfo as jsonb) WHERE uid in :trackedEntities</sql-query>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -78,6 +78,7 @@
import org.hisp.dhis.programrule.ProgramRuleVariableService;
import org.hisp.dhis.security.acl.AclService;
import org.hisp.dhis.test.TestBase;
import org.hisp.dhis.trackedentity.TrackedEntityType;
import org.hisp.dhis.user.User;
import org.hisp.dhis.user.UserDetails;
import org.hisp.dhis.user.sharing.Sharing;
Expand Down Expand Up @@ -130,10 +131,12 @@ void setup() {

@Test
void testCopyProgramFromUidWithValidProgram() throws NotFoundException, ForbiddenException {
OrganisationUnit orgUnit = createOrganisationUnit("New Org 1");
TrackedEntityType trackedEntityType = createTrackedEntityType('E');

OrganisationUnit orgUnit = createOrganisationUnit('A');
List<Enrollment> originalEnrollments =
List.of(createEnrollment(original, createTrackedEntity(orgUnit), orgUnit));
List.of(
createEnrollment(original, createTrackedEntity(orgUnit, trackedEntityType), orgUnit));
when(programService.getProgram(VALID_PROGRAM_UID)).thenReturn(original);

when(aclService.canWrite(UserDetails.fromUser(user), original)).thenReturn(true);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -521,17 +521,11 @@ private List<Enrollment> getEnrollments() {
}

private TrackedEntity getTrackedEntityA() {
TrackedEntity te = createTrackedEntity(organisationUnitA);
te.setTrackedEntityType(trackedEntityTypeA);

return te;
return createTrackedEntity(organisationUnitA, trackedEntityTypeA);
}

private TrackedEntity getTrackedEntityB() {
TrackedEntity te = createTrackedEntity(organisationUnitB);
te.setTrackedEntityType(trackedEntityTypeB);

return te;
return createTrackedEntity(organisationUnitB, trackedEntityTypeB);
}

private User getNoMergeAuthsUser() {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -91,7 +91,7 @@ public void setUp() {
Program program = createProgram('A');
ProgramStage programStage = createProgramStage('A', program);

trackedEntity = createTrackedEntity(organisationUnit);
trackedEntity = createTrackedEntity(organisationUnit, createTrackedEntityType('P'));
trackedEntity.setUid(TE_UID.getValue());
enrollment = createEnrollment(program, trackedEntity, organisationUnit);
enrollment.setUid(EN_UID.getValue());
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -109,11 +109,11 @@ public void setUp() {

OrganisationUnit organisationUnit = createOrganisationUnit('A');

teA = createTrackedEntity(organisationUnit);
teA = createTrackedEntity(organisationUnit, createTrackedEntityType('D'));
teA.setUid(TE_A_UID.getValue());
teB = createTrackedEntity(organisationUnit);
teB = createTrackedEntity(organisationUnit, createTrackedEntityType('E'));
teB.setUid(TE_B_UID.getValue());
teC = createTrackedEntity(organisationUnit);
teC = createTrackedEntity(organisationUnit, createTrackedEntityType('F'));
teC.setUid(TE_C_UID.getValue());

relationshipA =
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -97,7 +97,7 @@ public void setUp() {
OrganisationUnit orgUnit = createOrganisationUnit('A');
Program program = createProgram('A');
Attribute attribute = createAttribute('A');
trackedEntity = createTrackedEntity('A', orgUnit);
trackedEntity = createTrackedEntity('A', orgUnit, createTrackedEntityType('U'));
trackedEntity.setUid(TE_UID.getValue());
trackedEntity.setAttributeValues(AttributeValues.of(Map.of(attribute.getUid(), UNIQUE_VALUE)));
enrollment = createEnrollment(program, trackedEntity, orgUnit);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -327,7 +327,8 @@ private Attribute attribute(TrackedEntityAttribute trackedEntityAttribute, Strin
}

private TrackedEntity trackedEntity() {
return createTrackedEntity('I', organisationUnit, trackedEntityAttribute);
return createTrackedEntity(
'I', organisationUnit, trackedEntityAttribute, createTrackedEntityType('W'));
}

private org.hisp.dhis.tracker.imports.domain.Enrollment payloadEnrollment() {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -263,7 +263,7 @@ private TrackedEntityType trackedEntityType(String uid, char uniqueChar) {
}

private TrackedEntity trackedEntity(UID uid, TrackedEntityType type, OrganisationUnit orgUnit) {
TrackedEntity te = createTrackedEntity(orgUnit);
TrackedEntity te = createTrackedEntity(orgUnit, createTrackedEntityType('R'));
te.setUid(uid.getValue());
te.setTrackedEntityType(type);
zubaira marked this conversation as resolved.
Show resolved Hide resolved
return te;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -390,11 +390,10 @@ void verifyValidationFailsForEnrollmentDeletionAndUserWithoutTrackedEntityTypeRe
}

private TrackedEntity teWithNoEnrollments() {
TrackedEntity trackedEntity = createTrackedEntity(organisationUnit);
TrackedEntity trackedEntity =
createTrackedEntity(organisationUnit, createTrackedEntityType('C'));
trackedEntity.setUid(TE_ID.getValue());
trackedEntity.setEnrollments(Sets.newHashSet());
trackedEntity.setTrackedEntityType(trackedEntityType);

return trackedEntity;
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -487,7 +487,8 @@ void verifySuccessEventValidationWhenEventHasNoOrgUnitAssigned() {
}

private TrackedEntity teWithNoEnrollments() {
TrackedEntity trackedEntity = createTrackedEntity(organisationUnit);
TrackedEntity trackedEntity =
createTrackedEntity(organisationUnit, createTrackedEntityType('E'));
trackedEntity.setUid(TE_ID.getValue());
trackedEntity.setEnrollments(Sets.newHashSet());
trackedEntity.setTrackedEntityType(trackedEntityType);
zubaira marked this conversation as resolved.
Show resolved Hide resolved
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -300,7 +300,8 @@ void shouldFailWhenUpdateTEAndUserHasNoWriteAccess() {
}

private TrackedEntity teWithNoEnrollments() {
TrackedEntity trackedEntity = createTrackedEntity(organisationUnit);
TrackedEntity trackedEntity =
createTrackedEntity(organisationUnit, createTrackedEntityType('Z'));
trackedEntity.setUid(TE_ID.getValue());
trackedEntity.setEnrollments(Sets.newHashSet());
trackedEntity.setTrackedEntityType(trackedEntityType);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

setTrackedEntityType

Expand All @@ -312,16 +313,16 @@ private TrackedEntity teWithDeleteEnrollments() {
Enrollment enrollment = new Enrollment();
enrollment.setDeleted(true);

TrackedEntity trackedEntity = createTrackedEntity(organisationUnit);
TrackedEntity trackedEntity =
createTrackedEntity(organisationUnit, createTrackedEntityType('B'));
trackedEntity.setUid(TE_ID.getValue());
trackedEntity.setEnrollments(Sets.newHashSet(enrollment));
trackedEntity.setTrackedEntityType(trackedEntityType);

return trackedEntity;
}

private TrackedEntity teWithEnrollments() {
TrackedEntity trackedEntity = createTrackedEntity(organisationUnit);
TrackedEntity trackedEntity =
createTrackedEntity(organisationUnit, createTrackedEntityType('R'));
trackedEntity.setUid(TE_ID.getValue());
trackedEntity.setEnrollments(Sets.newHashSet(new Enrollment()));
trackedEntity.setTrackedEntityType(trackedEntityType);
zubaira marked this conversation as resolved.
Show resolved Hide resolved
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
DO $$
DECLARE
inconsistent_records_count INTEGER;
BEGIN
SELECT COUNT(*) INTO inconsistent_records_count FROM trackedentity WHERE trackedentitytypeid IS NULL;
IF inconsistent_records_count > 0 THEN
RAISE NOTICE 'Inconsistencies found: %', inconsistent_records_count;

UPDATE trackedentity
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Here we shouldn't update any record.
We should only apply the constraint.
The we can suggest such a script in the migration notes

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

there is a lower possibility of errors if we handle the task ourselves rather than leaving it to the user. Therefore, we should aim to complete as much of the migration process as possible and leave as little as necessary for the user to manage.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

For this kind of process that deal with data we need to be much more clear with the user and very very careful on any step.
Here you are fixing inconsistent data in a correct way, getting the tracked entity type from any underling enrollment but this is not the only issue that could happen and this SQL is running in a transaction (even without specifically start the transaction as done here, flyway will wrap this in a transaction), so if there is any inconsistent data that cannot be fixed in the way you provided, no data is fixed.
Than the user is sent to the guide and there we are not providing any explanation of what we attempted in the migration and we are not providing any option of fixing the data, only a script to delete tracked entitities and with all the enrollments and events.

To resume:
If the user has a number x of "fixable" tracked entities and one that is not automatically fixable, we are deleting ALL the tracked entities with inconsistent data.

SET trackedentitytypeid = (
SELECT program.trackedentitytypeid
FROM program
JOIN enrollment ON program.programid = enrollment.programid
WHERE enrollment.trackedentityid = trackedentity.trackedentityid
)
WHERE trackedentitytypeid IS NULL
AND EXISTS (
SELECT 1
FROM enrollment
JOIN program ON enrollment.programid = program.programid
WHERE enrollment.trackedentityid = trackedentity.trackedentityid
);
END IF;
ALTER TABLE IF EXISTS trackedentity ALTER COLUMN trackedentitytypeid SET NOT NULL;
EXCEPTION
WHEN not_null_violation THEN
RAISE EXCEPTION 'The database contains inconsistent data that must be resolved before re-running this migration. Please refer to the migration notes for detailed instructions: https://github.com/dhis2/dhis2-releases/blob/master/releases/2.42/migration-notes.md#null-tracked-entity-type. Detailed error message: %', SQLERRM;
END $$;
Original file line number Diff line number Diff line change
Expand Up @@ -2135,29 +2135,36 @@ public static TrackedEntityType createTrackedEntityType(char uniqueChar) {
return trackedEntityType;
}

public static TrackedEntity createTrackedEntity(OrganisationUnit organisationUnit) {
public static TrackedEntity createTrackedEntity(
OrganisationUnit organisationUnit, TrackedEntityType trackedEntityType) {
TrackedEntity trackedEntity = new TrackedEntity();
trackedEntity.setAutoFields();
trackedEntity.setOrganisationUnit(organisationUnit);
trackedEntity.setTrackedEntityType(trackedEntityType);

return trackedEntity;
}

public static TrackedEntity createTrackedEntity(
char uniqueChar, OrganisationUnit organisationUnit) {
char uniqueChar, OrganisationUnit organisationUnit, TrackedEntityType trackedEntityType) {
TrackedEntity trackedEntity = new TrackedEntity();
trackedEntity.setAutoFields();
trackedEntity.setOrganisationUnit(organisationUnit);
trackedEntity.setUid(BASE_TE_UID + uniqueChar);
trackedEntity.setTrackedEntityType(trackedEntityType);

return trackedEntity;
}

public static TrackedEntity createTrackedEntity(
char uniqueChar, OrganisationUnit organisationUnit, TrackedEntityAttribute attribute) {
char uniqueChar,
OrganisationUnit organisationUnit,
TrackedEntityAttribute attribute,
TrackedEntityType trackedEntityType) {
TrackedEntity trackedEntity = new TrackedEntity();
trackedEntity.setAutoFields();
trackedEntity.setOrganisationUnit(organisationUnit);
trackedEntity.setTrackedEntityType(trackedEntityType);

TrackedEntityAttributeValue attributeValue = new TrackedEntityAttributeValue();
attributeValue.setAttribute(attribute);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -411,9 +411,8 @@ void setUp() throws ConflictException {
manager.save(trackedEntityType);

// Tracked Entity Instances (Registrations)
TrackedEntity teiA = createTrackedEntity(ouD);
TrackedEntity teiA = createTrackedEntity(ouD, trackedEntityType);
teiA.setUid("trackEntInA");
teiA.setTrackedEntityType(trackedEntityType);
manager.save(teiA);

// Tracked Entity Attribute Values
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -56,6 +56,7 @@
import org.hisp.dhis.test.integration.PostgresIntegrationTestBase;
import org.hisp.dhis.trackedentity.TrackedEntity;
import org.hisp.dhis.trackedentity.TrackedEntityAttribute;
import org.hisp.dhis.trackedentity.TrackedEntityType;
import org.hisp.dhis.trackedentityattributevalue.TrackedEntityAttributeValue;
import org.hisp.dhis.tracker.trackedentityattributevalue.TrackedEntityAttributeValueService;
import org.junit.jupiter.api.Disabled;
Expand Down Expand Up @@ -108,7 +109,10 @@ void testSaveTrackedEntity() {
TrackedEntityAttribute attribute = createTrackedEntityAttribute('A');
manager.save(ou);
manager.save(attribute);
TrackedEntity trackedEntity = createTrackedEntity('A', ou, attribute);

TrackedEntityType trackedEntityType = createTrackedEntityType('O');
manager.save(trackedEntityType);
TrackedEntity trackedEntity = createTrackedEntity('A', ou, attribute, trackedEntityType);
manager.save(trackedEntity);
AuditQuery query = AuditQuery.builder().uid(Sets.newHashSet(trackedEntity.getUid())).build();
await().atMost(TIMEOUT, TimeUnit.SECONDS).until(() -> auditService.countAudits(query) >= 0);
Expand All @@ -127,7 +131,11 @@ void testSaveTrackedAttributeValue() {
TrackedEntityAttribute attribute = createTrackedEntityAttribute('A');
manager.save(ou);
manager.save(attribute);
TrackedEntity trackedEntity = createTrackedEntity('A', ou, attribute);

TrackedEntityType trackedEntityType = createTrackedEntityType('O');
manager.save(trackedEntityType);

TrackedEntity trackedEntity = createTrackedEntity('A', ou, attribute, trackedEntityType);
manager.save(trackedEntity);
TrackedEntityAttributeValue dataValue =
createTrackedEntityAttributeValue('A', trackedEntity, attribute);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -44,6 +44,7 @@
import org.hisp.dhis.test.integration.PostgresIntegrationTestBase;
import org.hisp.dhis.trackedentity.TrackedEntity;
import org.hisp.dhis.trackedentity.TrackedEntityAttribute;
import org.hisp.dhis.trackedentity.TrackedEntityType;
import org.junit.jupiter.api.Disabled;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
Expand All @@ -70,7 +71,9 @@ class HardDeleteAuditTest extends PostgresIntegrationTestBase {
void testHardDeleteTrackedEntity() {
OrganisationUnit ou = createOrganisationUnit('A');
TrackedEntityAttribute attribute = createTrackedEntityAttribute('A');
TrackedEntity trackedEntity = createTrackedEntity('A', ou, attribute);
TrackedEntityType trackedEntityType = createTrackedEntityType('O');
manager.save(trackedEntityType);
TrackedEntity trackedEntity = createTrackedEntity('A', ou, attribute, trackedEntityType);
transactionTemplate.execute(
status -> {
manager.save(ou);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -164,13 +164,11 @@ void setUp() {
programService.updateProgram(program);
TrackedEntityType trackedEntityType = createTrackedEntityType('A');
trackedEntityTypeService.addTrackedEntityType(trackedEntityType);
trackedEntity = createTrackedEntity(organisationUnit);
trackedEntity.setTrackedEntityType(trackedEntityType);
trackedEntity = createTrackedEntity(organisationUnit, trackedEntityType);
manager.save(trackedEntity);
trackedEntityB = createTrackedEntity(organisationUnit);
trackedEntityB.setTrackedEntityType(trackedEntityType);
trackedEntityB = createTrackedEntity(organisationUnit, trackedEntityType);
manager.save(trackedEntityB);
trackedEntityWithAssociations = createTrackedEntity('T', organisationUnit);
trackedEntityWithAssociations = createTrackedEntity('T', organisationUnit, trackedEntityType);
DateTime testDate1 = DateTime.now();
testDate1.withTimeAtStartOfDay();
testDate1 = testDate1.minusDays(70);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -95,6 +95,7 @@
import org.hisp.dhis.sms.command.hibernate.SMSCommandStore;
import org.hisp.dhis.test.integration.PostgresIntegrationTestBase;
import org.hisp.dhis.trackedentity.TrackedEntity;
import org.hisp.dhis.trackedentity.TrackedEntityType;
import org.hisp.dhis.util.DateUtils;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.DisplayName;
Expand Down Expand Up @@ -1385,7 +1386,9 @@ void dataApprovalMergeCocDiscardTest() throws ConflictException {
"Event attributeOptionCombo references to source COCs are replaced with target COC when using LAST_UPDATED, source COCs are not deleted")
void eventMergeTest() throws ConflictException {
// given
TrackedEntity trackedEntity = createTrackedEntity(ou1);
TrackedEntityType entityType = createTrackedEntityType('T');
manager.save(entityType);
TrackedEntity trackedEntity = createTrackedEntity(ou1, entityType);
manager.save(trackedEntity);
Enrollment enrollment = createEnrollment(program, trackedEntity, ou1);
manager.save(enrollment);
Expand Down Expand Up @@ -1437,7 +1440,9 @@ void eventMergeTest() throws ConflictException {
"Event eventDataValues references to source COCs are deleted using DISCARD, source COCs are deleted")
void eventMergeSourcesDeletedTest() throws ConflictException {
// given
TrackedEntity trackedEntity = createTrackedEntity(ou1);
TrackedEntityType entityType = createTrackedEntityType('T');
manager.save(entityType);
TrackedEntity trackedEntity = createTrackedEntity(ou1, entityType);
manager.save(trackedEntity);
Enrollment enrollment = createEnrollment(program, trackedEntity, ou1);
manager.save(enrollment);
Expand Down
Loading
Loading