com.nawforce.apexlink.org.OPM.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) 2022 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.analysis.OrgAnalysis
import com.nawforce.apexlink.api.{AvailableParser, BuildInfo, Org, Package, ServerOps, TypeSummary}
import com.nawforce.apexlink.cst.CompilationUnit
import com.nawforce.apexlink.deps.{DownWalker, MaxDependencyCountParser, TransitiveCollector}
import com.nawforce.apexlink.finding.TypeFinder
import com.nawforce.apexlink.finding.TypeResolver.TypeCache
import com.nawforce.apexlink.indexer.{Indexer, Monitor}
import com.nawforce.apexlink.names.TypeNames
import com.nawforce.apexlink.plugins.PluginsManager
import com.nawforce.apexlink.rpc._
import com.nawforce.apexlink.types.apex.{
ApexClassDeclaration,
ApexDeclaration,
ApexFullDeclaration,
TriggerDeclaration
}
import com.nawforce.apexlink.types.core.{TypeDeclaration, TypeId}
import com.nawforce.apexlink.types.other._
import com.nawforce.apexlink.types.platform.PlatformTypeDeclaration
import com.nawforce.apexlink.types.schema.{SObjectDeclaration, SchemaSObjectType}
import com.nawforce.pkgforce.diagnostics._
import com.nawforce.pkgforce.documents._
import com.nawforce.pkgforce.modifiers.ISTEST_ANNOTATION
import com.nawforce.pkgforce.names.{Name, TypeIdentifier, TypeName}
import com.nawforce.pkgforce.path.{Location, PathLike, PathLocation}
import com.nawforce.pkgforce.pkgs.TriHierarchy
import com.nawforce.pkgforce.stream._
import com.nawforce.pkgforce.workspace.{ModuleLayer, ProjectConfig, Workspace}
import com.nawforce.runtime.parsers.{CodeParser, SourceData}
import com.nawforce.runtime.platform.Path
import io.github.apexdevtools.apexparser.ApexParser
import java.io.{PrintWriter, StringWriter}
import java.nio.charset.StandardCharsets
import java.util
import java.util.concurrent.locks.ReentrantLock
import scala.annotation.unused
import scala.collection.immutable.ArraySeq
import scala.collection.mutable
import scala.collection.mutable.ArrayBuffer
import scala.util.hashing.MurmurHash3
/** Org/Package/Module hierarchy. This is based on generics to maintain consistency while we migrate
* features over to pkgforce. The generics force the use of inner classes which is not so
* desirable,
*/
object OPM extends TriHierarchy {
type TOrg = OrgImpl
type TPackage = PackageImpl
type TModule = Module
class OrgImpl(
val path: PathLike,
val issueManager: IssuesManager,
initWorkspace: Option[Workspace]
) extends TriOrg
with Org
with OrgTestClasses {
// Acquire lock for all operations that may be impacted by refresh
val refreshLock = new ReentrantLock(true)
// The workspace loaded into this Org
val workspace: Workspace = initWorkspace.getOrElse(new Workspace(issueManager, Seq()))
// Indexer monitor launcher that manages workspace watching
val monitorLauncher: Monitor = new Monitor(path)
/** Manager for post validation plugins */
private[nawforce] val pluginsManager = new PluginsManager
/** Parsed Apex data cache, the cache holds summary information about Apex types to speed
* startup
*/
private[nawforce] val parsedCache =
ParsedCache.create(MurmurHash3.stringHash(BuildInfo.implementationBuild)) match {
case Right(pc) => Some(pc)
case Left(err) => LoggerOps.info(err); None
}
/** Is this Org using auto-flushing of the parsedCache. */
private val autoFlush = ServerOps.isAutoFlushEnabled
/** The Org flusher. */
private val flusher =
if (autoFlush) new CacheFlusher(this, parsedCache) else new Flusher(this, parsedCache)
/* Parser type to use for org */
lazy val getParserType: AvailableParser = {
val parserType = ServerOps.getCurrentParser
LoggerOps.info(s"Using $parserType")
parserType
}
override def getProjectConfig(): Option[ProjectConfig] = workspace.projectConfig
/** Packages in org in deploy order, the last entry is the unmanaged package identified by
* namespace = None
*/
override val packages: ArraySeq[PackageImpl] = {
def createModule(
pkg: PackageImpl,
dependencies: ArraySeq[Module],
index: DocumentIndex
): Module = {
new Module(pkg, dependencies, index)
}
OrgInfo.current.withValue(this) {
// Fold over layers to create packages - with any package(namespace) dependencies linked to each package
// The workspace layers form a deploy ordering, so each is dependent on all previously created
val declared =
workspace.layers.foldLeft(ArraySeq[PackageImpl]())((acc, pkgLayer) => {
acc :+ new PackageImpl(
this,
pkgLayer.namespace,
pkgLayer.isGulped,
acc,
workspace,
ArraySeq.unsafeWrapArray(pkgLayer.layers.toArray),
createModule
)
})
// If no unmanaged, create it
val unmanaged =
if (declared.isEmpty || declared.lastOption.exists(_.namespace.nonEmpty))
Seq(
new PackageImpl(
this,
None,
isGulped = false,
declared,
workspace,
ArraySeq.empty,
createModule
)
)
else
Seq.empty
ArraySeq.unsafeWrapArray((declared ++ unmanaged).toArray)
}
}
// After packages/module setup load metadata & flush to cache
OrgInfo.current.withValue(this) {
packages.foreach(_.modules.foreach(_.freeze()))
CodeParser.clearCaches()
if (autoFlush)
flusher.refreshAndFlush()
}
// Once loaded we can run external analysis
OrgAnalysis.afterLoad(this)
/** Get all loaded packages. */
def getPackages(): Array[Package] = {
packages.toArray[Package]
}
/** Provide access to IssueManager for org */
override def issues: IssuesManager = issueManager
/** Check to see if cache has been flushed */
override def isDirty(): Boolean = flusher.isDirty
/** Write dirty metadata to the cache, only works for manual flush orgs */
def flush(): Boolean = {
if (!autoFlush)
flusher.refreshAndFlush()
else
false
}
/** Queue a metadata refresh request */
def queueMetadataRefresh(request: RefreshRequest): Unit = {
flusher.queue(request)
}
def queueMetadataRefresh(request: Iterable[RefreshRequest]): Unit = {
flusher.queueAll(request)
}
def getPackageForPath(path: String): Package = {
refreshLock.synchronized {
packages.find(_.isPackagePath(path)).orNull
}
}
/** Get a array of type identifiers available across all packages. */
def getTypeIdentifiers(apexOnly: Boolean): Array[TypeIdentifier] = {
refreshLock.synchronized {
OrgInfo.current.withValue(this) {
packages.foldLeft(Array[TypeIdentifier]())((acc, pkg) =>
acc ++ pkg.getTypeIdentifiers(apexOnly)
)
}
}
}
/** Extract all dependencies */
override def getDependencies: java.util.Map[String, Array[String]] = {
refreshLock.synchronized {
OrgInfo.current.withValue(this) {
val dependencies = new util.HashMap[String, Array[String]]()
packages
.filterNot(_.isGhosted)
.foreach(_.orderedModules.head.populateDependencies(dependencies))
dependencies
}
}
}
/** Find a location for an identifier */
override def getIdentifierLocation(identifier: TypeIdentifier): PathLocation = {
refreshLock.synchronized {
OrgInfo.current.withValue(this) {
(findTypeIdentifier(identifier) match {
case Some(ad: ApexDeclaration) => Some(PathLocation(ad.location.path, ad.idLocation))
case _ => None
}).orNull
}
}
}
/** Search package modules for the TypeDeclaration matching a TypeIdentifier. */
def findTypeIdentifier(identifier: TypeIdentifier): Option[TypeDeclaration] = {
packagesByNamespace
.get(identifier.namespace)
.flatMap(pkg => {
pkg.orderedModules.view.flatMap(_.moduleType(identifier.typeName)).headOption
})
}
def getDependencyGraph(
identifiers: Array[TypeIdentifier],
depth: Integer,
apexOnly: Boolean,
ignoring: Array[TypeIdentifier]
): DependencyGraph = {
refreshLock.synchronized {
OrgInfo.current.withValue(this) {
val depWalker = new DownWalker(this, apexOnly)
val nodeData = depWalker
.walk(identifiers, depth, ignoring)
.map(n => {
DependencyNode(
n.id,
nodeFileSize(n.id),
n.nature,
n.transitiveCount,
n.maxDependencyCount,
n.isEntryPoint,
n.extending,
n.implementing,
n.using
)
})
val nodeIndex = nodeData.map(_.identifier).zipWithIndex.toMap
val linkData = new ArrayBuffer[DependencyLink]()
nodeData.foreach(n => {
val source = nodeIndex(n.identifier)
def safeLink(nature: String)(identifier: TypeIdentifier): Unit = {
nodeIndex
.get(identifier)
.foreach(target =>
if (source != target) linkData += DependencyLink(source, target, nature)
)
}
n.extending.foreach(safeLink("extends"))
n.implementing.foreach(safeLink("implements"))
n.using.foreach(safeLink("uses"))
})
DependencyGraph(nodeData, linkData.toArray)
}
}
}
private def nodeFileSize(identifier: TypeIdentifier): Int = {
Option(getIdentifierLocation(identifier))
.map(location => location.path.size.toInt)
.getOrElse(0)
}
/** Locate a definition for a symbol */
override def getDefinition(
path: String,
line: Int,
offset: Int,
content: String
): Array[LocationLink] = {
if (path == null)
return Array.empty
refreshLock.synchronized {
OrgInfo.current.withValue(this) {
packages
.find(_.isPackagePath(path))
.map(_.getDefinition(Path(path), line, offset, Option(content)))
.getOrElse(Array.empty)
}
}
}
override def getImplementation(
path: String,
line: Int,
offset: Int,
content: String
): Array[LocationLink] = {
refreshLock.synchronized {
OrgInfo.current.withValue(this) {
packages
.find(_.isPackagePath(path))
.map(_.getImplementation(Path(path), line, offset, Option(content)))
.getOrElse(Array.empty)
}
}
}
override def getHover(path: String, line: Int, offset: Int, content: String): HoverItem = {
refreshLock.synchronized {
OrgInfo.current.withValue(this) {
packages
.find(_.isPackagePath(path))
.map(_.getHover(Path(path), line, offset, Option(content)))
.getOrElse(HoverItem(None, None))
}
}
}
override def getRenameLocations(
path: String,
line: Int,
offset: Int,
content: String
): Array[Rename] = {
refreshLock.synchronized {
OrgInfo.current.withValue(this) {
packages
.find(_.isPackagePath(path))
.map(_.getRenameLocations(Path(path), line, offset, Option(content)))
.getOrElse(Array.empty)
}
}
}
override def getReferences(path: String, line: Int, offset: Int): Array[TargetLocation] = {
if (path == null)
return Array.empty
refreshLock.synchronized {
OrgInfo.current.withValue(this) {
packages
.find(_.isPackagePath(path))
.map(_.getReferences(Path(path), line, offset))
.getOrElse(Array.empty)
}
}
}
override def getCompletionItems(
path: String,
line: Int,
offset: Int,
content: String
): Array[CompletionItemLink] = {
refreshLock.synchronized {
getCompletionItemsInternal(Path(path), line, offset, content)
}
}
def getCompletionItemsInternal(
path: PathLike,
line: Int,
offset: Int,
content: String
): Array[CompletionItemLink] = {
if (path == null || content == null)
return Array.empty
OrgInfo.current.withValue(this) {
packages
.find(_.isPackagePathInternal(path))
.map(_.getCompletionItems(path, line, offset, content))
.getOrElse(Array.empty)
}
}
def getDependencyBombs(count: Int): Array[BombScore] = {
refreshLock.synchronized {
val maxBombs = Math.max(0, count)
val allClasses = packages.flatMap(_.orderedModules.flatMap(_.nonTestClasses.toSeq))
val bombs = mutable.PriorityQueue[BombScore]()(Ordering.by(1000 - _.score))
allClasses.foreach(cls => {
if (!cls.inTest) {
val score = cls.bombScore(allClasses.size)
if (score._3 > 0)
bombs.enqueue(BombScore(cls.typeId.asTypeIdentifier, score._2, score._1, score._3))
if (bombs.size > maxBombs) {
bombs.dequeue()
}
}
})
bombs.dequeueAll.toArray.reverse
}
}
/** Returns dependency information for all classes & triggers
*
* @param scopeDirectory limit to classes & triggers under this directory, maybe relative to org path
* @param excludeTestClasses ignore @isTest classes
*/
def getAllDependencyCounts(
scopeDirectory: String,
excludeTestClasses: Boolean
): Array[DependencyCount] = {
val scopePath = path.join(scopeDirectory)
if (!scopePath.isDirectory) {
Array()
} else {
// Traverse indexes under or same as scopePath extracting all class files within scopePath
val paths = mutable.ArrayBuffer[PathLike]()
workspace.indexes
.foreach(layerAndIndex => {
paths.addAll(
layerAndIndex._2.get(ApexNature).values.flatten.filter(scopePath.isParentOf)
)
paths.addAll(
layerAndIndex._2.get(TriggerNature).values.flatten.filter(scopePath.isParentOf)
)
})
// Use paths to direct getDependencyCounts to do the work
getDependencyCountsInternal(paths.toArray, excludeTestClasses)
}
}
/** Returns dependency information for classes & triggers identified via paths
*
* @param paths paths of types to find dependency information for
* @param excludeTestClasses ignore @isTest classes
*/
def getDependencyCounts(
paths: Array[String],
excludeTestClasses: Boolean
): Array[DependencyCount] = {
getDependencyCountsInternal(paths.map(Path.safeApply), excludeTestClasses)
}
/** Returns dependency information for classes & triggers identified via paths
*
* @param paths paths of classes to find dependency information for
* @param excludeTestClasses ignore @isTest classes
*/
private def getDependencyCountsInternal(
paths: Array[PathLike],
excludeTestClasses: Boolean
): Array[DependencyCount] = {
def getTypeAndSummaryOfPath(path: PathLike): Option[(TypeIdentifier, TypeSummary)] =
packages.view
.flatMap(pkg =>
pkg
.getTypeOfPathInternal(path)
.map(_.asTypeIdentifier)
.flatMap(typeId =>
Option(pkg.getSummaryOfType(typeId))
.flatMap(summary => Option(typeId, summary))
)
)
.headOption
def countTransitiveDependencies(
typeId: TypeIdentifier,
transitiveDependencies: Array[TypeIdentifier]
): Int = {
transitiveDependencies.count(t => t != typeId)
}
def getTypeIdOfPath(path: PathLike): Option[TypeId] =
packages.view
.flatMap(pkg => pkg.getTypeOfPathInternal(path))
.headOption
val collector =
new TransitiveCollector(this, isSamePackage = !packages.exists(_.isGulped), true)
paths
.flatMap { path =>
getTypeAndSummaryOfPath(path)
.filter { case (typeId, summary) =>
typeId.toString
!excludeTestClasses || !summary.modifiers.contains(ISTEST_ANNOTATION)
}
.map { case (typeId, _) =>
(typeId, collector.transitives(typeId))
}
.map { case (typeId, transitiveDependencies) =>
DependencyCount(
path.toString,
countTransitiveDependencies(typeId, transitiveDependencies),
getTypeIdOfPath(path)
.map(id => new MaxDependencyCountParser(this).count(id))
.getOrElse(Left(Some(s"Could not find type for path $path")))
)
}
}
}
}
class PackageImpl(
override val org: OrgImpl,
override val namespace: Option[Name],
override val isGulped: Boolean,
override val basePackages: ArraySeq[PackageImpl],
workspace: Workspace,
layers: ArraySeq[ModuleLayer],
mdlFactory: (PackageImpl, ArraySeq[Module], DocumentIndex) => Module
) extends TriPackage
with PackageAPI
with DefinitionProvider
with CompletionProvider
with ImplementationProvider
with HoverProvider
with RenameProvider
with ReferenceProvider {
val modules: ArraySeq[Module] =
layers
.foldLeft(ArraySeq[Module]())((acc, layer) => {
acc :+ mdlFactory(this, acc, workspace.indexes(layer))
})
/** Is this or any base package of this a ghost package. */
lazy val hasGhosted: Boolean = isGhosted || basePackages.exists(_.hasGhosted)
/** Find module for a path. */
def getPackageModule(path: PathLike): Option[Module] = {
orderedModules.find(_.isVisibleFile(path)) match {
case Some(module) if MetadataDocument(path).nonEmpty => Some(module)
case _ => None
}
}
/** Get summary of package context containing namespace & base package namespace information. */
def packageContext: PackageContext = {
def toNSString(ns: Option[Name]): String = ns.map(_.value).getOrElse("")
val ghostedPackages = basePackages
.filterNot(_.isGulped)
.groupBy(_.isGhosted)
.map(kv => (kv._1, kv._2.map(pkg => toNSString(pkg.namespace)).sorted.toArray))
val additionalNamespaces =
basePackages.filter(_.isGulped).map(pkg => toNSString(pkg.namespace)).toArray
PackageContext(
namespace.map(_.value),
ghostedPackages.getOrElse(true, Array.empty),
ghostedPackages.getOrElse(false, Array.empty),
additionalNamespaces
)
}
/** Set of namespaces used by this package and its base packages. */
lazy val namespaces: Set[Name] = {
namespace.toSet ++
basePackages.flatMap(_.namespaces) ++
PlatformTypeDeclaration.namespaces
}
/** Load a class to obtain it's FullDeclaration, issues are not updated, this just returns a
* temporary version of the class so that it can be inspected.
*/
protected def loadClass(
path: PathLike,
source: String
): (Option[(ApexParser, ApexParser.CompilationUnitContext)], Option[ApexFullDeclaration]) = {
MetadataDocument(path) match {
case Some(doc: ApexClassDocument) =>
getPackageModule(path)
.map(module => {
val existingIssues = org.issueManager.pop(path)
val parser = CodeParser(doc.path, SourceData(source.getBytes(StandardCharsets.UTF_8)))
val result = parser.parseClassReturningParser()
try {
(
Some(result.value),
CompilationUnit
.construct(parser, module, doc.name, result.value._2)
.map(_.typeDeclaration)
)
} catch {
case ex: Throwable =>
LoggerOps.info(s"CST construction failed for ${doc.path}", ex)
(None, None)
} finally {
org.issueManager.push(path, existingIssues)
}
})
.getOrElse((None, None))
case _ => (None, None)
}
}
protected def loadTrigger(
path: PathLike,
source: String
): (Option[(ApexParser, ApexParser.TriggerUnitContext)], Option[ApexFullDeclaration]) = {
MetadataDocument(path) match {
case Some(doc: ApexTriggerDocument) =>
getPackageModule(path)
.map(module => {
val existingIssues = org.issueManager.pop(path)
val parser = CodeParser(doc.path, SourceData(source.getBytes(StandardCharsets.UTF_8)))
val result = parser.parseTriggerReturningParser()
try {
(Some(result.value), TriggerDeclaration.construct(parser, module, result.value._2))
} catch {
case ex: Throwable =>
LoggerOps.info(s"CST construction failed for ${doc.path}", ex)
(None, None)
} finally {
org.issueManager.push(path, existingIssues)
}
})
.getOrElse((None, None))
case _ => (None, None)
}
}
}
class Module(
override val pkg: PackageImpl,
override val dependents: ArraySeq[Module],
val index: DocumentIndex
) extends TriModule
with ModuleRefresh
with ModuleFind
with TypeFinder
with ModuleCompletions {
def namespaces: Set[Name] = pkg.namespaces
val isGulped: Boolean = pkg.isGulped
private[nawforce] val types = new TypeDeclarationCache()
private val schemaManager = SchemaSObjectType(this)
@unused
private val indexer = new Indexer(index.path, pkg.org.monitorLauncher) {
override def onFilesChanged(paths: Array[String], rescan: Boolean): Unit = {
val docPaths = paths.flatMap(path => MetadataDocument(Path(path)).map(_.path))
if (docPaths.nonEmpty) {
LoggerOps.debug(s"Indexer changed ${docPaths.length} files: ${docPaths.mkString(", ")}")
pkg.refreshAll(docPaths)
}
}
}
def freeze(): Unit = {
// FUTURE: Have return types, currently can't be done because class loading code needs access to in-flight types
upsertMetadata(AnyDeclaration(this))
upsertMetadata(schemaSObjectType)
upsertMetadata(schemaSObjectType, Some(TypeName(schemaSObjectType.name)))
new StreamDeployer(this, PackageStream.eventStream(index), types)
}
def schemaSObjectType: SchemaSObjectType = schemaManager
def any: AnyDeclaration = types.getUnsafe(TypeNames.Any).asInstanceOf[AnyDeclaration]
def labels: LabelDeclaration = types.getUnsafe(TypeNames.Label).asInstanceOf[LabelDeclaration]
def interviews: InterviewDeclaration =
types.getUnsafe(TypeNames.Interview).asInstanceOf[InterviewDeclaration]
def pages: PageDeclaration = types.getUnsafe(TypeNames.Page).asInstanceOf[PageDeclaration]
def components: ComponentDeclaration =
types.getUnsafe(TypeNames.Component).asInstanceOf[ComponentDeclaration]
def nonTestClasses: Iterable[ApexClassDeclaration] =
types.values().collect {
case ac: ApexClassDeclaration if !ac.inTest => ac
}
def testClasses: Iterable[ApexClassDeclaration] =
types.values().collect {
case ac: ApexClassDeclaration if ac.inTest => ac
}
override def isVisibleFile(path: PathLike): Boolean = {
index.isVisibleFile(path)
}
/* Iterator over available types */
def getTypes: Iterable[TypeDeclaration] = {
types.values()
}
/** Iterate metadata defined types, this will include referenced platform SObjects irrespective
* of if they have been extended or not which is perhaps not quite accurate to the method name.
*/
def getMetadataDefinedTypeIdentifiers(apexOnly: Boolean): Iterable[TypeIdentifier] = {
types
.values()
.collect {
case x: ApexDeclaration => x
case x: SObjectDeclaration if !apexOnly => x
}
.map(td => TypeIdentifier(namespace, td.typeName))
}
/* Search for a specific type, this has to be recursive to cope with double nesting of platform types,
* e.g. Component.. */
def moduleType(typeName: TypeName): Option[TypeDeclaration] = {
types
.get(typeName)
.orElse(
typeName.outer
.flatMap(types.get)
.flatMap(_.nestedTypes.find(_.typeName == typeName))
)
}
def replaceType(typeName: TypeName, typeDeclaration: Option[TypeDeclaration]): Unit = {
if (typeDeclaration.nonEmpty) {
val td = typeDeclaration.get
types.put(typeName, td)
// Labels must also be updated against just 'Label' to simulate the defaulting of the System namespace, this is
// a design flaw but I am leaving it for now, if this is skipped Label would resolve against an empty platform
// type.
typeName match {
case TypeNames.Label => types.put(TypeName(labels.name), labels)
case _ => ()
}
} else {
types.remove(typeName).foreach(_.dead = true)
}
}
// Upsert some metadata to the package
def upsertMetadata(td: TypeDeclaration, altTypeName: Option[TypeName] = None): Unit = {
types.put(altTypeName.getOrElse(td.typeName), td)
}
// Remove some metadata from the package
// Future: Support component & flow removal
def removeMetadata(td: TypeDeclaration): Unit = {
removeMetadata(td.typeName)
}
def removeMetadata(typeName: TypeName): Unit = {
types.remove(typeName).foreach(_.dead = true)
}
// Add dependencies for Apex types to a map
def populateDependencies(dependencies: java.util.Map[String, Array[String]]): Unit = {
val typeCache = new TypeCache()
types.values().foreach {
case td: ApexClassDeclaration =>
val depends = mutable.Set[TypeId]()
td.gatherDependencies(depends, apexOnly = false, outerTypesOnly = true, typeCache)
depends.remove(td.typeId)
if (depends.nonEmpty)
dependencies.put(td.typeName.toString, depends.map(_.typeName.toString).toArray)
case _ => ()
}
}
def log(issue: Issue): Unit = {
pkg.org.issueManager.log(issue)
}
def log(path: PathLike, message: String, ex: Throwable): Unit = {
val writer = new StringWriter
writer.append(message)
writer.append(": ")
ex.printStackTrace(new PrintWriter(writer))
log(Issue(ERROR_CATEGORY, PathLocation(path, Location.empty), writer.toString))
}
override def toString: String = {
s"Module(${index.path}}"
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy