storeMgr.getIdentifierFactory().newIdentifier(IdentifierFactory.TABLE,
"SOURCECLASS" + fmd.getAbsoluteFieldNumber() + "_" + i);
LogicSetExpression sourceTableExpr = qs.newTableExpression(sourceTables[i], sourceTableIdentifier, true)[0];
// Left outer join from target ID to the mapped-by field on the source class
ScalarExpression sourceExpr = sourceMappings[i].newScalarExpression(qs, sourceTableExpr);
ScalarExpression targetExpr;
if (tableIdentifier != null)
{
LogicSetExpression targetTableExpr;
if (qs.getTableExpression(tableIdentifier) == null)
{
targetTableExpr = qs.newTableExpression(m.getDatastoreContainer(), tableIdentifier);
}
else
{
targetTableExpr = qs.getTableExpression(tableIdentifier);
}
targetExpr = m.getDatastoreContainer().getIDMapping().newScalarExpression(qs, targetTableExpr);
}
else
{
targetExpr = m.getDatastoreContainer().getIDMapping().newScalarExpression(qs,
qs.getMainTableExpression());
}
qs.leftOuterJoin(sourceExpr, targetExpr, sourceTableExpr, true, true);
// Select the Id column(s) of the source class (via the Left Outer Join)
int[] columnNumbersByField = qs.select(sourceTableIdentifier, sourceTables[i].getIDMapping(), true);
// Copy these column numbers into our overall column numbers set
if (sourceMappings.length == 1)
{
// We only have one source so just reference these
colNums = columnNumbersByField;
}
else if (colNums != null)
{
// Append to the end of where we have got to
int[] tmpColNums = colNums;
colNums = new int[tmpColNums.length + columnNumbersByField.length];
for (int j=0;j<tmpColNums.length;j++)
{
colNums[j] = tmpColNums[j];
}
for (int j=0;j<columnNumbersByField.length;j++)
{
colNums[tmpColNums.length+j] = columnNumbersByField[j];
}
tmpColNums = null;
}
else
{
// Add to the start of our column numbers
colNums = new int[columnNumbersByField.length];
for (int j=0;j<columnNumbersByField.length;j++)
{
colNums[j] = columnNumbersByField[j];
}
}
}
statementExpressionIndex.setExpressionIndex(colNums);
}
}
else if (relationType == Relation.MANY_TO_ONE_BI)
{
AbstractMemberMetaData[] relatedMmds = fmd.getRelatedMemberMetaData(clr);
// TODO Cater for more than 1 related field
if (fmd.getJoinMetaData() != null || relatedMmds[0].getJoinMetaData() != null)
{
// 1-N bidirectional join table relation.
// Do a LEFT OUTER JOIN to the join table to pick up the value.
MappedStoreManager storeMgr = qs.getStoreManager();
DatastoreContainerObject joinTable = storeMgr.getDatastoreContainerObject(relatedMmds[0]);
JavaTypeMapping referenceMapping = null;
JavaTypeMapping selectMapping = null;
DatastoreElementContainer collTable = (DatastoreElementContainer)joinTable;
referenceMapping = collTable.getElementMapping();
selectMapping = collTable.getOwnerMapping();
DatastoreObject mainTable = qs.getMainTableExpression().getMainTable();
if (!mainTable.equals(joinTable))
{
// Statement selects the element table and the owner column should be selected
// Join across to the join table, and select the owner mapping of the join table
// TODO Move this hardcoded table name out into the calling class so it controls the names of all table aliases
DatastoreIdentifier joinTableIdentifier =
storeMgr.getIdentifierFactory().newIdentifier(IdentifierFactory.TABLE,
"JOINTABLE" + fmd.getAbsoluteFieldNumber());
LogicSetExpression table_expr_sub = qs.newTableExpression(joinTable, joinTableIdentifier, true)[0];
// Left outer join from our Id to the mapped-by field on the other class
ScalarExpression subExpr = referenceMapping.newScalarExpression(qs, table_expr_sub);
ScalarExpression schExpr = null;
if (tableIdentifier != null)
{
LogicSetExpression targetTableExpr;
if (qs.getTableExpression(tableIdentifier) == null)
{