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

core.pure.mapping.mappingExtension.pure Maven / Gradle / Ivy

There is a newer version: 4.67.9
Show newest version
// Copyright 2020 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::pure::router::metamodel::*;
import meta::pure::router::operations::*;
import meta::pure::router::routing::*;
import meta::pure::router::clustering::*;
import meta::pure::mapping::*;
import meta::pure::mapping::aggregationAware::*;
import meta::external::store::model::*;
import meta::pure::store::*;
import meta::core::runtime::*;
import meta::pure::mapping::*;

Class meta::pure::mapping::StaticMappingInstanceData
{
   mapping : Mapping[1];
   runtime : Runtime[1];
   systemMapping : StoreContract[1];
   setImplementation : InstanceSetImplementation[0..1];
   exeCtx:meta::pure::runtime::ExecutionContext[1];
   debug :DebugContext[1];
   extensions : meta::pure::extension::Extension[*];
}

Class meta::pure::mapping::MappingInstanceData
{
   static : StaticMappingInstanceData[1];
   mappingPropertyValues : KeyValue[*];
}

Class meta::pure::mapping::StoreQuery
{
    vs : ValueSpecification[1];
    inScopeVars : Map>[1];
    store : Store[1];
    //sets : Pair>[*]; //inputs to either cross store ( query )
}

Class meta::pure::mapping::TabularDataSetImplementation extends TabularDataSet
{
  store : 	meta::pure::store::Store[1];
}

function meta::pure::mapping::sourceClass(setImplementation:SetImplementation[1], extensions:meta::pure::extension::Extension[*]): Type[1]
{
   $setImplementation->match([
      pure: PureInstanceSetImplementation[1]             | $pure.srcClass->toOne()
   ]->concatenate($extensions.availableStores.extractSourceClassFromSetImplementation)->toOneMany());
}

function meta::pure::mapping::buildClassMappingsByClassMap(mapping:Mapping[1]):Map,List>[1]
{
   let allClassMappings = $mapping->classMappings();

   let mappingsInScope = $allClassMappings->filter(s|$s.root == true || $s->instanceOf(EmbeddedSetImplementation))
                           ->concatenate($allClassMappings->map(x | $x->match([
                              a : meta::pure::mapping::aggregationAware::AggregationAwareSetImplementation[1] | $a.aggregateSetImplementations.setImplementation->map(y | ^$y(class = $y.mappingClass->toOne())),
                              s : SetImplementation[1] | []
                           ])));

    $mappingsInScope->meta::pure::router::routing::potentiallyResolveOperation($mapping, $allClassMappings)->groupBy(i | $i.class);
}

function meta::pure::mapping::buildClassMappingsById(mapping:Mapping[1]):Map>[1]
{
   $mapping->classMappings()->groupBy(m|$m.id);
}

function meta::pure::mapping::getClassMappingById(idToClassMapping: Map>[1], id: String[1]): SetImplementation[*]
{
  $idToClassMapping->get($id).values;
}

function meta::pure::mapping::mergeMappings(mappings: Mapping[1..*]):Mapping[1]
{
   if($mappings->distinct()->size()==1,
   |$mappings->at(0),
   |let initialMapping = meta::pure::mapping::MergedMapping;
    let updatedMappings = $mappings->map(map | $map->updateImplementationIDForMapping('_'+$map->elementToPath()->toOne())
                                         );
  let setImplementationsInAUnion = $updatedMappings.classMappings->filter(c|$c->instanceOf(OperationSetImplementation))->cast(@OperationSetImplementation).parameters.setImplementation;
  let groupedClasses = $updatedMappings.classMappings->filter(c|$c->instanceOf(SetImplementation))->filter(c|!$c.id->in($setImplementationsInAUnion.id))->cast(@SetImplementation)->groupBy({r|$r.class});
  let newClasssMappings = $groupedClasses->keys()->map(c|
                                                          let setImplementations = $groupedClasses->get($c).values;
                                                          if( $setImplementations->size() >1,
                                                            | let noRootSets = $setImplementations ->map(s| ^$s(root=false)) ;
                                                              ^OperationSetImplementation( class = $c,
                                                                                            id=  $c->elementToPath()->replace('::','_'),  //Need version that has underscores
                                                                                            operation=   meta::pure::router::operations::union_OperationSetImplementation_1__SetImplementation_MANY_ ,
                                                                                            parameters = $noRootSets->map(s | ^SetImplementationContainer(id = $s.id,
                                                                                                                                                          setImplementation = $s)),
                                                                                            parent = $initialMapping,
                                                                                            root = true
                                                                                     )->concatenate($noRootSets);,
                                                             |   $setImplementations->at(0) ;
                                                           );
                                                       )->concatenate($setImplementationsInAUnion);

   ^$initialMapping( includes =   $mappings.includes->removeDuplicates(),
                     classMappings = $newClasssMappings,
                     enumerationMappings = $updatedMappings.enumerationMappings,
                     associationMappings = $updatedMappings.associationMappings
                 );
  );
}

function <> meta::pure::mapping::updateImplementationIDForMapping(mapping:Mapping[1], suffix:String[1]):Mapping[1]
{
   let toUpdateClassMappingIds = $mapping.classMappings.id;
   ^$mapping(classMappings = $mapping.classMappings->map(c|$c->updateImplementationID($suffix, $toUpdateClassMappingIds)->cast(@SetImplementation)),
             enumerationMappings = $mapping.enumerationMappings, //TODO: no IDs so we have possible conflict if we had two mappings with same enumeration name in each one
             associationMappings = $mapping.associationMappings->map(a|$a->updateImplementationID($suffix, $toUpdateClassMappingIds)->cast(@AssociationImplementation))
             );
}


function <> meta::pure::mapping::updateImplementationID(owner:PropertyOwnerImplementation[1],id:String[1], inScopeUpdatedClassMappingIds:String[*]):PropertyOwnerImplementation[1]
{
  $owner->match( [p:PropertyMappingsImplementation[1]|^$p(id = $p.id + $id,propertyMappings = $p.propertyMappings->map(p|$p->updatePropertyMappingIDWithSuffix($id, $inScopeUpdatedClassMappingIds))),
                  o:OperationSetImplementation[1] | ^$o(parameters = $o.parameters->map(p |^$p(id = $p.id + $id, setImplementation = $p.setImplementation->updateImplementationID($id, $inScopeUpdatedClassMappingIds)->cast(@SetImplementation)))),
                  s:SetImplementation[1]  |^$s(id= $s.id + $id)
                 ]);
}

function <> meta::pure::mapping::updatePropertyMappingIDWithSuffix(p:meta::pure::mapping::PropertyMapping[1], id:String[1], inScopeUpdatedClassMappingIds:String[*]):meta::pure::mapping::PropertyMapping[1]
{
  ^$p( sourceSetImplementationId = if($p.sourceSetImplementationId->isEmpty() || !$inScopeUpdatedClassMappingIds->contains($p.sourceSetImplementationId),|$p.sourceSetImplementationId ,| $p.sourceSetImplementationId + $id),
       targetSetImplementationId = if($p.targetSetImplementationId->isEmpty() || !$inScopeUpdatedClassMappingIds->contains($p.targetSetImplementationId),|$p.targetSetImplementationId ,| $p.targetSetImplementationId + $id)
     );
}

function meta::pure::mapping::allEnumerationMappings(m: Mapping[1]): EnumerationMapping[*]
{
  $m.enumerationMappings->concatenate($m.includes->map(i | $i.included->allEnumerationMappings()))
}

function meta::pure::mapping::valueTransformerEquality(a:ValueTransformer[0..1], b:ValueTransformer[0..1]):Boolean[1]
{
  ($a->isEmpty() && $b->isEmpty()) || ($a->isNotEmpty() && $b->isNotEmpty() && $a->type() == $b->type() && $a->match([
                                                                                                                      i:EnumerationMapping[1]| $i.enumeration == $b->cast(@EnumerationMapping).enumeration && enumValueMappingsEquality($i.enumValueMappings->zip($b->cast(@EnumerationMapping).enumValueMappings));,
                                                                                                                      v:ValueTransformer[1]| fail('UnSupported Operation: Only supports EnumerationMapping')
                                                                                                                    ]))
}

function meta::pure::mapping::enumValueMappingsEquality(pairs:Pair[*]):Boolean[1]
{  
  $pairs.first.enum == $pairs.second.enum && $pairs.first.sourceValues == $pairs.second.sourceValues
}

function meta::pure::mapping::allSuperSetImplementations(set :PropertyMappingsImplementation[1], m:Mapping[1]):PropertyMappingsImplementation[*]
{
      if ($set.superSetImplementationId->isEmpty(),
       | [],
       | let superMapping = $m->classMappingById($set.superSetImplementationId->toOne());
         if($superMapping->isEmpty(),
            | [],
            |$superMapping->toOne()->cast(@PropertyMappingsImplementation)->allSuperSetImplementations($m)->concatenate($superMapping->toOne()->cast(@PropertyMappingsImplementation))
         );
    );
}

function meta::pure::router::clustering::resolveInstanceSetImplementations(o:OperationSetImplementation[1]):InstanceSetImplementation[*]
{
   let sets = $o.operation->eval($o);
   $sets->map(s | $s->match([
                              o:OperationSetImplementation[1] | $o->resolveInstanceSetImplementations(),
                              e:EmbeddedSetImplementation[1] | $e->resolveRootFromEmbedded(),
                              i:InstanceSetImplementation[1] | $i
                            ]));
}

function meta::pure::router::clustering::resolveRootFromEmbedded(o:EmbeddedSetImplementation[1]):InstanceSetImplementation[1]
{
   $o.owner->match([
                     e:EmbeddedSetImplementation[1]|$e->resolveRootFromEmbedded(),
                     i:InstanceSetImplementation[1]|$i
                  ]);
}

function meta::pure::router::routing::getStoreFromSet(set: SetImplementation[1], mapping: Mapping[1], extensions:meta::pure::extension::Extension[*]):Store[1]
{
  $set->storeContractForSetImplementation($mapping, $extensions).second;
}

function meta::pure::router::routing::findMappingsFromProperty(p:AbstractProperty[1], mapping:Mapping[1], sourceMappings:SetImplementation[*], state:RoutingState[1], extensions:meta::pure::extension::Extension[*]):SetImplementation[*]
{
   let c = $p->functionReturnType().rawType->toOne();
   if ($c->instanceOf(DataType),
      |[],
      |// Get potential embedded mappings (only for concrete SetImplementation). Operations are still possible in the old flow
       let embeddedMappings = if($sourceMappings->isEmpty(),
                                 |[],
                                 |$sourceMappings->filter(s|$s->instanceOf(PropertyMappingsImplementation))->cast(@PropertyMappingsImplementation)->map(s|$s->_propertyMappingsByPropertyName($p.name->toOne())->map(pm | $pm->reprocessAggregationAwarePropertyMapping())->filter(p|$p->instanceOf(EmbeddedSetImplementation)))->cast(@SetImplementation)
                              );

      let embeddedExistsInOperation = !$sourceMappings->resolveOperation($mapping)->cast(@PropertyMappingsImplementation)->map(s|$s->_propertyMappingsByPropertyName($p.name->toOne())->map(pm | $pm->reprocessAggregationAwarePropertyMapping())->filter(p|$p->instanceOf(EmbeddedSetImplementation)))->isEmpty();

      let classMapping = if($embeddedMappings->isEmpty(),
                                   |  if ($embeddedExistsInOperation,
                                          | let embeddedMappings = $sourceMappings->resolveOperation($mapping)->cast(@PropertyMappingsImplementation)->map(s|$s->_propertyMappingsByPropertyName($p.name->toOne())->map(pm | $pm->reprocessAggregationAwarePropertyMapping())->filter(p|$p->instanceOf(EmbeddedSetImplementation)))->cast(@EmbeddedSetImplementation);
                                            let containers = $embeddedMappings->map(e | ^SetImplementationContainer(id=$e.id, setImplementation=$e));
                                            ^OperationSetImplementation(id='embedded_operation', root=true, class=$c->cast(@Class), parent=$mapping, parameters=$containers, operation=$sourceMappings->at(0)->cast(@OperationSetImplementation).operation);,
                                          | let targetSetIds = $sourceMappings->resolveOperation($mapping)->cast(@PropertyMappingsImplementation)->map(s|$s->_propertyMappingsByPropertyName($p.name->toOne())).targetSetImplementationId;
                                            if(!$targetSetIds->isEmpty(),
                                               | let classMappingsById = $mapping->_classMappingByIdRecursive($targetSetIds)->map(x | $mapping->classMappingById($x.id));
                                                 if($classMappingsById->isEmpty(),
                                                    |$mapping->rootClassMappingByClass($c->cast(@Class))->potentiallyResolveOperation($mapping),
                                                    |$classMappingsById
                                                 );,
                                               | let otherwiseTargetIds = $sourceMappings->filter(s|$s->instanceOf(OtherwiseEmbeddedSetImplementation))->cast(@OtherwiseEmbeddedSetImplementation).otherwisePropertyMapping.targetSetImplementationId;
                                                 let classMappingsById = if ($otherwiseTargetIds->isEmpty(),|[],|findMappingsFromProperty($p, $mapping, $mapping->_classMappingByIdRecursive($otherwiseTargetIds), $state, $extensions));
                                                 if($classMappingsById->isEmpty(),
                                                    |$mapping->rootClassMappingByClass($c->cast(@Class))->potentiallyResolveOperation($mapping),
                                                    |$classMappingsById
                                                 );
                                            );
                                     ),
                                   | if($embeddedMappings->at(0)->instanceOf(InlineEmbeddedSetImplementation),
                                          |$embeddedMappings->at(0)->cast(@InlineEmbeddedSetImplementation)->inlineEmbeddedMapping($mapping),
                                          |
                                     if($state.graphFetchFlow == true && $embeddedMappings->at(0)->instanceOf(OtherwiseEmbeddedSetImplementation),
                                          | let otherwiseTargetIds = $embeddedMappings->at(0)->cast(@OtherwiseEmbeddedSetImplementation).otherwisePropertyMapping.targetSetImplementationId;
                                            let classMappingsById = if ($otherwiseTargetIds->isEmpty(),|[],|findMappingsFromProperty($p, $mapping, $mapping->_classMappingByIdRecursive($otherwiseTargetIds), $state, $extensions));
                                            $classMappingsById;,
                                          | $embeddedMappings
                                       ))
                          );

       // We don't find anything in the old flow (if the source is an operation and the target type is not directly mapped...
       if ($classMapping->isEmpty(),
          |let mappings = $state.routingStrategy->cast(@meta::pure::router::store::metamodel::StoreMappingRoutingStrategy).classMappingsForClass($p->functionReturnType().rawType->toOne()->cast(@Class));
           if ($mappings->isEmpty(),
               |^M2MEmbeddedSetImplementation(id='embdedded_todo', root=true, class=$p->functionReturnType().rawType->toOne()->cast(@Class), parent = $mapping, property=$p->cast(@Property), targetSetImplementationId='todo', sourceSetImplementationId='todo'),
               |$mappings
           );,
          |$classMapping
       );
   );
}

function meta::pure::router::routing::allSuperSetImplementationIds(idToClassMapping:Map>[1], id:String[1]):String[*]
{
    let setImplementation = getClassMappingById($idToClassMapping, $id);
    $id->concatenate(if ($setImplementation->isEmpty() || $setImplementation.superSetImplementationId->isEmpty(),
                         | [],
                         | $idToClassMapping->allSuperSetImplementationIds($setImplementation.superSetImplementationId->toOne())));
}

function meta::pure::router::routing::inlineEmbeddedMapping(i:InlineEmbeddedSetImplementation[1], m:Mapping[1]):EmbeddedSetImplementation[1]
{
    let cm = $m->classMappingById($i.inlineSetImplementationId);
    assertEquals(1, $cm->size(), | 'Found too many or not enough matches ['+$cm.id->makeString(',')+'] for inline implementation Set Id [' + $i.inlineSetImplementationId+']');
    let pmaps = $cm->cast(@InstanceSetImplementation)->toOne()->allPropertyMappings()->map(pm | ^$pm(owner = $i.owner, sourceSetImplementationId = $i.sourceSetImplementationId));
    ^$i(propertyMappings=$pmaps);
}

function meta::pure::router::routing::reprocessAggregationAwarePropertyMapping(pm: PropertyMapping[1]):PropertyMapping[1]
{
   $pm->match([
      agg:AggregationAwarePropertyMapping[1]| $agg.owner->cast(@AggregationAwareSetImplementation).mainSetImplementation->toOne()->_propertyMappingsByPropertyName($agg.property.name->toOne())->filter(p | $p.sourceSetImplementationId == $p.sourceSetImplementationId && $p.targetSetImplementationId == $p.targetSetImplementationId)->toOne(),
      m:PropertyMapping[1] | $m
   ])
}

function meta::pure::router::operations::getMappedLeafClasses(class:Class[1], mapping:Mapping[1], state: RoutingState[1]):Type[*]
{
    let thisMapping = $state.routingStrategy->cast(@meta::pure::router::store::metamodel::StoreMappingRoutingStrategy).classMappingsForClass($class);
    let subTypes = $class.specializations.specific->cast(@Class)->map(c | $c->getMappedLeafClasses_recursive([], $mapping, $state));
    if($subTypes->isEmpty(), | if($thisMapping->isEmpty(), | [], |$class), | $subTypes);
}

function <> meta::pure::router::operations::getMappedLeafClasses_recursive(class:Class[1], leafMostMappedClass:Class[0..1], mapping:Mapping[1], state: RoutingState[1]):Class[*]
{
    let specs = $class.specializations.specific->cast(@Class);
    let foundClassMapping = $state.routingStrategy->cast(@meta::pure::router::store::metamodel::StoreMappingRoutingStrategy).classMappingsForClass($class);
    let mappedClass = if($foundClassMapping->isEmpty(), | $leafMostMappedClass, | $class);
    if($specs->isEmpty(), | $mappedClass, | $specs->map(s | $s->getMappedLeafClasses_recursive($mappedClass, $mapping, $state)));
}

function <> meta::pure::mapping::from(t:T[m], runtime:Runtime[1]):T[m]
{
   $t
}

function <> meta::pure::mapping::from(t:T[m], m:meta::pure::mapping::Mapping[1], runtime:Runtime[1]):T[m]
{
   $t
}

function <> meta::pure::mapping::from(tds:TabularDataSet[1], mapping:Mapping[1], runtime:Runtime[1]):TabularDataSet[1]
{
   $tds;
}

function <> meta::pure::mapping::from(tds:TabularDataSet[1], mapping:Mapping[1], runtime:Runtime[1], executionContext:meta::pure::runtime::ExecutionContext[1]):TabularDataSet[1]
{
   $tds;
}


###Mapping
Mapping meta::pure::mapping::MergedMapping
()




© 2015 - 2025 Weber Informatics LLC | Privacy Policy