com.nawforce.apexlink.cst.Expressions.scala Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of apex-ls_2.13 Show documentation
Show all versions of apex-ls_2.13 Show documentation
Salesforce Apex static analysis toolkit
The newest version!
/*
Copyright (c) 2019 Kevin Jones, All rights reserved.
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions
are met:
1. Redistributions of source code must retain the above copyright
notice, this list of conditions and the following disclaimer.
2. Redistributions in binary form must reproduce the above copyright
notice, this list of conditions and the following disclaimer in the
documentation and/or other materials provided with the distribution.
3. The name of the author may not be used to endorse or promote products
derived from this software without specific prior written permission.
*/
package com.nawforce.apexlink.cst
import com.nawforce.apexlink.cst.Expression.mergeableSObjects
import com.nawforce.apexlink.diagnostics.IssueOps
import com.nawforce.apexlink.finding.TypeResolver
import com.nawforce.apexlink.names.TypeNames
import com.nawforce.apexlink.names.TypeNames._
import com.nawforce.apexlink.org.{OPM, Referenceable}
import com.nawforce.apexlink.types.apex.{ApexClassDeclaration, ApexConstructorLike}
import com.nawforce.apexlink.types.core.{FieldDeclaration, MethodDeclaration, TypeDeclaration}
import com.nawforce.apexlink.types.other.{AnyDeclaration, RecordSetDeclaration}
import com.nawforce.apexlink.types.platform.{PlatformTypeDeclaration, PlatformTypes}
import com.nawforce.apexlink.types.synthetic.CustomConstructorDeclaration
import com.nawforce.pkgforce.diagnostics.{ERROR_CATEGORY, Issue, WARNING_CATEGORY}
import com.nawforce.pkgforce.names.{EncodedName, Name, Names, TypeName}
import com.nawforce.pkgforce.path.{Locatable, Location, PathLocation}
import com.nawforce.runtime.parsers.CodeParser
import io.github.apexdevtools.apexparser.ApexParser._
import scala.collection.immutable.ArraySeq
/** Context used during expression verification to indicate focus & return state.
*
* Declaration provides the current context TypeDeclaration. isStatic=None is used as a marker that we have yet to
* enter an explicit static/instance context as you find on the outermost expression used in an instance method. In
* this state declaration == this & both static/instance resolution is allowed. If isStatic is set the specific
* expression should become restricted to either static or instance resolution.
*
* @param isStatic static or instance or either context
* @param declaration input/return type declaration, for return None is used to mean unknown/indeterminable
* @param locatable position of code that generated this context to support introspection
*/
case class ExprContext(
isStatic: Option[Boolean],
declaration: Option[TypeDeclaration],
locatable: Option[Locatable] = None
) {
def isVoid = false
def isDefined: Boolean =
declaration.nonEmpty && !declaration.exists(_.isInstanceOf[AnyDeclaration])
def typeDeclaration: TypeDeclaration = declaration.get
def typeName: TypeName = declaration.get.typeName
}
/** ExprContext of type 'void'
*
* void is not commonly used in expressions, essentially just as method return type so we use simple
* null-object style implementation for it.
*
* @param locatable position of code that generated this context to support introspection
*/
class VoidExprContext(locatable: Option[Locatable]) extends ExprContext(None, None, locatable) {
override def isVoid = true
}
object VoidExprContext {
def apply(locatable: Any): ExprContext = {
locatable match {
case l: Locatable => new VoidExprContext(Some(l))
case _ => new VoidExprContext(None)
}
}
}
object ExprContext {
val empty: ExprContext = new ExprContext(None, None)
def apply(isStatic: Option[Boolean], declaration: TypeDeclaration): ExprContext = {
new ExprContext(isStatic, Some(declaration))
}
/* We allow flex here on the locatable type as it a cross cutting concern of many sorts of things */
def apply(
isStatic: Option[Boolean],
declaration: Option[TypeDeclaration],
locatable: Any
): ExprContext = {
locatable match {
case l: Locatable => new ExprContext(isStatic, declaration, Some(l))
case _ => new ExprContext(isStatic, declaration, None)
}
}
}
/** base for any type of expression, provides helpers for type validation */
sealed abstract class Expression extends CST {
def verify(input: ExprContext, context: ExpressionVerifyContext): ExprContext
def verify(context: BlockVerifyContext): ExprContext = {
val staticContext = if (context.isStatic) Some(true) else None
verify(ExprContext(staticContext, context.thisType), new ExpressionVerifyContext(context))
}
/** Verify an expression result type is an exception
*
* @param context verify context to use
* @param prefix for the log issue
*/
def verifyIsExceptionInstance(
context: BlockVerifyContext,
prefix: String
): (Boolean, ExprContext) = {
val verifyResult = verify(context)
if (
verifyResult.isDefined && (verifyResult.isStatic
.contains(true) || !verifyResult.typeName.name.endsWith(Names.Exception))
) {
val resultQualifier = if (verifyResult.isStatic.contains(true)) "type" else "instance"
context.log(
Issue(
ERROR_CATEGORY,
location,
s"$prefix expression should return an Exception instance, not a '${verifyResult.typeName}' $resultQualifier"
)
)
(false, verifyResult)
} else {
(true, verifyResult)
}
}
/** Verify an expression result type matches a specific type logging an issue if not
*
* @param context verify context to use
* @param typeNames set of permitted types
* @param isStatic check for static or instance value
* @param prefix for the log issue
*/
def verifyIs(
context: BlockVerifyContext,
typeNames: Set[TypeName],
isStatic: Boolean,
prefix: String
): (Boolean, ExprContext) = {
val verifyResult = verify(context)
if (
verifyResult.isDefined && (!verifyResult.isStatic.contains(isStatic) ||
!typeNames.contains(verifyResult.typeName))
) {
val resultQualifier = if (verifyResult.isStatic.contains(true)) "type" else "instance"
val qualifier = if (isStatic) "type" else "instance"
val requiredTypes = if (typeNames.size == 1) {
s"a '${typeNames.head}' $qualifier"
} else {
typeNames.map(n => s"'$n'").mkString("one of ", " or ", s" ${qualifier}s")
}
context.log(
Issue(
ERROR_CATEGORY,
location,
s"$prefix expression should return $requiredTypes, not a '${verifyResult.typeName}' $resultQualifier"
)
)
(false, verifyResult)
} else {
(true, verifyResult)
}
}
/** Verify an expression result type is an SObject or SObject List/RecordSet
*
* @param context verify context to use
* @param prefix for the log issue
*/
def verifyIsSObjectOrSObjectList(
context: BlockVerifyContext,
prefix: String
): (Boolean, ExprContext) = {
val verifyResult = verify(context)
if (
verifyResult.isDefined &&
(verifyResult.isStatic
.contains(true) || !isSObjectOrSObjectList(verifyResult.typeName, context))
) {
val resultQualifier = if (verifyResult.isStatic.contains(true)) "type" else "instance"
context.log(
Issue(
ERROR_CATEGORY,
location,
s"$prefix expression should return an SObject or list of SObjects, not a '${verifyResult.typeName}' $resultQualifier"
)
)
(false, verifyResult)
} else {
(true, verifyResult)
}
}
private def isSObjectOrSObjectList(typeName: TypeName, context: VerifyContext): Boolean = {
if (
typeName == TypeNames.SObject || typeName == TypeNames.listOf(
SObject
) || typeName == TypeNames.recordSetOf(SObject)
)
return true
val sObjectTypeName =
if (typeName.isList || typeName.isRecordSet) typeName.params.head else typeName
context.getTypeFor(sObjectTypeName, context.thisType).toOption.forall(_.isSObject)
}
/** Verify an expression result type is an SObject suitable for merging. Only leads, contacts,
* cases, and accounts can be merged. Returns typeName on success.
*
* @param context verify context to use
* @param prefix for the log issue
*/
def verifyIsMergeableSObject(context: BlockVerifyContext, prefix: String): Option[TypeName] = {
val verifyResult = verify(context)
if (
verifyResult.isDefined &&
(verifyResult.isStatic.contains(true) ||
!verifyResult.typeName.outer.contains(TypeName.Schema) ||
!mergeableSObjects.contains(verifyResult.typeName.name))
) {
val resultQualifier = if (verifyResult.isStatic.contains(true)) "type" else "instance"
val mergeableSObjectNames = mergeableSObjects.map(_.toString).mkString(" or ")
context.log(
Issue(
ERROR_CATEGORY,
location,
s"$prefix expression should return a $mergeableSObjectNames SObject, not a '${verifyResult.typeName}' $resultQualifier"
)
)
None
} else {
verifyResult.declaration.map(_.typeName)
}
}
/** Verify an expression result type is an SObject or SObject List/RecordSet suitable for merging. Only
* leads, contacts, cases, and accounts can be merged. Returns typeName on success.
*
* @param context verify context to use
* @param prefix for the log issue
*/
def verifyIsMergeableSObjectOrSObjectList(
context: BlockVerifyContext,
prefix: String
): Option[TypeName] = {
val verifyResult = verify(context)
val mergeableTypeName = getMergeableTypeNameFromSObjectOrSObjectList(verifyResult.typeName)
if (
verifyResult.isDefined && verifyResult.isStatic.contains(true) || mergeableTypeName.isEmpty
) {
val resultQualifier = if (verifyResult.isStatic.contains(true)) "type" else "instance"
val mergeableSObjectNames = mergeableSObjects.map(_.toString).mkString(" or ")
context.log(
Issue(
ERROR_CATEGORY,
location,
s"$prefix expression should return a $mergeableSObjectNames SObject or list of SObjects, not a '${verifyResult.typeName}' $resultQualifier"
)
)
None
} else {
mergeableTypeName
}
}
private def getMergeableTypeNameFromSObjectOrSObjectList(typeName: TypeName): Option[TypeName] = {
val sObjectTypeName = {
if (typeName.isList || typeName.isRecordSet) typeName.params.head else typeName
}
if (
sObjectTypeName.outer.contains(
TypeNames.Schema
) && sObjectTypeName.params.isEmpty && mergeableSObjects
.contains(sObjectTypeName.name)
) Some(sObjectTypeName)
else None
}
}
final case class EmptyExpr() extends Expression {
override def verify(input: ExprContext, context: ExpressionVerifyContext): ExprContext = {
ExprContext.empty
}
}
object DotExpression {
def findField(
name: Name,
td: TypeDeclaration,
module: OPM.Module,
staticContext: Option[Boolean]
): Option[FieldDeclaration] = {
val encodedName = EncodedName(name)
val namespaceName = encodedName.defaultNamespace(module.namespace)
td.findField(namespaceName.fullName, staticContext)
.orElse({
if (encodedName != namespaceName) td.findField(encodedName.fullName, staticContext)
else None
})
}
}
final case class DotExpressionWithId(expression: Expression, safeNavigation: Boolean, target: Id)
extends Expression {
override def verify(input: ExprContext, context: ExpressionVerifyContext): ExprContext = {
assert(input.declaration.nonEmpty)
// When we have a leading IdPrimary there are a couple of special cases to handle, we could with a better
// understanding of how these are handled, likely it via some parser hack
getInterceptPrimary
.flatMap(primary => {
val td = input.declaration.get
val localVar = context.isVar(primary.id.name)
val field = DotExpression
.findField(primary.id.name, td, context.module, input.isStatic)
if (localVar.isEmpty && field.isEmpty) {
interceptMissingId(input, context, primary.id)
} else if (localVar.isEmpty && field.nonEmpty) {
// Handle cases of shadowing
interceptFoundField(input, context, primary.id, field.get)
} else {
None
}
})
.map(result => context.saveResult(this)(result))
.getOrElse(verifyInternal(input, context))
}
private def getInterceptPrimary: Option[IdPrimary] = {
expression match {
case PrimaryExpression(primary: IdPrimary) if !safeNavigation => Some(primary)
case _ => None
}
}
private def interceptMissingId(
input: ExprContext,
context: ExpressionVerifyContext,
id: Id
): Option[ExprContext] = {
// It might be a namespace
if (isNamespace(id.name, input.declaration.get)) {
val typeName = TypeName(target.name, Nil, Some(TypeName(id.name))).intern
val td = context.getTypeAndAddDependency(typeName, context.thisType).toOption
td.map(td =>
context.saveResult(this) {
ExprContext(isStatic = Some(true), Some(td), td)
}
)
} else {
TypeResolver(TypeName(id.name), context.module).toOption match {
// It might be a static reference to an outer class that failed normal analysis due to class name shadowing
// This occurs where say A has an inner of B and in C with an inner of A we reference 'A.B' which is valid.
case Some(td: ApexClassDeclaration) => verifyShadowedStatic(td, context)
case _ => None
}
}
}
private def interceptFoundField(
input: ExprContext,
context: ExpressionVerifyContext,
id: Id,
field: FieldDeclaration
): Option[ExprContext] = {
// It may be a Platform reference shadowed by an existing field
// e.g. String contact; contact.Name -> "String has no property Name"
// Test the target field exists on the shadowed field or try fallback to the static
val targetField = context
.getTypeFor(field.typeName, input.declaration.get)
.toOption
.flatMap(f => DotExpression.findField(target.name, f, context.module, Some(false)))
if (targetField.isEmpty) {
TypeResolver(TypeName(id.name), context.module).toOption match {
case Some(td: PlatformTypeDeclaration) => verifyShadowedStatic(td, context)
case Some(td) if td.isSObject => verifyShadowedStatic(td, context)
case _ => None
}
} else {
None
}
}
private def verifyShadowedStatic(
td: TypeDeclaration,
context: ExpressionVerifyContext
): Option[ExprContext] = {
val result = verifyWithId(ExprContext(isStatic = Some(true), td), context)
if (result.isDefined) {
context.addDependency(td)
Some(result)
} else {
None
}
}
private def verifyInternal(input: ExprContext, context: ExpressionVerifyContext): ExprContext = {
val inter = expression.verify(input, context)
if (inter.isDefined) {
if (inter.isStatic.contains(true) && safeNavigation) {
context.logError(
location,
"Safe navigation operator (?.) can not be used on static references"
)
ExprContext.empty
} else {
context.saveResult(this) {
verifyWithId(inter, context)
}
}
} else {
ExprContext.empty
}
}
private def isNamespace(name: Name, td: TypeDeclaration): Boolean = {
if (td.moduleDeclaration.nonEmpty)
td.moduleDeclaration.get.pkg.namespaces.contains(name)
else
PlatformTypeDeclaration.namespaces.contains(name)
}
private def verifyWithId(input: ExprContext, context: ExpressionVerifyContext): ExprContext = {
val inputType = input.declaration.get
val name = target.name
val field: Option[FieldDeclaration] =
DotExpression.findField(name, inputType, context.module, input.isStatic)
if (field.nonEmpty) {
context.addDependency(field.get)
val target = context.getTypeAndAddDependency(field.get.typeName, inputType).toOption
if (target.isEmpty) {
context.missingType(location, field.get.typeName)
return ExprContext.empty
}
return ExprContext(isStatic = Some(false), target, field.get)
}
// TODO: Private/protected types?
if (input.isStatic.contains(true)) {
val nt = inputType.findLocalType(TypeName(name))
if (nt.nonEmpty) {
return ExprContext(isStatic = Some(true), nt.get)
}
}
// Field is missing
if (inputType.isSObject || inputType.isInstanceOf[RecordSetDeclaration]) {
// For SObject or RecordSet being used as an SObject, ignore if we not have a
// complete type or the field is using a ghosted namespace
if (inputType.isComplete && !context.module.isGhostedFieldName(name)) {
context.log(IssueOps.unknownFieldOnSObject(location, name, inputType.typeName))
}
} else if (inputType.isComplete) {
// For other types, if complete we should error
context.log(IssueOps.unknownFieldOrType(location, name, inputType.typeName))
}
ExprContext.empty
}
}
final case class DotExpressionWithMethod(
expression: Expression,
safeNavigation: Boolean,
target: Option[MethodCallWithId]
) extends Expression {
override def verify(input: ExprContext, context: ExpressionVerifyContext): ExprContext = {
assert(input.declaration.nonEmpty)
interceptAmbiguousMethodCall(input, context)
.getOrElse({
val inter = expression.verify(input, context)
if (inter.isDefined) {
if (inter.isStatic.contains(true) && safeNavigation) {
context.logError(
location,
"Safe navigation operator (?.) can not be used on static references"
)
ExprContext.empty
} else {
target
.map(target =>
target.verify(location, inter.typeDeclaration, inter.isStatic, input, context)
)
.getOrElse(ExprContext.empty)
}
} else {
// When we can't find method we should still verify args for dependency side-effects
target.map(target => target.arguments.map(_.verify(input, context)))
ExprContext.empty
}
})
}
/** Intercept static method call to BusinessHours or Site as these operate on System.* rather than
* Schema.* classes. This hack avoids having to pass additional context into platform type
* loading to disambiguate.
*/
private def interceptAmbiguousMethodCall(
input: ExprContext,
context: ExpressionVerifyContext
): Option[ExprContext] = {
expression match {
case PrimaryExpression(primary: IdPrimary)
if DotExpressionWithMethod.isAmbiguousName.contains(primary.id.name) &&
context.isVar(primary.id.name).isEmpty &&
DotExpression
.findField(primary.id.name, input.typeDeclaration, context.module, None)
.isEmpty =>
context
.getTypeAndAddDependency(
TypeName(primary.id.name, Nil, Some(TypeNames.System)),
context.thisType
)
.toOption
.flatMap(td =>
target.map(target => target.verify(location, td, Some(true), input, context))
)
case _ => None
}
}
}
object DotExpressionWithMethod {
private val isAmbiguousName = Set(Name("BusinessHours"), Name("Site"), Name("Network"))
}
final case class ArrayExpression(expression: Expression, arrayExpression: Expression)
extends Expression {
override def verify(input: ExprContext, context: ExpressionVerifyContext): ExprContext = {
val index =
arrayExpression.verify(ExprContext(input.isStatic, context.thisType), context)
if (index.declaration.isEmpty)
return ExprContext.empty
if (index.typeName != TypeNames.Integer) {
context.logError(
arrayExpression.location,
s"Array indexes must be Integers, found '${index.typeName}'"
)
return ExprContext.empty
}
val inter = expression.verify(input, context)
if (!inter.isDefined)
return ExprContext.empty
val listType = inter.typeName.getArrayType
if (inter.isStatic.contains(true) || listType.isEmpty) {
context.logError(
location,
s"Only Lists can be de-referenced as an array, found '${inter.typeName}'"
)
return ExprContext.empty
}
context.getTypeAndAddDependency(listType.get, context.thisType) match {
case Left(_) =>
context.missingType(location, listType.get)
ExprContext.empty
case Right(td) =>
ExprContext(isStatic = Some(false), td)
}
}
}
abstract class MethodCall extends Expression
final case class MethodCallWithId(target: Id, arguments: ArraySeq[Expression]) extends MethodCall {
var cachedMethod: Option[MethodDeclaration] = None
override def verify(input: ExprContext, context: ExpressionVerifyContext): ExprContext = {
verify(location, input.typeDeclaration, input.isStatic, input, context)
}
def verify(
location: PathLocation,
callee: TypeDeclaration,
staticContext: Option[Boolean],
input: ExprContext,
context: ExpressionVerifyContext
): ExprContext = {
val argTypes = arguments.map(arg => {
val argExpr = arg.verify(input, context)
interceptFoundArg(input, context, arg, argExpr)
})
callee.findMethod(target.name, argTypes, staticContext, context) match {
case Right(method) =>
cachedMethod = Some(method)
context.addDependency(method)
method match {
case ref: Referenceable => ref.addLocation(location)
case _ =>
}
if (method.typeName != TypeNames.Void) {
val td = context.getTypeAndAddDependency(method.typeName, context.thisType)
td match {
case Left(error) =>
if (!context.module.isGhostedType(method.typeName))
context.log(error.asIssue(location))
context.saveResult(this, target.location.location) {
ExprContext(None, None, method)
}
case Right(td) =>
context.saveResult(this, target.location.location) {
ExprContext(isStatic = Some(false), Some(td), method)
}
}
} else {
context.saveResult(this, target.location.location) {
VoidExprContext(method)
}
}
case Left(err) =>
if (callee.isComplete) {
if (argTypes.contains(TypeNames.Any)) {
context.log(Issue(WARNING_CATEGORY, location, s"$err, likely due to unknown type"))
} else {
context.logMissing(location, s"$err")
}
}
ExprContext.empty
}
}
/** Given a method declaration, if this method callout belongs to that method declaration then returns the location
* if the callout. If the callout does not match to the method declaration, returns None.
*/
def getTargetLocationForMethodCallOut(md: ApexMethodDeclaration): Option[Location] = {
cachedMethod match {
case Some(mdFromCallout: ApexMethodDeclaration) =>
if (mdFromCallout.idPathLocation == md.idPathLocation) {
Some(target.location.location)
} else {
None
}
case _ => None
}
}
private def interceptFoundArg(
input: ExprContext,
context: ExpressionVerifyContext,
arg: Expression,
foundType: ExprContext
): TypeName = {
if (foundType.isDefined) {
// handle static type reference, expecting instance but most likely missing variable
if (foundType.isStatic.contains(true)) {
arg match {
case PrimaryExpression(IdPrimary(id)) =>
context.missingIdentifier(arg.location, input.typeName, id.name)
case _ =>
}
}
foundType.typeName
} else {
// If we failed to get argument type (maybe due to ghosting), use null as assignable to anything
TypeNames.Any
}
}
}
final case class MethodCallCtor(isSuper: Boolean, arguments: ArraySeq[Expression])
extends MethodCall {
override def verify(input: ExprContext, context: ExpressionVerifyContext): ExprContext = {
// Verify args so vars don't show as unused and map to typeNames
val args = arguments.map(_.verify(input, context))
if (args.forall(_.isDefined)) {
val ctorSearchContext = if (isSuper) context.superType else Some(context.thisType)
ctorSearchContext match {
case Some(td) =>
td.findConstructor(args.map(arg => arg.typeName), context) match {
case Left(error) =>
context.logError(location, error)
ExprContext.empty
case Right(ctor: ApexConstructorLike) =>
context.saveResult(this, ctor.idLocation) {
ExprContext(Some(false), Some(td), ctor)
}
case Right(ctor: CustomConstructorDeclaration) =>
context.saveResult(this, ctor.nameLocation) {
ExprContext(Some(false), Some(td), ctor)
}
case Right(ctor) =>
ExprContext(Some(false), Some(td), ctor)
}
case _ => ExprContext.empty
}
} else ExprContext.empty
}
}
object MethodCall {
def construct(from: MethodCallContext): MethodCall = {
CodeParser
.toScala(from.id())
.map(id => {
MethodCallWithId(Id.construct(id), expressions(from.expressionList())).withContext(from)
})
.getOrElse({
MethodCallCtor(
CodeParser.toScala(from.SUPER()).nonEmpty,
expressions(from.expressionList())
).withContext(from)
})
}
def construct(from: DotMethodCallContext): MethodCallWithId = {
MethodCallWithId(Id.constructAny(from.anyId()), expressions(from.expressionList()))
.withContext(from)
}
private def expressions(from: ExpressionListContext): ArraySeq[Expression] = {
CodeParser
.toScala(from)
.map(el => CodeParser.toScala(el.expression()).map(e => Expression.construct(e)))
.getOrElse(Expression.emptyExpressions)
}
}
final case class NewExpression(creator: Creator) extends Expression {
override def verify(input: ExprContext, context: ExpressionVerifyContext): ExprContext = {
context.saveResult(this) {
creator.verify(input, context)
}
}
}
final case class CastExpression(typeName: TypeName, expression: Expression) extends Expression {
override def verify(input: ExprContext, context: ExpressionVerifyContext): ExprContext = {
expression.verify(input, context)
val castType = context.getTypeAndAddDependency(typeName, context.thisType).toOption
if (castType.isEmpty) {
context.missingType(location, typeName)
ExprContext.empty
} else {
ExprContext(isStatic = Some(false), castType.get)
}
}
}
final case class SubExpression(expression: Expression) extends Expression {
override def verify(input: ExprContext, context: ExpressionVerifyContext): ExprContext = {
expression.verify(input, context)
}
}
final case class PostfixExpression(expression: Expression, op: String) extends Expression {
override def verify(input: ExprContext, context: ExpressionVerifyContext): ExprContext = {
val inter = expression.verify(input, context)
if (!inter.isDefined)
return inter
val td = inter.declaration.get
td.typeName match {
case TypeNames.Integer | TypeNames.Long | TypeNames.Decimal | TypeNames.Double
if inter.isStatic.contains(false) =>
inter
case _ =>
context.logError(
location,
s"Postfix increment/decrement is not supported on type '${td.typeName}'"
)
ExprContext.empty
}
}
}
final case class PrefixExpression(expression: Expression, op: String) extends Expression {
def isAssignmentOperation: Boolean = op == "++" || op == "--"
override def verify(input: ExprContext, context: ExpressionVerifyContext): ExprContext = {
val inter = expression.verify(input, context)
if (!inter.isDefined)
return inter
val td = inter.declaration.get
td.typeName match {
case TypeNames.Integer | TypeNames.Long | TypeNames.Decimal | TypeNames.Double
if inter.isStatic.contains(false) =>
inter
case _ if inter.isStatic.contains(false) && op == "+" =>
ExprContext(isStatic = Some(false), PlatformTypes.stringType)
case _ =>
context.logError(location, s"Prefix operations are not supported on type '${td.typeName}'")
ExprContext.empty
}
}
}
final case class NegationExpression(expression: Expression, isBitwise: Boolean) extends Expression {
override def verify(input: ExprContext, context: ExpressionVerifyContext): ExprContext = {
val inter = expression.verify(input, context)
if (!inter.isDefined)
return inter
val td = inter.declaration.get
td.typeName match {
case TypeNames.Boolean if !isBitwise && inter.isStatic.contains(false) => inter
case TypeNames.Integer if isBitwise && inter.isStatic.contains(false) => inter
case TypeNames.Long if isBitwise && inter.isStatic.contains(false) => inter
case _ =>
context.logError(location, s"Negation operations is not supported on type '${td.typeName}'")
ExprContext.empty
}
}
}
final case class BinaryExpression(lhs: Expression, rhs: Expression, op: String) extends Expression {
private lazy val operation: Operation = op match {
case "=" => AssignmentOperation
case "&&" => LogicalOperation
case "||" => LogicalOperation
case "??" => ConditionalOperation
case "==" => EqualityOperation
case "!=" => EqualityOperation
case "<>" => EqualityOperation
case "===" => ExactEqualityOperation
case "!==" => ExactEqualityOperation
case "<" => CompareOperation
case ">" => CompareOperation
case "<=" => CompareOperation
case ">=" => CompareOperation
case "+" => PlusOperation
case "-" => ArithmeticOperation
case "*" => ArithmeticOperation
case "/" => ArithmeticOperation
case "+=" => ArithmeticAddSubtractAssignmentOperation
case "-=" => ArithmeticAddSubtractAssignmentOperation
case "*=" => ArithmeticMultiplyDivideAssignmentOperation
case "/=" => ArithmeticMultiplyDivideAssignmentOperation
case "&" => BitwiseOperation
case "|" => BitwiseOperation
case "^" => BitwiseOperation
case "<<" => BitwiseOperation
case ">>" => BitwiseOperation
case ">>>" => BitwiseOperation
case "^=" => BitwiseAssignmentOperation
case "&=" => BitwiseAssignmentOperation
case "|=" => BitwiseAssignmentOperation
case "<<=" => BitwiseAssignmentOperation
case ">>=" => BitwiseAssignmentOperation
case ">>>=" => BitwiseAssignmentOperation
}
def isAssignmentOperation: Boolean = operation.isAssignmentOperation
override def verify(input: ExprContext, context: ExpressionVerifyContext): ExprContext = {
val leftInter = lhs.verify(input, context)
val rightInter = rhs.verify(input, context)
if (!leftInter.isDefined || !rightInter.isDefined)
return ExprContext.empty
if (leftInter.isStatic.contains(true))
context.logError(
location,
s"Expecting instance for operation, not type '${leftInter.typeName}'"
)
if (rightInter.isStatic.contains(true))
context.logError(
location,
s"Expecting instance for operation, not type '${rightInter.typeName}'"
)
// TODO Remove temporary warning, add getReadOnlyError calls back to each operation verify
operation match {
case AssignmentOperation | BitwiseAssignmentOperation |
ArithmeticAddSubtractAssignmentOperation | ArithmeticMultiplyDivideAssignmentOperation =>
operation
.getReadOnlyError(leftInter, context)
.foreach(msg => context.log(Issue(WARNING_CATEGORY, location, msg)))
case _ =>
}
operation.verify(leftInter, rightInter, op, context) match {
case Left(error) =>
context.logError(location, error)
ExprContext.empty
case Right(context) => context
}
}
}
final case class InstanceOfExpression(expression: Expression, typeName: TypeName)
extends Expression {
override def verify(input: ExprContext, context: ExpressionVerifyContext): ExprContext = {
val instanceOfType = context.getTypeAndAddDependency(typeName, context.thisType).toOption
if (instanceOfType.isEmpty)
context.missingType(location, typeName)
expression.verify(input, context)
ExprContext(isStatic = Some(false), PlatformTypes.booleanType)
}
}
final case class QueryExpression(query: Expression, lhs: Expression, rhs: Expression)
extends Expression {
override def verify(input: ExprContext, context: ExpressionVerifyContext): ExprContext = {
query.verify(input, context)
val leftInter = lhs.verify(input, context)
val rightInter = rhs.verify(input, context)
if (!leftInter.isDefined || !rightInter.isDefined)
return ExprContext.empty
ConditionalOperation.verify(leftInter, rightInter, "?", context) match {
case Left(error) =>
context.logError(location, error)
ExprContext.empty
case Right(context) => context
}
}
}
final case class PrimaryExpression(var primary: Primary) extends Expression {
override def verify(input: ExprContext, context: ExpressionVerifyContext): ExprContext = {
context.saveResult(this) {
primary.verify(ExprContext(isStatic = input.isStatic, context.thisType), context)
}
}
}
object Expression {
val emptyExpressions: ArraySeq[Expression] = ArraySeq()
val mergeableSObjects: Set[Name] = Set("Lead", "Contact", "Case", "Account").map(Name(_))
def construct(from: ExpressionContext): Expression = {
val cst: Expression = {
from match {
case expr: DotExpressionContext =>
CodeParser
.toScala(expr.anyId())
.map(id => {
DotExpressionWithId(
Expression.construct(expr.expression()),
CodeParser.toScala(expr.DOT()).isEmpty,
Id.constructAny(id)
)
})
.getOrElse({
DotExpressionWithMethod(
Expression.construct(expr.expression()),
CodeParser.toScala(expr.DOT()).isEmpty,
CodeParser
.toScala(expr.dotMethodCall())
.map(method => {
MethodCall.construct(method)
})
)
})
case expr: ArrayExpressionContext =>
val expressions = CodeParser.toScala(expr.expression())
if (expressions.length == 2)
ArrayExpression(
Expression.construct(expressions.head),
Expression.construct(expressions(1))
)
else
EmptyExpr()
case expr: MethodCallExpressionContext =>
MethodCall.construct(expr.methodCall())
case expr: NewExpressionContext =>
NewExpression(Creator.construct(expr.creator()))
case expr: CastExpressionContext =>
CastExpression(
TypeReference.construct(expr.typeRef()),
Expression.construct(expr.expression())
)
case expr: SubExpressionContext =>
SubExpression(Expression.construct(expr.expression()))
case expr: PostOpExpressionContext =>
val op = CodeParser
.toScala(expr.INC())
.orElse(CodeParser.toScala(expr.DEC()))
PostfixExpression(Expression.construct(expr.expression()), CodeParser.getText(op.get))
case expr: PreOpExpressionContext =>
val op = CodeParser
.toScala(expr.ADD())
.orElse(CodeParser.toScala(expr.DEC()))
.orElse(CodeParser.toScala(expr.INC()))
.orElse(CodeParser.toScala(expr.SUB()))
PrefixExpression(Expression.construct(expr.expression()), CodeParser.getText(op.get))
case expr: NegExpressionContext =>
val op = CodeParser
.toScala(expr.BANG())
.orElse(CodeParser.toScala(expr.TILDE()))
NegationExpression(
Expression.construct(expr.expression()),
CodeParser.getText(op.get) == "~"
)
case expr: Arth1ExpressionContext =>
val op = CodeParser
.toScala(expr.DIV())
.orElse(CodeParser.toScala(expr.MUL()))
val expressions = CodeParser.toScala(expr.expression())
if (expressions.length == 2) {
BinaryExpression(
Expression.construct(expressions.head),
Expression.construct(expressions(1)),
CodeParser.getText(op.get)
)
} else {
EmptyExpr()
}
case expr: Arth2ExpressionContext =>
val op = CodeParser
.toScala(expr.ADD())
.orElse(CodeParser.toScala(expr.SUB()))
val expressions = CodeParser.toScala(expr.expression())
if (expressions.length == 2) {
BinaryExpression(
Expression.construct(expressions.head),
Expression.construct(expressions(1)),
CodeParser.getText(op.get)
)
} else {
EmptyExpr()
}
case expr: BitExpressionContext =>
val gt = ">" * CodeParser.toScala(expr.GT()).size
val lt = "<" * CodeParser.toScala(expr.LT()).size
assert(gt.nonEmpty != lt.nonEmpty)
val expressions = CodeParser.toScala(expr.expression())
if (expressions.length == 2) {
BinaryExpression(
Expression.construct(expressions.head),
Expression.construct(expressions(1)),
gt + lt
)
} else {
EmptyExpr()
}
case expr: CmpExpressionContext =>
val assign = CodeParser.toScala(expr.ASSIGN()).nonEmpty
val op = CodeParser.getText(
CodeParser.toScala(expr.GT()).orElse(CodeParser.toScala(expr.LT())).get
)
val opText = (assign, op) match {
case (true, ">") => ">="
case (true, "<") => "<="
case (false, op) => op
case _ => assert(false); ""
}
val expressions = CodeParser.toScala(expr.expression())
if (expressions.length == 2) {
BinaryExpression(
Expression.construct(expressions.head),
Expression.construct(expressions(1)),
opText
)
} else {
EmptyExpr()
}
case expr: InstanceOfExpressionContext =>
InstanceOfExpression(
Expression.construct(expr.expression()),
TypeReference.construct(expr.typeRef())
)
case expr: EqualityExpressionContext =>
val op = CodeParser
.toScala(expr.EQUAL())
.orElse(CodeParser.toScala(expr.LESSANDGREATER()))
.orElse(CodeParser.toScala(expr.NOTEQUAL()))
.orElse(CodeParser.toScala(expr.TRIPLEEQUAL()))
.orElse(CodeParser.toScala(expr.TRIPLENOTEQUAL()))
val expressions = CodeParser.toScala(expr.expression())
if (expressions.length == 2) {
BinaryExpression(
Expression.construct(expressions.head),
Expression.construct(expressions(1)),
CodeParser.getText(op.get)
)
} else {
EmptyExpr()
}
case expr: BitAndExpressionContext =>
val expressions = CodeParser.toScala(expr.expression())
if (expressions.length == 2) {
BinaryExpression(
Expression.construct(expressions.head),
Expression.construct(expressions(1)),
"&"
)
} else {
EmptyExpr()
}
case expr: BitNotExpressionContext =>
val expressions = CodeParser.toScala(expr.expression())
if (expressions.length == 2) {
BinaryExpression(
Expression.construct(expressions.head),
Expression.construct(expressions(1)),
"^"
)
} else {
EmptyExpr()
}
case expr: BitOrExpressionContext =>
val expressions = CodeParser.toScala(expr.expression())
if (expressions.length == 2) {
BinaryExpression(
Expression.construct(expressions.head),
Expression.construct(expressions(1)),
"|"
)
} else {
EmptyExpr()
}
case expr: LogAndExpressionContext =>
val expressions = CodeParser.toScala(expr.expression())
if (expressions.length == 2) {
BinaryExpression(
Expression.construct(expressions.head),
Expression.construct(expressions(1)),
"&&"
)
} else {
EmptyExpr()
}
case expr: LogOrExpressionContext =>
val expressions = CodeParser.toScala(expr.expression())
if (expressions.length == 2) {
BinaryExpression(
Expression.construct(expressions.head),
Expression.construct(expressions(1)),
"||"
)
} else {
EmptyExpr()
}
case expr: CoalExpressionContext =>
val expressions = CodeParser.toScala(expr.expression())
if (expressions.length == 2) {
BinaryExpression(
Expression.construct(expressions.head),
Expression.construct(expressions(1)),
"??"
)
} else {
EmptyExpr()
}
case expr: CondExpressionContext =>
val expressions = CodeParser.toScala(expr.expression())
if (expressions.length == 3) {
QueryExpression(
Expression.construct(expressions.head),
Expression.construct(expressions(1)),
Expression.construct(expressions(2))
)
} else {
EmptyExpr()
}
case expr: AssignExpressionContext =>
val op = CodeParser
.toScala(expr.ADD_ASSIGN())
.orElse(CodeParser.toScala(expr.AND_ASSIGN()))
.orElse(CodeParser.toScala(expr.ASSIGN()))
.orElse(CodeParser.toScala(expr.DIV_ASSIGN()))
.orElse(CodeParser.toScala(expr.LSHIFT_ASSIGN()))
.orElse(CodeParser.toScala(expr.MUL_ASSIGN()))
.orElse(CodeParser.toScala(expr.OR_ASSIGN()))
.orElse(CodeParser.toScala(expr.RSHIFT_ASSIGN()))
.orElse(CodeParser.toScala(expr.SUB_ASSIGN()))
.orElse(CodeParser.toScala(expr.URSHIFT_ASSIGN()))
.orElse(CodeParser.toScala(expr.XOR_ASSIGN()))
val expressions = CodeParser.toScala(expr.expression())
BinaryExpression(
Expression.construct(expressions.head),
Expression.construct(expressions(1)),
CodeParser.getText(op.get)
)
case expr: PrimaryExpressionContext =>
PrimaryExpression(Primary.construct(expr.primary()))
case _ => EmptyExpr()
}
}
cst.withContext(from)
}
def construct(expression: ArraySeq[ExpressionContext]): ArraySeq[Expression] = {
expression.map(x => Expression.construct(x))
}
}
object Arguments {
def construct(from: ArgumentsContext): ArraySeq[Expression] = {
val el = CodeParser.toScala(from.expressionList())
if (el.nonEmpty) {
Expression.construct(CodeParser.toScala(el.get.expression()))
} else {
Expression.emptyExpressions
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy