com.nawforce.apexlink.org.StreamDeployer.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) 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.apexlink.org
import com.nawforce.apexlink.api._
import com.nawforce.apexlink.finding.TypeResolver.TypeCache
import com.nawforce.apexlink.names.TypeNames.TypeNameUtils
import com.nawforce.apexlink.opcst.OutlineParserFullDeclaration
import com.nawforce.apexlink.types.apex.{FullDeclaration, SummaryApex, TriggerDeclaration}
import com.nawforce.apexlink.types.other._
import com.nawforce.apexlink.types.platform.PlatformTypes
import com.nawforce.pkgforce.diagnostics._
import com.nawforce.pkgforce.documents._
import com.nawforce.pkgforce.names._
import com.nawforce.pkgforce.stream._
import java.util.concurrent.{ConcurrentHashMap, ConcurrentLinkedQueue}
import scala.collection.immutable.ArraySeq
import scala.collection.parallel.CollectionConverters._
import scala.collection.{BufferedIterator, mutable}
import scala.jdk.CollectionConverters._
import scala.util.hashing.MurmurHash3
/** 'Deploy' a module from a stream of PackageEvents. Deploying here really means constructing a set
* of TypeDeclarations and validating them against each other. This process mutates the passed
* types Map for compatibility with dependency analysis code. The class handling code here is
* performance sensitive so this may also aid with efficiency.
*
* FUTURE: Remove Module dependency.
*/
class StreamDeployer(
module: OPM.Module,
events: Iterator[PackageEvent],
types: TypeDeclarationCache
) {
load()
private def load(): Unit = {
val start = java.lang.System.currentTimeMillis()
val basicTypesSize = types.size
// Process package events, these must follow the publishing order from pkgforce
PlatformTypes.withLoadingObserver(module.schemaSObjectType) {
val bufferedIterator = events.buffered
consumeLabels(bufferedIterator)
val components = consumeComponents(bufferedIterator)
val pages = consumePages(bufferedIterator)
consumeFlows(bufferedIterator)
consumeSObjects(bufferedIterator)
consumeClasses(bufferedIterator)
consumeTriggers(bufferedIterator)
components.safeValidate()
pages.safeValidate()
}
// Run plugins over loaded types DependentTypes
// This has to be done post loading to allow dependencies to be established
module.pkg.org.pluginsManager.closePlugins()
// Report progress and tidy up
if (types.size > basicTypesSize) {
val total = (java.lang.System.currentTimeMillis() - start).toDouble
val avg = total / types.size
LoggerOps.info(f"$module loaded ${types.size} types in ${total}ms, average $avg%.1f ms/type")
}
}
private def consumeLabels(events: BufferedIterator[PackageEvent]): Unit = {
val labelRelatedEvents = bufferEvents(Set(classOf[LabelFileEvent], classOf[LabelEvent]), events)
val labelFileEvents = labelRelatedEvents.collect { case e: LabelFileEvent => e }
val labelEvents = labelRelatedEvents.collect { case e: LabelEvent => e }
val labels = LabelDeclaration(module).merge(labelFileEvents, labelEvents)
types.put(labels)
types.put(labels, Some(TypeName(labels.name)))
}
private def consumeComponents(events: BufferedIterator[PackageEvent]): ComponentDeclaration = {
val componentDeclaration =
ComponentDeclaration(module).merge(bufferEvents[ComponentEvent](events))
types.put(componentDeclaration)
componentDeclaration
}
private def consumePages(events: BufferedIterator[PackageEvent]): PageDeclaration = {
val pageDeclaration = PageDeclaration(module).merge(bufferEvents[PageEvent](events))
types.put(pageDeclaration)
pageDeclaration
}
private def consumeFlows(events: BufferedIterator[PackageEvent]): Unit = {
types.put(InterviewDeclaration(module).merge(bufferEvents[FlowEvent](events)))
}
private def consumeSObjects(events: BufferedIterator[PackageEvent]): Unit = {
// Create and register SObjects
val deployer = new SObjectDeployer(module)
val sobjects = deployer.createSObjects(events)
sobjects.foreach(sobject => {
types.put(sobject)
module.schemaSObjectType.add(sobject.typeName.name, hasFieldSets = true)
})
// Run custom validation to setup dependencies
sobjects.foreach(_.safeValidate())
}
/** Consume Apex class events, this is a bit more involved as we try and load first via cache and
* then fallback to reading the source and parsing.
*/
private def consumeClasses(events: BufferedIterator[PackageEvent]): Unit = {
val docs = bufferEvents[ApexEvent](events).map(e => ApexClassDocument(e.path))
// Load summary classes from the cache
LoggerOps.debugTime(s"Loaded summary classes", docs.nonEmpty) {
validateSummaryClasses(loadClassesFromCache(docs))
}
// Load any classes not found via cache or that have been rejected
val missingClasses =
docs.filterNot(doc => types.contains(TypeName(doc.name).withNamespace(module.namespace)))
LoggerOps.debug(s"${missingClasses.length} of ${docs.length} classes not available from cache")
module.pkg.org.getParserType match {
case ANTLRParser =>
parseAndValidateClasses(missingClasses)
case OutlineParserSingleThreaded | OutlineParserMultithreaded =>
val failures = loadClassesWithOutlineParser(ServerOps.getCurrentParser, missingClasses)
if (failures.nonEmpty)
parseAndValidateClasses(failures)
}
}
/** Parse a collection of Apex classes, insert them and validate them. */
private def parseAndValidateClasses(docs: ArraySeq[ClassDocument]): Unit = {
LoggerOps.debugTime(s"Parsed ${docs.length} classes", docs.nonEmpty) {
val decls = docs
.flatMap(doc =>
doc.path.readSourceData() match {
case Left(_) => None
case Right(data) =>
LoggerOps.debugTime(s"Parsed ${doc.path}") {
FullDeclaration
.create(module, doc, data, forceConstruct = false)
.map(td => {
types.put(td)
td
})
}
}
)
// Validate the classes, this must be last due to mutual dependence
decls.foreach { _.safeValidate() }
}
}
/** Validate summary classes & log diagnostics, those with any invalid dependents are discarded.
*/
private def validateSummaryClasses(summaryClasses: Iterator[SummaryApex]): Unit = {
val classes = summaryClasses.toArray
val rejected = mutable.Set[SummaryApex]()
val typeCache = new TypeCache()
var hasRejections = true
var rejectCycles = 0
// Collect rejected in a set, we have to multi-pass to make sure all are found
while (hasRejections) {
val rejects = classes.filterNot(cls =>
rejected.contains(cls) || cls.declaration.hasValidDependencies(typeCache)
)
// Tidy up rejected so they can't be found
rejects.foreach(reject => {
val typeName = reject.declaration.typeName
types.remove(typeName)
typeCache.remove((typeName, module))
LoggerOps.info(s"Cached type $typeName rejected due to invalid dependencies")
})
rejected.addAll(rejects)
hasRejections = rejects.nonEmpty
rejectCycles += 1
}
if (rejectCycles > 1)
LoggerOps.info(s"Used $rejectCycles rejection cycles")
// For those not rejected, complete processing
classes
.filterNot(rejected.contains)
.foreach(cls => {
// Re-establish dependencies
cls.declaration.propagateOuterDependencies(typeCache)
cls.declaration.propagateDependencies()
// Report any (existing) diagnostics
val path = cls.declaration.location.path
cls.diagnostics.foreach(diagnostic => module.pkg.org.issues.add(Issue(path, diagnostic)))
})
}
/** Load classes from the code cache as types returning TypeNames of those available. Benchmarking
* has shown running this in parallel helps performance quite a bit with SSDs.
*/
private def loadClassesFromCache(classes: ArraySeq[ApexClassDocument]): Iterator[SummaryApex] = {
module.pkg.org.parsedCache
.map(parsedCache => {
val pkgContext = module.pkg.packageContext
val localAccum = new ConcurrentHashMap[TypeName, SummaryApex]()
classes.par.foreach(doc => {
val value = parsedCache.get(pkgContext, doc.name.value, contentHash(doc))
val ad = value.map(v => SummaryApex(doc.path, module, v))
if (ad.nonEmpty && !ad.get.diagnostics.exists(_.category == MISSING_CATEGORY)) {
localAccum.put(ad.get.declaration.typeName, ad.get)
}
})
localAccum.entrySet.forEach(kv => {
types.put(kv.getKey, kv.getValue.declaration)
})
localAccum.values().iterator().asScala
})
.getOrElse(Iterator())
}
private def contentHash(doc: ApexClassDocument): Int = {
ParsedCache.classMetaHash(
doc.path.parent.join(s"${doc.name.toString}.cls-meta.xml"),
MurmurHash3.bytesHash(doc.path.readBytes().getOrElse(Array.empty))
)
}
private def loadClassesWithOutlineParser(
selectedParser: AvailableParser,
classes: ArraySeq[ClassDocument]
): ArraySeq[ClassDocument] = {
val localAccum = new ConcurrentHashMap[TypeName, FullDeclaration]()
val failedDocuments = new ConcurrentLinkedQueue[ClassDocument]()
val clsItr =
if (selectedParser == OutlineParserSingleThreaded) classes.iterator else classes.par.iterator
LoggerOps.debugTime(s"Parsed ${classes.length} classes", classes.nonEmpty) {
clsItr.foreach(cls => {
cls.path.readSourceData() match {
case Left(error) =>
LoggerOps.info(s"Failed reading source $error")
case Right(srcData) =>
LoggerOps.debugTime(s"Parsed ${cls.path}") {
val td = OutlineParserFullDeclaration
.toFullDeclaration(cls, srcData, module)
.map(td => {
localAccum.put(td.typeName, td)
})
if (td.isEmpty) failedDocuments.add(cls)
}
}
})
localAccum.entrySet.forEach(kv => {
types.put(kv.getKey, kv.getValue)
})
localAccum.values().asScala.foreach(_.safeValidate())
}
ArraySeq.from(failedDocuments.asScala.toSeq)
}
/** Consume trigger events, these could be cached, but they don't consume much time to for we load
* from disk and parse each time.
*/
private def consumeTriggers(events: BufferedIterator[PackageEvent]): Unit = {
val docs = bufferEvents[TriggerEvent](events).map(e => ApexTriggerDocument(e.path))
LoggerOps.debugTime(s"Parsed ${docs.length} triggers", docs.nonEmpty) {
docs
.flatMap(doc => {
doc.path.readSourceData() match {
case Left(_) => None
case Right(data) =>
TriggerDeclaration
.create(module, doc.path, data)
.map(td => {
types.put(td)
td.safeValidate()
})
}
})
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy