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

com.nawforce.apexlink.cst.ConstructorMap.scala Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (c) 2022 FinancialForce.com, inc. All rights reserved
 */

package com.nawforce.apexlink.cst
import com.nawforce.apexlink.cst.AssignableSupport.{AssignableOptions, isAssignable}
import com.nawforce.apexlink.names.TypeNames
import com.nawforce.apexlink.types.apex.{ApexClassDeclaration, ApexConstructorLike, ApexDeclaration}
import com.nawforce.apexlink.types.core.{ConstructorDeclaration, TypeDeclaration}
import com.nawforce.apexlink.types.synthetic.{
  CustomConstructorDeclaration,
  CustomParameterDeclaration
}
import com.nawforce.pkgforce.diagnostics.Duplicates.IterableOps
import com.nawforce.pkgforce.diagnostics.{Diagnostic, ERROR_CATEGORY, Issue}
import com.nawforce.pkgforce.modifiers._
import com.nawforce.pkgforce.names.{Name, TypeName}
import com.nawforce.pkgforce.parsers.CLASS_NATURE
import com.nawforce.pkgforce.path.PathLocation

import scala.collection.immutable.ArraySeq
import scala.collection.mutable

final case class ConstructorMap(
  typeName: Option[TypeName],
  td: Option[ApexClassDeclaration],
  constructorsByParam: Map[Int, Array[ConstructorDeclaration]],
  superConstructorsByParam: Option[ConstructorMap],
  errors: List[Issue]
) {
  val deepHash: Int = td.map(_.deepHash).getOrElse(0)

  def allConstructors: ArraySeq[ConstructorDeclaration] = {
    val buffer = new mutable.ArrayBuffer[ConstructorDeclaration]()
    constructorsByParam.values.foreach(ctor => buffer.addAll(ctor))
    ArraySeq.unsafeWrapArray(buffer.toArray)
  }

  def findConstructorByParams(
    params: ArraySeq[TypeName],
    context: VerifyContext
  ): Either[String, ConstructorDeclaration] = {
    constructorsByParam.get(params.length) match {
      case Some(potential) => findPotentialMatch(potential, params, context)
      case None            => Left(s"No constructor defined with ${params.length} arguments")
    }
  }

  private def findPotentialMatch(
    matches: Array[ConstructorDeclaration],
    params: ArraySeq[TypeName],
    context: VerifyContext
  ): Either[String, ConstructorDeclaration] = {
    def getCtorString: String = {
      typeName match {
        case Some(name) =>
          s"$name.(${params.mkString(",")})"
        case None => s"(${params.mkString(",")})"
      }
    }

    val potential = findMostSpecificMatch(strict = true, matches, params, context)
      .orElse(findMostSpecificMatch(strict = false, matches, params, context))

    potential match {
      case Some(Right(ctor)) =>
        if (ConstructorMap.isCtorAccessible(ctor, context.thisType, context.superType)) {
          Right(ctor)
        } else {
          // Check the rest of assignable for accessible ctors, if not return the original error
          findPotentialMatch(matches.filterNot(_ == ctor), params, context) match {
            case Right(ctor) => Right(ctor)
            case _           => Left(s"Constructor is not visible: $getCtorString")
          }
        }
      case Some(Left(error)) => Left(s"$error: $getCtorString")
      case None              => Left(s"Constructor not defined: $getCtorString")
    }
  }

  private def findMostSpecificMatch(
    strict: Boolean,
    matches: Array[ConstructorDeclaration],
    params: ArraySeq[TypeName],
    context: VerifyContext
  ): Option[Either[String, ConstructorDeclaration]] = {
    val assignable = matches.filter(c => {
      val argZip = c.parameters.map(_.typeName).zip(params)
      argZip.forall(argPair =>
        isAssignable(
          argPair._1,
          argPair._2,
          context,
          AssignableOptions(strict, narrowSObjects = true)
        )
      )
    })
    if (assignable.isEmpty)
      None
    else if (assignable.length == 1)
      Some(Right(assignable.head))
    else {
      Some(
        assignable
          .find(ctor =>
            assignable.forall(c =>
              c == ctor || ctor
                .hasMoreSpecificParams(c.parameters, params, context)
                .contains(true)
            )
          )
          .map(Right(_))
          .getOrElse(Left("Ambiguous constructor call"))
      )
    }
  }

}

object ConstructorMap {
  type WorkingMap = mutable.HashMap[Int, List[ConstructorDeclaration]]

  def isCtorAccessible(
    ctor: ConstructorDeclaration,
    thisType: TypeDeclaration,
    superType: Option[TypeDeclaration]
  ): Boolean = {
    lazy val isTestVisible            = ctor.isTestVisible && thisType.inTest
    lazy val isAccessedInThisContext  = areInSameApexFile(ctor, thisType)
    lazy val isAccessedInSuperContext = superType.nonEmpty && areInSameApexFile(ctor, superType.get)

    ctor.visibility match {
      case PUBLIC_MODIFIER => true
      case GLOBAL_MODIFIER => true
      case PROTECTED_MODIFIER =>
        isAccessedInThisContext || isAccessedInSuperContext || isTestVisible
      case PRIVATE_MODIFIER => isAccessedInThisContext || isTestVisible
      case _                => false
    }
  }

  def apply(td: TypeDeclaration): ConstructorMap = {
    val workingMap = new WorkingMap()
    td.constructors.foreach(ctor => {
      val key = ctor.parameters.length
      workingMap.put(key, ctor :: workingMap.getOrElse(key, Nil))
    })
    new ConstructorMap(Some(td.typeName), None, toMap(workingMap), None, Nil)
  }

  def apply(
    td: TypeDeclaration,
    errorLocation: Option[PathLocation],
    ctors: ArraySeq[ConstructorDeclaration],
    superClassMap: ConstructorMap
  ): ConstructorMap = {
    val workingMap = new WorkingMap()
    val errors     = mutable.Buffer[Issue]()

    val deduped = deDupeConstructors(ctors, errors)

    deduped.foreach(ctor => {
      val key                      = ctor.parameters.length
      val ctorsWithSameParamLength = workingMap.getOrElse(key, Nil)
      val platformGenericDupes =
        ctorsWithSameParamLength.find(x =>
          x.hasSameParameters(ctor, allowPlatformGenericEquivalence = true)
        )
      if (platformGenericDupes.nonEmpty)
        setConstructorDuplicateError(ctor, platformGenericDupes.head, errors)
      workingMap.put(key, ctor :: ctorsWithSameParamLength)
    })

    applySyntheticsCtors(errorLocation, td, workingMap, superClassMap, errors)

    td match {
      case td: ApexClassDeclaration =>
        new ConstructorMap(
          Some(td.typeName),
          Some(td),
          toMap(workingMap),
          Some(superClassMap),
          errors.toList
        )
      case td: TypeDeclaration =>
        new ConstructorMap(
          Some(td.typeName),
          None,
          toMap(workingMap),
          Some(superClassMap),
          errors.toList
        )
    }
  }

  def empty: ConstructorMap = {
    new ConstructorMap(None, None, Map(), None, Nil)
  }

  private def applySyntheticsCtors(
    errorLocation: Option[PathLocation],
    td: TypeDeclaration,
    workingMap: WorkingMap,
    superClassMap: ConstructorMap,
    errors: mutable.Buffer[Issue]
  ): Unit = {
    if (td.isCustomException) applyCustomExceptionsCtors(errorLocation, td, workingMap)
    else applyDefaultCtor(errorLocation, td, workingMap, superClassMap, errors)
  }

  private def applyDefaultCtor(
    errorLocation: Option[PathLocation],
    td: TypeDeclaration,
    workingMap: WorkingMap,
    superClassMap: ConstructorMap,
    errors: mutable.Buffer[Issue]
  ): Unit = {
    if (workingMap.keys.isEmpty) {
      (td, td.nature) match {
        case (ad: ApexDeclaration, CLASS_NATURE) =>
          val ctor = CustomConstructorDeclaration(errorLocation, ArraySeq())
          if (superClassMap.td.isEmpty)
            workingMap.put(0, List(ctor))
          else if (superClassMap.td.get.nature == CLASS_NATURE)
            superClassMap
              .findConstructorByParams(
                ArraySeq.empty,
                new TypeVerifyContext(None, ad, None, enablePlugins = false)
              ) match {
              case Left(error) =>
                val msg =
                  if (superClassMap.constructorsByParam.contains(0)) error
                  else
                    s"No default constructor available in super type: ${superClassMap.typeName.get}"
                setClassError(td, errors, msg)
              case _ => workingMap.put(0, List(ctor))
            }
        case _ =>
      }
    }
  }

  private def applyCustomExceptionsCtors(
    errorLocation: Option[PathLocation],
    td: TypeDeclaration,
    workingMap: WorkingMap
  ): Unit = {
    def toParam(id: String, typeName: TypeName): CustomParameterDeclaration = {
      CustomParameterDeclaration(Name(id), typeName)
    }
    val synthetics = td.nature match {
      case CLASS_NATURE =>
        ArraySeq(
          CustomConstructorDeclaration(errorLocation, ArraySeq()),
          CustomConstructorDeclaration(
            errorLocation,
            ArraySeq(toParam("param1", TypeNames.String))
          ),
          CustomConstructorDeclaration(
            errorLocation,
            ArraySeq(toParam("param1", TypeNames.Exception))
          ),
          CustomConstructorDeclaration(
            errorLocation,
            ArraySeq(toParam("param1", TypeNames.String), toParam("param2", TypeNames.Exception))
          )
        )

      case _ => ArraySeq.empty
    }

    synthetics.foreach(s => {
      val key = s.parameters.length
      workingMap.put(key, s :: workingMap.getOrElse(key, Nil))
    })

  }

  private def deDupeConstructors(
    ctors: ArraySeq[ConstructorDeclaration],
    errors: mutable.Buffer[Issue]
  ): ArraySeq[ConstructorDeclaration] = {
    val dupes = ctors.duplicates(_.parameters.map(_.typeName.toString()).mkString(","))
    dupes.foreach(duplicates => {
      duplicates._2.foreach(dup => {
        setConstructorDuplicateError(dup, duplicates._1, errors)
      })
    })
    ctors.filterNot(dupes.values.flatten.toSeq.contains)
  }

  private def setConstructorDuplicateError(
    constructor: ConstructorDeclaration,
    duplicateOf: ConstructorDeclaration,
    errors: mutable.Buffer[Issue]
  ): Unit = {
    (constructor, duplicateOf) match {
      case (ac: ApexConstructorLike, dup: ApexConstructorLike) =>
        errors.append(
          new Issue(
            ac.location.path,
            Diagnostic(
              ERROR_CATEGORY,
              ac.idLocation,
              s"Constructor is a duplicate of an earlier constructor at ${dup.idLocation.displayPosition}"
            )
          )
        )
      case _ =>
    }
  }

  private def setClassError(td: TypeDeclaration, errors: mutable.Buffer[Issue], message: String) = {
    td match {
      case ad: ApexDeclaration =>
        errors.append(
          new Issue(ad.location.path, Diagnostic(ERROR_CATEGORY, ad.idLocation, message))
        )
      case _ =>
    }
  }

  private def toMap(workingMap: WorkingMap): Map[Int, Array[ConstructorDeclaration]] = {
    workingMap.map(kv => (kv._1, kv._2.toArray)).toMap
  }

  private def areInSameApexFile(
    ctor: ConstructorDeclaration,
    calledFrom: TypeDeclaration
  ): Boolean = {
    (ctor, calledFrom) match {
      case (acl: ApexConstructorLike, ad: ApexDeclaration) => ad.location.path == acl.location.path
      case _                                               => false
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy