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

core_analytics_quality.functionChecks.pure Maven / Gradle / Ivy

There is a newer version: 4.68.0
Show newest version
import meta::analytics::quality::*;
import meta::analytics::quality::model::*;
import meta::pure::functions::collection::*;
import meta::pure::functions::meta::applications::*;
import meta::pure::metamodel::serialization::grammar::*;
import meta::analytics::quality::model::domain::*;

function meta::analytics::quality::model::domain::functionRules():Rule>>[*]
{

   [
     meta::analytics::quality::model::domain::validEqualityComparisons_List_1__Pair_MANY_,
     meta::analytics::quality::model::domain::invalidContainsComparisons_List_1__Pair_MANY_,
     meta::analytics::quality::model::domain::badInstanceOfChecks_List_1__Pair_MANY_,
     meta::analytics::quality::model::domain::invalidMatchUsages_List_1__Pair_MANY_,
     meta::analytics::quality::model::domain::findUnnecessaryComparisonsToTrue_List_1__Pair_MANY_,
     meta::analytics::quality::model::domain::findInvalidCastBugs_List_1__Pair_MANY_,
     meta::analytics::quality::model::domain::findUnusedPrivateProtectedFunctionBugs_List_1__Pair_MANY_,
     meta::analytics::quality::model::domain::findUnnecessaryIfBugs_List_1__Pair_MANY_,
     meta::analytics::quality::model::domain::findUnnecessaryIfBugs2_List_1__Pair_MANY_,
     // apps::pure::quality::findUnnecessaryToOneBugs_List_1__Pair_MANY_, enable when compile mode is fixed
     meta::analytics::quality::model::domain::findUnnecessaryCasts_List_1__Pair_MANY_,
     meta::analytics::quality::model::domain::findUnnecessaryLetFunctionsCheck_List_1__Pair_MANY_,
     meta::analytics::quality::model::domain::findPotentiallyExpensiveAssertions_List_1__Pair_MANY_
    ]->map(rule|createRule($rule)->cast(@Rule>>));


}

function meta::analytics::quality::model::domain::getUsages(srcFns:List>[1],fns:Function[*]):FunctionExpression[*]
{
   $srcFns.values->map(f|
     $f->evaluateAndDeactivate()->meta::analytics::quality::model::domain::applicationsMatchLocal({vs : FunctionExpression[1] | $vs.func->in($fns)});
   );
}

function meta::analytics::quality::model::domain::applicationsMatchLocal(expr : FunctionDefinition[1], matchFunc:Function<{FunctionExpression[1]->Boolean[1]}>[1]):FunctionExpression[*]
{
   let fes = $expr->evaluateAndDeactivate().expressionSequence->meta::analytics::quality::model::domain::getLocalFunctionExpressions();
   $fes->filter(fe|$matchFunc->eval($fe));
}

function <> meta::analytics::quality::model::domain::getLocalFunctionExpressions(vs : ValueSpecification[*]) : FunctionExpression[*]
{
   $vs->map(x|
         $x->match([
            sfe:SimpleFunctionExpression[1]|
               $sfe->concatenate($sfe.parametersValues->meta::analytics::quality::model::domain::getLocalFunctionExpressions()),
            v : VariableExpression[1]|[],
            iv : InstanceValue[1]|$iv.values->map(v|
                  $v->match([
                     subVS:ValueSpecification[1]|$subVS->meta::analytics::quality::model::domain::getLocalFunctionExpressions(),
                     lambda:LambdaFunction[1]|$lambda.expressionSequence->meta::analytics::quality::model::domain::getLocalFunctionExpressions(),
                     a:Any[*]| []
                  ])
                  )
         ]));
}

function  {rule.rule = 'Invalid Contains',
           rule.description ='Check for contains / containsAll / containsAny checks that will always result in false due to type mismatches (e.g. [\'abc\']->contains(123))',
           rule.severity = 'High',
           rule.category = 'Correctness'}
meta::analytics::quality::model::domain::invalidContainsComparisons(fns:List>[1]) : Pair[*]
{


   meta::analytics::quality::model::domain::findInvalidContainsComparisons($fns)
      ->map(ie|
         let t1 = $ie->meta::analytics::quality::model::domain::resolveParameterType(0);
         let t2 = $ie->meta::analytics::quality::model::domain::resolveParameterType(1);
         let message = ('Possible invalid ' + $ie.functionName->makeString() + ' check (type mismatch, ' + $t1->elementToPath() + ' vs ' + $t2->elementToPath() + ')');
         let result = ^CheckResult(isValid=false,message=$message);
         pair($ie,$result);
         );
}


function  {rule.rule = 'Invalid Equality',
           rule.description ='Check for equality checks that will always result in false due to type mismatches (e.g. \'abc\' == 123)',
           rule.severity = 'High',
           rule.category = 'Correctness'}
meta::analytics::quality::model::domain::validEqualityComparisons(fns:List>[1]) : Pair[*]
{

   let applications = [equal_Any_MANY__Any_MANY__Boolean_1_,
                       is_Any_1__Any_1__Boolean_1_,
                       eq_Any_1__Any_1__Boolean_1_];

   let usages = $fns->meta::analytics::quality::model::domain::getUsages($applications);

   $usages->evaluateAndDeactivate()
        ->map(ie|
            let t1 = $ie->meta::analytics::quality::model::domain::resolveParameterType(0);
            let t2 = $ie->meta::analytics::quality::model::domain::resolveParameterType(1);

            let valid = meta::analytics::quality::model::domain::equalityCompatibleTypes($t1, $t2);

            let message = if($valid,
                             |'Valid ' + $ie.functionName->toOne() + ' check (' + $t1->elementToPath() + ' vs ' + $t2->elementToPath() + ')',
                             | 'Function (' + $ie.functionName->toOne() + ') does not match required standards: Possible invalid ' + $ie.functionName->toOne() + ' check (type mismatch, ' + $t1->elementToPath() + ' vs ' + $t2->elementToPath() + ')');
            let result = ^CheckResult(isValid=$valid, message=$message);
            pair($ie, $result);
         );

}


function <> meta::analytics::quality::model::domain::equalityCompatibleTypes(t1 : Type[1], t2 : Type[1]) : Boolean[1]
{
   castCompatibleTypes($t1, $t2) || ($t1->subTypeOf(Number) && $t2->subTypeOf(Number));
}



// Nested filters
function meta::analytics::quality::model::domain::findAllNestedFilterExpressions():FunctionExpression[*]
{
	   meta::pure::functions::collection::filter_T_MANY__Function_1__T_MANY_.applications
	       ->evaluateAndDeactivate()
	       ->filter(fe | let arg1 = $fe.parametersValues->evaluateAndDeactivate()->at(0);
	                     $arg1->instanceOf(FunctionExpression) &&
	                       is($arg1->cast(@FunctionExpression).func,
	                          meta::pure::functions::collection::filter_T_MANY__Function_1__T_MANY_);)
}


function  {rule.rule = 'Invalid Instance Of',
           rule.description ='Check for Match functions that will always fail (e.g. \'abc\'->match([ i: Integer[1]|$i))',
           rule.severity = 'High',
           rule.category = 'Correctness'}
meta::analytics::quality::model::domain::badInstanceOfChecks(fns:List>[1]) : Pair[*]
{
   meta::analytics::quality::model::domain::findBadInstanceOfChecks($fns)
      ->map(ie|
         let t1 = $ie->meta::analytics::quality::model::domain::resolveParameterType(0);
         let t2 = $ie.parametersValues->at(1)->meta::analytics::quality::model::domain::resolveValueType();
         let message = ('Invalid instanceOf check, always ' + meta::analytics::quality::model::domain::castNotNecessary($t1, $t2)->makeString() + ' (' + $t1->elementToPath() + ' vs ' + $t2->elementToPath() + ')');
         let result = ^CheckResult(isValid=false,message=$message);
         pair($ie,$result);
         );
}

function  {rule.rule = 'Invalid Match',
           rule.description ='Check for invalid instanceOf checks (e.g. \'hello\'->instanceOf(Float) or \'hello\'->instanceOf(String))',
           rule.severity = 'High',
           rule.category = 'Correctness'}
meta::analytics::quality::model::domain::invalidMatchUsages(fns:List>[1]) : Pair[*]
{
   meta::analytics::quality::model::domain::findInvalidMatchUsage($fns)
      ->map(ie|
         let t1 = $ie->meta::analytics::quality::model::domain::resolveParameterType(0);
         let message = ('Possible invalid Match usage, no type matches for ' + $t1->elementToPath() + ')');
         let result = ^CheckResult(isValid=false,message=$message);
         pair($ie,$result);
         );
}

function  {rule.rule = 'Bad Size',
           rule.description ='Check that code uses $collection->isEmpty() / isNotEmpty() instead of $collection->size() == 0 / size() > 0 / size != 0',
           rule.severity = 'Low',
           rule.category = 'Quality'}
meta::analytics::quality::model::domain::badSizeComparisons(fns:List>[1]) : Pair[*]
{
   meta::analytics::quality::model::domain::findBadSizeComparisons($fns)
      ->map(ie|
            let message = ('Use isEmpty() / isNotEmpty() check, rather than size() ' + $ie.functionName->makeString() + ' 0');
            let result = ^CheckResult(isValid=false,message=$message);
            pair($ie,$result);
         );
}


function  {rule.rule = 'Unnecessary Comparison To True',
           rule.description = 'Check for unnecessary comparison to true (e.g. if(myBooleanFunc() == true, .., ...))',
           rule.severity = 'Medium',
           rule.category = 'Quality'}
meta::analytics::quality::model::domain::findUnnecessaryComparisonsToTrue(fns:List>[1]) : Pair[*]
{
   meta::analytics::quality::model::domain::unnecessaryTrueComparisons($fns)
      ->map(ie|
         let  message = ('Avoid unnecessary comparison to true for boolean values');
         let result = ^CheckResult(isValid=false,message=$message);
         pair($ie,$result);
         );
}

function  {rule.rule = 'Invalid Cast',
           rule.description = 'Check for invalid cast operations (e.g. \'hello\'->cast(@Float))',
           rule.severity = 'High',
           rule.category = 'Correctness'}
meta::analytics::quality::model::domain::findInvalidCastBugs(fns:List>[1]) : Pair[*]
{
   meta::analytics::quality::model::domain::findInvalidCasts($fns)
      ->map(ie|
         let t1 = $ie->meta::analytics::quality::model::domain::resolveParameterType(0);
         let t2 = $ie.parametersValues->at(1)->meta::analytics::quality::model::domain::resolveValueType();
         let message = ('Possible invalid cast (from ' + $t1->elementToPath() + ' to ' + $t2->elementToPath() + ')');
         let result = ^CheckResult(isValid=false,message=$message);
         pair($ie,$result);
         );
}

function  {rule.rule = 'Unused Private Protected',
           rule.description =  'Check for unused private or protected functions (they could be removed)',
           rule.severity = 'Low',
           rule.category = 'Quality'}
meta::analytics::quality::model::domain::findUnusedPrivateProtectedFunctionBugs(fns:List>[1]) : Pair,CheckResult>[*]
{
   $fns.values->map(f|
      if(meta::analytics::quality::model::domain::isPrivateProtectedFunctionUnused($f)
      ,| let message = ('Unused private/protected function ' + $f.functionName->toOne());
         let result = ^CheckResult(isValid=false,message=$message);
         [pair($f,$result)];
      ,|[])
    );
}

function meta::analytics::quality::model::domain::isPrivateProtectedFunctionUnused(f:Function[1]) :Boolean[1]
{
     $f->isFunctionUncalled() &&
     ($f->instanceOf(AnnotatedElement) && ($f->cast(@AnnotatedElement)->hasStereotype('private', meta::pure::profiles::access)  || $f->cast(@AnnotatedElement)->hasStereotype('protected', meta::pure::profiles::access)));

}

function {doc.doc = 'Find if defined functions is not called anywhere.'}
meta::analytics::quality::model::domain::isFunctionUncalled(f:Function[1]):Boolean[1]
{
    !$f->isTestElement() && !$f->isFunctionReferenced()
}

function <> meta::analytics::quality::model::domain::isFunctionReferenced(func:Function[1]):Boolean[1]
{
    !$func.applications->evaluateAndDeactivate()->isEmpty() || !$func.referenceUsages->isEmpty()
}

function  {rule.rule = 'Unnecessary If Condition',
           rule.description =  'Check for unnecessary if statements, with constant condition (e.g. if(true, |abc, |def)))',
           rule.severity = 'Medium',
           rule.category = 'Quality'}
meta::analytics::quality::model::domain::findUnnecessaryIfBugs(fns:List>[1]) : Pair[*]
{
   meta::analytics::quality::model::domain::findUnnecessaryIfs($fns)
      ->map(f| let message = ('If statement with constant expression');
         let result = ^CheckResult(isValid=false,message=$message);
         pair($f,$result);
         )

}

function  {rule.rule = 'Unnecessary If Return',
           rule.description = 'Check for unnecessary if statements, with boolean result (e.g. if(2 > 1, |true, |false)))',
           rule.severity = 'Medium',
           rule.category = 'Quality'}
meta::analytics::quality::model::domain::findUnnecessaryIfBugs2(fns:List>[1]) : Pair[*]
{
   meta::analytics::quality::model::domain::findUnnecessaryIfs2($fns)
      ->map(f|let message = ('If statement with true/false returns');
         let result = ^CheckResult(isValid=false,message=$message);
         pair($f,$result);
         )

}

function  {rule.rule = 'Unnecessary To One',
           rule.description =  'Check for unnecessary toOne() operations (e.g. \'hello\'->toOne())',
           rule.severity = 'Medium',
           rule.category = 'Quality'}
meta::analytics::quality::model::domain::findUnnecessaryToOneBugs(fns:List>[1]) : Pair[*]
{
   meta::analytics::quality::model::domain::findUnnecessaryToOneFEs($fns)
      ->map(ie|
            let message = ('Avoid unnecessary toOne() on item that is already [1..1]');
            let result = ^CheckResult(isValid=false,message=$message);
            pair($ie,$result);
         );
}

function  {rule.rule = 'Unnecessary Map Property',
           rule.description = 'Check for unnecessary map to property usage (e.g. [pair(1,2), pair(3,4)]->map(p|$p.first))',
           rule.severity = 'Medium',
           rule.category = 'Quality'}
meta::analytics::quality::model::domain::findUnnecessaryMapToPropertyBugs(fns:List>[1]) : Pair[*]
{
   meta::analytics::quality::model::domain::findUnnecessaryMapToProperty($fns)
      ->map(ie|let message = ('Use .property syntax directly against the collection rather than mapping to extract single property');
         let result = ^CheckResult(isValid=false,message=$message);
         pair($ie,$result);
           );
}




function meta::analytics::quality::model::domain::findBadSizeComparisons(fns:List>[1]) : FunctionExpression[*]
{
   let applications = [equal_Any_MANY__Any_MANY__Boolean_1_,greaterThan_Number_1__Number_1__Boolean_1_];


   let usages = $fns->getUsages($applications);


   let invalidApplications =  $usages->evaluateAndDeactivate()->filter({i|
      let p1 = $i.parametersValues->at(0);
      let p2 = $i.parametersValues->at(1);

      $p1->instanceOf(SimpleFunctionExpression) && $p1->cast(@SimpleFunctionExpression).func == size_Any_MANY__Integer_1_
         && $p2->instanceOf(InstanceValue) && $p2->cast(@InstanceValue).values == [0];
      });
}

function meta::analytics::quality::model::domain::findInvalidContainsComparisons(fns:List>[1]) : FunctionExpression[*]
{
   let applications = [contains_Any_MANY__Any_1__Boolean_1_,
                       containsAny_Any_MANY__Any_MANY__Boolean_1_,
                       containsAll_Any_MANY__Any_MANY__Boolean_1_];

   let usages = $fns->getUsages($applications);

   $usages->evaluateAndDeactivate()
          ->filter({i|
                   let t1 = $i->resolveParameterType(0);
                   let t2 = $i.parametersValues->at(1)->resolveValueType();

                   let valid = equalityCompatibleTypes($t1, $t2);

                   !$valid;
            });
}

function meta::analytics::quality::model::domain::findInvalidMatchUsage(fns:List>[1]) : FunctionExpression[*]
{

   let applications = [match_Any_MANY__Function_$1_MANY$__T_m_];

   let usages = $fns->getUsages($applications);

   let invalidApplications = $usages->evaluateAndDeactivate()->filter({i|

      let t1 = $i->resolveParameterType(0);
      let lambdasSpec = $i.parametersValues->at(1);

      //TODO: Handle the case where the match handler values are passed in by a variable
      //TODO: Handle the case where the match handlers are functions (not lambdas)

      if($lambdasSpec->instanceOf(InstanceValue) == false
         || !$lambdasSpec->cast(@InstanceValue).values->forAll(v|$v->instanceOf(LambdaFunction)),
         | true,
         |
            let lambdas = $lambdasSpec->cast(@InstanceValue).values->cast(@LambdaFunction);

            let matchLambdaParamTypes = $lambdas->map(l|$l->functionType().parameters->toOne());

            let valid = $matchLambdaParamTypes->fold({pt,r|if ($r, |$r, {|
               $t1->castCompatibleTypes($pt.genericType.rawType->toOne());
            })}, false);

            !$valid;
         );
      });

      $invalidApplications;
}

function meta::analytics::quality::model::domain::findInvalidCasts(fns:List>[1]) : FunctionExpression[*]
{
  let applications = [cast_Any_m__T_1__T_m_];

  let usages = $fns->getUsages($applications);

  let invalidApplications = $usages->evaluateAndDeactivate()->filter({i|
      let p1 = $i->resolveParameterType(0);
      let p2 = $i.parametersValues->at(1)->resolveValueType();

      !$p1->castCompatibleTypes($p2);
      });
}

function meta::analytics::quality::model::domain::findBadInstanceOfChecks(fns:List>[1]) : FunctionExpression[*]
{

   let applications = [instanceOf_Any_1__Type_1__Boolean_1_];

   let usages = $fns->getUsages($applications);

   let invalidApplications = $usages->evaluateAndDeactivate()->filter({i|
      let p1 = $i.parametersValues->at(0);
      let p2 = $i.parametersValues->at(1);

      let p1t= $i->resolveParameterType(0);

      if(!$p2->instanceOf(InstanceValue),
         | false,
         {|
            let targetType = $p2->cast(@InstanceValue).values->toOne()->cast(@Type);

            (!$p1t->castCompatibleTypes($targetType));
         });
      });
}

function meta::analytics::quality::model::domain::resolveParameterType(fe : FunctionExpression[1], index : Integer[1]) : Type[1]
{
   let fParams = $fe.func->functionType().parameters;
   assert($fParams->size() > $index);

   let pGT = $fParams->at($index).genericType;

   let gt = if($pGT.rawType->isNotEmpty() && $pGT.rawType != Any,
                  | $pGT,
                  | ^GenericType(rawType = $fe.parametersValues->at($index)->resolveValueType())
                  );

   if($gt->isEmpty() || $gt.rawType->isEmpty(),
      | Any,
      | $gt.rawType->toOne()
      );
}

function meta::analytics::quality::model::domain::resolveValueType(vs : ValueSpecification[1]) : Type[1]
{
   let t = $vs->match([
         v : ValueSpecification[1]| $v.genericType.rawType
         ]);

   if($t->isEmpty(),
      | Any,
      | $t->toOne()
      );
}


