core_analytics_mapping.modelCoverage.mappedEntityBuilder.pure Maven / Gradle / Ivy
// Copyright 2022 Goldman Sachs
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
import meta::analytics::class::modelCoverage::utility::*;
import meta::relational::metamodel::*;
import meta::relational::mapping::*;
import meta::analytics::mapping::modelCoverage::*;
import meta::pure::mapping::*;
import meta::external::store::model::*;
import meta::analytics::mapping::modelCoverage::utility::*;
Class meta::analytics::mapping::modelCoverage::AnalysisConfiguration
{
returnMappedEntityInfo: Boolean[1];
returnMappedPropertyInfo: Boolean[1];
}
Class meta::analytics::mapping::modelCoverage::AutoMappedHelperClass
{
mappedProperties: MappedProperty[*];
mappedEntities: MappedEntity[*];
}
function <> meta::analytics::mapping::modelCoverage::getPropertyTargetType(p:PropertyMapping[1], rootClassMappings:SetImplementation[*]):Type[1]
{
let targetId = $p.targetSetImplementationId;
if ($targetId->isEmpty() || $targetId == '',
| $p.property.genericType.rawType->toOne(),
|
$p->toOne()->match([
i:InlineEmbeddedSetImplementation[1] |
$rootClassMappings->filter(x | $x.id == $i.inlineSetImplementationId)->toOne().class;,
p:EmbeddedSetImplementation[1] |
$p.property.genericType.rawType->toOne();,
e:PropertyMapping[1] |
$rootClassMappings->filter(x | $x.id == $targetId)->toOne().class;
]));
}
function <> meta::analytics::mapping::modelCoverage::getPropertyTargetName(p:PropertyMapping[1], type:Type[1], rootClassMappings:SetImplementation[*]):String[0..1]
{
$p->toOne()->match([
i:InlineEmbeddedSetImplementation[1] |
let root = $rootClassMappings->filter(r | $r.id == $i.inlineSetImplementationId);
if ($root->isNotEmpty() && $root->toOne().root,
| $root->toOne().class->elementToPath(),
| $i.inlineSetImplementationId);,
p:EmbeddedSetImplementation[1] |
$p.targetSetImplementationId,
e:PropertyMapping[1] |
let root = $rootClassMappings->filter(r | $r.id == $p.targetSetImplementationId);
if ($root->isNotEmpty() && $root->toOne().root,
| $root->toOne().class->elementToPath(),
| $p.targetSetImplementationId);
,
a:Any[1] | []
])
}
function <> meta::analytics::mapping::modelCoverage::buildProperty(
p:AbstractProperty[1],
type:Type[1],
targetName:String[0..1],
propertyInfo:PropertyInfo[1],
config: AnalysisConfiguration[1]
): MappedProperty[1]
{
$type->match([
pt:PrimitiveType[1] |
^MappedProperty(name = $propertyInfo.name, info = if($config.returnMappedPropertyInfo == true, | ^MappedPropertyInfo(type = $type->mapType(), multiplicity = $propertyInfo.multiplicity), | [])),
et:Enumeration[1] |
^EnumMappedProperty(
name = $propertyInfo.name, enumPath = $type->elementToPath(), info = if($config.returnMappedPropertyInfo == true, | ^MappedPropertyInfo(type = MappedPropertyType.Enumeration, multiplicity = $propertyInfo.multiplicity), | [])),
ot:Any[1] |
let subType = if ($p.genericType.rawType != $type,
| $type->elementToPath(),
| []);
$p.owner->match([
a:Association[1] |
^AssociatedMappedProperty(
name = $propertyInfo.name,
info = if($config.returnMappedPropertyInfo == true, | ^MappedPropertyInfo(type = MappedPropertyType.Entity, multiplicity = $propertyInfo.multiplicity), | []),
entityPath = $targetName->toOne(),
association = $a->elementToPath(),
subType = $subType),
a:Any[*] |
^EntityMappedProperty(
name = $propertyInfo.name,
info = if($config.returnMappedPropertyInfo == true, | ^MappedPropertyInfo(type = MappedPropertyType.Entity, multiplicity = $propertyInfo.multiplicity), | []),
entityPath = $targetName->toOne(),
subType = $subType)
]);
]);
}
function <> meta::analytics::mapping::modelCoverage::processPropertyMapping(
p:PropertyMapping[1],
rootClassMappings:SetImplementation[*],
propertyMap:Map, PropertyInfo>[1],
config: AnalysisConfiguration[1]
):MappedProperty[1]
{
let type = getPropertyTargetType($p, $rootClassMappings);
let name = getPropertyTargetName($p, $type, $rootClassMappings);
$p.property->buildProperty($type, $name, $p.property->getPropertyInfo($propertyMap), $config);
}
function <> meta::analytics::mapping::modelCoverage::processMultiPropertyMapping(
p:PropertyMapping[*],
inheritanceMap:Map>[1],
propertyMap:Map, PropertyInfo>[1],
config: AnalysisConfiguration[1]
):MappedProperty[1]
{
let property = $p->at(0).property;
let type = $property.genericType.rawType->toOne();
let dataSetProperty = $property->buildProperty($type, $type->elementToPath(), $property->getPropertyInfo($propertyMap), $config);
let inheritanceKeyValue = $inheritanceMap->keyValues()
->filter(i | $i.first.class == $type)->last()
->sort({x,y|$x.first.class->elementToPath()->compare($y.first.class->elementToPath())});
if ($inheritanceKeyValue->isNotEmpty(),
|
let inheritanceMappings = $inheritanceKeyValue.second;
let implementations = $inheritanceMappings.values->filter(c | $c.id->in($p.targetSetImplementationId)).class->map(c | $c->elementToPath());
let entityPath = $p.targetSetImplementationId->sort()->joinStrings(',') + '@' + $type->elementToPath();
let targetSetImplementationIds = $p->map(pm | $pm->match([
o:OtherwiseEmbeddedSetImplementation[1] | $o.otherwisePropertyMapping.targetSetImplementationId,
i:InlineEmbeddedSetImplementation[1] | $i.inlineSetImplementationId,
p:PropertyMapping[1] | $p.targetSetImplementationId
]));
//if the inheritance mapping exactly matches the targets then we can just point to that instead of requiring inheritance node to be built as it would be identical
let isCorrectInheritanceType = $inheritanceMappings.values.id->containsAll($targetSetImplementationIds) && $targetSetImplementationIds->containsAll($inheritanceMappings.values.id);
if ($isCorrectInheritanceType,
| $dataSetProperty->match([
e:EntityMappedProperty[1] | ^$e(entityPath = $inheritanceKeyValue->toOne().first->buildEntityName()),
d:MappedProperty[1] | $d
]),
| $dataSetProperty->match([
a:AssociatedMappedProperty[1] |
^MultiInheritanceAssociatedMappedProperty(
name = $a.name, entityPath = $entityPath, info = $a.info,
association = $a.association, subClasses = $implementations, inheritanceEntityPath = $type->elementToPath()),
e:EntityMappedProperty[1] |
^MultiInheritanceEntityMappedProperty(
name = $e.name, info = $e.info, entityPath = $entityPath,
subClasses = $implementations, inheritanceEntityPath = $type->elementToPath()),
d:MappedProperty[1] | $d
])
);,
|
$dataSetProperty
);
}
function <> meta::analytics::mapping::modelCoverage::processPropertyMappings(
p:PropertyMapping[*],
rootClassMappings:SetImplementation[*],
inheritanceMap:Map>[1],
propertyMap:Map, PropertyInfo>[1],
config: AnalysisConfiguration[1]
): MappedProperty[*]
{
let grouped = $p->groupBy(p | $p.property);
$grouped->keyValues()->sort({x,y|$x.first.name->toOne()->compare($y.first.name->toOne())})->map(kv |
$kv.second.values->match([
e:PropertyMapping[1] | $e->processPropertyMapping($rootClassMappings, $propertyMap, $config),
e:PropertyMapping[*] | $e->processMultiPropertyMapping($inheritanceMap, $propertyMap, $config)
]);
);
}
function <> meta::analytics::mapping::modelCoverage::getQualifiedPropertyTarget(
qp:QualifiedProperty[1], classMappings:SetImplementation[*],
currentSetImplementation:SetImplementation[1], currentClass:Class[1], currentTargetName:String[1],
rootClassMappings:SetImplementation[*]):Pair[0..1]
{
let qualifiedPropertyType = $qp.genericType.rawType->toOne();
$qualifiedPropertyType->match([
p:PrimitiveType[1] | [],
e:Enumeration[1] | [],
c:Class[1] |
//we preference root mapped targets if any exist.
let potentialTargets = $classMappings->filter(cm | $cm.class == $qualifiedPropertyType &&
$cm->match([
e:EmbeddedSetImplementation[1] | false,
s:SetImplementation[1] | true
]));
let potentialRootTarget = $potentialTargets->filter(t | $t.root)->first();
let potentialTarget = if ($potentialRootTarget->isEmpty(), | $potentialTargets->first(), | $potentialRootTarget->toOne());
if ($qualifiedPropertyType == $currentClass,
| pair($currentSetImplementation, $currentTargetName),
| if ($potentialTarget->isEmpty() || $currentSetImplementation->instanceOf(OperationSetImplementation) || $potentialTarget->toOne()->instanceOf(OperationSetImplementation),
| pair($rootClassMappings->filter(cm | $cm.class == $qualifiedPropertyType)->at(0),
$qualifiedPropertyType->elementToPath());,
| pair($potentialTarget->toOne(),
$potentialTarget->toOne()->buildEntityName())
));
]);
}
function <> meta::analytics::mapping::modelCoverage::buildEntity(
class:Class[1],
path:String[1],
properties:MappedProperty[*],
isRootEntity:Boolean[1],
classInfo:ClassInfo[1],
config: AnalysisConfiguration[1]
): MappedEntity[1]
{
^MappedEntity(
path = $path,
properties = $properties->removeDuplicatesBy(x | $x.name),
info = if($config.returnMappedEntityInfo == true, | ^MappedEntityInfo(isRootEntity = $isRootEntity, classPath = $class->elementToPath()), | [])
);
}
function meta::analytics::mapping::modelCoverage::buildEntity(
class:Class[1],
target:String[1],
setImplementation:SetImplementation[1],
propertyMappings:PropertyMapping[*],
propertyFilter:Function<{AbstractProperty[1]->Boolean[1]}>[1],
rootClassMappings:SetImplementation[*],
classMap:Map, ClassInfo>[1],
propertiesMap:Map, PropertyInfo>[1],
inheritanceMap:Map>[1],
mappingClassMappings:Map>[1],
isRootEntity:Boolean[1],
config: AnalysisConfiguration[1]
): MappedEntity[*]
{
let classInfo = $class->getClassInfo($classMap);
let supportedPropertyMappings = $propertyMappings
->filter(pm |
!$pm.localMappingProperty->isTrue()
&& !$pm.property->meta::pure::milestoning::isRangeMilestoningProperty()
&& ($pm.property->isPrimitiveValueProperty()
|| $pm->match([semi: meta::relational::mapping::SemiStructuredRelationalPropertyMapping[1] | false, rpm:InstanceSetImplementation[1] | true, a: PropertyMapping[1] | false])
// handle SemiStructuredRelationalPropertyMapping manually to take care of automapped properties
|| $pm.targetSetImplementationId->in($rootClassMappings.id))
)->filter(pm | $propertyFilter->eval($pm.property));
let properties = $supportedPropertyMappings->processPropertyMappings($rootClassMappings, $inheritanceMap, $propertiesMap, $config);
//TODO this should take into account how the return type is constructed. this currently would miss out
//any qp that returns an embedded type within the class or one which returns a Class type which is
//not a root mapped entity. Ideally we should not even look in root mapped entities. We should only look
//in property types which have been mapped recursively. In the perfect world we would undertand exactly
//how the return type is mapped.
let qualifiedProperties = $classInfo.qualifiedProperties
->filter(p |
let returnType = $p->functionReturnType().rawType->toOne();
$p->isSupportedProperty()
&& !$p->meta::pure::milestoning::isRangeMilestoningProperty()
&& ($rootClassMappings.class->contains($returnType)
|| $returnType == $class
|| $returnType->instanceOf(PrimitiveType))
&& $propertyFilter->eval($p);
)
->map(qp |
let qualifiedPropertyType = $qp.genericType.rawType->toOne();
let classMappings = $mappingClassMappings->get($setImplementation.parent).values;
let qpTarget = getQualifiedPropertyTarget($qp, $classMappings, $setImplementation, $class, $target, $rootClassMappings);
$qp->buildProperty(
$qualifiedPropertyType->toOne(), $qpTarget.second, $qp->getPropertyInfo($propertiesMap), $config
); //TODO check how types work with qp's.we n eed to filter on whether the type is root supported currently
);
// handle auto mapped properties
let autoMappedPropertyMappings = $propertyMappings->removeAll($supportedPropertyMappings)
->filter(pm| $pm.owner->isNotEmpty() && $pm.owner->toOne()->instanceOf(InstanceSetImplementation) && meta::pure::router::store::routing::isPropertyAutoMapped($pm.owner->toOne()->cast(@InstanceSetImplementation), $pm.property, meta::relational::extension::relationalExtensions()));
let autoMappedHelperClasses = $autoMappedPropertyMappings->map(pm |
let id = if($pm.owner->isNotEmpty() && $pm.owner->toOne()->instanceOf(PureInstanceSetImplementation),
| let pureInstanceSetImplementation = $pm.owner->toOne()->cast(@PureInstanceSetImplementation);
let targetClassPath = if($pureInstanceSetImplementation.class->isNotEmpty(),|$pureInstanceSetImplementation.class->toOne()->cast(@Class)->elementToPath(),|'');
let srcClassPath = if($pureInstanceSetImplementation.srcClass->isNotEmpty(),|$pureInstanceSetImplementation.srcClass->toOne()->cast(@Class)->elementToPath(),|'');
$targetClassPath + '_' + $srcClassPath;,
|$pm.targetSetImplementationId);
processAutoMappedPropertyMapping($pm.property->cast(@AbstractProperty), $id + '_autoMapped_', $isRootEntity, $config, []););
let autoMappedPrimitiveProperties = if($setImplementation->instanceOf(PureInstanceSetImplementation),
|buildAutoMappedProperty($class, if($setImplementation->cast(@PureInstanceSetImplementation).srcClass->isEmpty(),|[],|$setImplementation->cast(@PureInstanceSetImplementation).srcClass->cast(@Class)->toOne()), $properties, $isRootEntity, $config),
|[]);
buildEntity(
$class,
$target,
$properties->concatenate($qualifiedProperties)->concatenate($autoMappedPrimitiveProperties)->concatenate($autoMappedHelperClasses->map(c|$c.mappedProperties)),
$isRootEntity,
$classInfo,
$config
)->concatenate($autoMappedHelperClasses->map(c|$c.mappedEntities));
}
function meta::analytics::mapping::modelCoverage::processAutoMappedPropertyMapping(
property: AbstractProperty[1],
prefix: String[1],
isRootEntity: Boolean[1],
config: AnalysisConfiguration[1],
visited: Pair, String>[*]
): AutoMappedHelperClass[1]
{
if($property->isPrimitiveValueProperty(),
|let mappedProp = ^MappedProperty(name = $property.name->toOne(),
info = if($config.returnMappedPropertyInfo == true,
| ^MappedPropertyInfo(type = $property.genericType.rawType->toOne()->mapType(), multiplicity = $property.multiplicity->toOne()),
| []));
^AutoMappedHelperClass(mappedProperties = [$mappedProp], mappedEntities=[]);,
|
if($visited->map(p|$p.first->elementToPath())->contains($property.genericType.rawType->toOne()->elementToPath()),
|let entityMappedProp = ^EntityMappedProperty(entityPath=$visited->filter(p|$p.first->elementToPath()==$property.genericType.rawType->toOne()->elementToPath())->at(0).second,
name=$property.name->toOne(),
info = if($config.returnMappedPropertyInfo == true,
| ^MappedPropertyInfo(type = $property.genericType.rawType->toOne()->mapType(), multiplicity = $property.multiplicity->toOne()),
| []));
^AutoMappedHelperClass(mappedProperties = [$entityMappedProp], mappedEntities=[]);,
|let entityPath = $prefix + $property.name->toOne();
let entityMappedProp = ^EntityMappedProperty(entityPath=$entityPath,name=$property.name->toOne(),
info = if($config.returnMappedPropertyInfo == true,
| ^MappedPropertyInfo(type = $property.genericType.rawType->toOne()->mapType(), multiplicity = $property.multiplicity->toOne()),
| []));
let newVisited = $visited->concatenate([^Pair, String>(first = $property.genericType.rawType->toOne()->cast(@Class), second = $entityPath)]);
let propertyType = $property.genericType.rawType->toOne()->cast(@Class);
let children = $propertyType.properties->map(p | $p->processAutoMappedPropertyMapping($entityPath + '_' , $isRootEntity, $config, $newVisited));
let currentMappedEntity = ^MappedEntity(path=$entityPath, properties = $children->map(c|$c.mappedProperties), info = if($config.returnMappedEntityInfo == true, | ^MappedEntityInfo(isRootEntity = $isRootEntity, classPath = $propertyType->elementToPath()), | []));
^AutoMappedHelperClass(mappedProperties = [$entityMappedProp], mappedEntities=[$currentMappedEntity]->concatenate($children->map(c|$c.mappedEntities)));
);
)
}
function meta::analytics::mapping::modelCoverage::buildAutoMappedProperty(
class:Class[1],
srcClass: Class[0..1],
mappedProperties: MappedProperty[*],
isRootEntity: Boolean[1],
config: AnalysisConfiguration[1]
): MappedProperty[*]
{
let autoMappedProperties = $class.properties->filter(p|!$p.name->in($mappedProperties.name))
->filter(p|$srcClass.properties->exists(sp|$sp.name == $p.name && $sp.genericType.rawType == $p.genericType.rawType && $sp.multiplicity == $p.multiplicity));
let primitiveProperties = $autoMappedProperties->filter(p|$p->isPrimitiveValueProperty());
$primitiveProperties->map(p|^MappedProperty(name = $p.name->toOne(), info = if($config.returnMappedPropertyInfo == true, | ^MappedPropertyInfo(type = $p.genericType.rawType->toOne()->mapType(), multiplicity = $p.multiplicity->toOne()), | [])));
}
function <> meta::analytics::mapping::modelCoverage::buildInheritanceEntitiesImpl(
baseMapping:SetImplementation[1],
mapping:SetImplementation[1],
seenInheritanceTypes: Class[*],
mappings:SetImplementation[*],
namePrefix: String[1],
originalPropertyFilter:Function<{AbstractProperty[1]->Boolean[1]}>[1],
rootClassMappings:SetImplementation[*],
classMap:Map, ClassInfo>[1],
propertiesMap:Map, PropertyInfo>[1],
inheritanceMap:Map>[1],
mappingClassMappings:Map>[1],
isRootEntity:Boolean[1],
isBase:Boolean[1],
config: AnalysisConfiguration[1]
): MappedEntity[*]
{
let class = $mapping.class;
let specs = $class.specializations.specific->cast(@Class);
let mappedProperties = $mapping->match([
o:OperationSetImplementation[1] |
let properties = $mappings
->filter(c | $c->instanceOf(InstanceSetImplementation))
->cast(@InstanceSetImplementation)->map(x|$x->allPropertyMappings());
if ($isRootEntity,
| $properties->groupBy(p | $p.property)->keyValues()->sort({x,y|$x.first.name->toOne()->compare($y.first.name->toOne())})->map(kv |
$kv.second.values->match([
pm:PropertyMapping[1] | $pm,
pm:PropertyMapping[*] | $pm->fold({p, acc |
$p->match([
e:EmbeddedSetImplementation[1] |
if ($acc->exists(a | $a.property == $e.property), | $acc, | $acc->add($p));,
p:PropertyMapping[1] |
let root = $rootClassMappings->exists(rm | $rm.root && $rm.class == $p.property.genericType.rawType);
if (!$root,
| if ($acc->exists(pm | $pm.property == $p.property), | $acc, | $acc->add($p)),
| $acc->add($p));
])
}, []->cast(@PropertyMapping))
])),
| $properties->removeDuplicatesBy(pm | $pm.property));,
i:InstanceSetImplementation[1] | $i->allPropertyMappings()
]);
let propertyFilter = {a:AbstractProperty[1] |
let ownerClass = $a->ownerClass();
if ($mapping.class == $baseMapping.class,
| $ownerClass->in($mapping.class->superTypes()->concatenate($mapping.class)),
| $ownerClass == $mapping.class || $ownerClass->in($mapping.class->superTypes(false)->removeAll($seenInheritanceTypes)))
&& $originalPropertyFilter->eval($a);
};
let inheritanceTypes = $mapping->match([
o:OperationSetImplementation[1] | $seenInheritanceTypes->add($o.class),
i:InstanceSetImplementation[1] | $seenInheritanceTypes
]);
let name = $namePrefix + if ($isBase, | $mapping->buildEntityName(),| $class->createInheritanceName());
let entities = buildEntity($class, $name, $mapping,
$mappedProperties, $propertyFilter, $rootClassMappings, $classMap, $propertiesMap, $inheritanceMap, $mappingClassMappings, $isRootEntity, $config);
let rootEntity = $entities->at(0);
let currentMappings = $mappings->filter(m | $m.class->in($specs));
let inheritanceEntities = $currentMappings->map(cm |
$baseMapping->buildInheritanceEntitiesImpl($cm, $inheritanceTypes, $mappings, $namePrefix, $originalPropertyFilter, $rootClassMappings, $classMap, $propertiesMap,
$inheritanceMap, $mappingClassMappings, false, false, $config);
);
^$rootEntity(properties = $rootEntity.properties->concatenate($currentMappings->map(cm |
let name = $cm.class.name->toOne();
^InheritanceMappedProperty(
name = $name->substring(0, 1)->toLower() + $name->substring(1),
info = if($config.returnMappedPropertyInfo == true, | ^MappedPropertyInfo(type = MappedPropertyType.Entity, multiplicity = PureOne), | []),
entityPath = $namePrefix + $cm.class->createInheritanceName(),
subType = $cm.class->elementToPath()
);
)))->concatenate($entities->removeAll($rootEntity))->concatenate($inheritanceEntities);
}
function <> meta::analytics::mapping::modelCoverage::buildInheritanceEntities(
baseMapping:SetImplementation[1],
mapping:SetImplementation[1],
seenInheritanceTypes: Class[*],
mappings:SetImplementation[*],
namePrefix: String[1],
originalPropertyFilter:Function<{AbstractProperty[1]->Boolean[1]}>[1],
rootClassMappings:SetImplementation[*],
classMap:Map, ClassInfo>[1],
propertiesMap:Map, PropertyInfo>[1],
inheritanceMap:Map>[1],
mappingClassMappings:Map>[1],
isRootEntity:Boolean[1],
isBase:Boolean[1],
config: AnalysisConfiguration[1]
):MappedEntity[*]
{
let entities = buildInheritanceEntitiesImpl($baseMapping, $mapping, $seenInheritanceTypes, $mappings,
$namePrefix, $originalPropertyFilter, $rootClassMappings, $classMap, $propertiesMap, $inheritanceMap, $mappingClassMappings, $isRootEntity, $isBase, $config);
$entities->map(e | ^$e(properties = $e.properties->removeDuplicatesBy(x | $x.name)));
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy