if (rightExpr instanceof JoinExpression)
{
JoinExpression joinExpr = (JoinExpression)rightExpr;
JoinType joinType = joinExpr.getType();
String joinAlias = joinExpr.getAlias();
PrimaryExpression joinPrimExpr = joinExpr.getPrimaryExpression();
Iterator<String> iter = joinPrimExpr.getTuples().iterator();
String rootId = iter.next();
String joinTableGroupName = null;
if (rootId.equalsIgnoreCase(candidateAlias))
{
// Join relative to the candidate
// Name table group of joined-to as per the relation
// Note : this will only work for one level out from the candidate TODO Extend this
joinTableGroupName = joinPrimExpr.getId();
}
else
{
// Join relative to some other alias
SQLTableMapping sqlTblMapping = getSQLTableMappingForAlias(rootId);
if (sqlTblMapping != null)
{
cmd = sqlTblMapping.cmd;
joinTableGroupName = sqlTblMapping.table.getGroupName();
sqlTbl = sqlTblMapping.table;
}
else
{
throw new NucleusUserException("Query has " + joinPrimExpr.getId() + " yet the first component "+
rootId + " is unknown!");
}
}
while (iter.hasNext())
{
String id = iter.next();
AbstractMemberMetaData mmd = cmd.getMetaDataForMember(id);
int relationType = mmd.getRelationType(clr);
DatastoreClass relTable = null;
AbstractMemberMetaData relMmd = null;
// TODO Cater for map in 1-N relations?
switch (relationType)
{
case Relation.ONE_TO_ONE_UNI:
relTable = storeMgr.getDatastoreClass(mmd.getTypeName(), clr);
cmd = mmgr.getMetaDataForClass(mmd.getType(), clr);
if (joinType == JoinType.JOIN_INNER || joinType == JoinType.JOIN_INNER_FETCH)
{
sqlTbl = stmt.innerJoin(sqlTbl, sqlTbl.getTable().getMemberMapping(mmd),
relTable, null, relTable.getIdMapping(), null, joinTableGroupName);
}
else
{
sqlTbl = stmt.leftOuterJoin(sqlTbl, sqlTbl.getTable().getMemberMapping(mmd),
relTable, null, relTable.getIdMapping(), null, joinTableGroupName);
}
break;
case Relation.ONE_TO_ONE_BI:
relTable = storeMgr.getDatastoreClass(mmd.getTypeName(), clr);
cmd = storeMgr.getMetaDataManager().getMetaDataForClass(mmd.getType(), clr);
if (mmd.getMappedBy() != null)
{
relMmd = mmd.getRelatedMemberMetaData(clr)[0];
if (joinType == JoinType.JOIN_INNER || joinType == JoinType.JOIN_INNER_FETCH)
{
sqlTbl = stmt.innerJoin(sqlTbl, sqlTbl.getTable().getIdMapping(),
relTable, null, relTable.getMemberMapping(relMmd), null, joinTableGroupName);
}
else
{
sqlTbl = stmt.leftOuterJoin(sqlTbl, sqlTbl.getTable().getIdMapping(),
relTable, null, relTable.getMemberMapping(relMmd), null, joinTableGroupName);
}
}
else
{
if (joinType == JoinType.JOIN_INNER || joinType == JoinType.JOIN_INNER_FETCH)
{
sqlTbl = stmt.innerJoin(sqlTbl, sqlTbl.getTable().getMemberMapping(mmd), relTable, null,
relTable.getIdMapping(), null, joinTableGroupName);
}
else
{
sqlTbl = stmt.leftOuterJoin(sqlTbl, sqlTbl.getTable().getMemberMapping(mmd), relTable, null,
relTable.getIdMapping(), null, joinTableGroupName);
}
}
break;
case Relation.ONE_TO_MANY_BI:
relTable = storeMgr.getDatastoreClass(mmd.getCollection().getElementType(), clr);
cmd = mmd.getCollection().getElementClassMetaData(clr, mmgr);
relMmd = mmd.getRelatedMemberMetaData(clr)[0];
if (mmd.getJoinMetaData() != null || relMmd.getJoinMetaData() != null)
{
// Join to join table, then to related table
ElementContainerTable joinTbl = (ElementContainerTable)storeMgr.getDatastoreContainerObject(mmd);
if (joinType == JoinType.JOIN_INNER || joinType == JoinType.JOIN_INNER_FETCH)
{
SQLTable joinSqlTbl = stmt.innerJoin(sqlTbl, sqlTbl.getTable().getIdMapping(),
joinTbl, null, joinTbl.getOwnerMapping(), null, null);
sqlTbl = stmt.innerJoin(joinSqlTbl, joinTbl.getElementMapping(),
relTable, null, relTable.getIdMapping(), null, joinTableGroupName);
}
else
{
SQLTable joinSqlTbl = stmt.leftOuterJoin(sqlTbl, sqlTbl.getTable().getIdMapping(),
joinTbl, null, joinTbl.getOwnerMapping(), null, null);
sqlTbl = stmt.leftOuterJoin(joinSqlTbl, joinTbl.getElementMapping(),
relTable, null, relTable.getIdMapping(), null, joinTableGroupName);
}
}
else
{
// Join to related table
if (joinType == JoinType.JOIN_INNER || joinType == JoinType.JOIN_INNER_FETCH)
{
sqlTbl = stmt.innerJoin(sqlTbl, sqlTbl.getTable().getIdMapping(),
relTable, null, relTable.getMemberMapping(relMmd), null, joinTableGroupName);
}
else
{
sqlTbl = stmt.leftOuterJoin(sqlTbl, sqlTbl.getTable().getIdMapping(),
relTable, null, relTable.getMemberMapping(relMmd), null, joinTableGroupName);
}
}
break;
case Relation.ONE_TO_MANY_UNI:
relTable = storeMgr.getDatastoreClass(mmd.getCollection().getElementType(), clr);
cmd = mmd.getCollection().getElementClassMetaData(clr, mmgr);
if (mmd.getJoinMetaData() != null)
{
// Join to join table, then to related table
ElementContainerTable joinTbl = (ElementContainerTable)storeMgr.getDatastoreContainerObject(mmd);
if (joinType == JoinType.JOIN_INNER || joinType == JoinType.JOIN_INNER_FETCH)
{
SQLTable joinSqlTbl = stmt.innerJoin(sqlTbl, sqlTbl.getTable().getIdMapping(),
joinTbl, null, joinTbl.getOwnerMapping(), null, null);
sqlTbl = stmt.innerJoin(joinSqlTbl, joinTbl.getElementMapping(),
relTable, null, relTable.getIdMapping(), null, joinTableGroupName);
}
else
{
SQLTable joinSqlTbl = stmt.leftOuterJoin(sqlTbl, sqlTbl.getTable().getIdMapping(),
joinTbl, null, joinTbl.getOwnerMapping(), null, null);
sqlTbl = stmt.leftOuterJoin(joinSqlTbl, joinTbl.getElementMapping(),
relTable, null, relTable.getIdMapping(), null, joinTableGroupName);
}
}
else
{
// Join to related table
JavaTypeMapping relMapping = relTable.getExternalMapping(mmd, MappingConsumer.MAPPING_TYPE_EXTERNAL_FK);
if (joinType == JoinType.JOIN_INNER || joinType == JoinType.JOIN_INNER_FETCH)
{
sqlTbl = stmt.innerJoin(sqlTbl, sqlTbl.getTable().getIdMapping(),
relTable, null, relMapping, null, joinTableGroupName);
}
else
{
sqlTbl = stmt.leftOuterJoin(sqlTbl, sqlTbl.getTable().getIdMapping(),
relTable, null, relMapping, null, joinTableGroupName);
}
}
break;
case Relation.MANY_TO_MANY_BI:
relTable = storeMgr.getDatastoreClass(mmd.getCollection().getElementType(), clr);
cmd = mmd.getCollection().getElementClassMetaData(clr, mmgr);
relMmd = mmd.getRelatedMemberMetaData(clr)[0];
// Join to join table, then to related table
CollectionTable joinTbl = (CollectionTable)storeMgr.getDatastoreContainerObject(mmd);
if (joinType == JoinType.JOIN_INNER || joinType == JoinType.JOIN_INNER_FETCH)
{
SQLTable joinSqlTbl = stmt.innerJoin(sqlTbl, sqlTbl.getTable().getIdMapping(),
joinTbl, null, joinTbl.getOwnerMapping(), null, null);
sqlTbl = stmt.innerJoin(joinSqlTbl, joinTbl.getElementMapping(),
relTable, null, relTable.getIdMapping(), null, joinTableGroupName);
}
else
{
SQLTable joinSqlTbl = stmt.leftOuterJoin(sqlTbl, sqlTbl.getTable().getIdMapping(),
joinTbl, null, joinTbl.getOwnerMapping(), null, null);
sqlTbl = stmt.leftOuterJoin(joinSqlTbl, joinTbl.getElementMapping(),
relTable, null, relTable.getIdMapping(), null, joinTableGroupName);
}
break;
case Relation.MANY_TO_ONE_BI:
relTable = storeMgr.getDatastoreClass(mmd.getTypeName(), clr);
cmd = storeMgr.getMetaDataManager().getMetaDataForClass(mmd.getType(), clr);
relMmd = mmd.getRelatedMemberMetaData(clr)[0];
if (mmd.getJoinMetaData() != null || relMmd.getJoinMetaData() != null)
{
// Join to join table, then to related table
joinTbl = (CollectionTable)storeMgr.getDatastoreContainerObject(relMmd);
if (joinType == JoinType.JOIN_INNER || joinType == JoinType.JOIN_INNER_FETCH)
{
SQLTable joinSqlTbl = stmt.innerJoin(sqlTbl, sqlTbl.getTable().getIdMapping(),
joinTbl, null, joinTbl.getElementMapping(), null, null);
sqlTbl = stmt.innerJoin(joinSqlTbl, joinTbl.getOwnerMapping(),
relTable, null, relTable.getIdMapping(), null, joinTableGroupName);
}
else
{
SQLTable joinSqlTbl = stmt.leftOuterJoin(sqlTbl, sqlTbl.getTable().getIdMapping(),
joinTbl, null, joinTbl.getElementMapping(), null, null);
sqlTbl = stmt.leftOuterJoin(joinSqlTbl, joinTbl.getOwnerMapping(),
relTable, null, relTable.getIdMapping(), null, joinTableGroupName);
}
}
else
{
// Join to owner table
JavaTypeMapping fkMapping = sqlTbl.getTable().getMemberMapping(mmd);
if (joinType == JoinType.JOIN_INNER || joinType == JoinType.JOIN_INNER_FETCH)
{
sqlTbl = stmt.innerJoin(sqlTbl, fkMapping,
relTable, null, relTable.getIdMapping(), null, joinTableGroupName);
}
else
{
sqlTbl = stmt.leftOuterJoin(sqlTbl, fkMapping,
relTable, null, relTable.getIdMapping(), null, joinTableGroupName);
}
}
break;
default:
break;
}
}
if (joinAlias != null)
{
if (explicitJoinPrimaryByAlias == null)
{
explicitJoinPrimaryByAlias = new HashMap<String, String>();
}
explicitJoinPrimaryByAlias.put(joinAlias, joinPrimExpr.getId());
}
SQLTableMapping tblMapping = new SQLTableMapping(sqlTbl, cmd, sqlTbl.getTable().getIdMapping());
setSQLTableMappingForAlias(joinAlias, tblMapping);
}