function meta::analytics::quality::model::domain::findUnnecessaryToOneFEs(fns:List>[1]) : FunctionExpression[*]
{
   let applications = [toOne_T_MANY__T_1_];

   let usages = $fns->getUsages($applications);


   $usages->evaluateAndDeactivate()
      ->filter({i|
      $i.parametersValues->toOne().multiplicity == PureOne;
      });
}

function <> meta::analytics::quality::model::domain::castCompatibleTypes(from: Type[1], to: Type[1]) : Boolean[1]
{
   if($from->in([$to, Any]) || $from == Nil || $to == Any,
    | true,
    |
      if ($from.generalizations->size() == 0 && $to.specializations->size() == 0,
          | false,
          | $to->_subTypeOf($from) || $from->_subTypeOf($to) || $from->getLeafTypes()->remove($from)->exists(x|$x->_subTypeOf($to))
          )
    )
}

function meta::analytics::quality::model::domain::castNotNecessary(from: Type[1], to: Type[1]) : Boolean[1]
{
   if($to->in([$from, Any]),
    | true,
    |
      $from->_subTypeOf($to)
    );
}


function meta::analytics::quality::model::domain::findUnnecessaryIfs(fns:List>[1]):FunctionExpression[*]
{

   let applications = [if_Boolean_1__Function_1__Function_1__T_m_];

   let usages = $fns->getUsages($applications);

   $usages->evaluateAndDeactivate()->filter(a |
      let condition = $a.parametersValues->at(0);
      $condition->match([
         i:InstanceValue[1] | $i.values->at(0)->instanceOf(Boolean),
         a:Any[1] | false
      ]);
   );
}

function meta::analytics::quality::model::domain::findUnnecessaryIfs2(fns:List>[1]):FunctionExpression[*]
{
   let applications = [if_Boolean_1__Function_1__Function_1__T_m_];

   let usages = $fns->getUsages($applications);

   $usages->evaluateAndDeactivate()->filter(a |
      let statements = $a.parametersValues->tail()->cast(@InstanceValue).values->cast(@LambdaFunction).expressionSequence;
      let truth = $statements->at(0);
      let falsy = $statements->at(1);

      // If the return values of the if conditions are constant booleans and
      // 1) They are the same, then the If is unnecessary
      // 2) They are different to each other, we should just return the condition or the !condition

      $truth->match([
         i:InstanceValue[1] | $i.values == true || $i.values == false,
         a:Any[*] | false
      ])
      &&
      $falsy->match([
         i:InstanceValue[1] | $i.values == true || $i.values == false,
         a:Any[*] | false
      ]);
   );
}


function meta::analytics::quality::model::domain::unnecessaryTrueComparisons(fns:List>[1]):FunctionExpression[*]
{
   let applications = [equal_Any_MANY__Any_MANY__Boolean_1_];

   let usages = $fns->getUsages($applications);

   $usages->evaluateAndDeactivate()->filter(a |
      let p1 = $a.parametersValues->at(0);
      let p2 = $a.parametersValues->at(1);

      let isBooleanFunction = $p1->match([
         f:SimpleFunctionExpression[1] | $f.genericType.rawType == Boolean,//.typeArguments.rawType->cast(@FunctionType).returnType.rawType == Boolean,
         a:Any[1] | false
      ]);

      let isTrueValue =  $p2->match([
         i:InstanceValue[1] | $i.values->size() == 1 && $i.values->at(0) == true,
         a:Any[1] | false
      ]);

      $isBooleanFunction && $isTrueValue;
   );
}


function meta::analytics::quality::model::domain::findUnnecessaryMapToProperty(fns:List>[1]):FunctionExpression[*]
{
   let applications = [map_T_m__Function_1__V_m_];

   let usages = $fns->getUsages($applications);

   $usages->evaluateAndDeactivate()->filter(m|
          let param0 = $m.parametersValues->at(0);
          let param1 = $m.parametersValues->at(1);
          if (!$param0.genericType.typeArguments->isEmpty() || $param1->instanceOf(InstanceValue) == false,
             | false,
             |
                let param1Value = $param1->cast(@InstanceValue).values->toOne();
                if ($param1Value->instanceOf(LambdaFunction) == false,
                   | false,
                   | let lambda = $param1Value->cast(@LambdaFunction);
                      let firstExpr = $lambda.expressionSequence->first();
                      if ($lambda.expressionSequence->size() != 1 || !$firstExpr->toOne()->instanceOf(SimpleFunctionExpression),
                         | false,
                         | let f = $firstExpr->cast(@SimpleFunctionExpression).func->toOne();
                            $f->instanceOf(Property) && !$f->instanceOf(QualifiedProperty); //Do qualified properties with no args work?
                         );
                   );
              );
          );
}



function  {rule.rule = 'Unnecessary Cast',
           rule.description = 'Check for unnecessary casts (e.g. \'hello\'->cast(@String))',
           rule.severity = 'Low',
           rule.category = 'Quality'}
meta::analytics::quality::model::domain::findUnnecessaryCasts(fns:List>[1]) : Pair[*]
{


   let applications = [cast_Any_m__T_1__T_m_];

   let usages = $fns->meta::analytics::quality::model::domain::getUsages($applications);

   $usages->evaluateAndDeactivate()->map({i|
      let p1t = $i->meta::analytics::quality::model::domain::resolveParameterType(0);
      let p2t = $i.parametersValues->at(1)->meta::analytics::quality::model::domain::resolveValueType();

      if(!$p1t->meta::analytics::quality::model::domain::castNotNecessary($p2t),
         | [],
         |
             let message = ('Possible unnecesary cast (from ' + $p1t->makeString() + ' to ' + $p2t->makeString() + ')');
             let result = ^CheckResult(isValid=false,message=$message);
             pair($i,$result);
         );
      });
}


function  {rule.rule = 'Unnecessary let',
           rule.description ='Check for functions where last statement is a let',
           rule.severity = 'Medium',
           rule.category = 'Quality'}
meta::analytics::quality::model::domain::findUnnecessaryLetFunctionsCheck(fns:List>[1]) : Pair,CheckResult>[*]
{
 $fns.values->map(f|
   let last = $f->evaluateAndDeactivate().expressionSequence->evaluateAndDeactivate()->last()->toOne();
   let doesEndInLet = $last->instanceOf(FunctionExpression) && $last->cast(@FunctionExpression).func == letFunction_String_1__T_m__T_m_;
   let message =  'function ' + $f.functionName->makeString() + ' ends with an unnecessary let statement';
   let result = ^CheckResult(isValid = !$doesEndInLet, message = $message);
   pair($f,$result);
  );
}


function {doc.doc = 'Find all concretely defined functions which are not called anywhere.'}
meta::analytics::quality::model::domain::findAllUncalledFunctions():ConcreteFunctionDefinition[*]
{
    ConcreteFunctionDefinition.all()->filter(f | !$f->isTestElement() && !$f->isFunctionReferenced())
}

function  <>
          {rule.rule = 'Assert with dynamic message',
           rule.description ='Avoid using dynamic messages as they can be "expensive", even if the assertion passes',
           rule.severity = 'Medium',
           rule.category = 'Quality'}
meta::analytics::quality::model::domain::findPotentiallyExpensiveAssertions(fns:List>[1]) : Pair[*]
{
   let applications = [assert_Boolean_1__String_1__Boolean_1_];

   let usages = $fns->meta::analytics::quality::model::domain::getUsages($applications);

   let expensiveUsages = $usages->evaluateAndDeactivate()->filter(m|
          let param1 = $m.parametersValues->at(1);
          !$param1->instanceOf(InstanceValue);
          );

    $expensiveUsages->map(f|
      let message =  'Assertion message is dynamically built';
      let result = ^CheckResult(isValid = false, message = $message);
      pair($f,$result);
     );
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy