com.nawforce.pkgforce.parsers.ApexClassSummary.scala Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of apex-ls_sjs1_2.13 Show documentation
Show all versions of apex-ls_sjs1_2.13 Show documentation
Salesforce Apex static analysis toolkit
The newest version!
/*
Copyright (c) 2020 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.pkgforce.parsers
import com.nawforce.pkgforce.diagnostics.Duplicates.IterableOps
import com.nawforce.pkgforce.diagnostics.{Diagnostic, ERROR_CATEGORY, Issue}
import com.nawforce.pkgforce.modifiers.{GLOBAL_MODIFIER, Modifier, WEBSERVICE_MODIFIER}
import com.nawforce.pkgforce.names.{Name, Names}
import com.nawforce.pkgforce.path.{IdLocatable, Location, PathLocation}
import com.nawforce.runtime.parsers.CodeParser
import io.github.apexdevtools.apexparser.ApexParser._
import scala.collection.compat.immutable.ArraySeq
import scala.collection.mutable.ArrayBuffer
sealed abstract class Nature(val value: String)
case object CLASS_NATURE extends Nature("class")
case object INTERFACE_NATURE extends Nature("interface")
case object ENUM_NATURE extends Nature("enum")
case object TRIGGER_NATURE extends Nature(value = "trigger")
case object CONSTRUCTOR_NATURE extends Nature(value = "constructor")
case object METHOD_NATURE extends Nature(value = "method")
case object FIELD_NATURE extends Nature(value = "field")
case object PROPERTY_NATURE extends Nature(value = "property")
case object ENUM_CONSTANT_NATURE extends Nature(value = "enum constant")
case object INIT_NATURE extends Nature(value = "")
object Nature {
def forType(value: String): Nature = {
value match {
case CLASS_NATURE.value => CLASS_NATURE
case INTERFACE_NATURE.value => INTERFACE_NATURE
case ENUM_NATURE.value => ENUM_NATURE
case TRIGGER_NATURE.value => TRIGGER_NATURE
}
}
}
trait ApexNode extends IdLocatable {
val nature: Nature
val name: Name
val children: ArraySeq[ApexNode]
val modifiers: ArraySeq[Modifier]
val parseIssues: ArraySeq[Issue]
val signature: String
val description: String
def collectIssues(): ArraySeq[Issue] = {
val issues = new ArrayBuffer[Issue]()
collectIssues(issues)
ArraySeq.unsafeWrapArray(issues.toArray)
}
protected def collectIssues(issues: ArrayBuffer[Issue]): Unit = {
issues.addAll(parseIssues)
localIssues.foreach(issue => issues.append(issue))
children.foreach(_.collectIssues(issues))
}
def localIssues: Seq[Issue] = {
if (nature == CLASS_NATURE) {
checkNeedsGlobalOrWebService() ++
checkMisnamedConstructors() ++
checkDuplicateConstructors() ++
checkDuplicateMethods()
} else {
Seq.empty
}
}
private def checkNeedsGlobalOrWebService(): Seq[Issue] = {
if (!modifiers.contains(GLOBAL_MODIFIER)) {
children
.filter(_.modifiers.intersect(Seq(GLOBAL_MODIFIER, WEBSERVICE_MODIFIER)).nonEmpty)
.map(child =>
new Issue(
location.path,
Diagnostic(
ERROR_CATEGORY,
child.idLocation,
"Enclosing class must be declared global to use global or webservice modifiers"
)
)
)
} else {
Seq.empty
}
}
private def checkMisnamedConstructors(): Seq[Issue] = {
children
.filter(child => child.nature == CONSTRUCTOR_NATURE && child.name != name)
.map(misnamed => {
new Issue(
location.path,
Diagnostic(
ERROR_CATEGORY,
misnamed.idLocation,
s"Constructors should have same name as the class, maybe method return type is missing?"
)
)
})
}
// This is a weak test as types have not been normalised
private def checkDuplicateConstructors(): Seq[Issue] = {
children
.collect { case n: ApexConstructorNode => n }
.duplicates(_.compareString)
.flatMap(duplicates => {
duplicates._2.map(dup => {
new Issue(
location.path,
Diagnostic(
ERROR_CATEGORY,
dup.idLocation,
s"Constructor is a duplicate of an earlier constructor at ${duplicates._1.idLocation.displayPosition}"
)
)
})
})
.toSeq
}
private def checkDuplicateMethods(): Seq[Issue] = {
children
.collect { case n: ApexMethodNode => n }
.duplicates(_.compareString)
.flatMap(duplicates => {
duplicates._2.map(dup => {
new Issue(
location.path,
Diagnostic(
ERROR_CATEGORY,
dup.idLocation,
s"Method is a duplicate of an earlier method at ${duplicates._1.idLocation.displayPosition}"
)
)
})
})
.toSeq
}
}
object ApexNode {
def apply(parser: CodeParser, ctx: CompilationUnitContext): Option[ApexNode] = {
val visitor = new ApexClassVisitor(parser)
visitor.visit(ctx).headOption
}
def apply(parser: CodeParser, ctx: TriggerUnitContext): Option[ApexNode] = {
val visitor = new ApexClassVisitor(parser)
visitor.visit(ctx).headOption
}
def appendSpace(str: String): String = {
if (str.nonEmpty)
str + " "
else
str
}
}
case class ExtendsType(name: Name, location: Location)
class ApexLightNode(
val location: PathLocation,
val nature: Nature,
val name: Name,
val idLocation: Location,
val typeRef: Option[ExtendsType],
val children: ArraySeq[ApexNode],
val modifiers: ArraySeq[Modifier],
override val signature: String,
override val description: String,
val parseIssues: ArraySeq[Issue]
) extends ApexNode {
override def localIssues: Seq[Issue] = super.localIssues ++ checkCustomException()
private def checkCustomException(): Seq[Issue] = {
val isClass = nature == CLASS_NATURE || nature == INTERFACE_NATURE
lazy val isNamedCorrectly = name.endsWith(Names.Exception)
lazy val isExtendedCorrectly = typeRef.exists(_.name.endsWith(Names.Exception))
if (isClass) {
(isNamedCorrectly, isExtendedCorrectly) match {
case (true, false) =>
ArraySeq(
new Issue(
location.path,
Diagnostic(
ERROR_CATEGORY,
typeRef.map(_.location).getOrElse(idLocation),
s"Exception class '$name' must extend another Exception class"
)
)
)
case (false, true) =>
ArraySeq(
new Issue(
location.path,
Diagnostic(
ERROR_CATEGORY,
idLocation,
s"Class '$name' extending an Exception must have a name ending in Exception"
)
)
)
case _ => ArraySeq.empty
}
} else
ArraySeq.empty
}
}
case class ApexFormalParameter(
modifiers: ArraySeq[Modifier],
typeName: String,
name: String,
parseIssues: ArraySeq[Issue]
) {
def toStringNoName: String =
s"${ApexNode.appendSpace(modifiers.map(_.name).sorted.mkString(" "))}$typeName"
override def toString: String = s"$toStringNoName $name"
}
case class ApexConstructorNode(
location: PathLocation,
name: Name,
idLocation: Location,
children: ArraySeq[ApexLightNode],
modifiers: ArraySeq[Modifier],
parseIssues: ArraySeq[Issue],
params: ArraySeq[ApexFormalParameter]
) extends ApexNode {
override val nature: Nature = CONSTRUCTOR_NATURE
override lazy val signature: String =
s"${ApexNode.appendSpace(modifiers.mkString(" "))}$name(${params.mkString(", ")})"
override lazy val description: String = s"(${params.mkString(", ")}) ${modifiers.mkString(" ")}"
lazy val compareString: String =
s"$name(${params.map(_.toStringNoName).mkString(", ")}".toLowerCase()
}
case class ApexMethodNode(
location: PathLocation,
name: Name,
idLocation: Location,
children: ArraySeq[ApexLightNode],
modifiers: ArraySeq[Modifier],
parseIssues: ArraySeq[Issue],
returnType: String,
params: ArraySeq[ApexFormalParameter]
) extends ApexNode {
override val nature: Nature = METHOD_NATURE
override lazy val signature: String =
s"${ApexNode.appendSpace(modifiers.mkString(" "))}$returnType $name(${params.mkString(", ")})"
override lazy val description: String =
s"$returnType (${params.mkString(", ")}) ${modifiers.mkString(" ")}"
lazy val compareString: String =
s"${ApexNode.appendSpace(modifiers.map(_.name).sorted.mkString(" "))} " +
s"$returnType $name(${params.map(_.toStringNoName).mkString(", ")}".toLowerCase()
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy