com.nawforce.apexlink.org.SObjectDeployer.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.diagnostics.IssueOps
import com.nawforce.apexlink.names.TypeNames.TypeNameUtils
import com.nawforce.apexlink.names.XNames.NameUtils
import com.nawforce.apexlink.names.{TypeNames, XNames}
import com.nawforce.apexlink.org.SObjectDeployer.{feedFieldsFor, historyFieldsFor, shareFieldsFor}
import com.nawforce.apexlink.types.core.{FieldDeclaration, TypeDeclaration}
import com.nawforce.apexlink.types.platform.PlatformTypes
import com.nawforce.apexlink.types.schema.{SObjectNature, _}
import com.nawforce.apexlink.types.synthetic.{
CustomFieldDeclaration,
LocatableCustomFieldDeclaration
}
import com.nawforce.pkgforce.documents._
import com.nawforce.pkgforce.names._
import com.nawforce.pkgforce.stream._
import scala.collection.immutable.ArraySeq
import scala.collection.mutable.ArrayBuffer
import scala.collection.{BufferedIterator, mutable}
/** '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 SObjectDeployer(module: OPM.Module) {
def createSObjects(events: BufferedIterator[PackageEvent]): Array[SObjectLikeDeclaration] = {
val objectsEvents = bufferEvents(
Set(
classOf[SObjectEvent],
classOf[CustomFieldEvent],
classOf[FieldsetEvent],
classOf[SharingReasonEvent]
),
events
).iterator.buffered
val createdSObjects = mutable.Map[TypeName, SObjectLikeDeclaration]()
val derivedFields = ArrayBuffer[(TypeName, CustomFieldEvent)]()
while (objectsEvents.hasNext) {
val sObjectEvent = objectsEvents.next().asInstanceOf[SObjectEvent]
val encodedName = EncodedName(sObjectEvent.name, module.namespace)
val typeName = TypeName(encodedName.fullName, Nil, Some(TypeNames.Schema))
val nature = SObjectNature(sObjectEvent.name, sObjectEvent)
val fieldEvents = bufferEvents[CustomFieldEvent](objectsEvents)
val fieldSetEvents = bufferEvents[FieldsetEvent](objectsEvents)
val sharingReasonEvents = bufferEvents[SharingReasonEvent](objectsEvents)
val fieldSets = fieldSetEvents.map(_.name)
val sharingReasons = sharingReasonEvents.map(_.name)
val sources: Array[SourceInfo] =
sObjectEvent.sourceInfo.toArray ++
fieldEvents.map(_.sourceInfo) ++
fieldSetEvents.map(_.sourceInfo) ++
sharingReasonEvents.map(_.sourceInfo)
val (derived, nonDerived) = fieldEvents.partitionMap {
case e if SObjectDeployer.derivedFieldTypes.contains(e.rawType) =>
Left((typeName, e))
case f => Right(f)
}
derivedFields.addAll(derived)
val fields = nonDerived.flatMap(createCustomField)
val sobjects =
if (encodedName.ext.nonEmpty)
createCustomObject(
sources,
sObjectEvent,
typeName,
nature,
fields,
fieldSets,
sharingReasons
)
else
createReplacementSObject(sources, typeName, nature, fields, fieldSets, sharingReasons)
sobjects.foreach(sobject => createdSObjects.put(sobject.typeName, sobject))
}
addDerivedFieldsToObjects(derivedFields, createdSObjects)
createdSObjects.values.toArray
}
private def createCustomField(field: CustomFieldEvent): Array[FieldDeclaration] = {
val name = defaultNamespace(field.name)
val location = field.sourceInfo.location
val maybeRefType = field.referenceTo.map(to => schemaTypeNameOf(to._1))
val fieldType = SObjectDeployer.platformTypeOfFieldType(field).typeName
val fieldDeclaration = LocatableCustomFieldDeclaration(location, name, fieldType, maybeRefType)
// Create additional fields & lookup relationships for special fields types
field.rawType.value match {
case "Lookup" | "MasterDetail" | "MetadataRelationship" =>
Array(
fieldDeclaration,
LocatableCustomFieldDeclaration(
location,
name.replaceAll("__c$", "__r"),
maybeRefType.get,
None,
asStatic = false,
Some(field.referenceTo.get._2.toString)
)
)
case "Location" =>
Array(
fieldDeclaration,
LocatableCustomFieldDeclaration(
location,
name.replaceAll("__c$", "__latitude__s"),
TypeNames.Double,
None
),
LocatableCustomFieldDeclaration(
location,
name.replaceAll("__c$", "__longitude__s"),
TypeNames.Double,
None
)
)
case _ => Array(fieldDeclaration)
}
}
private def addDerivedFieldsToObjects(
fields: ArrayBuffer[(TypeName, CustomFieldEvent)],
createdSObjects: mutable.Map[TypeName, SObjectLikeDeclaration]
): Unit = {
fields
.groupMap(_._1) {
// group fields by obj name to overwrite each obj only once
case (objName, field) =>
val ghosted = field.relatedField.exists(f => module.isGhostedFieldName(f._2))
val relField =
field.relatedField.flatMap {
case _ if ghosted => None
case f => findRelatedField(f, createdSObjects, fields)
}
createDerivedField(objName, field, relField, ghosted)
}
.foreach(a => addFieldsToSObject(a._1, a._2.toArray, createdSObjects))
}
private def createDerivedField(
objectName: TypeName,
field: CustomFieldEvent,
relatedField: Option[(TypeDeclaration, FieldDeclaration)],
ghostRelated: Boolean
): FieldDeclaration = {
val location = field.sourceInfo.location
val relatedFieldType = relatedField.map(_._2.typeName)
if (relatedField.isEmpty && field.relatedField.nonEmpty && !ghostRelated) {
val (relObj, relField) = field.relatedField.get
OrgInfo.logError(
location,
s"Related field '$relObj.$relField' required by $objectName is not defined"
)
}
LocatableCustomFieldDeclaration(
location,
defaultNamespace(field.name),
relatedFieldType.getOrElse(SObjectDeployer.platformTypeOfFieldType(field).typeName),
idTarget = None,
relationshipName = None,
derivedFrom = relatedField.map(_._1.typeName).toList
)
}
private def findRelatedField(
field: (Name, Name),
createdSObjects: mutable.Map[TypeName, SObjectLikeDeclaration],
derivedFields: ArrayBuffer[(TypeName, CustomFieldEvent)]
): Option[(TypeDeclaration, FieldDeclaration)] = {
val objectTypeName = schemaTypeNameOf(field._1)
val relObj = createdSObjects.get(objectTypeName).orElse(module.moduleType(objectTypeName))
relObj
.flatMap(
_.findField(defaultNamespace(field._2), Some(false)).map(field => (relObj.get, field))
)
.orElse(
// rare case of chained related fields
derivedFields
.find { case (obj, f) =>
obj == objectTypeName && f.name == field._2
}
.flatMap(rf =>
rf._2.relatedField.flatMap(findRelatedField(_, createdSObjects, derivedFields))
)
)
}
private def addFieldsToSObject(
objectName: TypeName,
fields: Array[FieldDeclaration],
createdSObjects: mutable.Map[TypeName, SObjectLikeDeclaration]
): Unit = {
createdSObjects.get(objectName).foreach { case td: SObjectDeclaration =>
val newFields = fields.filter(f => !td.fields.exists(_.name == f.name))
createdSObjects.put(
td.typeName,
extendExistingSObject(
Some(td),
Array(),
td.typeName,
td.sobjectNature,
ArraySeq.unsafeWrapArray(newFields),
ArraySeq(),
ArraySeq()
)
)
}
}
private def schemaTypeNameOf(name: Name): TypeName = {
TypeName(defaultNamespace(name), Nil, Some(TypeNames.Schema))
}
private def createCustomObject(
sources: Array[SourceInfo],
event: SObjectEvent,
typeName: TypeName,
nature: SObjectNature,
fields: ArraySeq[FieldDeclaration],
fieldSets: ArraySeq[Name],
sharingReasons: ArraySeq[Name]
): Array[SObjectDeclaration] = {
if (module.isGhostedType(typeName))
Array(
extendExistingSObject(None, sources, typeName, nature, fields, fieldSets, sharingReasons)
)
else {
val sobjectType = resolveBaseType(typeName)
(event.isDefining, sobjectType.isEmpty) match {
case (true, true) =>
createNewSObject(
sources,
typeName,
nature,
fields,
fieldSets,
sharingReasons,
event.sharingModel
)
case (false, false) =>
createReplacementSObject(sources, typeName, nature, fields, fieldSets, sharingReasons)
case (false, true) =>
// Limit logging to once per file
val paths = sources.map(_.location.path).distinct
paths.foreach(path => {
sources
.find(_.location.path == path)
.foreach(source => {
OrgInfo.log(IssueOps.extendingUnknownSObject(source.location, typeName.name))
})
})
Array.empty
case (true, false) =>
OrgInfo.log(IssueOps.redefiningSObject(sources.head.location, event.name))
createReplacementSObject(sources, typeName, nature, fields, fieldSets, sharingReasons)
}
}
}
/** Create a new SObject along with it's supporting objects as needed. */
private def createNewSObject(
sources: Array[SourceInfo],
typeName: TypeName,
nature: SObjectNature,
fields: ArraySeq[FieldDeclaration],
fieldSets: ArraySeq[Name],
sharingReasons: ArraySeq[Name],
sharingModel: Option[SharingModel]
): Array[SObjectDeclaration] = {
val syntheticSObjects =
if (nature == CustomObjectNature) {
// FUTURE: Check fields & when these should be available
Array(
createShare(sources, typeName, sharingReasons),
createFeed(sources, typeName),
createHistory(sources, typeName)
)
} else {
Array[SObjectDeclaration]()
}
val hasOwner: Boolean = sharingModel match {
case Some(ControlledByParentSharingModel) => false
case Some(ControlledByLeadOrContractSharingModel) => false
case Some(ControlledByCampaignSharingModel) => false
case _ => true
}
syntheticSObjects :+
new SObjectDeclaration(
sources,
module,
typeName,
nature,
fieldSets,
ArraySeq(),
collectFields(typeName, nature, fields, hasOwner)
)
}
private def createShare(
sources: Array[SourceInfo],
typeName: TypeName,
sharingReasons: ArraySeq[Name]
): SObjectDeclaration = {
SObjectDeclaration(
sources,
module,
typeName.withNameReplace("__c$", "__Share"),
CustomObjectNature,
ArraySeq(),
sharingReasons,
shareFieldsFor(typeName)
)
}
private def createFeed(sources: Array[SourceInfo], typeName: TypeName): SObjectDeclaration = {
SObjectDeclaration(
sources,
module,
typeName.withNameReplace("__c$", "__Feed"),
CustomObjectNature,
ArraySeq(),
ArraySeq(),
feedFieldsFor(typeName)
)
}
private def createHistory(sources: Array[SourceInfo], typeName: TypeName): SObjectDeclaration = {
SObjectDeclaration(
sources,
module,
typeName.withNameReplace("__c$", "__History"),
CustomObjectNature,
ArraySeq(),
ArraySeq(),
historyFieldsFor(typeName)
)
}
/** Create an SObject to replace some existing SObject so that it can be extended. If no existing
* can be found then an error is raised and the operation is new SObject is not created.
*/
private def createReplacementSObject(
sources: Array[SourceInfo],
typeName: TypeName,
nature: SObjectNature,
fields: ArraySeq[FieldDeclaration],
fieldSets: ArraySeq[Name],
sharingReasons: ArraySeq[Name]
): Array[SObjectDeclaration] = {
if (typeName == TypeNames.Activity) {
// Fake Activity as applying to Task & Event, how bizarre is that
createReplacementSObject(
sources,
TypeNames.Task,
nature,
fields,
fieldSets,
sharingReasons
) ++
createReplacementSObject(
sources,
TypeNames.Event,
nature,
fields,
fieldSets,
sharingReasons
)
} else {
val sobjectType = resolveBaseType(typeName)
if (
sobjectType.isEmpty || !sobjectType.get.superClassDeclaration
.exists(superClass => superClass.typeName == TypeNames.SObject)
) {
sources.headOption.foreach(source => {
OrgInfo.logError(source.location, s"No SObject declaration found for '$typeName'")
})
return Array()
}
Array(
extendExistingSObject(
sobjectType,
sources,
typeName,
nature,
fields,
fieldSets,
sharingReasons
)
)
}
}
/** Search for a type in dependent modules. Note. this skips the SObject handling in TypeFinder
* deliberately.
*/
private def resolveBaseType(typeName: TypeName): Option[TypeDeclaration] = {
val td = module.baseModules.headOption.flatMap(_.findType(typeName).toOption)
if (td.nonEmpty)
return td
val pkgTd = module.basePackages.headOption.flatMap(basePkg =>
basePkg.modules.headOption.flatMap(_.findType(typeName).toOption)
)
if (pkgTd.nonEmpty)
return pkgTd
PlatformTypes.get(typeName, None).toOption
}
/** Create an SObject by extending a base SObject with new fields, fieldSets and sharing reasons.
* If you don't pass a base object than this will extend System.SObject.
*/
def extendExistingSObject(
base: Option[TypeDeclaration],
sources: Array[SourceInfo],
typeName: TypeName,
nature: SObjectNature,
fields: ArraySeq[FieldDeclaration],
fieldSets: ArraySeq[Name],
sharingReasons: ArraySeq[Name]
): SObjectDeclaration = {
// FUTURE: Add type for platform sobjects so we don't need this hackery
def baseAsSObject: Option[SObjectDeclaration] = {
if (base.nonEmpty && base.get.isInstanceOf[SObjectDeclaration])
base.map(_.asInstanceOf[SObjectDeclaration])
else {
None
}
}
// Prefer base type nature, more accurate e.g. custom setting
val definedNature = baseAsSObject.map(_.sobjectNature).getOrElse(nature)
// Check we are not trying to extend a non-extensible over a namespace boundary
val crossModule = base.flatMap(_.moduleDeclaration).exists(_ != module)
if (crossModule && !definedNature.extendable) {
val baseNS = base.flatMap(_.moduleDeclaration).flatMap(_.namespace)
if (baseNS != module.namespace) {
OrgInfo.log(
IssueOps.extendingOverNamespace(
sources.head.location,
definedNature,
baseNS.getOrElse(Names.Empty),
module.namespace.getOrElse(Names.Empty)
)
)
}
}
val extend = base.getOrElse(PlatformTypes.sObjectType)
val combinedSources = baseAsSObject.map(_.sources).getOrElse(Array()) ++ sources
val combinedFields =
collectFields(typeName, definedNature, extend.fields ++ fields, hasOwner = base.isEmpty)
val combinedFieldsets = ArraySeq.unsafeWrapArray(
fieldSets
.foldLeft(baseAsSObject.map(_.fieldSets).getOrElse(ArraySeq()).toSet)((acc, fieldset) =>
acc + fieldset
)
.toArray
)
val combinedSharingReasons = ArraySeq.unsafeWrapArray(
sharingReasons
.foldLeft(baseAsSObject.map(_.sharingReasons).getOrElse(ArraySeq()).toSet)(
(acc, sharingReason) => acc + sharingReason
)
.toArray
)
val newSObject = new SObjectDeclaration(
combinedSources,
module,
typeName,
definedNature,
combinedFieldsets,
combinedSharingReasons,
combinedFields,
isComplete = baseAsSObject.nonEmpty,
if (crossModule) baseAsSObject else None
)
// If we are extending over a module boundary then link via dependencies for refresh handling
if (crossModule) {
newSObject.addDependency(base.get)
}
newSObject
}
/** Collect fields for an SObject by folding custom fields over standard fields. */
def collectFields(
typeName: TypeName,
nature: SObjectNature,
fields: ArraySeq[FieldDeclaration],
hasOwner: Boolean
): ArraySeq[FieldDeclaration] = {
nature match {
case ListCustomSettingNature => customObjectFields(typeName, nature, fields, hasOwner)
case HierarchyCustomSettingsNature => customObjectFields(typeName, nature, fields, hasOwner)
case CustomObjectNature => customObjectFields(typeName, nature, fields, hasOwner)
case CustomMetadataNature => customMetadataFields(typeName, fields)
case BigObjectNature => fields
case PlatformEventNature => platformEventFields(typeName, fields)
case PlatformObjectNature => fields
}
}
/** Construct a full set of fields for a custom objects from the custom fields defined in the
* event.
*/
private def customObjectFields(
typeName: TypeName,
nature: SObjectNature,
fields: ArraySeq[FieldDeclaration],
hasOwner: Boolean
): ArraySeq[FieldDeclaration] = {
def fieldFilter(field: FieldDeclaration): Boolean = {
if (hasOwner)
false
else
field.name.value == "OwnerId" || field.name.value == "Owner"
}
deDuplicateFields(
ArraySeq(
CustomFieldDeclaration(
Names.SObjectType,
TypeNames.sObjectType$(typeName),
None,
asStatic = true
),
CustomFieldDeclaration(
Names.Fields,
TypeNames.sObjectFields$(typeName),
None,
asStatic = true
),
CustomFieldDeclaration(Names.Id, TypeNames.IdType, Some(typeName))
) ++
SObjectDeployer.standardCustomObjectFields.filterNot(fieldFilter) ++
fields ++
(if (nature == HierarchyCustomSettingsNature)
ArraySeq(CustomFieldDeclaration(Names.SetupOwnerId, PlatformTypes.idType.typeName, None))
else
ArraySeq[FieldDeclaration]())
)
}
/** Construct a full set of fields for a custom metadata from the custom fields defined in the
* event.
*/
private def customMetadataFields(
typeName: TypeName,
fields: ArraySeq[FieldDeclaration]
): ArraySeq[FieldDeclaration] = {
deDuplicateFields(
ArraySeq(
CustomFieldDeclaration(Names.Id, TypeNames.IdType, Some(typeName)),
CustomFieldDeclaration(
Names.SObjectType,
TypeNames.sObjectType$(typeName),
None,
asStatic = true
)
) ++
SObjectDeployer.standardCustomMetadataFields ++
fields
)
}
/** Construct a full set of fields for a platform event from the custom fields defined in the
* event.
*/
private def platformEventFields(
typeName: TypeName,
fields: ArraySeq[FieldDeclaration]
): ArraySeq[FieldDeclaration] = {
deDuplicateFields(
SObjectDeployer.standardPlatformEventFields ++
fields :+
CustomFieldDeclaration(
Names.SObjectType,
TypeNames.sObjectType$(typeName),
None,
asStatic = true
)
)
}
private def deDuplicateFields(fields: ArraySeq[FieldDeclaration]): ArraySeq[FieldDeclaration] = {
// FUTURE: Duplicates are likely OK due to deploy handling, but we should warn?
ArraySeq.unsafeWrapArray(fields.map(field => (field.name, field)).toMap.values.toArray)
}
private def defaultNamespace(name: Name): Name = {
EncodedName(name, module.namespace).fullName
}
}
object SObjectDeployer {
/** Standard fields for custom objects, this is a superset, filtering may be needed to trim do to
* available.
*/
val standardCustomObjectFields: ArraySeq[FieldDeclaration] = {
PlatformTypes.sObjectType.fields ++
ArraySeq(
CustomFieldDeclaration(Names.Id, TypeNames.IdType, None),
CustomFieldDeclaration(Names.NameName, TypeNames.String, None),
CustomFieldDeclaration(XNames.RecordTypeId, TypeNames.IdType, None),
CustomFieldDeclaration(XNames.RecordType, TypeNames.RecordType, None),
CustomFieldDeclaration(XNames.OwnerId, TypeNames.IdType, None),
CustomFieldDeclaration(XNames.Owner, TypeNames.NameSObject, None),
CustomFieldDeclaration(XNames.CurrencyIsoCode, TypeNames.String, None),
CustomFieldDeclaration(XNames.CreatedBy, TypeNames.NameSObject, None),
CustomFieldDeclaration(XNames.CreatedById, TypeNames.IdType, None),
CustomFieldDeclaration(XNames.CreatedDate, TypeNames.Datetime, None),
CustomFieldDeclaration(XNames.LastModifiedBy, TypeNames.User, None),
CustomFieldDeclaration(XNames.LastModifiedById, TypeNames.IdType, None),
CustomFieldDeclaration(XNames.LastModifiedDate, TypeNames.Datetime, None),
CustomFieldDeclaration(XNames.LastReferencedDate, TypeNames.Datetime, None),
CustomFieldDeclaration(XNames.LastViewedDate, TypeNames.Datetime, None),
CustomFieldDeclaration(XNames.LastActivityDate, TypeNames.Datetime, None),
CustomFieldDeclaration(XNames.Tasks, TypeNames.listOf(TypeNames.Task), None),
CustomFieldDeclaration(XNames.Notes, TypeNames.listOf(TypeNames.Note), None),
CustomFieldDeclaration(
XNames.NotesAndAttachments,
TypeNames.listOf(TypeNames.NoteAndAttachment),
None
),
CustomFieldDeclaration(XNames.Attachments, TypeNames.listOf(TypeNames.Attachment), None),
CustomFieldDeclaration(
XNames.ContentDocumentLinks,
TypeNames.listOf(TypeNames.ContentDocumentLink),
None
),
CustomFieldDeclaration(
XNames.ProcessSteps,
TypeNames.listOf(TypeNames.ProcessInstanceHistory),
None
),
CustomFieldDeclaration(XNames.IsDeleted, TypeNames.Boolean, None),
CustomFieldDeclaration(XNames.SystemModstamp, TypeNames.Datetime, None)
)
}
/** Standard fields for platform events. */
private val standardPlatformEventFields: ArraySeq[FieldDeclaration] = {
ArraySeq(
CustomFieldDeclaration(Names.ReplayId, TypeNames.String, None),
CustomFieldDeclaration(Names.EventUuid, TypeNames.String, None),
CustomFieldDeclaration(XNames.CreatedBy, TypeNames.User, None),
CustomFieldDeclaration(XNames.CreatedById, TypeNames.IdType, None),
CustomFieldDeclaration(XNames.CreatedDate, TypeNames.Datetime, None)
)
}
/** Standard fields for custom metadata. */
private val standardCustomMetadataFields: ArraySeq[FieldDeclaration] = {
ArraySeq(
CustomFieldDeclaration(Names.DeveloperName, TypeNames.String, None),
CustomFieldDeclaration(Names.IsProtected, TypeNames.Boolean, None),
CustomFieldDeclaration(Names.Label, TypeNames.String, None),
CustomFieldDeclaration(Names.Language, TypeNames.String, None),
CustomFieldDeclaration(Names.MasterLabel, TypeNames.String, None),
CustomFieldDeclaration(Names.NamespacePrefix, TypeNames.String, None),
CustomFieldDeclaration(Names.QualifiedAPIName, TypeNames.String, None)
)
}
/** Standard fields for a \_\_Share SObject. */
def shareFieldsFor(typeName: TypeName): ArraySeq[FieldDeclaration] = {
shareFields ++ commonFieldsFor(typeName)
}
private val shareFields: ArraySeq[FieldDeclaration] =
PlatformTypes.sObjectType.fields ++ ArraySeq(
CustomFieldDeclaration(XNames.IsDeleted, TypeNames.Boolean, None),
CustomFieldDeclaration(XNames.LastModifiedBy, TypeNames.User, None),
CustomFieldDeclaration(XNames.LastModifiedById, TypeNames.IdType, None),
CustomFieldDeclaration(XNames.LastModifiedDate, TypeNames.Datetime, None),
CustomFieldDeclaration(Names.AccessLevel, PlatformTypes.stringType.typeName, None),
CustomFieldDeclaration(Names.ParentId, PlatformTypes.idType.typeName, None),
CustomFieldDeclaration(Names.RowCause, PlatformTypes.stringType.typeName, None),
CustomFieldDeclaration(Names.UserOrGroupId, PlatformTypes.idType.typeName, None)
)
def feedFieldsFor(typeName: TypeName): ArraySeq[FieldDeclaration] = {
feedFields ++ commonFieldsFor(typeName)
}
private def commonFieldsFor(typeName: TypeName): ArraySeq[FieldDeclaration] = {
ArraySeq(
CustomFieldDeclaration(
Names.SObjectType,
TypeNames.sObjectType$(typeName),
None,
asStatic = true
),
CustomFieldDeclaration(
Names.Fields,
TypeNames.sObjectFields$(typeName),
None,
asStatic = true
)
)
}
/** Standard fields for a \_\_Feed SObject. */
private val feedFields: ArraySeq[FieldDeclaration] = PlatformTypes.sObjectType.fields ++ ArraySeq(
CustomFieldDeclaration(XNames.BestCommentId, TypeNames.IdType, None),
CustomFieldDeclaration(XNames.Body, TypeNames.String, None),
CustomFieldDeclaration(XNames.CommentCount, TypeNames.Decimal, None),
CustomFieldDeclaration(Names.ConnectionId, TypeNames.IdType, None),
CustomFieldDeclaration(XNames.CreatedBy, TypeNames.NameSObject, None),
CustomFieldDeclaration(XNames.CreatedById, TypeNames.IdType, None),
CustomFieldDeclaration(XNames.CreatedDate, TypeNames.Datetime, None),
CustomFieldDeclaration(XNames.InsertedById, TypeNames.IdType, None),
CustomFieldDeclaration(XNames.InsertedBy, TypeNames.NameSObject, None),
CustomFieldDeclaration(XNames.IsDeleted, TypeNames.Boolean, None),
CustomFieldDeclaration(XNames.IsRichText, TypeNames.Boolean, None),
CustomFieldDeclaration(XNames.LastModifiedBy, TypeNames.User, None),
CustomFieldDeclaration(XNames.LastModifiedById, TypeNames.IdType, None),
CustomFieldDeclaration(XNames.LastModifiedDate, TypeNames.Datetime, None),
CustomFieldDeclaration(XNames.LikeCount, TypeNames.Decimal, None),
CustomFieldDeclaration(XNames.LinkUrl, TypeNames.String, None),
CustomFieldDeclaration(XNames.NetworkScope, TypeNames.String, None),
CustomFieldDeclaration(XNames.ParentId, TypeNames.IdType, None),
CustomFieldDeclaration(XNames.RelatedRecordId, TypeNames.IdType, None),
CustomFieldDeclaration(XNames.RelatedRecord, TypeNames.SObject, None),
CustomFieldDeclaration(XNames.Title, TypeNames.String, None),
CustomFieldDeclaration(XNames.Type, TypeNames.String, None),
CustomFieldDeclaration(XNames.Visibility, TypeNames.String, None)
)
def historyFieldsFor(typeName: TypeName): ArraySeq[FieldDeclaration] =
historyFields ++ Array(
CustomFieldDeclaration(
Names.SObjectType,
TypeNames.sObjectType$(typeName),
None,
asStatic = true
),
CustomFieldDeclaration(
Names.Fields,
TypeNames.sObjectFields$(typeName),
None,
asStatic = true
)
)
private val historyFields: ArraySeq[FieldDeclaration] =
PlatformTypes.sObjectType.fields ++ ArraySeq(
CustomFieldDeclaration(XNames.CreatedBy, TypeNames.NameSObject, None),
CustomFieldDeclaration(XNames.CreatedById, TypeNames.IdType, None),
CustomFieldDeclaration(XNames.CreatedDate, TypeNames.Datetime, None),
CustomFieldDeclaration(XNames.DataType, TypeNames.String, None),
CustomFieldDeclaration(XNames.Field, TypeNames.String, None),
CustomFieldDeclaration(XNames.IsDeleted, TypeNames.Boolean, None),
CustomFieldDeclaration(XNames.NewValue, TypeNames.InternalObject, None),
CustomFieldDeclaration(XNames.OldValue, TypeNames.InternalObject, None),
CustomFieldDeclaration(XNames.ParentId, TypeNames.IdType, None)
)
private val derivedFieldTypes: Set[Name] =
Set(Name("Summary"))
/** Convert a field type string to the platform type used for it in Apex. */
private def platformTypeOfFieldType(field: CustomFieldEvent): TypeDeclaration = {
field.rawType.value match {
case "MasterDetail" => PlatformTypes.idType
case "Lookup" => PlatformTypes.idType
case "AutoNumber" => PlatformTypes.stringType
case "Checkbox" => PlatformTypes.booleanType
case "Currency" => PlatformTypes.decimalType
case "Date" => PlatformTypes.dateType
case "DateTime" => PlatformTypes.datetimeType
case "Email" => PlatformTypes.stringType
case "EncryptedText" => PlatformTypes.stringType
case "Number" => PlatformTypes.decimalType
case "Percent" => PlatformTypes.decimalType
case "Phone" => PlatformTypes.stringType
case "Picklist" => PlatformTypes.stringType
case "MultiselectPicklist" => PlatformTypes.stringType
case "Summary" => PlatformTypes.decimalType
case "Text" => PlatformTypes.stringType
case "TextArea" => PlatformTypes.stringType
case "LongTextArea" => PlatformTypes.stringType
case "Url" => PlatformTypes.stringType
case "File" => PlatformTypes.stringType
case "Location" => PlatformTypes.locationType
case "Time" => PlatformTypes.timeType
case "Html" => PlatformTypes.stringType
case "MetadataRelationship"
if field.referenceTo
.map(_._1.value)
.exists(to => to == "FieldDefinition" || to == "EntityDefinition") =>
PlatformTypes.stringType
case "MetadataRelationship" => PlatformTypes.idType
// pkgforce validates on loading, so need for default handling here
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy