mirror of
https://github.com/Kingsrook/qqq.git
synced 2025-07-18 21:20:45 +00:00
Implement CHILD_POINTS_AT_PARENT use-case
This commit is contained in:
@ -29,6 +29,7 @@ import java.util.List;
|
|||||||
import com.kingsrook.qqq.backend.core.actions.tables.InsertAction;
|
import com.kingsrook.qqq.backend.core.actions.tables.InsertAction;
|
||||||
import com.kingsrook.qqq.backend.core.actions.tables.UpdateAction;
|
import com.kingsrook.qqq.backend.core.actions.tables.UpdateAction;
|
||||||
import com.kingsrook.qqq.backend.core.exceptions.QException;
|
import com.kingsrook.qqq.backend.core.exceptions.QException;
|
||||||
|
import com.kingsrook.qqq.backend.core.exceptions.QRuntimeException;
|
||||||
import com.kingsrook.qqq.backend.core.model.actions.tables.insert.InsertInput;
|
import com.kingsrook.qqq.backend.core.model.actions.tables.insert.InsertInput;
|
||||||
import com.kingsrook.qqq.backend.core.model.actions.tables.insert.InsertOutput;
|
import com.kingsrook.qqq.backend.core.model.actions.tables.insert.InsertOutput;
|
||||||
import com.kingsrook.qqq.backend.core.model.actions.tables.update.UpdateInput;
|
import com.kingsrook.qqq.backend.core.model.actions.tables.update.UpdateInput;
|
||||||
@ -42,18 +43,16 @@ import com.kingsrook.qqq.backend.core.utils.CollectionUtils;
|
|||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
** Standard/re-usable post-insert customizer, for the use case where, when we
|
** Standard/re-usable post-insert customizer, for the use case where, when we
|
||||||
** do an insert into table "parent", we want a record automatically inserted into
|
** do an insert into table "parent", we want a record automatically inserted into
|
||||||
** table "child", and there's a foreign key in "parent", pointed at "child"
|
** table "child". Optionally (based on RelationshipType), there can be a foreign
|
||||||
** e.g., named: "parent.childId".
|
** key in "parent", pointed at "child". e.g., named: "parent.childId".
|
||||||
**
|
**
|
||||||
** A similar use-case would have the foreign key in the child table - in which case,
|
|
||||||
** we could add a "Type" enum, plus abstract method to get our "Type", then logic
|
|
||||||
** to switch behavior based on type. See existing type enum, but w/ only 1 case :)
|
|
||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
public abstract class ChildInserterPostInsertCustomizer extends AbstractPostInsertCustomizer
|
public abstract class ChildInserterPostInsertCustomizer extends AbstractPostInsertCustomizer
|
||||||
{
|
{
|
||||||
public enum RelationshipType
|
public enum RelationshipType
|
||||||
{
|
{
|
||||||
PARENT_POINTS_AT_CHILD
|
PARENT_POINTS_AT_CHILD,
|
||||||
|
CHILD_POINTS_AT_PARENT
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -68,10 +67,17 @@ public abstract class ChildInserterPostInsertCustomizer extends AbstractPostInse
|
|||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
public abstract String getChildTableName();
|
public abstract String getChildTableName();
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
**
|
**
|
||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
public abstract String getForeignKeyFieldName();
|
public String getForeignKeyFieldName()
|
||||||
|
{
|
||||||
|
return (null);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
**
|
**
|
||||||
@ -88,7 +94,7 @@ public abstract class ChildInserterPostInsertCustomizer extends AbstractPostInse
|
|||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
List<QRecord> rs = new ArrayList<>();
|
List<QRecord> rs = records;
|
||||||
List<QRecord> childrenToInsert = new ArrayList<>();
|
List<QRecord> childrenToInsert = new ArrayList<>();
|
||||||
QTableMetaData table = getInsertInput().getTable();
|
QTableMetaData table = getInsertInput().getTable();
|
||||||
QTableMetaData childTable = getInsertInput().getInstance().getTable(getChildTableName());
|
QTableMetaData childTable = getInsertInput().getInstance().getTable(getChildTableName());
|
||||||
@ -97,12 +103,37 @@ public abstract class ChildInserterPostInsertCustomizer extends AbstractPostInse
|
|||||||
// iterate over the inserted records, building a list child records to insert //
|
// iterate over the inserted records, building a list child records to insert //
|
||||||
// for ones missing a value in the foreign key field. //
|
// for ones missing a value in the foreign key field. //
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
for(QRecord record : records)
|
switch(getRelationshipType())
|
||||||
{
|
{
|
||||||
if(record.getValue(getForeignKeyFieldName()) == null)
|
case PARENT_POINTS_AT_CHILD ->
|
||||||
{
|
{
|
||||||
childrenToInsert.add(buildChildForRecord(record));
|
String foreignKeyFieldName = getForeignKeyFieldName();
|
||||||
|
try
|
||||||
|
{
|
||||||
|
table.getField(foreignKeyFieldName);
|
||||||
|
}
|
||||||
|
catch(Exception e)
|
||||||
|
{
|
||||||
|
throw new QRuntimeException("For RelationshipType.PARENT_POINTS_AT_CHILD, a valid foreignKeyFieldName in the parent table must be given. "
|
||||||
|
+ "[" + foreignKeyFieldName + "] is not a valid field name in table [" + table.getName() + "]");
|
||||||
|
}
|
||||||
|
|
||||||
|
for(QRecord record : records)
|
||||||
|
{
|
||||||
|
if(record.getValue(foreignKeyFieldName) == null)
|
||||||
|
{
|
||||||
|
childrenToInsert.add(buildChildForRecord(record));
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
case CHILD_POINTS_AT_PARENT ->
|
||||||
|
{
|
||||||
|
for(QRecord record : records)
|
||||||
|
{
|
||||||
|
childrenToInsert.add(buildChildForRecord(record));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
default -> throw new IllegalStateException("Unexpected value: " + getRelationshipType());
|
||||||
}
|
}
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////////
|
||||||
@ -129,51 +160,70 @@ public abstract class ChildInserterPostInsertCustomizer extends AbstractPostInse
|
|||||||
/////////////////////////////////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
// for the PARENT_POINTS_AT_CHILD relationship type:
|
||||||
// iterate over the original list of records again - for any that need a child (e.g., are missing //
|
// iterate over the original list of records again - for any that need a child (e.g., are missing //
|
||||||
// foreign key), set their foreign key to a newly inserted child's key, and add them to be updated. //
|
// foreign key), set their foreign key to a newly inserted child's key, and add them to be updated. //
|
||||||
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
List<QRecord> recordsToUpdate = new ArrayList<>();
|
switch(getRelationshipType())
|
||||||
for(QRecord record : records)
|
|
||||||
{
|
{
|
||||||
Serializable primaryKey = record.getValue(table.getPrimaryKeyField());
|
case PARENT_POINTS_AT_CHILD ->
|
||||||
if(record.getValue(getForeignKeyFieldName()) == null)
|
|
||||||
{
|
{
|
||||||
///////////////////////////////////////////////////////////////////////////////////////////////////
|
rs = new ArrayList<>();
|
||||||
// get the corresponding child record, if it has any errors, set that as a warning in the parent //
|
List<QRecord> recordsToUpdate = new ArrayList<>();
|
||||||
///////////////////////////////////////////////////////////////////////////////////////////////////
|
for(QRecord record : records)
|
||||||
QRecord childRecord = insertedRecordIterator.next();
|
|
||||||
if(CollectionUtils.nullSafeHasContents(childRecord.getErrors()))
|
|
||||||
{
|
{
|
||||||
for(QStatusMessage error : childRecord.getErrors())
|
Serializable primaryKey = record.getValue(table.getPrimaryKeyField());
|
||||||
|
if(record.getValue(getForeignKeyFieldName()) == null)
|
||||||
{
|
{
|
||||||
record.addWarning(new QWarningMessage("Error creating child " + childTable.getLabel() + " (" + error.toString() + ")"));
|
///////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
// get the corresponding child record, if it has any errors, set that as a warning in the parent //
|
||||||
|
///////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
QRecord childRecord = insertedRecordIterator.next();
|
||||||
|
if(CollectionUtils.nullSafeHasContents(childRecord.getErrors()))
|
||||||
|
{
|
||||||
|
for(QStatusMessage error : childRecord.getErrors())
|
||||||
|
{
|
||||||
|
record.addWarning(new QWarningMessage("Error creating child " + childTable.getLabel() + " (" + error.toString() + ")"));
|
||||||
|
}
|
||||||
|
rs.add(record);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
Serializable foreignKey = childRecord.getValue(childTable.getPrimaryKeyField());
|
||||||
|
recordsToUpdate.add(new QRecord().withValue(table.getPrimaryKeyField(), primaryKey).withValue(getForeignKeyFieldName(), foreignKey));
|
||||||
|
record.setValue(getForeignKeyFieldName(), foreignKey);
|
||||||
|
rs.add(record);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
rs.add(record);
|
||||||
}
|
}
|
||||||
rs.add(record);
|
|
||||||
continue;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Serializable foreignKey = childRecord.getValue(childTable.getPrimaryKeyField());
|
////////////////////////////////////////////////////////////////////////////
|
||||||
recordsToUpdate.add(new QRecord().withValue(table.getPrimaryKeyField(), primaryKey).withValue(getForeignKeyFieldName(), foreignKey));
|
// update the originally inserted records to reference their new children //
|
||||||
record.setValue(getForeignKeyFieldName(), foreignKey);
|
////////////////////////////////////////////////////////////////////////////
|
||||||
rs.add(record);
|
UpdateInput updateInput = new UpdateInput();
|
||||||
|
updateInput.setTableName(getInsertInput().getTableName());
|
||||||
|
updateInput.setRecords(recordsToUpdate);
|
||||||
|
updateInput.setTransaction(this.insertInput.getTransaction());
|
||||||
|
new UpdateAction().execute(updateInput);
|
||||||
}
|
}
|
||||||
else
|
case CHILD_POINTS_AT_PARENT ->
|
||||||
{
|
{
|
||||||
rs.add(record);
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
// todo - some version of looking at the inserted children to confirm that they were inserted, and updating the parents with warnings if they weren't //
|
||||||
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
}
|
}
|
||||||
|
default -> throw new IllegalStateException("Unexpected value: " + getRelationshipType());
|
||||||
}
|
}
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////
|
|
||||||
// update the originally inserted records to reference their new children //
|
|
||||||
////////////////////////////////////////////////////////////////////////////
|
|
||||||
UpdateInput updateInput = new UpdateInput();
|
|
||||||
updateInput.setTableName(getInsertInput().getTableName());
|
|
||||||
updateInput.setRecords(recordsToUpdate);
|
|
||||||
updateInput.setTransaction(this.insertInput.getTransaction());
|
|
||||||
new UpdateAction().execute(updateInput);
|
|
||||||
|
|
||||||
return (rs);
|
return (rs);
|
||||||
}
|
}
|
||||||
|
catch(RuntimeException re)
|
||||||
|
{
|
||||||
|
throw (re);
|
||||||
|
}
|
||||||
catch(Exception e)
|
catch(Exception e)
|
||||||
{
|
{
|
||||||
throw new RuntimeException("Error inserting new child records for new parent records", e);
|
throw new RuntimeException("Error inserting new child records for new parent records", e);
|
||||||
|
@ -70,7 +70,7 @@ class ChildInserterPostInsertCustomizerTest extends BaseTest
|
|||||||
void testEmptyCases() throws QException
|
void testEmptyCases() throws QException
|
||||||
{
|
{
|
||||||
QInstance qInstance = QContext.getQInstance();
|
QInstance qInstance = QContext.getQInstance();
|
||||||
addPostInsertActionToTable(qInstance);
|
addPostInsertActionToPersonTable(qInstance);
|
||||||
|
|
||||||
InsertInput insertInput = new InsertInput();
|
InsertInput insertInput = new InsertInput();
|
||||||
insertInput.setTableName(TestUtils.TABLE_NAME_PERSON_MEMORY);
|
insertInput.setTableName(TestUtils.TABLE_NAME_PERSON_MEMORY);
|
||||||
@ -95,10 +95,21 @@ class ChildInserterPostInsertCustomizerTest extends BaseTest
|
|||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
**
|
**
|
||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
private static void addPostInsertActionToTable(QInstance qInstance)
|
private static void addPostInsertActionToPersonTable(QInstance qInstance)
|
||||||
{
|
{
|
||||||
qInstance.getTable(TestUtils.TABLE_NAME_PERSON_MEMORY)
|
qInstance.getTable(TestUtils.TABLE_NAME_PERSON_MEMORY)
|
||||||
.withCustomizer(TableCustomizers.POST_INSERT_RECORD.getRole(), new QCodeReference(PersonPostInsertAddFavoriteShapeCustomizer.class));
|
.withCustomizer(TableCustomizers.POST_INSERT_RECORD, new QCodeReference(PersonPostInsertAddFavoriteShapeCustomizer.class));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
private static void addPostInsertActionToShapeTable(QInstance qInstance)
|
||||||
|
{
|
||||||
|
qInstance.getTable(TestUtils.TABLE_NAME_SHAPE)
|
||||||
|
.withCustomizer(TableCustomizers.POST_INSERT_RECORD, new QCodeReference(ShapePostInsertAddPersonCustomizer.class));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -107,10 +118,10 @@ class ChildInserterPostInsertCustomizerTest extends BaseTest
|
|||||||
**
|
**
|
||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
@Test
|
@Test
|
||||||
void testSimpleCase() throws QException
|
void testSimpleParentPointsAtChildCase() throws QException
|
||||||
{
|
{
|
||||||
QInstance qInstance = QContext.getQInstance();
|
QInstance qInstance = QContext.getQInstance();
|
||||||
addPostInsertActionToTable(qInstance);
|
addPostInsertActionToPersonTable(qInstance);
|
||||||
|
|
||||||
assertEquals(0, TestUtils.queryTable(qInstance, TestUtils.TABLE_NAME_SHAPE).size());
|
assertEquals(0, TestUtils.queryTable(qInstance, TestUtils.TABLE_NAME_SHAPE).size());
|
||||||
|
|
||||||
@ -135,10 +146,10 @@ class ChildInserterPostInsertCustomizerTest extends BaseTest
|
|||||||
**
|
**
|
||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
@Test
|
@Test
|
||||||
void testComplexCase() throws QException
|
void testComplexParentPointsAtChildCase() throws QException
|
||||||
{
|
{
|
||||||
QInstance qInstance = QContext.getQInstance();
|
QInstance qInstance = QContext.getQInstance();
|
||||||
addPostInsertActionToTable(qInstance);
|
addPostInsertActionToPersonTable(qInstance);
|
||||||
|
|
||||||
assertEquals(0, TestUtils.queryTable(qInstance, TestUtils.TABLE_NAME_SHAPE).size());
|
assertEquals(0, TestUtils.queryTable(qInstance, TestUtils.TABLE_NAME_SHAPE).size());
|
||||||
|
|
||||||
@ -169,6 +180,34 @@ class ChildInserterPostInsertCustomizerTest extends BaseTest
|
|||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
**
|
**
|
||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
|
@Test
|
||||||
|
void testSimpleChildPointsAtParentCase() throws QException
|
||||||
|
{
|
||||||
|
QInstance qInstance = QContext.getQInstance();
|
||||||
|
addPostInsertActionToShapeTable(qInstance);
|
||||||
|
|
||||||
|
assertEquals(0, TestUtils.queryTable(qInstance, TestUtils.TABLE_NAME_PERSON_MEMORY).size());
|
||||||
|
assertEquals(0, TestUtils.queryTable(qInstance, TestUtils.TABLE_NAME_SHAPE).size());
|
||||||
|
|
||||||
|
InsertInput insertInput = new InsertInput();
|
||||||
|
insertInput.setTableName(TestUtils.TABLE_NAME_SHAPE);
|
||||||
|
insertInput.setRecords(List.of(
|
||||||
|
new QRecord().withValue("name", "Circle")
|
||||||
|
));
|
||||||
|
InsertOutput insertOutput = new InsertAction().execute(insertInput);
|
||||||
|
Integer shapeId = insertOutput.getRecords().get(0).getValueInteger("id");
|
||||||
|
|
||||||
|
List<QRecord> personRecords = TestUtils.queryTable(qInstance, TestUtils.TABLE_NAME_PERSON_MEMORY);
|
||||||
|
assertEquals(1, personRecords.size());
|
||||||
|
assertEquals(shapeId, personRecords.get(0).getValue("favoriteShapeId"));
|
||||||
|
assertEquals("loves Circle", personRecords.get(0).getValue("lastName"));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** for the person table - where we do PARENT_POINTS_AT_CHILD
|
||||||
|
*******************************************************************************/
|
||||||
public static class PersonPostInsertAddFavoriteShapeCustomizer extends ChildInserterPostInsertCustomizer
|
public static class PersonPostInsertAddFavoriteShapeCustomizer extends ChildInserterPostInsertCustomizer
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -215,4 +254,47 @@ class ChildInserterPostInsertCustomizerTest extends BaseTest
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** for the shape table - where we do CHILD_POINTS_AT_PARENT
|
||||||
|
*******************************************************************************/
|
||||||
|
public static class ShapePostInsertAddPersonCustomizer extends ChildInserterPostInsertCustomizer
|
||||||
|
{
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
@Override
|
||||||
|
public QRecord buildChildForRecord(QRecord parentRecord) throws QException
|
||||||
|
{
|
||||||
|
return (new QRecord()
|
||||||
|
.withValue("firstName", "Someone who")
|
||||||
|
.withValue("lastName", "loves " + parentRecord.getValue("name"))
|
||||||
|
.withValue("favoriteShapeId", parentRecord.getValue("id")));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
@Override
|
||||||
|
public String getChildTableName()
|
||||||
|
{
|
||||||
|
return (TestUtils.TABLE_NAME_PERSON_MEMORY);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
@Override
|
||||||
|
public RelationshipType getRelationshipType()
|
||||||
|
{
|
||||||
|
return (RelationshipType.CHILD_POINTS_AT_PARENT);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
Reference in New Issue
Block a user