All Downloads are FREE. Search and download functionalities are using the official Maven repository.

core_analytics_mapping.modelCoverage.mappedEntityBuilder.pure Maven / Gradle / Ivy

There is a newer version: 4.68.0
Show newest version
// 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